mirror of
https://github.com/KarolS/millfork.git
synced 2025-01-10 20:29:35 +00:00
Preliminary segment support; C16/+4 fixes
This commit is contained in:
parent
9e8a125487
commit
8a347e5058
@ -4,6 +4,8 @@
|
|||||||
|
|
||||||
* **Breaking change!** Renamed `inline` to `macro`.
|
* **Breaking change!** Renamed `inline` to `macro`.
|
||||||
|
|
||||||
|
* **Breaking change!** Added support for memory segments. Changed the platform definition file syntax.
|
||||||
|
|
||||||
* Added preliminary support for 65CE02, HuC6280 and 65816 processors.
|
* Added preliminary support for 65CE02, HuC6280 and 65816 processors.
|
||||||
|
|
||||||
* Added new `-O1` optimization preset; old `-O1` became `-O2`, old `-O2` became `-O3` and so on.
|
* Added new `-O1` optimization preset; old `-O1` became `-O2`, old `-O2` became `-O3` and so on.
|
||||||
|
@ -96,30 +96,49 @@ Every platform is defined in an `.ini` file with an appropriate name.
|
|||||||
|
|
||||||
#### `[allocation]` section
|
#### `[allocation]` section
|
||||||
|
|
||||||
* `main_org` – the address for the `main` function; all the other functions will be placed after it
|
|
||||||
|
|
||||||
* `zp_pointers` – either a list of comma separated zeropage addresses that can be used by the program as zeropage pointers, or `all` for all. Each value should be the address of the first of two free bytes in the zeropage.
|
* `zp_pointers` – either a list of comma separated zeropage addresses that can be used by the program as zeropage pointers, or `all` for all. Each value should be the address of the first of two free bytes in the zeropage.
|
||||||
|
|
||||||
* `himem_style` – not yet supported
|
* `segments` – a comma-separated list of segment names.
|
||||||
|
A segment named `default` is always required.
|
||||||
|
Default: `default`. In all options below, `NAME` refers to a segment name.
|
||||||
|
|
||||||
* `himem_start` – the first address used for non-zeropage variables, or `after_code` if the variables should be allocated after the code
|
* `default_code_segment` – the default segment for code and initialized arrays.
|
||||||
|
Note that the default segment for uninitialized arrays and variables is always `default`.
|
||||||
|
Default: `default`
|
||||||
|
|
||||||
* `himem_end` – the last address available for non-zeropage variables
|
* `segment_NAME_start` – the first address used for automatic allocation in the segment.
|
||||||
|
Note that the `default` segment shouldn't start before $200, as the $0-$1FF range is reserved for the zeropage and the stack.
|
||||||
|
The `main` function will be placed as close to the beginning of its segment as possible, but not necessarily at `segment_NAME_start`
|
||||||
|
|
||||||
|
* `segment_NAME_end` – the last address in the segment
|
||||||
|
|
||||||
|
* `segment_NAME_codeend` – the last address in the segment for code and initialized arrays.
|
||||||
|
Only uninitialized variables are allowed between `segment_NAME_codeend` and `segment_NAME_end`.
|
||||||
|
Default: the same as `segment_NAME_end`.
|
||||||
|
|
||||||
|
* `segment_NAME_datastart` – the first address used for non-zeropage variables, or `after_code` if the variables should be allocated after the code.
|
||||||
|
Default: `after_code`.
|
||||||
|
|
||||||
#### `[output]` section
|
#### `[output]` section
|
||||||
|
|
||||||
* `style` – not yet supported
|
* `style` – how multi-segment programs should be output:
|
||||||
|
|
||||||
|
* `single` – output a single file, based mostly, but not necessarily only on data in the `default` segment (the default)
|
||||||
|
|
||||||
|
* `per_segment` – generate a separate file with each segment
|
||||||
|
|
||||||
* `format` – output file format; a comma-separated list of tokens:
|
* `format` – output file format; a comma-separated list of tokens:
|
||||||
|
|
||||||
* literal byte values
|
* literal byte values
|
||||||
|
|
||||||
* `startaddr` – little-endian 16-bit address of the first used byte of the compiled output
|
* `startaddr` – little-endian 16-bit address of the first used byte of the compiled output (not necessarily the segment start)
|
||||||
|
|
||||||
* `endaddr` – little-endian 16-bit address of the last used byte of the compiled output
|
* `endaddr` – little-endian 16-bit address of the last used byte of the compiled output (usually not the segment end)
|
||||||
|
|
||||||
* `allocated` – all used bytes
|
* `allocated` – all used bytes
|
||||||
|
|
||||||
* `<addr>:<addr>` - inclusive range of bytes
|
* `<addr>:<addr>` - inclusive range of bytes
|
||||||
|
|
||||||
|
* `<segment>:<addr>:<addr>` - inclusive range of bytes in a given segment
|
||||||
|
|
||||||
* `extension` – target file extension, with or without the dot
|
* `extension` – target file extension, with or without the dot
|
@ -2,9 +2,11 @@
|
|||||||
|
|
||||||
Syntax:
|
Syntax:
|
||||||
|
|
||||||
`[<modifiers>] <return_type> <name> ( <params> ) [@ <address>] { <body> }`
|
`[segment (<segment>)] [<modifiers>] <return_type> <name> ( <params> ) [@ <address>] { <body> }`
|
||||||
|
|
||||||
`asm <return_type> <name> ( <params> ) @ <address> extern`
|
`[segment (<segment>)] asm <return_type> <name> ( <params> ) @ <address> extern`
|
||||||
|
|
||||||
|
* `<segment>`: segment name; if absent, then defaults to `default_code_segment` as defined for the platform
|
||||||
|
|
||||||
* `<modifiers>`: zero or more of the following:
|
* `<modifiers>`: zero or more of the following:
|
||||||
|
|
||||||
|
@ -18,7 +18,9 @@ or a top level of a function (*local* variables).
|
|||||||
|
|
||||||
Syntax:
|
Syntax:
|
||||||
|
|
||||||
`[<storage>] <type> <name> [@<address>] [= <initial_value>]`
|
`[segment(<segment>)] [<storage>] <type> <name> [@<address>] [= <initial_value>]`
|
||||||
|
|
||||||
|
* `<segment>`: segment name; if absent, then defaults to `default`.
|
||||||
|
|
||||||
* `<storage>` can be only specified for local variables. It can be either `stack`, `static`, `register` or nothing.
|
* `<storage>` can be only specified for local variables. It can be either `stack`, `static`, `register` or nothing.
|
||||||
`register` is only a hint for the optimizer.
|
`register` is only a hint for the optimizer.
|
||||||
@ -44,7 +46,11 @@ An array is a continuous sequence of bytes in memory.
|
|||||||
|
|
||||||
Syntax:
|
Syntax:
|
||||||
|
|
||||||
`array <name> [[<size>]] [@<address>] [= <initial_values>]`
|
`[segment(<segment>)] array <name> [[<size>]] [@<address>] [= <initial_values>]`
|
||||||
|
|
||||||
|
* `<segment>`: segment name; if absent,
|
||||||
|
then defaults to `default_code_segment` as defined for the platform if the array has initial values,
|
||||||
|
or to `default` if it doesn't.
|
||||||
|
|
||||||
TODO
|
TODO
|
||||||
|
|
||||||
|
@ -4,18 +4,15 @@ modules=a8_kernel,default_panic
|
|||||||
|
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
main_org=$2000
|
|
||||||
; TODO
|
; TODO
|
||||||
zp_pointers=$80,$82,$84,$86,$88,$8a,$8c,$8e,$90,$92,$94,$96,$98,$9a,$9c,$9e,$a0,$a2,$a4
|
zp_pointers=$80,$82,$84,$86,$88,$8a,$8c,$8e,$90,$92,$94,$96,$98,$9a,$9c,$9e,$a0,$a2,$a4
|
||||||
;TODO
|
segment_default_start=$2000
|
||||||
himem_style=per_bank
|
; TODO
|
||||||
himem_start=after_code
|
segment_default_end=$3fff
|
||||||
;TODO
|
|
||||||
himem_end=$3FFF
|
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
;TODO
|
;TODO
|
||||||
style=per_bank
|
style=single
|
||||||
format=$FF,$FF,$E0,$02,$E1,$02,startaddr,startaddr,endaddr,allocated
|
format=$FF,$FF,$E0,$02,$E1,$02,startaddr,startaddr,endaddr,allocated
|
||||||
extension=xex
|
extension=xex
|
||||||
|
|
||||||
|
@ -4,17 +4,14 @@ modules=apple2_kernel,default_panic
|
|||||||
|
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
main_org=$0C00
|
|
||||||
; TODO
|
; TODO
|
||||||
zp_pointers=6,8,$EB,$ED,$FA,$FC
|
zp_pointers=6,8,$EB,$ED,$FA,$FC
|
||||||
;TODO
|
segment_default_start=$0C00
|
||||||
himem_style=per_bank
|
segment_default_end=$95FF
|
||||||
himem_start=after_code
|
|
||||||
himem_end=$95FF
|
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
;TODO
|
;TODO
|
||||||
style=per_bank
|
style=single
|
||||||
format=allocated
|
format=allocated
|
||||||
extension=a2
|
extension=a2
|
||||||
|
|
||||||
|
@ -4,16 +4,13 @@ modules=c128_hardware,loader_1c01,c128_kernal,default_panic
|
|||||||
|
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
main_org=$1C0D
|
|
||||||
; TODO
|
; TODO
|
||||||
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$45,$47,$4B
|
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$45,$47,$4B
|
||||||
himem_style=per_bank
|
segment_default_start=$1C0D
|
||||||
himem_start=after_code
|
segment_default_end=$FEFF
|
||||||
; TODO
|
|
||||||
himem_end=$FEFF
|
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
style=per_bank
|
style=single
|
||||||
format=startaddr,allocated
|
format=startaddr,allocated
|
||||||
extension=prg
|
extension=prg
|
||||||
|
|
||||||
|
@ -4,15 +4,13 @@ modules=loader_1001,c264_kernal,c264_hardware,default_panic
|
|||||||
|
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
main_org=$100D
|
|
||||||
; TODO
|
; TODO
|
||||||
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
||||||
himem_style=per_bank
|
segment_default_start=$100D
|
||||||
himem_start=after_code
|
segment_default_end=$3FFF
|
||||||
himem_end=$3FFF
|
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
style=per_bank
|
style=single
|
||||||
format=startaddr,allocated
|
format=startaddr,allocated
|
||||||
extension=prg
|
extension=prg
|
||||||
|
|
||||||
|
@ -1 +1 @@
|
|||||||
import c16_ted
|
import c264_ted
|
@ -13,8 +13,8 @@ const byte light_red = $32
|
|||||||
const byte dark_grey = $21
|
const byte dark_grey = $21
|
||||||
const byte dark_gray = $21
|
const byte dark_gray = $21
|
||||||
const byte medium_grey = $31
|
const byte medium_grey = $31
|
||||||
const byte medium gray = $31
|
const byte medium_gray = $31
|
||||||
const byte light_green = $55
|
const byte light_green = $55
|
||||||
const byte light_blue = $36
|
const byte light_blue = $36
|
||||||
const byte light_grey = $41
|
const byte light_grey = $41
|
||||||
const byte light_gray = $41
|
const byte light_gray = $41
|
||||||
|
@ -7,22 +7,23 @@ arch=nmos
|
|||||||
; modules to load
|
; modules to load
|
||||||
modules=c64_hardware,loader_0801,c64_kernal,c64_panic,stdlib
|
modules=c64_hardware,loader_0801,c64_kernal,c64_panic,stdlib
|
||||||
; optionally: default flags
|
; optionally: default flags
|
||||||
emit_illegals=false
|
emit_illegals=true
|
||||||
|
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
; where the main function should be allocated, also the start of bank 0
|
|
||||||
main_org=$80D
|
|
||||||
; list of free zp pointer locations (these assume that some BASIC routines will keep working)
|
; list of free zp pointer locations (these assume that some BASIC routines will keep working)
|
||||||
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$45,$47,$4B
|
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$45,$47,$4B
|
||||||
; where to allocate non-zp variables
|
segments=default
|
||||||
himem_style=per_bank
|
default_code_segment=default
|
||||||
himem_start=after_code
|
segment_default_start=$80D
|
||||||
himem_end=$9FFF
|
segment_default_codeend=$9fff
|
||||||
|
segment_default_datastart=after_code
|
||||||
|
segment_default_end=$cfff
|
||||||
|
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
; how the banks are laid out in the output files; so far, there is no bank support in the compiler yet
|
; how the banks are laid out in the output files; so far, there is no bank support in the compiler yet
|
||||||
style=per_bank
|
style=single
|
||||||
; output file format
|
; output file format
|
||||||
; startaddr - little-endian address of the first used byte in the bank
|
; startaddr - little-endian address of the first used byte in the bank
|
||||||
; endaddr - little-endian address of the last used byte in the bank
|
; endaddr - little-endian address of the last used byte in the bank
|
||||||
|
@ -7,14 +7,13 @@ modules=c64_hardware,loader_0801,c64_kernal,c64_panic,stdlib
|
|||||||
emit_65816=emulation
|
emit_65816=emulation
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
main_org=$80D
|
|
||||||
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$45,$47,$4B
|
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$45,$47,$4B
|
||||||
himem_style=per_bank
|
segment_default_start=$80D
|
||||||
himem_start=after_code
|
segment_default_codeend=$9fff
|
||||||
himem_end=$9FFF
|
segment_default_end=$cfff
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
style=per_bank
|
style=single
|
||||||
format=startaddr,allocated
|
format=startaddr,allocated
|
||||||
extension=prg
|
extension=prg
|
||||||
|
|
||||||
|
@ -10,12 +10,12 @@ emit_65816=native
|
|||||||
[allocation]
|
[allocation]
|
||||||
main_org=$811
|
main_org=$811
|
||||||
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$45,$47,$4B
|
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$45,$47,$4B
|
||||||
himem_style=per_bank
|
segment_default_start=$80D
|
||||||
himem_start=after_code
|
segment_default_codeend=$9fff
|
||||||
himem_end=$9FFF
|
segment_default_end=$cfff
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
style=per_bank
|
style=single
|
||||||
format=startaddr,allocated
|
format=startaddr,allocated
|
||||||
extension=prg
|
extension=prg
|
||||||
|
|
||||||
|
@ -4,15 +4,13 @@ modules=loader_0401,pet_kernal,default_panic
|
|||||||
|
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
main_org=$40D
|
|
||||||
; TODO
|
; TODO
|
||||||
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
||||||
himem_style=per_bank
|
segment_default_start=$40D
|
||||||
himem_start=after_code
|
segment_default_end=$FFF
|
||||||
himem_end=$FFF
|
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
style=per_bank
|
style=single
|
||||||
format=startaddr,allocated
|
format=startaddr,allocated
|
||||||
extension=prg
|
extension=prg
|
||||||
|
|
||||||
|
@ -1,15 +1,13 @@
|
|||||||
[compilation]
|
[compilation]
|
||||||
arch=nmos
|
arch=nmos
|
||||||
modules=c264_loader,c264_kernal,c264_hardware,default_panic
|
modules=loader_1001,c264_kernal,c264_hardware,default_panic
|
||||||
|
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
main_org=$100D
|
|
||||||
; TODO
|
; TODO
|
||||||
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
||||||
himem_style=per_bank
|
segment_default_start=$100D
|
||||||
himem_start=after_code
|
segment_default_end=$3FFF
|
||||||
himem_end=$7FFF
|
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
style=per_bank
|
style=per_bank
|
||||||
|
@ -4,15 +4,13 @@ modules=loader_1001,vic20_kernal,default_panic
|
|||||||
|
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
main_org=$100D
|
|
||||||
; TODO
|
; TODO
|
||||||
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
||||||
himem_style=per_bank
|
segment_default_start=$100D
|
||||||
himem_start=after_code
|
segment_default_end=$1CFF
|
||||||
himem_end=$1CFF
|
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
style=per_bank
|
style=single
|
||||||
format=startaddr,allocated
|
format=startaddr,allocated
|
||||||
extension=prg
|
extension=prg
|
||||||
|
|
||||||
|
@ -4,15 +4,13 @@ modules=loader_0401,vic20_kernal,default_panic
|
|||||||
|
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
main_org=$40D
|
|
||||||
; TODO
|
; TODO
|
||||||
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
||||||
himem_style=per_bank
|
segment_default_start=$40D
|
||||||
himem_start=after_code
|
segment_default_end=$1CFF
|
||||||
himem_end=$1CFF
|
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
style=per_bank
|
style=single
|
||||||
format=startaddr,allocated
|
format=startaddr,allocated
|
||||||
extension=prg
|
extension=prg
|
||||||
|
|
||||||
|
@ -4,15 +4,13 @@ modules=loader_1201,vic20_kernal,default_panic
|
|||||||
|
|
||||||
|
|
||||||
[allocation]
|
[allocation]
|
||||||
main_org=$120D
|
|
||||||
; TODO
|
; TODO
|
||||||
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
zp_pointers=$C1,$C3,$FB,$FD,$39,$3B,$3D,$43,$4B
|
||||||
himem_style=per_bank
|
segment_default_start=$120D
|
||||||
himem_start=after_code
|
segment_default_end=$1FFF
|
||||||
himem_end=$1FFF
|
|
||||||
|
|
||||||
[output]
|
[output]
|
||||||
style=per_bank
|
style=single
|
||||||
format=startaddr,allocated
|
format=startaddr,allocated
|
||||||
extension=prg
|
extension=prg
|
||||||
|
|
||||||
|
@ -73,7 +73,16 @@ object Main {
|
|||||||
val output = c.outputFileName.getOrElse("a")
|
val output = c.outputFileName.getOrElse("a")
|
||||||
val assOutput = output + ".asm"
|
val assOutput = output + ".asm"
|
||||||
val labelOutput = output + ".lbl"
|
val labelOutput = output + ".lbl"
|
||||||
val prgOutput = if (!output.endsWith(platform.fileExtension)) output + platform.fileExtension else output
|
// val prgOutputs = (platform.outputStyle match {
|
||||||
|
// case OutputStyle.Single => List("default")
|
||||||
|
// case OutputStyle.PerBank => platform.bankNumbers.keys.toList
|
||||||
|
// }).map(bankName => bankName -> {
|
||||||
|
// if (bankName == "default") {
|
||||||
|
// if (output.endsWith(platform.fileExtension)) output else output + platform.fileExtension
|
||||||
|
// } else {
|
||||||
|
// s"${output.stripSuffix(platform.fileExtension)}.$bankName${platform.fileExtension}"
|
||||||
|
// }
|
||||||
|
// }).toMap
|
||||||
|
|
||||||
val unoptimized = new SourceLoadingQueue(
|
val unoptimized = new SourceLoadingQueue(
|
||||||
initialFilenames = c.inputFileNames,
|
initialFilenames = c.inputFileNames,
|
||||||
@ -114,7 +123,7 @@ object Main {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// compile
|
// compile
|
||||||
val assembler = new Assembler(program, env)
|
val assembler = new Assembler(program, env, platform)
|
||||||
val result = assembler.assemble(callGraph, assemblyOptimizations, options)
|
val result = assembler.assemble(callGraph, assemblyOptimizations, options)
|
||||||
ErrorReporting.assertNoErrors("Codegen failed")
|
ErrorReporting.assertNoErrors("Codegen failed")
|
||||||
ErrorReporting.debug(f"Unoptimized code size: ${assembler.unoptimizedCodeSize}%5d B")
|
ErrorReporting.debug(f"Unoptimized code size: ${assembler.unoptimizedCodeSize}%5d B")
|
||||||
@ -140,13 +149,22 @@ object Main {
|
|||||||
s"al ${a.toHexString} .$normalized"
|
s"al ${a.toHexString} .$normalized"
|
||||||
}.mkString("\n").getBytes(StandardCharsets.UTF_8))
|
}.mkString("\n").getBytes(StandardCharsets.UTF_8))
|
||||||
}
|
}
|
||||||
val path = Paths.get(prgOutput)
|
val defaultPrgOutput = if (output.endsWith(platform.fileExtension)) output else output + platform.fileExtension
|
||||||
ErrorReporting.debug("Writing output to " + path.toAbsolutePath)
|
result.code.foreach{
|
||||||
ErrorReporting.debug(s"Total output size: ${result.code.length} bytes")
|
case (bankName, code) =>
|
||||||
Files.write(path, result.code)
|
val prgOutput = if (bankName == "default") {
|
||||||
|
defaultPrgOutput
|
||||||
|
} else {
|
||||||
|
s"${output.stripSuffix(platform.fileExtension)}.$bankName${platform.fileExtension}"
|
||||||
|
}
|
||||||
|
val path = Paths.get(prgOutput)
|
||||||
|
ErrorReporting.debug("Writing output to " + path.toAbsolutePath)
|
||||||
|
ErrorReporting.debug(s"Total output size: ${code.length} bytes")
|
||||||
|
Files.write(path, code)
|
||||||
|
}
|
||||||
ErrorReporting.debug(s"Total time: ${Math.round((System.nanoTime() - startTime)/1e6)} ms")
|
ErrorReporting.debug(s"Total time: ${Math.round((System.nanoTime() - startTime)/1e6)} ms")
|
||||||
c.runFileName.foreach(program =>
|
c.runFileName.foreach(program =>
|
||||||
new ProcessBuilder(program, path.toAbsolutePath.toString).start()
|
new ProcessBuilder(program, Paths.get(defaultPrgOutput).toAbsolutePath.toString).start()
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -12,32 +12,25 @@ import org.apache.commons.configuration2.INIConfiguration
|
|||||||
* @author Karol Stasiak
|
* @author Karol Stasiak
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
object OutputStyle extends Enumeration {
|
||||||
|
val Single, PerBank = Value
|
||||||
|
}
|
||||||
|
|
||||||
class Platform(
|
class Platform(
|
||||||
val cpu: Cpu.Value,
|
val cpu: Cpu.Value,
|
||||||
val flagOverrides: Map[CompilationFlag.Value, Boolean],
|
val flagOverrides: Map[CompilationFlag.Value, Boolean],
|
||||||
val startingModules: List[String],
|
val startingModules: List[String],
|
||||||
val outputPackager: OutputPackager,
|
val outputPackager: OutputPackager,
|
||||||
val codeAllocator: UpwardByteAllocator,
|
val codeAllocators: Map[String, UpwardByteAllocator],
|
||||||
val variableAllocator: VariableAllocator,
|
val variableAllocators: Map[String, VariableAllocator],
|
||||||
val fileExtension: String,
|
val fileExtension: String,
|
||||||
var defaultCodeBank: Int = 0,
|
val bankNumbers: Map[String, Int],
|
||||||
|
val defaultCodeBank: String,
|
||||||
|
val outputStyle: OutputStyle.Value
|
||||||
)
|
)
|
||||||
|
|
||||||
object Platform {
|
object Platform {
|
||||||
|
|
||||||
val C64 = new Platform(
|
|
||||||
Cpu.Mos,
|
|
||||||
Map(),
|
|
||||||
List("c64_hardware", "c64_loader"),
|
|
||||||
SequenceOutput(List(StartAddressOutput, AllocatedDataOutput)),
|
|
||||||
new UpwardByteAllocator(0x80D, 0xA000),
|
|
||||||
new VariableAllocator(
|
|
||||||
List(0xC1, 0xC3, 0xFB, 0xFD, 0x39, 0x3B, 0x3D, 0x43, 0x4B),
|
|
||||||
new AfterCodeByteAllocator(0xA000)
|
|
||||||
),
|
|
||||||
".prg"
|
|
||||||
)
|
|
||||||
|
|
||||||
def lookupPlatformFile(includePath: List[String], platformName: String): Platform = {
|
def lookupPlatformFile(includePath: List[String], platformName: String): Platform = {
|
||||||
includePath.foreach { dir =>
|
includePath.foreach { dir =>
|
||||||
val file = Paths.get(dir, platformName + ".ini").toFile
|
val file = Paths.get(dir, platformName + ".ini").toFile
|
||||||
@ -79,7 +72,7 @@ object Platform {
|
|||||||
}).toMap ++ CompilationFlag.fromString.flatMap { case (k, f) =>
|
}).toMap ++ CompilationFlag.fromString.flatMap { case (k, f) =>
|
||||||
val value = cs.get(classOf[String], k, "")
|
val value = cs.get(classOf[String], k, "")
|
||||||
value.toLowerCase match {
|
value.toLowerCase match {
|
||||||
case "" => None
|
case "" | null => None
|
||||||
case "false" | "off" | "no" | "0" => Some(f -> false)
|
case "false" | "off" | "no" | "0" => Some(f -> false)
|
||||||
case "true" | "on" | "yes" | "1" => Some(f -> true)
|
case "true" | "on" | "yes" | "1" => Some(f -> true)
|
||||||
case _ =>
|
case _ =>
|
||||||
@ -90,23 +83,66 @@ object Platform {
|
|||||||
val startingModules = cs.get(classOf[String], "modules", "").split("[, ]+").filter(_.nonEmpty).toList
|
val startingModules = cs.get(classOf[String], "modules", "").split("[, ]+").filter(_.nonEmpty).toList
|
||||||
|
|
||||||
val as = conf.getSection("allocation")
|
val as = conf.getSection("allocation")
|
||||||
val org = as.get(classOf[String], "main_org", "") match {
|
|
||||||
case "" => ErrorReporting.fatal(s"Undefined main_org")
|
val banks = as.get(classOf[String], "segments", "default").split("[, ]+").filter(_.nonEmpty).toList
|
||||||
case m => parseNumber(m)
|
if (!banks.contains("default")) {
|
||||||
|
ErrorReporting.error("A segment named `default` is required")
|
||||||
}
|
}
|
||||||
|
if (banks.toSet.size != banks.length) {
|
||||||
|
ErrorReporting.error("Duplicate segment name")
|
||||||
|
}
|
||||||
|
val BankRegex = """\A[A-Za-z0-9_]+\z""".r
|
||||||
|
banks.foreach {
|
||||||
|
case BankRegex(_*) => // ok
|
||||||
|
case b => ErrorReporting.error(s"Invalid segment name: `$b`")
|
||||||
|
}
|
||||||
|
|
||||||
|
val bankStarts = banks.map(b => b -> (as.get(classOf[String], s"segment_${b}_start") match {
|
||||||
|
case "" | null => ErrorReporting.error(s"Undefined segment_${b}_start"); 0
|
||||||
|
case x => parseNumber(x)
|
||||||
|
})).toMap
|
||||||
|
val bankDataStarts = banks.map(b => b -> (as.get(classOf[String], s"segment_${b}_datastart", "after_code") match {
|
||||||
|
case "" | "after_code" => None
|
||||||
|
case x => Some(parseNumber(x))
|
||||||
|
})).toMap
|
||||||
|
val bankEnds = banks.map(b => b -> (as.get(classOf[String], s"segment_${b}_end") match {
|
||||||
|
case "" | null => ErrorReporting.error(s"Undefined segment_${b}_end"); 0xffff
|
||||||
|
case x => parseNumber(x)
|
||||||
|
})).toMap
|
||||||
|
val bankCodeEnds = banks.map(b => b -> (as.get(classOf[String], s"segment_${b}_codeend", "") match {
|
||||||
|
case "" => bankEnds(b)
|
||||||
|
case x => parseNumber(x)
|
||||||
|
})).toMap
|
||||||
|
val defaultCodeBank = as.get(classOf[String], "default_code_segment") match {
|
||||||
|
case "" | null => "default"
|
||||||
|
case x => x
|
||||||
|
}
|
||||||
|
// used by 65816:
|
||||||
|
val bankNumbers = banks.map(b => b -> (as.get(classOf[String], s"segment_${b}_bank", "00") match {
|
||||||
|
case "" => 0
|
||||||
|
case x => parseNumber(x)
|
||||||
|
})).toMap
|
||||||
|
|
||||||
|
// TODO: validate stuff
|
||||||
|
banks.foreach(b => {
|
||||||
|
if (bankNumbers(b) < 0 || bankNumbers(b) > 255) ErrorReporting.error(s"Segment $b has invalid bank")
|
||||||
|
if (bankStarts(b) >= bankCodeEnds(b)) ErrorReporting.error(s"Segment $b has invalid range")
|
||||||
|
if (bankCodeEnds(b) > bankEnds(b)) ErrorReporting.error(s"Segment $b has invalid range")
|
||||||
|
if (bankStarts(b) >= bankEnds(b)) ErrorReporting.error(s"Segment $b has invalid range")
|
||||||
|
bankDataStarts(b).foreach(dataStarts => if (dataStarts >= bankEnds(b)) ErrorReporting.error(s"Segment $b has invalid range"))
|
||||||
|
})
|
||||||
|
|
||||||
val freePointers = as.get(classOf[String], "zp_pointers", "all") match {
|
val freePointers = as.get(classOf[String], "zp_pointers", "all") match {
|
||||||
case "all" => List.tabulate(128)(_ * 2)
|
case "all" => List.tabulate(128)(_ * 2)
|
||||||
case xs => xs.split("[, ]+").map(parseNumber).toList
|
case xs => xs.split("[, ]+").map(parseNumber).toList
|
||||||
}
|
}
|
||||||
val himemEnd = as.get(classOf[String], "himem_end", "") match {
|
|
||||||
case "" => ErrorReporting.fatal(s"Undefined himem_end")
|
val codeAllocators = banks.map(b => b -> new UpwardByteAllocator(bankStarts(b), bankCodeEnds(b)))
|
||||||
case end => parseNumber(end) + 1
|
val variableAllocators = banks.map(b => b -> new VariableAllocator(
|
||||||
}
|
if (b == "default") freePointers else Nil, bankDataStarts(b) match {
|
||||||
val byteAllocator = as.get(classOf[String], "himem_start", "") match {
|
case None => new AfterCodeByteAllocator(bankEnds(b))
|
||||||
case "" => ErrorReporting.fatal(s"Undefined himem_start")
|
case Some(start) => new UpwardByteAllocator(start, bankEnds(b))
|
||||||
case "after_code" => new AfterCodeByteAllocator(himemEnd)
|
}))
|
||||||
case start => new UpwardByteAllocator(parseNumber(start), himemEnd)
|
|
||||||
}
|
|
||||||
|
|
||||||
val os = conf.getSection("output")
|
val os = conf.getSection("output")
|
||||||
val outputPackager = SequenceOutput(os.get(classOf[String], "format", "").split("[, ]+").filter(_.nonEmpty).map {
|
val outputPackager = SequenceOutput(os.get(classOf[String], "format", "").split("[, ]+").filter(_.nonEmpty).map {
|
||||||
@ -114,18 +150,26 @@ object Platform {
|
|||||||
case "endaddr" => EndAddressOutput
|
case "endaddr" => EndAddressOutput
|
||||||
case "allocated" => AllocatedDataOutput
|
case "allocated" => AllocatedDataOutput
|
||||||
case n => n.split(":").filter(_.nonEmpty) match {
|
case n => n.split(":").filter(_.nonEmpty) match {
|
||||||
case Array(b, s, e) => BankFragmentOutput(parseNumber(b), parseNumber(s), parseNumber(e))
|
case Array(b, s, e) => BankFragmentOutput(b, parseNumber(s), parseNumber(e))
|
||||||
case Array(s, e) => CurrentBankFragmentOutput(parseNumber(s), parseNumber(e))
|
case Array(s, e) => CurrentBankFragmentOutput(parseNumber(s), parseNumber(e))
|
||||||
case Array(b) => ConstOutput(parseNumber(b).toByte)
|
case Array(b) => ConstOutput(parseNumber(b).toByte)
|
||||||
case x => ErrorReporting.fatal(s"Invalid output format: `$x`")
|
case x => ErrorReporting.fatal(s"Invalid output format: `$x`")
|
||||||
}
|
}
|
||||||
}.toList)
|
}.toList)
|
||||||
var fileExtension = os.get(classOf[String], "extension", ".bin")
|
val fileExtension = os.get(classOf[String], "extension", ".bin")
|
||||||
|
val outputStyle = os.get(classOf[String], "style", "single") match {
|
||||||
|
case "" | "single" => OutputStyle.Single
|
||||||
|
case "per_bank" | "per_segment" => OutputStyle.PerBank
|
||||||
|
case x => ErrorReporting.fatal(s"Invalid output style: `$x`")
|
||||||
|
}
|
||||||
|
|
||||||
new Platform(cpu, flagOverrides, startingModules, outputPackager,
|
new Platform(cpu, flagOverrides, startingModules, outputPackager,
|
||||||
new UpwardByteAllocator(org, 0xffff),
|
codeAllocators.toMap,
|
||||||
new VariableAllocator(freePointers, byteAllocator),
|
variableAllocators.toMap,
|
||||||
if (fileExtension.startsWith(".")) fileExtension else "." + fileExtension)
|
if (fileExtension.startsWith(".")) fileExtension else "." + fileExtension,
|
||||||
|
bankNumbers,
|
||||||
|
defaultCodeBank,
|
||||||
|
outputStyle)
|
||||||
}
|
}
|
||||||
|
|
||||||
def parseNumber(s: String): Int = {
|
def parseNumber(s: String): Int = {
|
||||||
|
@ -386,7 +386,7 @@ object ExpressionCompiler {
|
|||||||
|
|
||||||
def callingContext(ctx: CompilationContext, v: MemoryVariable): CompilationContext = {
|
def callingContext(ctx: CompilationContext, v: MemoryVariable): CompilationContext = {
|
||||||
val result = new Environment(Some(ctx.env), "")
|
val result = new Environment(Some(ctx.env), "")
|
||||||
result.registerVariable(VariableDeclarationStatement(v.name, v.typ.name, stack = false, global = false, constant = false, volatile = false, register = false, initialValue = None, address = None), ctx.options)
|
result.registerVariable(VariableDeclarationStatement(v.name, v.typ.name, stack = false, global = false, constant = false, volatile = false, register = false, initialValue = None, address = None, bank = v.declaredBank), ctx.options)
|
||||||
ctx.copy(env = result)
|
ctx.copy(env = result)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -126,7 +126,8 @@ object ReturnDispatch {
|
|||||||
val paramArrays = stmt.params.indices.map { ix =>
|
val paramArrays = stmt.params.indices.map { ix =>
|
||||||
val a = InitializedArray(label + "$" + ix + ".array", None, (paramMins(ix) to paramMaxes(ix)).map { key =>
|
val a = InitializedArray(label + "$" + ix + ".array", None, (paramMins(ix) to paramMaxes(ix)).map { key =>
|
||||||
map(key)._2.lift(ix).getOrElse(Constant.Zero)
|
map(key)._2.lift(ix).getOrElse(Constant.Zero)
|
||||||
}.toList)
|
}.toList,
|
||||||
|
ctx.function.declaredBank)
|
||||||
env.registerUnnamedArray(a)
|
env.registerUnnamedArray(a)
|
||||||
a
|
a
|
||||||
}
|
}
|
||||||
@ -146,7 +147,7 @@ object ReturnDispatch {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (useJmpaix) {
|
if (useJmpaix) {
|
||||||
val jumpTable = InitializedArray(label + "$jt.array", None, (actualMin to actualMax).flatMap(i => List(map(i)._1.loByte, map(i)._1.hiByte)).toList)
|
val jumpTable = InitializedArray(label + "$jt.array", None, (actualMin to actualMax).flatMap(i => List(map(i)._1.loByte, map(i)._1.hiByte)).toList, ctx.function.declaredBank)
|
||||||
env.registerUnnamedArray(jumpTable)
|
env.registerUnnamedArray(jumpTable)
|
||||||
if (copyParams.isEmpty) {
|
if (copyParams.isEmpty) {
|
||||||
val loadIndex = ExpressionCompiler.compile(ctx, stmt.indexer, Some(b -> RegisterVariable(Register.A, b)), BranchSpec.None)
|
val loadIndex = ExpressionCompiler.compile(ctx, stmt.indexer, Some(b -> RegisterVariable(Register.A, b)), BranchSpec.None)
|
||||||
@ -161,8 +162,8 @@ object ReturnDispatch {
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
val loadIndex = ExpressionCompiler.compile(ctx, stmt.indexer, Some(b -> RegisterVariable(Register.X, b)), BranchSpec.None)
|
val loadIndex = ExpressionCompiler.compile(ctx, stmt.indexer, Some(b -> RegisterVariable(Register.X, b)), BranchSpec.None)
|
||||||
val jumpTableLo = InitializedArray(label + "$jl.array", None, (actualMin to actualMax).map(i => (map(i)._1 - 1).loByte).toList)
|
val jumpTableLo = InitializedArray(label + "$jl.array", None, (actualMin to actualMax).map(i => (map(i)._1 - 1).loByte).toList, ctx.function.declaredBank)
|
||||||
val jumpTableHi = InitializedArray(label + "$jh.array", None, (actualMin to actualMax).map(i => (map(i)._1 - 1).hiByte).toList)
|
val jumpTableHi = InitializedArray(label + "$jh.array", None, (actualMin to actualMax).map(i => (map(i)._1 - 1).hiByte).toList, ctx.function.declaredBank)
|
||||||
env.registerUnnamedArray(jumpTableLo)
|
env.registerUnnamedArray(jumpTableLo)
|
||||||
env.registerUnnamedArray(jumpTableHi)
|
env.registerUnnamedArray(jumpTableHi)
|
||||||
loadIndex ++ copyParams ++ List(
|
loadIndex ++ copyParams ++ List(
|
||||||
|
94
src/main/scala/millfork/env/Environment.scala
vendored
94
src/main/scala/millfork/env/Environment.scala
vendored
@ -23,7 +23,7 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
private val relVarId = new AtomicLong
|
private val relVarId = new AtomicLong
|
||||||
|
|
||||||
def genRelativeVariable(constant: Constant, typ: Type, zeropage: Boolean): RelativeVariable = {
|
def genRelativeVariable(constant: Constant, typ: Type, zeropage: Boolean): RelativeVariable = {
|
||||||
val variable = RelativeVariable(".rv__" + relVarId.incrementAndGet().formatted("%06d"), constant, typ, zeropage = zeropage)
|
val variable = RelativeVariable(".rv__" + relVarId.incrementAndGet().formatted("%06d"), constant, typ, zeropage = zeropage, declaredBank = None /*TODO*/)
|
||||||
addThing(variable, None)
|
addThing(variable, None)
|
||||||
variable
|
variable
|
||||||
}
|
}
|
||||||
@ -68,19 +68,19 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
case _ => Nil
|
case _ => Nil
|
||||||
}.toList
|
}.toList
|
||||||
|
|
||||||
def getAllFixedAddressObjects: List[(Int, Int)] = {
|
def getAllFixedAddressObjects: List[(String, Int, Int)] = {
|
||||||
things.values.flatMap {
|
things.values.flatMap {
|
||||||
case RelativeArray(_, NumericConstant(addr, _), size) =>
|
case RelativeArray(_, NumericConstant(addr, _), size, declaredBank) =>
|
||||||
List(addr.toInt -> size)
|
List((declaredBank.getOrElse("default"), addr.toInt, size))
|
||||||
case RelativeVariable(_, NumericConstant(addr, _), typ, _) =>
|
case RelativeVariable(_, NumericConstant(addr, _), typ, _, declaredBank) =>
|
||||||
List(addr.toInt -> typ.size)
|
List((declaredBank.getOrElse("default"), addr.toInt, typ.size))
|
||||||
case f: NormalFunction =>
|
case f: NormalFunction =>
|
||||||
f.environment.getAllFixedAddressObjects
|
f.environment.getAllFixedAddressObjects
|
||||||
case _ => Nil
|
case _ => Nil
|
||||||
}.toList
|
}.toList
|
||||||
}
|
}
|
||||||
|
|
||||||
def allocateVariables(nf: Option[NormalFunction], mem: MemoryBank, callGraph: CallGraph, allocator: VariableAllocator, options: CompilationOptions, onEachVariable: (String, Int) => Unit): Unit = {
|
def allocateVariables(nf: Option[NormalFunction], mem: CompiledMemory, callGraph: CallGraph, allocators: Map[String, VariableAllocator], options: CompilationOptions, onEachVariable: (String, Int) => Unit): Unit = {
|
||||||
val b = get[Type]("byte")
|
val b = get[Type]("byte")
|
||||||
val p = get[Type]("pointer")
|
val p = get[Type]("pointer")
|
||||||
val params = nf.fold(List[String]()) { f =>
|
val params = nf.fold(List[String]()) { f =>
|
||||||
@ -104,6 +104,7 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else GlobalVertex
|
} else GlobalVertex
|
||||||
|
val bank = m.bank(options)
|
||||||
m.alloc match {
|
m.alloc match {
|
||||||
case VariableAllocationMethod.None =>
|
case VariableAllocationMethod.None =>
|
||||||
Nil
|
Nil
|
||||||
@ -111,7 +112,7 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
m.sizeInBytes match {
|
m.sizeInBytes match {
|
||||||
case 2 =>
|
case 2 =>
|
||||||
val addr =
|
val addr =
|
||||||
allocator.allocatePointer(mem, callGraph, vertex)
|
allocators(bank).allocatePointer(mem.banks(bank), callGraph, vertex)
|
||||||
onEachVariable(m.name, addr)
|
onEachVariable(m.name, addr)
|
||||||
List(
|
List(
|
||||||
ConstantThing(m.name.stripPrefix(prefix) + "`", NumericConstant(addr, 2), p)
|
ConstantThing(m.name.stripPrefix(prefix) + "`", NumericConstant(addr, 2), p)
|
||||||
@ -125,13 +126,13 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
case 0 => Nil
|
case 0 => Nil
|
||||||
case 2 =>
|
case 2 =>
|
||||||
val addr =
|
val addr =
|
||||||
allocator.allocateBytes(mem, callGraph, vertex, options, 2, initialized = false, writeable = true)
|
allocators(bank).allocateBytes(mem.banks(bank), callGraph, vertex, options, 2, initialized = false, writeable = true)
|
||||||
onEachVariable(m.name, addr)
|
onEachVariable(m.name, addr)
|
||||||
List(
|
List(
|
||||||
ConstantThing(m.name.stripPrefix(prefix) + "`", NumericConstant(addr, 2), p)
|
ConstantThing(m.name.stripPrefix(prefix) + "`", NumericConstant(addr, 2), p)
|
||||||
)
|
)
|
||||||
case count =>
|
case count =>
|
||||||
val addr = allocator.allocateBytes(mem, callGraph, vertex, options, count, initialized = false, writeable = true)
|
val addr = allocators(bank).allocateBytes(mem.banks(bank), callGraph, vertex, options, count, initialized = false, writeable = true)
|
||||||
onEachVariable(m.name, addr)
|
onEachVariable(m.name, addr)
|
||||||
List(
|
List(
|
||||||
ConstantThing(m.name.stripPrefix(prefix) + "`", NumericConstant(addr, 2), p)
|
ConstantThing(m.name.stripPrefix(prefix) + "`", NumericConstant(addr, 2), p)
|
||||||
@ -139,7 +140,7 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
case f: NormalFunction =>
|
case f: NormalFunction =>
|
||||||
f.environment.allocateVariables(Some(f), mem, callGraph, allocator, options, onEachVariable)
|
f.environment.allocateVariables(Some(f), mem, callGraph, allocators, options, onEachVariable)
|
||||||
Nil
|
Nil
|
||||||
case _ => Nil
|
case _ => Nil
|
||||||
}.toList
|
}.toList
|
||||||
@ -200,9 +201,9 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
InitializedMemoryVariable
|
InitializedMemoryVariable
|
||||||
UninitializedMemoryVariable
|
UninitializedMemoryVariable
|
||||||
getArrayOrPointer(name) match {
|
getArrayOrPointer(name) match {
|
||||||
case th@InitializedArray(_, _, cs) => ConstantPointy(th.toAddress, Some(cs.length))
|
case th@InitializedArray(_, _, cs, _) => ConstantPointy(th.toAddress, Some(cs.length))
|
||||||
case th@UninitializedArray(_, size) => ConstantPointy(th.toAddress, Some(size))
|
case th@UninitializedArray(_, size, _) => ConstantPointy(th.toAddress, Some(size))
|
||||||
case th@RelativeArray(_, _, size) => ConstantPointy(th.toAddress, Some(size))
|
case th@RelativeArray(_, _, size, _) => ConstantPointy(th.toAddress, Some(size))
|
||||||
case ConstantThing(_, value, typ) if typ.size <= 2 => ConstantPointy(value, None)
|
case ConstantThing(_, value, typ) if typ.size <= 2 => ConstantPointy(value, None)
|
||||||
case th:VariableInMemory => VariablePointy(th.toAddress)
|
case th:VariableInMemory => VariablePointy(th.toAddress)
|
||||||
case _ =>
|
case _ =>
|
||||||
@ -480,7 +481,8 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
resultType,
|
resultType,
|
||||||
params,
|
params,
|
||||||
addr,
|
addr,
|
||||||
env
|
env,
|
||||||
|
stmt.bank
|
||||||
)
|
)
|
||||||
addThing(mangled, stmt.position)
|
addThing(mangled, stmt.position)
|
||||||
registerAddressConstant(mangled, stmt.position)
|
registerAddressConstant(mangled, stmt.position)
|
||||||
@ -498,12 +500,15 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
}
|
}
|
||||||
val needsExtraRTS = !stmt.isMacro && !stmt.assembly && (statements.isEmpty || !statements.last.isInstanceOf[ReturnStatement])
|
val needsExtraRTS = !stmt.isMacro && !stmt.assembly && (statements.isEmpty || !statements.last.isInstanceOf[ReturnStatement])
|
||||||
if (stmt.isMacro) {
|
if (stmt.isMacro) {
|
||||||
|
if (stmt.bank.isDefined) {
|
||||||
|
ErrorReporting.error("Macro functions cannot be in a defined segment", stmt.position)
|
||||||
|
}
|
||||||
val mangled = MacroFunction(
|
val mangled = MacroFunction(
|
||||||
name,
|
name,
|
||||||
resultType,
|
resultType,
|
||||||
params,
|
params,
|
||||||
env,
|
env,
|
||||||
executableStatements ++ (if (needsExtraRTS) List(AssemblyStatement.implied(Opcode.RTS, elidable = true)) else Nil),
|
executableStatements ++ (if (needsExtraRTS) List(AssemblyStatement.implied(Opcode.RTS, elidable = true)) else Nil)
|
||||||
)
|
)
|
||||||
addThing(mangled, stmt.position)
|
addThing(mangled, stmt.position)
|
||||||
} else {
|
} else {
|
||||||
@ -522,7 +527,8 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
interrupt = stmt.interrupt,
|
interrupt = stmt.interrupt,
|
||||||
kernalInterrupt = stmt.kernalInterrupt,
|
kernalInterrupt = stmt.kernalInterrupt,
|
||||||
reentrant = stmt.reentrant,
|
reentrant = stmt.reentrant,
|
||||||
position = stmt.position
|
position = stmt.position,
|
||||||
|
declaredBank = stmt.bank
|
||||||
)
|
)
|
||||||
addThing(mangled, stmt.position)
|
addThing(mangled, stmt.position)
|
||||||
registerAddressConstant(mangled, stmt.position)
|
registerAddressConstant(mangled, stmt.position)
|
||||||
@ -544,13 +550,13 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
stmt.assemblyParamPassingConvention match {
|
stmt.assemblyParamPassingConvention match {
|
||||||
case ByVariable(name) =>
|
case ByVariable(name) =>
|
||||||
val zp = typ.name == "pointer" // TODO
|
val zp = typ.name == "pointer" // TODO
|
||||||
val v = UninitializedMemoryVariable(prefix + name, typ, if (zp) VariableAllocationMethod.Zeropage else VariableAllocationMethod.Auto)
|
val v = UninitializedMemoryVariable(prefix + name, typ, if (zp) VariableAllocationMethod.Zeropage else VariableAllocationMethod.Auto, None)
|
||||||
addThing(v, stmt.position)
|
addThing(v, stmt.position)
|
||||||
registerAddressConstant(v, stmt.position)
|
registerAddressConstant(v, stmt.position)
|
||||||
if (typ.size == 2) {
|
if (typ.size == 2) {
|
||||||
val addr = v.toAddress
|
val addr = v.toAddress
|
||||||
addThing(RelativeVariable(v.name + ".hi", addr + 1, b, zeropage = zp), stmt.position)
|
addThing(RelativeVariable(v.name + ".hi", addr + 1, b, zeropage = zp, None), stmt.position)
|
||||||
addThing(RelativeVariable(v.name + ".lo", addr, b, zeropage = zp), stmt.position)
|
addThing(RelativeVariable(v.name + ".lo", addr, b, zeropage = zp, None), stmt.position)
|
||||||
}
|
}
|
||||||
case ByRegister(_) => ()
|
case ByRegister(_) => ()
|
||||||
case ByConstant(name) =>
|
case ByConstant(name) =>
|
||||||
@ -558,10 +564,10 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
addThing(v, stmt.position)
|
addThing(v, stmt.position)
|
||||||
case ByReference(name) =>
|
case ByReference(name) =>
|
||||||
val addr = UnexpandedConstant(prefix + name, typ.size)
|
val addr = UnexpandedConstant(prefix + name, typ.size)
|
||||||
val v = RelativeVariable(prefix + name, addr, p, zeropage = false)
|
val v = RelativeVariable(prefix + name, addr, p, zeropage = false, None)
|
||||||
addThing(v, stmt.position)
|
addThing(v, stmt.position)
|
||||||
addThing(RelativeVariable(v.name + ".hi", addr + 1, b, zeropage = false), stmt.position)
|
addThing(RelativeVariable(v.name + ".hi", addr + 1, b, zeropage = false, None), stmt.position)
|
||||||
addThing(RelativeVariable(v.name + ".lo", addr, b, zeropage = false), stmt.position)
|
addThing(RelativeVariable(v.name + ".lo", addr, b, zeropage = false, None), stmt.position)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -589,16 +595,19 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
case NumericConstant(length, _) =>
|
case NumericConstant(length, _) =>
|
||||||
if (length > 0xffff || length < 0) ErrorReporting.error(s"Array `${stmt.name}` has invalid length", stmt.position)
|
if (length > 0xffff || length < 0) ErrorReporting.error(s"Array `${stmt.name}` has invalid length", stmt.position)
|
||||||
val array = address match {
|
val array = address match {
|
||||||
case None => UninitializedArray(stmt.name + ".array", length.toInt)
|
case None => UninitializedArray(stmt.name + ".array", length.toInt,
|
||||||
case Some(aa) => RelativeArray(stmt.name + ".array", aa, length.toInt)
|
declaredBank = stmt.bank)
|
||||||
|
case Some(aa) => RelativeArray(stmt.name + ".array", aa, length.toInt,
|
||||||
|
declaredBank = stmt.bank)
|
||||||
}
|
}
|
||||||
addThing(array, stmt.position)
|
addThing(array, stmt.position)
|
||||||
registerAddressConstant(UninitializedMemoryVariable(stmt.name, p, VariableAllocationMethod.None), stmt.position)
|
registerAddressConstant(UninitializedMemoryVariable(stmt.name, p, VariableAllocationMethod.None, stmt.bank), stmt.position)
|
||||||
val a = address match {
|
val a = address match {
|
||||||
case None => array.toAddress
|
case None => array.toAddress
|
||||||
case Some(aa) => aa
|
case Some(aa) => aa
|
||||||
}
|
}
|
||||||
addThing(RelativeVariable(stmt.name + ".first", a, b, zeropage = false), stmt.position)
|
addThing(RelativeVariable(stmt.name + ".first", a, b, zeropage = false,
|
||||||
|
declaredBank = stmt.bank), stmt.position)
|
||||||
addThing(ConstantThing(stmt.name, a, p), stmt.position)
|
addThing(ConstantThing(stmt.name, a, p), stmt.position)
|
||||||
addThing(ConstantThing(stmt.name + ".hi", a.hiByte.quickSimplify, b), stmt.position)
|
addThing(ConstantThing(stmt.name + ".hi", a.hiByte.quickSimplify, b), stmt.position)
|
||||||
addThing(ConstantThing(stmt.name + ".lo", a.loByte.quickSimplify, b), stmt.position)
|
addThing(ConstantThing(stmt.name + ".lo", a.loByte.quickSimplify, b), stmt.position)
|
||||||
@ -625,14 +634,17 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
if (length > 0xffff || length < 0) ErrorReporting.error(s"Array `${stmt.name}` has invalid length", stmt.position)
|
if (length > 0xffff || length < 0) ErrorReporting.error(s"Array `${stmt.name}` has invalid length", stmt.position)
|
||||||
val address = stmt.address.map(a => eval(a).getOrElse(Constant.error(s"Array `${stmt.name}` has non-constant address", stmt.position)))
|
val address = stmt.address.map(a => eval(a).getOrElse(Constant.error(s"Array `${stmt.name}` has non-constant address", stmt.position)))
|
||||||
val data = contents.map(x => eval(x).getOrElse(Constant.error(s"Array `${stmt.name}` has non-constant contents", stmt.position)))
|
val data = contents.map(x => eval(x).getOrElse(Constant.error(s"Array `${stmt.name}` has non-constant contents", stmt.position)))
|
||||||
val array = InitializedArray(stmt.name + ".array", address, data)
|
val array = InitializedArray(stmt.name + ".array", address, data,
|
||||||
|
declaredBank = stmt.bank)
|
||||||
addThing(array, stmt.position)
|
addThing(array, stmt.position)
|
||||||
registerAddressConstant(UninitializedMemoryVariable(stmt.name, p, VariableAllocationMethod.None), stmt.position)
|
registerAddressConstant(UninitializedMemoryVariable(stmt.name, p, VariableAllocationMethod.None,
|
||||||
|
declaredBank = stmt.bank), stmt.position)
|
||||||
val a = address match {
|
val a = address match {
|
||||||
case None => array.toAddress
|
case None => array.toAddress
|
||||||
case Some(aa) => aa
|
case Some(aa) => aa
|
||||||
}
|
}
|
||||||
addThing(RelativeVariable(stmt.name + ".first", a, b, zeropage = false), stmt.position)
|
addThing(RelativeVariable(stmt.name + ".first", a, b, zeropage = false,
|
||||||
|
declaredBank = stmt.bank), stmt.position)
|
||||||
addThing(ConstantThing(stmt.name, a, p), stmt.position)
|
addThing(ConstantThing(stmt.name, a, p), stmt.position)
|
||||||
addThing(ConstantThing(stmt.name + ".hi", a.hiByte.quickSimplify, b), stmt.position)
|
addThing(ConstantThing(stmt.name + ".hi", a.hiByte.quickSimplify, b), stmt.position)
|
||||||
addThing(ConstantThing(stmt.name + ".lo", a.loByte.quickSimplify, b), stmt.position)
|
addThing(ConstantThing(stmt.name + ".lo", a.loByte.quickSimplify, b), stmt.position)
|
||||||
@ -704,12 +716,14 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
if (alloc != VariableAllocationMethod.Static && stmt.initialValue.isDefined) {
|
if (alloc != VariableAllocationMethod.Static && stmt.initialValue.isDefined) {
|
||||||
ErrorReporting.error(s"`$name` cannot be preinitialized`", position)
|
ErrorReporting.error(s"`$name` cannot be preinitialized`", position)
|
||||||
}
|
}
|
||||||
val v = stmt.initialValue.fold[MemoryVariable](UninitializedMemoryVariable(prefix + name, typ, alloc)){ive =>
|
val v = stmt.initialValue.fold[MemoryVariable](UninitializedMemoryVariable(prefix + name, typ, alloc,
|
||||||
|
declaredBank = stmt.bank)){ive =>
|
||||||
if (options.flags(CompilationFlag.ReadOnlyArrays)) {
|
if (options.flags(CompilationFlag.ReadOnlyArrays)) {
|
||||||
ErrorReporting.warn("Initialized variable in read-only segment", options, position)
|
ErrorReporting.warn("Initialized variable in read-only segment", options, position)
|
||||||
}
|
}
|
||||||
val ivc = eval(ive).getOrElse(Constant.error(s"Initial value of `$name` is not a constant", position))
|
val ivc = eval(ive).getOrElse(Constant.error(s"Initial value of `$name` is not a constant", position))
|
||||||
InitializedMemoryVariable(name, None, typ, ivc)
|
InitializedMemoryVariable(name, None, typ, ivc,
|
||||||
|
declaredBank = stmt.bank)
|
||||||
}
|
}
|
||||||
registerAddressConstant(v, stmt.position)
|
registerAddressConstant(v, stmt.position)
|
||||||
(v, v.toAddress)
|
(v, v.toAddress)
|
||||||
@ -719,7 +733,8 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
case NumericConstant(n, _) => n < 0x100
|
case NumericConstant(n, _) => n < 0x100
|
||||||
case _ => false
|
case _ => false
|
||||||
}
|
}
|
||||||
val v = RelativeVariable(prefix + name, addr, typ, zeropage = zp)
|
val v = RelativeVariable(prefix + name, addr, typ, zeropage = zp,
|
||||||
|
declaredBank = stmt.bank)
|
||||||
registerAddressConstant(v, stmt.position)
|
registerAddressConstant(v, stmt.position)
|
||||||
(v, addr)
|
(v, addr)
|
||||||
})
|
})
|
||||||
@ -728,8 +743,10 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
addThing(ConstantThing(v.name + "`", addr, b), stmt.position)
|
addThing(ConstantThing(v.name + "`", addr, b), stmt.position)
|
||||||
}
|
}
|
||||||
if (typ.size == 2) {
|
if (typ.size == 2) {
|
||||||
addThing(RelativeVariable(prefix + name + ".hi", addr + 1, b, zeropage = v.zeropage), stmt.position)
|
addThing(RelativeVariable(prefix + name + ".hi", addr + 1, b, zeropage = v.zeropage,
|
||||||
addThing(RelativeVariable(prefix + name + ".lo", addr, b, zeropage = v.zeropage), stmt.position)
|
declaredBank = stmt.bank), stmt.position)
|
||||||
|
addThing(RelativeVariable(prefix + name + ".lo", addr, b, zeropage = v.zeropage,
|
||||||
|
declaredBank = stmt.bank), stmt.position)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -771,7 +788,8 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
|
|
||||||
def collectDeclarations(program: Program, options: CompilationOptions): Unit = {
|
def collectDeclarations(program: Program, options: CompilationOptions): Unit = {
|
||||||
if (options.flag(CompilationFlag.OptimizeForSonicSpeed)) {
|
if (options.flag(CompilationFlag.OptimizeForSonicSpeed)) {
|
||||||
addThing(InitializedArray("identity$", None, List.tabulate(256)(n => NumericConstant(n, 1))), None)
|
addThing(InitializedArray("identity$", None, List.tabulate(256)(n => NumericConstant(n, 1)),
|
||||||
|
declaredBank = None), None)
|
||||||
}
|
}
|
||||||
program.declarations.foreach {
|
program.declarations.foreach {
|
||||||
case f: FunctionDeclarationStatement => registerFunction(f, options)
|
case f: FunctionDeclarationStatement => registerFunction(f, options)
|
||||||
@ -782,6 +800,7 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
if (options.flag(CompilationFlag.ZeropagePseudoregister) && !things.contains("__reg")) {
|
if (options.flag(CompilationFlag.ZeropagePseudoregister) && !things.contains("__reg")) {
|
||||||
registerVariable(VariableDeclarationStatement(
|
registerVariable(VariableDeclarationStatement(
|
||||||
name = "__reg",
|
name = "__reg",
|
||||||
|
bank = None,
|
||||||
typ = "pointer",
|
typ = "pointer",
|
||||||
global = true,
|
global = true,
|
||||||
stack = false,
|
stack = false,
|
||||||
@ -792,7 +811,8 @@ class Environment(val parent: Option[Environment], val prefix: String) {
|
|||||||
address = None), options)
|
address = None), options)
|
||||||
}
|
}
|
||||||
if (!things.contains("__constant8")) {
|
if (!things.contains("__constant8")) {
|
||||||
things("__constant8") = InitializedArray("__constant8", None, List(NumericConstant(8, 1)))
|
things("__constant8") = InitializedArray("__constant8", None, List(NumericConstant(8, 1)),
|
||||||
|
declaredBank = None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
38
src/main/scala/millfork/env/Thing.scala
vendored
38
src/main/scala/millfork/env/Thing.scala
vendored
@ -81,10 +81,10 @@ sealed trait ThingInMemory extends Thing {
|
|||||||
def toAddress: Constant
|
def toAddress: Constant
|
||||||
|
|
||||||
var farFlag: Option[Boolean] = None
|
var farFlag: Option[Boolean] = None
|
||||||
var declaredBank: Option[Int] = None
|
val declaredBank: Option[String]
|
||||||
|
|
||||||
def isFar(compilationOptions: CompilationOptions): Boolean
|
def isFar(compilationOptions: CompilationOptions): Boolean
|
||||||
def bank(compilationOptions: CompilationOptions): Int
|
def bank(compilationOptions: CompilationOptions): String
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed trait PreallocableThing extends ThingInMemory {
|
sealed trait PreallocableThing extends ThingInMemory {
|
||||||
@ -101,8 +101,10 @@ case class Label(name: String) extends ThingInMemory {
|
|||||||
override def isFar(compilationOptions: CompilationOptions): Boolean =
|
override def isFar(compilationOptions: CompilationOptions): Boolean =
|
||||||
compilationOptions.flag(CompilationFlag.LargeCode) && farFlag.getOrElse(true)
|
compilationOptions.flag(CompilationFlag.LargeCode) && farFlag.getOrElse(true)
|
||||||
|
|
||||||
override def bank(compilationOptions: CompilationOptions): Int =
|
override def bank(compilationOptions: CompilationOptions): String =
|
||||||
declaredBank.getOrElse(compilationOptions.platform.defaultCodeBank)
|
declaredBank.getOrElse(compilationOptions.platform.defaultCodeBank)
|
||||||
|
|
||||||
|
override val declaredBank: Option[String] = None
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed trait Variable extends TypedThing with VariableLikeThing
|
sealed trait Variable extends TypedThing with VariableLikeThing
|
||||||
@ -117,8 +119,8 @@ sealed trait VariableInMemory extends Variable with ThingInMemory with Indexable
|
|||||||
override def isFar(compilationOptions: CompilationOptions): Boolean =
|
override def isFar(compilationOptions: CompilationOptions): Boolean =
|
||||||
!zeropage && farFlag.getOrElse(false)
|
!zeropage && farFlag.getOrElse(false)
|
||||||
|
|
||||||
override def bank(compilationOptions: CompilationOptions): Int =
|
override def bank(compilationOptions: CompilationOptions): String =
|
||||||
declaredBank.getOrElse(0)
|
declaredBank.getOrElse("default")
|
||||||
}
|
}
|
||||||
|
|
||||||
case class RegisterVariable(register: Register.Value, typ: Type) extends Variable {
|
case class RegisterVariable(register: Register.Value, typ: Type) extends Variable {
|
||||||
@ -149,7 +151,7 @@ abstract class MemoryVariable extends VariableInMemory {
|
|||||||
def alloc: VariableAllocationMethod.Value
|
def alloc: VariableAllocationMethod.Value
|
||||||
}
|
}
|
||||||
|
|
||||||
case class UninitializedMemoryVariable(name: String, typ: Type, alloc: VariableAllocationMethod.Value) extends MemoryVariable with UninitializedMemory {
|
case class UninitializedMemoryVariable(name: String, typ: Type, alloc: VariableAllocationMethod.Value, declaredBank: Option[String]) extends MemoryVariable with UninitializedMemory {
|
||||||
override def sizeInBytes: Int = typ.size
|
override def sizeInBytes: Int = typ.size
|
||||||
|
|
||||||
override def zeropage: Boolean = alloc == VariableAllocationMethod.Zeropage
|
override def zeropage: Boolean = alloc == VariableAllocationMethod.Zeropage
|
||||||
@ -157,7 +159,7 @@ case class UninitializedMemoryVariable(name: String, typ: Type, alloc: VariableA
|
|||||||
override def toAddress: MemoryAddressConstant = MemoryAddressConstant(this)
|
override def toAddress: MemoryAddressConstant = MemoryAddressConstant(this)
|
||||||
}
|
}
|
||||||
|
|
||||||
case class InitializedMemoryVariable(name: String, address: Option[Constant], typ: Type, initialValue: Constant) extends MemoryVariable with PreallocableThing {
|
case class InitializedMemoryVariable(name: String, address: Option[Constant], typ: Type, initialValue: Constant, declaredBank: Option[String]) extends MemoryVariable with PreallocableThing {
|
||||||
override def zeropage: Boolean = false
|
override def zeropage: Boolean = false
|
||||||
|
|
||||||
override def toAddress: MemoryAddressConstant = MemoryAddressConstant(this)
|
override def toAddress: MemoryAddressConstant = MemoryAddressConstant(this)
|
||||||
@ -169,33 +171,33 @@ case class InitializedMemoryVariable(name: String, address: Option[Constant], ty
|
|||||||
|
|
||||||
trait MfArray extends ThingInMemory with IndexableThing
|
trait MfArray extends ThingInMemory with IndexableThing
|
||||||
|
|
||||||
case class UninitializedArray(name: String, sizeInBytes: Int) extends MfArray with UninitializedMemory {
|
case class UninitializedArray(name: String, sizeInBytes: Int, declaredBank: Option[String]) extends MfArray with UninitializedMemory {
|
||||||
override def toAddress: MemoryAddressConstant = MemoryAddressConstant(this)
|
override def toAddress: MemoryAddressConstant = MemoryAddressConstant(this)
|
||||||
|
|
||||||
override def alloc = VariableAllocationMethod.Static
|
override def alloc = VariableAllocationMethod.Static
|
||||||
|
|
||||||
override def isFar(compilationOptions: CompilationOptions): Boolean = farFlag.getOrElse(false)
|
override def isFar(compilationOptions: CompilationOptions): Boolean = farFlag.getOrElse(false)
|
||||||
|
|
||||||
override def bank(compilationOptions: CompilationOptions): Int = declaredBank.getOrElse(0)
|
override def bank(compilationOptions: CompilationOptions): String = declaredBank.getOrElse("default")
|
||||||
}
|
}
|
||||||
|
|
||||||
case class RelativeArray(name: String, address: Constant, sizeInBytes: Int) extends MfArray {
|
case class RelativeArray(name: String, address: Constant, sizeInBytes: Int, declaredBank: Option[String]) extends MfArray {
|
||||||
override def toAddress: Constant = address
|
override def toAddress: Constant = address
|
||||||
|
|
||||||
override def isFar(compilationOptions: CompilationOptions): Boolean = farFlag.getOrElse(false)
|
override def isFar(compilationOptions: CompilationOptions): Boolean = farFlag.getOrElse(false)
|
||||||
|
|
||||||
override def bank(compilationOptions: CompilationOptions): Int = declaredBank.getOrElse(0)
|
override def bank(compilationOptions: CompilationOptions): String = declaredBank.getOrElse("default")
|
||||||
}
|
}
|
||||||
|
|
||||||
case class InitializedArray(name: String, address: Option[Constant], contents: List[Constant]) extends MfArray with PreallocableThing {
|
case class InitializedArray(name: String, address: Option[Constant], contents: List[Constant], declaredBank: Option[String]) extends MfArray with PreallocableThing {
|
||||||
override def shouldGenerate = true
|
override def shouldGenerate = true
|
||||||
|
|
||||||
override def isFar(compilationOptions: CompilationOptions): Boolean = farFlag.getOrElse(false)
|
override def isFar(compilationOptions: CompilationOptions): Boolean = farFlag.getOrElse(false)
|
||||||
|
|
||||||
override def bank(compilationOptions: CompilationOptions): Int = declaredBank.getOrElse(0)
|
override def bank(compilationOptions: CompilationOptions): String = declaredBank.getOrElse(compilationOptions.platform.defaultCodeBank)
|
||||||
}
|
}
|
||||||
|
|
||||||
case class RelativeVariable(name: String, address: Constant, typ: Type, zeropage: Boolean) extends VariableInMemory {
|
case class RelativeVariable(name: String, address: Constant, typ: Type, zeropage: Boolean, declaredBank: Option[String]) extends VariableInMemory {
|
||||||
override def toAddress: Constant = address
|
override def toAddress: Constant = address
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -231,7 +233,7 @@ sealed trait FunctionInMemory extends MangledFunction with ThingInMemory {
|
|||||||
override def isFar(compilationOptions: CompilationOptions): Boolean =
|
override def isFar(compilationOptions: CompilationOptions): Boolean =
|
||||||
compilationOptions.flag(CompilationFlag.LargeCode) && farFlag.getOrElse(true)
|
compilationOptions.flag(CompilationFlag.LargeCode) && farFlag.getOrElse(true)
|
||||||
|
|
||||||
override def bank(compilationOptions: CompilationOptions): Int =
|
override def bank(compilationOptions: CompilationOptions): String =
|
||||||
declaredBank.getOrElse(compilationOptions.platform.defaultCodeBank)
|
declaredBank.getOrElse(compilationOptions.platform.defaultCodeBank)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -239,7 +241,8 @@ case class ExternFunction(name: String,
|
|||||||
returnType: Type,
|
returnType: Type,
|
||||||
params: ParamSignature,
|
params: ParamSignature,
|
||||||
address: Constant,
|
address: Constant,
|
||||||
environment: Environment) extends FunctionInMemory {
|
environment: Environment,
|
||||||
|
declaredBank: Option[String]) extends FunctionInMemory {
|
||||||
override def toAddress: Constant = address
|
override def toAddress: Constant = address
|
||||||
|
|
||||||
override def interrupt = false
|
override def interrupt = false
|
||||||
@ -255,7 +258,8 @@ case class NormalFunction(name: String,
|
|||||||
interrupt: Boolean,
|
interrupt: Boolean,
|
||||||
kernalInterrupt: Boolean,
|
kernalInterrupt: Boolean,
|
||||||
reentrant: Boolean,
|
reentrant: Boolean,
|
||||||
position: Option[Position]) extends FunctionInMemory with PreallocableThing {
|
position: Option[Position],
|
||||||
|
declaredBank: Option[String]) extends FunctionInMemory with PreallocableThing {
|
||||||
override def shouldGenerate = true
|
override def shouldGenerate = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -93,6 +93,7 @@ case class TypeDefinitionStatement(name: String, parent: String) extends Declara
|
|||||||
|
|
||||||
case class VariableDeclarationStatement(name: String,
|
case class VariableDeclarationStatement(name: String,
|
||||||
typ: String,
|
typ: String,
|
||||||
|
bank: Option[String],
|
||||||
global: Boolean,
|
global: Boolean,
|
||||||
stack: Boolean,
|
stack: Boolean,
|
||||||
constant: Boolean,
|
constant: Boolean,
|
||||||
@ -104,6 +105,7 @@ case class VariableDeclarationStatement(name: String,
|
|||||||
}
|
}
|
||||||
|
|
||||||
case class ArrayDeclarationStatement(name: String,
|
case class ArrayDeclarationStatement(name: String,
|
||||||
|
bank: Option[String],
|
||||||
length: Option[Expression],
|
length: Option[Expression],
|
||||||
address: Option[Expression],
|
address: Option[Expression],
|
||||||
elements: Option[List[Expression]]) extends DeclarationStatement {
|
elements: Option[List[Expression]]) extends DeclarationStatement {
|
||||||
@ -120,6 +122,7 @@ case class ImportStatement(filename: String) extends DeclarationStatement {
|
|||||||
case class FunctionDeclarationStatement(name: String,
|
case class FunctionDeclarationStatement(name: String,
|
||||||
resultType: String,
|
resultType: String,
|
||||||
params: List[ParameterDeclaration],
|
params: List[ParameterDeclaration],
|
||||||
|
bank: Option[String],
|
||||||
address: Option[Expression],
|
address: Option[Expression],
|
||||||
statements: Option[List[Statement]],
|
statements: Option[List[Statement]],
|
||||||
isMacro: Boolean,
|
isMacro: Boolean,
|
||||||
|
@ -6,7 +6,7 @@ import millfork.compiler.{CompilationContext, MfCompiler}
|
|||||||
import millfork.env._
|
import millfork.env._
|
||||||
import millfork.error.ErrorReporting
|
import millfork.error.ErrorReporting
|
||||||
import millfork.node.{CallGraph, Program}
|
import millfork.node.{CallGraph, Program}
|
||||||
import millfork.{CompilationFlag, CompilationOptions, Tarjan}
|
import millfork._
|
||||||
|
|
||||||
import scala.collection.mutable
|
import scala.collection.mutable
|
||||||
|
|
||||||
@ -14,41 +14,41 @@ import scala.collection.mutable
|
|||||||
* @author Karol Stasiak
|
* @author Karol Stasiak
|
||||||
*/
|
*/
|
||||||
|
|
||||||
case class AssemblerOutput(code: Array[Byte], asm: Array[String], labels: List[(String, Int)])
|
case class AssemblerOutput(code: Map[String, Array[Byte]], asm: Array[String], labels: List[(String, Int)])
|
||||||
|
|
||||||
class Assembler(private val program: Program, private val rootEnv: Environment) {
|
class Assembler(private val program: Program, private val rootEnv: Environment, private val platform: Platform) {
|
||||||
|
|
||||||
private var env = rootEnv.allThings
|
private var env = rootEnv.allThings
|
||||||
var unoptimizedCodeSize: Int = 0
|
var unoptimizedCodeSize: Int = 0
|
||||||
var optimizedCodeSize: Int = 0
|
var optimizedCodeSize: Int = 0
|
||||||
var initializedVariablesSize: Int = 0
|
var initializedVariablesSize: Int = 0
|
||||||
|
|
||||||
val mem = new CompiledMemory
|
val mem = new CompiledMemory(platform.bankNumbers.keys.toList)
|
||||||
val labelMap: mutable.Map[String, Int] = mutable.Map()
|
val labelMap: mutable.Map[String, Int] = mutable.Map()
|
||||||
private val bytesToWriteLater = mutable.ListBuffer[(Int, Int, Constant)]()
|
private val bytesToWriteLater = mutable.ListBuffer[(String, Int, Constant)]()
|
||||||
private val wordsToWriteLater = mutable.ListBuffer[(Int, Int, Constant)]()
|
private val wordsToWriteLater = mutable.ListBuffer[(String, Int, Constant)]()
|
||||||
|
|
||||||
def writeByte(bank: Int, addr: Int, value: Byte): Unit = {
|
def writeByte(bank: String, addr: Int, value: Byte): Unit = {
|
||||||
mem.banks(bank).occupied(addr) = true
|
mem.banks(bank).occupied(addr) = true
|
||||||
mem.banks(bank).initialized(addr) = true
|
mem.banks(bank).initialized(addr) = true
|
||||||
mem.banks(bank).readable(addr) = true
|
mem.banks(bank).readable(addr) = true
|
||||||
mem.banks(bank).output(addr) = value.toByte
|
mem.banks(bank).output(addr) = value.toByte
|
||||||
}
|
}
|
||||||
|
|
||||||
def writeByte(bank: Int, addr: Int, value: Constant): Unit = {
|
def writeByte(bank: String, addr: Int, value: Constant): Unit = {
|
||||||
mem.banks(bank).occupied(addr) = true
|
mem.banks(bank).occupied(addr) = true
|
||||||
mem.banks(bank).initialized(addr) = true
|
mem.banks(bank).initialized(addr) = true
|
||||||
mem.banks(bank).readable(addr) = true
|
mem.banks(bank).readable(addr) = true
|
||||||
value match {
|
value match {
|
||||||
case NumericConstant(x, _) =>
|
case NumericConstant(x, _) =>
|
||||||
if (x > 0xffff) ErrorReporting.error("Byte overflow")
|
if (x > 0xff) ErrorReporting.error("Byte overflow")
|
||||||
mem.banks(0).output(addr) = x.toByte
|
mem.banks(bank).output(addr) = x.toByte
|
||||||
case _ =>
|
case _ =>
|
||||||
bytesToWriteLater += ((bank, addr, value))
|
bytesToWriteLater += ((bank, addr, value))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
def writeWord(bank: Int, addr: Int, value: Constant): Unit = {
|
def writeWord(bank: String, addr: Int, value: Constant): Unit = {
|
||||||
mem.banks(bank).occupied(addr) = true
|
mem.banks(bank).occupied(addr) = true
|
||||||
mem.banks(bank).occupied(addr + 1) = true
|
mem.banks(bank).occupied(addr + 1) = true
|
||||||
mem.banks(bank).initialized(addr) = true
|
mem.banks(bank).initialized(addr) = true
|
||||||
@ -202,15 +202,15 @@ class Assembler(private val program: Program, private val rootEnv: Environment)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
val bank0 = mem.banks(0)
|
|
||||||
|
|
||||||
env.allPreallocatables.foreach {
|
env.allPreallocatables.foreach {
|
||||||
case InitializedArray(name, Some(NumericConstant(address, _)), items) =>
|
case thing@InitializedArray(name, Some(NumericConstant(address, _)), items, _) =>
|
||||||
|
val bank = thing.bank(options)
|
||||||
|
val bank0 = mem.banks(bank)
|
||||||
var index = address.toInt
|
var index = address.toInt
|
||||||
assembly.append("* = $" + index.toHexString)
|
assembly.append("* = $" + index.toHexString)
|
||||||
assembly.append(name)
|
assembly.append(name)
|
||||||
for (item <- items) {
|
for (item <- items) {
|
||||||
writeByte(0, index, item)
|
writeByte(bank, index, item)
|
||||||
bank0.occupied(index) = true
|
bank0.occupied(index) = true
|
||||||
bank0.initialized(index) = true
|
bank0.initialized(index) = true
|
||||||
bank0.writeable(index) = true
|
bank0.writeable(index) = true
|
||||||
@ -221,13 +221,15 @@ class Assembler(private val program: Program, private val rootEnv: Environment)
|
|||||||
assembly.append(" !byte " + group.mkString(", "))
|
assembly.append(" !byte " + group.mkString(", "))
|
||||||
}
|
}
|
||||||
initializedVariablesSize += items.length
|
initializedVariablesSize += items.length
|
||||||
case InitializedArray(name, Some(_), items) => ???
|
case thing@InitializedArray(name, Some(_), items, _) => ???
|
||||||
case f: NormalFunction if f.address.isDefined =>
|
case f: NormalFunction if f.address.isDefined =>
|
||||||
|
val bank = f.bank(options)
|
||||||
|
val bank0 = mem.banks(bank)
|
||||||
val index = f.address.get.asInstanceOf[NumericConstant].value.toInt
|
val index = f.address.get.asInstanceOf[NumericConstant].value.toInt
|
||||||
val code = compiledFunctions(f.name)
|
val code = compiledFunctions(f.name)
|
||||||
if (code.nonEmpty) {
|
if (code.nonEmpty) {
|
||||||
labelMap(f.name) = index
|
labelMap(f.name) = index
|
||||||
val end = outputFunction(code, index, assembly, options)
|
val end = outputFunction(bank, code, index, assembly, options)
|
||||||
for(i <- index until end) {
|
for(i <- index until end) {
|
||||||
bank0.occupied(index) = true
|
bank0.occupied(index) = true
|
||||||
bank0.initialized(index) = true
|
bank0.initialized(index) = true
|
||||||
@ -237,68 +239,77 @@ class Assembler(private val program: Program, private val rootEnv: Environment)
|
|||||||
case _ =>
|
case _ =>
|
||||||
}
|
}
|
||||||
|
|
||||||
val codeAllocator = new VariableAllocator(Nil, platform.codeAllocator)
|
val codeAllocators = platform.codeAllocators.mapValues(new VariableAllocator(Nil, _))
|
||||||
var justAfterCode = platform.codeAllocator.startAt
|
var justAfterCode = platform.codeAllocators.mapValues(a => a.startAt)
|
||||||
env.allPreallocatables.foreach {
|
env.allPreallocatables.foreach {
|
||||||
case f: NormalFunction if f.address.isEmpty && f.name == "main" =>
|
case f: NormalFunction if f.address.isEmpty && f.name == "main" =>
|
||||||
|
val bank = f.bank(options)
|
||||||
|
|
||||||
val code = compiledFunctions(f.name)
|
val code = compiledFunctions(f.name)
|
||||||
if (code.nonEmpty) {
|
if (code.nonEmpty) {
|
||||||
val size = code.map(_.sizeInBytes).sum
|
val size = code.map(_.sizeInBytes).sum
|
||||||
val index = codeAllocator.allocateBytes(bank0, options, size, initialized = true, writeable = false)
|
val index = codeAllocators(bank).allocateBytes(mem.banks(bank), options, size, initialized = true, writeable = false)
|
||||||
labelMap(f.name) = index
|
labelMap(f.name) = index
|
||||||
justAfterCode = outputFunction(code, index, assembly, options)
|
justAfterCode += bank -> outputFunction(bank, code, index, assembly, options)
|
||||||
}
|
}
|
||||||
case _ =>
|
case _ =>
|
||||||
}
|
}
|
||||||
env.allPreallocatables.foreach {
|
env.allPreallocatables.foreach {
|
||||||
case f: NormalFunction if f.address.isEmpty && f.name != "main" =>
|
case f: NormalFunction if f.address.isEmpty && f.name != "main" =>
|
||||||
|
val bank = f.bank(options)
|
||||||
|
val bank0 = mem.banks(bank)
|
||||||
val code = compiledFunctions(f.name)
|
val code = compiledFunctions(f.name)
|
||||||
if (code.nonEmpty) {
|
if (code.nonEmpty) {
|
||||||
val size = code.map(_.sizeInBytes).sum
|
val size = code.map(_.sizeInBytes).sum
|
||||||
val index = codeAllocator.allocateBytes(bank0, options, size, initialized = true, writeable = false)
|
val index = codeAllocators(bank).allocateBytes(bank0, options, size, initialized = true, writeable = false)
|
||||||
labelMap(f.name) = index
|
labelMap(f.name) = index
|
||||||
justAfterCode = outputFunction(code, index, assembly, options)
|
justAfterCode += bank -> outputFunction(bank, code, index, assembly, options)
|
||||||
}
|
}
|
||||||
case _ =>
|
case _ =>
|
||||||
}
|
}
|
||||||
env.allPreallocatables.foreach {
|
env.allPreallocatables.foreach {
|
||||||
case InitializedArray(name, None, items) =>
|
case thing@InitializedArray(name, None, items, _) =>
|
||||||
var index = codeAllocator.allocateBytes(bank0, options, items.size, initialized = true, writeable = true)
|
val bank = thing.bank(options)
|
||||||
|
val bank0 = mem.banks(bank)
|
||||||
|
var index = codeAllocators(bank).allocateBytes(bank0, options, items.size, initialized = true, writeable = true)
|
||||||
labelMap(name) = index
|
labelMap(name) = index
|
||||||
assembly.append("* = $" + index.toHexString)
|
assembly.append("* = $" + index.toHexString)
|
||||||
assembly.append(name)
|
assembly.append(name)
|
||||||
for (item <- items) {
|
for (item <- items) {
|
||||||
writeByte(0, index, item)
|
writeByte(bank, index, item)
|
||||||
index += 1
|
index += 1
|
||||||
}
|
}
|
||||||
items.grouped(16).foreach {group =>
|
items.grouped(16).foreach {group =>
|
||||||
assembly.append(" !byte " + group.mkString(", "))
|
assembly.append(" !byte " + group.mkString(", "))
|
||||||
}
|
}
|
||||||
initializedVariablesSize += items.length
|
initializedVariablesSize += items.length
|
||||||
justAfterCode = index
|
justAfterCode += bank -> index
|
||||||
case m@InitializedMemoryVariable(name, None, typ, value) =>
|
case m@InitializedMemoryVariable(name, None, typ, value, _) =>
|
||||||
var index = codeAllocator.allocateBytes(bank0, options, typ.size, initialized = true, writeable = true)
|
val bank = m.bank(options)
|
||||||
|
val bank0 = mem.banks(bank)
|
||||||
|
var index = codeAllocators(bank).allocateBytes(bank0, options, typ.size, initialized = true, writeable = true)
|
||||||
labelMap(name) = index
|
labelMap(name) = index
|
||||||
val altName = m.name.stripPrefix(env.prefix) + "`"
|
val altName = m.name.stripPrefix(env.prefix) + "`"
|
||||||
env.things += altName -> ConstantThing(altName, NumericConstant(index, 2), env.get[Type]("pointer"))
|
env.things += altName -> ConstantThing(altName, NumericConstant(index, 2), env.get[Type]("pointer"))
|
||||||
assembly.append("* = $" + index.toHexString)
|
assembly.append("* = $" + index.toHexString)
|
||||||
assembly.append(name)
|
assembly.append(name)
|
||||||
for (i <- 0 until typ.size) {
|
for (i <- 0 until typ.size) {
|
||||||
writeByte(0, index, value.subbyte(i))
|
writeByte(bank, index, value.subbyte(i))
|
||||||
assembly.append(" !byte " + value.subbyte(i).quickSimplify)
|
assembly.append(" !byte " + value.subbyte(i).quickSimplify)
|
||||||
index += 1
|
index += 1
|
||||||
}
|
}
|
||||||
initializedVariablesSize += typ.size
|
initializedVariablesSize += typ.size
|
||||||
justAfterCode = index
|
justAfterCode += bank -> index
|
||||||
case _ =>
|
case _ =>
|
||||||
}
|
}
|
||||||
env.getAllFixedAddressObjects.foreach {
|
env.getAllFixedAddressObjects.foreach {
|
||||||
case (addr, size) =>
|
case (bank, addr, size) =>
|
||||||
|
val bank0 = mem.banks(bank)
|
||||||
for(i <- 0 until size) bank0.occupied(addr + i) = true
|
for(i <- 0 until size) bank0.occupied(addr + i) = true
|
||||||
}
|
}
|
||||||
val variableAllocator = platform.variableAllocator
|
val variableAllocators = platform.variableAllocators
|
||||||
variableAllocator.notifyAboutEndOfCode(justAfterCode)
|
variableAllocators.foreach{case (b,a) => a.notifyAboutEndOfCode(justAfterCode(b))}
|
||||||
env.allocateVariables(None, bank0, callGraph, variableAllocator, options, labelMap.put)
|
env.allocateVariables(None, mem, callGraph, variableAllocators, options, labelMap.put)
|
||||||
|
|
||||||
env = rootEnv.allThings
|
env = rootEnv.allThings
|
||||||
|
|
||||||
@ -327,7 +338,12 @@ class Assembler(private val program: Program, private val rootEnv: Environment)
|
|||||||
assembly += f" ; $$$v%04X = $l%s"
|
assembly += f" ; $$$v%04X = $l%s"
|
||||||
}
|
}
|
||||||
|
|
||||||
AssemblerOutput(platform.outputPackager.packageOutput(mem, 0), assembly.toArray, labelMap.toList)
|
// TODO:
|
||||||
|
val code = (platform.outputStyle match {
|
||||||
|
case OutputStyle.Single => List("default")
|
||||||
|
case OutputStyle.PerBank => platform.bankNumbers.keys.toList
|
||||||
|
}).map(b => b -> platform.outputPackager.packageOutput(mem, b)).toMap
|
||||||
|
AssemblerOutput(code, assembly.toArray, labelMap.toList)
|
||||||
}
|
}
|
||||||
|
|
||||||
private def compileFunction(f: NormalFunction, optimizations: Seq[AssemblyOptimization], options: CompilationOptions, inlinedFunctions: Map[String, List[AssemblyLine]]): List[AssemblyLine] = {
|
private def compileFunction(f: NormalFunction, optimizations: Seq[AssemblyOptimization], options: CompilationOptions, inlinedFunctions: Map[String, List[AssemblyLine]]): List[AssemblyLine] = {
|
||||||
@ -363,7 +379,7 @@ class Assembler(private val program: Program, private val rootEnv: Environment)
|
|||||||
else code
|
else code
|
||||||
}
|
}
|
||||||
|
|
||||||
private def outputFunction(code: List[AssemblyLine], startFrom: Int, assOut: mutable.ArrayBuffer[String], options: CompilationOptions): Int = {
|
private def outputFunction(bank: String, code: List[AssemblyLine], startFrom: Int, assOut: mutable.ArrayBuffer[String], options: CompilationOptions): Int = {
|
||||||
var index = startFrom
|
var index = startFrom
|
||||||
assOut.append("* = $" + startFrom.toHexString)
|
assOut.append("* = $" + startFrom.toHexString)
|
||||||
import millfork.assembly.AddrMode._
|
import millfork.assembly.AddrMode._
|
||||||
@ -378,24 +394,24 @@ class Assembler(private val program: Program, private val rootEnv: Environment)
|
|||||||
case AssemblyLine(_, DoesNotExist, _, _) =>
|
case AssemblyLine(_, DoesNotExist, _, _) =>
|
||||||
()
|
()
|
||||||
case AssemblyLine(op, Implied, _, _) =>
|
case AssemblyLine(op, Implied, _, _) =>
|
||||||
writeByte(0, index, Assembler.opcodeFor(op, Implied, options))
|
writeByte(bank, index, Assembler.opcodeFor(op, Implied, options))
|
||||||
index += 1
|
index += 1
|
||||||
case AssemblyLine(op, Relative, param, _) =>
|
case AssemblyLine(op, Relative, param, _) =>
|
||||||
writeByte(0, index, Assembler.opcodeFor(op, Relative, options))
|
writeByte(bank, index, Assembler.opcodeFor(op, Relative, options))
|
||||||
writeByte(0, index + 1, param - (index + 2))
|
writeByte(bank, index + 1, param - (index + 2))
|
||||||
index += 2
|
index += 2
|
||||||
case AssemblyLine(op, am@(Immediate | ZeroPage | ZeroPageX | ZeroPageY | IndexedY | IndexedX | IndexedZ | LongIndexedY | LongIndexedZ | Stack), param, _) =>
|
case AssemblyLine(op, am@(Immediate | ZeroPage | ZeroPageX | ZeroPageY | IndexedY | IndexedX | IndexedZ | LongIndexedY | LongIndexedZ | Stack), param, _) =>
|
||||||
writeByte(0, index, Assembler.opcodeFor(op, am, options))
|
writeByte(bank, index, Assembler.opcodeFor(op, am, options))
|
||||||
writeByte(0, index + 1, param)
|
writeByte(bank, index + 1, param)
|
||||||
index += 2
|
index += 2
|
||||||
case AssemblyLine(op, am@(WordImmediate | Absolute | AbsoluteY | AbsoluteX | Indirect | AbsoluteIndexedX), param, _) =>
|
case AssemblyLine(op, am@(WordImmediate | Absolute | AbsoluteY | AbsoluteX | Indirect | AbsoluteIndexedX), param, _) =>
|
||||||
writeByte(0, index, Assembler.opcodeFor(op, am, options))
|
writeByte(bank, index, Assembler.opcodeFor(op, am, options))
|
||||||
writeWord(0, index + 1, param)
|
writeWord(bank, index + 1, param)
|
||||||
index += 3
|
index += 3
|
||||||
case AssemblyLine(op, am@(LongAbsolute | LongAbsoluteX | LongIndirect), param, _) =>
|
case AssemblyLine(op, am@(LongAbsolute | LongAbsoluteX | LongIndirect), param, _) =>
|
||||||
writeByte(0, index, Assembler.opcodeFor(op, am, options))
|
writeByte(bank, index, Assembler.opcodeFor(op, am, options))
|
||||||
writeWord(0, index + 1, param)
|
writeWord(bank, index + 1, param)
|
||||||
writeByte(0, index + 3, extractBank(param, options))
|
writeByte(bank, index + 3, extractBank(param, options))
|
||||||
index += 4
|
index += 4
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,8 +5,8 @@ import scala.collection.mutable
|
|||||||
/**
|
/**
|
||||||
* @author Karol Stasiak
|
* @author Karol Stasiak
|
||||||
*/
|
*/
|
||||||
class CompiledMemory {
|
class CompiledMemory(bankNames: List[String]) {
|
||||||
val banks = mutable.Map(0 -> new MemoryBank)
|
val banks = mutable.Map(bankNames.map(_ -> new MemoryBank): _*)
|
||||||
}
|
}
|
||||||
|
|
||||||
class MemoryBank {
|
class MemoryBank {
|
||||||
|
@ -6,11 +6,11 @@ import java.io.ByteArrayOutputStream
|
|||||||
* @author Karol Stasiak
|
* @author Karol Stasiak
|
||||||
*/
|
*/
|
||||||
trait OutputPackager {
|
trait OutputPackager {
|
||||||
def packageOutput(mem: CompiledMemory, bank: Int): Array[Byte]
|
def packageOutput(mem: CompiledMemory, bank: String): Array[Byte]
|
||||||
}
|
}
|
||||||
|
|
||||||
case class SequenceOutput(children: List[OutputPackager]) extends OutputPackager {
|
case class SequenceOutput(children: List[OutputPackager]) extends OutputPackager {
|
||||||
def packageOutput(mem: CompiledMemory, bank: Int): Array[Byte] = {
|
def packageOutput(mem: CompiledMemory, bank: String): Array[Byte] = {
|
||||||
val baos = new ByteArrayOutputStream
|
val baos = new ByteArrayOutputStream
|
||||||
children.foreach { c =>
|
children.foreach { c =>
|
||||||
val a = c.packageOutput(mem, bank)
|
val a = c.packageOutput(mem, bank)
|
||||||
@ -21,39 +21,39 @@ case class SequenceOutput(children: List[OutputPackager]) extends OutputPackager
|
|||||||
}
|
}
|
||||||
|
|
||||||
case class ConstOutput(byte: Byte) extends OutputPackager {
|
case class ConstOutput(byte: Byte) extends OutputPackager {
|
||||||
def packageOutput(mem: CompiledMemory, bank: Int): Array[Byte] = Array(byte)
|
def packageOutput(mem: CompiledMemory, bank: String): Array[Byte] = Array(byte)
|
||||||
}
|
}
|
||||||
|
|
||||||
case class CurrentBankFragmentOutput(start: Int, end: Int) extends OutputPackager {
|
case class CurrentBankFragmentOutput(start: Int, end: Int) extends OutputPackager {
|
||||||
def packageOutput(mem: CompiledMemory, bank: Int): Array[Byte] = {
|
def packageOutput(mem: CompiledMemory, bank: String): Array[Byte] = {
|
||||||
val b = mem.banks(bank)
|
val b = mem.banks(bank)
|
||||||
b.output.slice(start, end + 1)
|
b.output.slice(start, end + 1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
case class BankFragmentOutput(alwaysBank: Int, start: Int, end: Int) extends OutputPackager {
|
case class BankFragmentOutput(alwaysBank: String, start: Int, end: Int) extends OutputPackager {
|
||||||
def packageOutput(mem: CompiledMemory, bank: Int): Array[Byte] = {
|
def packageOutput(mem: CompiledMemory, bank: String): Array[Byte] = {
|
||||||
val b = mem.banks(alwaysBank)
|
val b = mem.banks(alwaysBank)
|
||||||
b.output.slice(start, end + 1)
|
b.output.slice(start, end + 1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
object StartAddressOutput extends OutputPackager {
|
object StartAddressOutput extends OutputPackager {
|
||||||
def packageOutput(mem: CompiledMemory, bank: Int): Array[Byte] = {
|
def packageOutput(mem: CompiledMemory, bank: String): Array[Byte] = {
|
||||||
val b = mem.banks(bank)
|
val b = mem.banks(bank)
|
||||||
Array(b.start.toByte, b.start.>>(8).toByte)
|
Array(b.start.toByte, b.start.>>(8).toByte)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
object EndAddressOutput extends OutputPackager {
|
object EndAddressOutput extends OutputPackager {
|
||||||
def packageOutput(mem: CompiledMemory, bank: Int): Array[Byte] = {
|
def packageOutput(mem: CompiledMemory, bank: String): Array[Byte] = {
|
||||||
val b = mem.banks(bank)
|
val b = mem.banks(bank)
|
||||||
Array(b.end.toByte, b.end.>>(8).toByte)
|
Array(b.end.toByte, b.end.>>(8).toByte)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
object AllocatedDataOutput extends OutputPackager {
|
object AllocatedDataOutput extends OutputPackager {
|
||||||
def packageOutput(mem: CompiledMemory, bank: Int): Array[Byte] = {
|
def packageOutput(mem: CompiledMemory, bank: String): Array[Byte] = {
|
||||||
val b = mem.banks(bank)
|
val b = mem.banks(bank)
|
||||||
b.output.slice(b.start, b.end + 1)
|
b.output.slice(b.start, b.end + 1)
|
||||||
}
|
}
|
||||||
|
@ -140,6 +140,7 @@ case class MfParser(filename: String, input: String, currentDirectory: String, o
|
|||||||
|
|
||||||
def variableDefinition(implicitlyGlobal: Boolean): P[Seq[DeclarationStatement]] = for {
|
def variableDefinition(implicitlyGlobal: Boolean): P[Seq[DeclarationStatement]] = for {
|
||||||
p <- position()
|
p <- position()
|
||||||
|
bank <- bankDeclaration
|
||||||
flags <- flags("const", "static", "volatile", "stack", "register") ~ HWS
|
flags <- flags("const", "static", "volatile", "stack", "register") ~ HWS
|
||||||
typ <- identifier ~ SWS
|
typ <- identifier ~ SWS
|
||||||
name <- identifier ~/ HWS ~/ Pass
|
name <- identifier ~/ HWS ~/ Pass
|
||||||
@ -148,6 +149,7 @@ case class MfParser(filename: String, input: String, currentDirectory: String, o
|
|||||||
_ <- &(EOL) ~/ ""
|
_ <- &(EOL) ~/ ""
|
||||||
} yield {
|
} yield {
|
||||||
Seq(VariableDeclarationStatement(name, typ,
|
Seq(VariableDeclarationStatement(name, typ,
|
||||||
|
bank,
|
||||||
global = implicitlyGlobal || flags("static"),
|
global = implicitlyGlobal || flags("static"),
|
||||||
stack = flags("stack"),
|
stack = flags("stack"),
|
||||||
constant = flags("const"),
|
constant = flags("const"),
|
||||||
@ -229,11 +231,12 @@ case class MfParser(filename: String, input: String, currentDirectory: String, o
|
|||||||
|
|
||||||
def arrayDefinition: P[Seq[ArrayDeclarationStatement]] = for {
|
def arrayDefinition: P[Seq[ArrayDeclarationStatement]] = for {
|
||||||
p <- position()
|
p <- position()
|
||||||
|
bank <- bankDeclaration
|
||||||
name <- "array" ~ !letterOrDigit ~/ SWS ~ identifier ~ HWS
|
name <- "array" ~ !letterOrDigit ~/ SWS ~ identifier ~ HWS
|
||||||
length <- ("[" ~/ AWS ~/ mlExpression(nonStatementLevel) ~ AWS ~ "]").? ~ HWS
|
length <- ("[" ~/ AWS ~/ mlExpression(nonStatementLevel) ~ AWS ~ "]").? ~ HWS
|
||||||
addr <- ("@" ~/ HWS ~/ mlExpression(1)).? ~/ HWS
|
addr <- ("@" ~/ HWS ~/ mlExpression(1)).? ~/ HWS
|
||||||
contents <- ("=" ~/ HWS ~/ arrayContents).? ~/ HWS
|
contents <- ("=" ~/ HWS ~/ arrayContents).? ~/ HWS
|
||||||
} yield Seq(ArrayDeclarationStatement(name, length, addr, contents).pos(p))
|
} yield Seq(ArrayDeclarationStatement(name, bank, length, addr, contents).pos(p))
|
||||||
|
|
||||||
def tightMlExpression: P[Expression] = P(mlParenExpr | functionCall | mlIndexedExpression | atom) // TODO
|
def tightMlExpression: P[Expression] = P(mlParenExpr | functionCall | mlIndexedExpression | atom) // TODO
|
||||||
|
|
||||||
@ -463,8 +466,14 @@ case class MfParser(filename: String, input: String, currentDirectory: String, o
|
|||||||
condition <- "while" ~ !letterOrDigit ~/ HWS ~/ mlExpression(nonStatementLevel)
|
condition <- "while" ~ !letterOrDigit ~/ HWS ~/ mlExpression(nonStatementLevel)
|
||||||
} yield Seq(DoWhileStatement(body.toList, Nil, condition))
|
} yield Seq(DoWhileStatement(body.toList, Nil, condition))
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def bankDeclaration: P[Option[String]] = ("segment" ~/ AWS ~/ "(" ~/ AWS ~/ identifier ~/ AWS ~/ ")" ~/ AWS).?
|
||||||
|
|
||||||
def functionDefinition: P[Seq[DeclarationStatement]] = for {
|
def functionDefinition: P[Seq[DeclarationStatement]] = for {
|
||||||
p <- position()
|
p <- position()
|
||||||
|
bank <- bankDeclaration
|
||||||
flags <- flags("asm", "inline", "interrupt", "macro", "noinline", "reentrant", "kernal_interrupt") ~ HWS
|
flags <- flags("asm", "inline", "interrupt", "macro", "noinline", "reentrant", "kernal_interrupt") ~ HWS
|
||||||
returnType <- identifier ~ SWS
|
returnType <- identifier ~ SWS
|
||||||
name <- identifier ~ HWS
|
name <- identifier ~ HWS
|
||||||
@ -510,6 +519,7 @@ case class MfParser(filename: String, input: String, currentDirectory: String, o
|
|||||||
case None => ()
|
case None => ()
|
||||||
}
|
}
|
||||||
Seq(FunctionDeclarationStatement(name, returnType, params.toList,
|
Seq(FunctionDeclarationStatement(name, returnType, params.toList,
|
||||||
|
bank,
|
||||||
addr,
|
addr,
|
||||||
statements,
|
statements,
|
||||||
flags("macro"),
|
flags("macro"),
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
package millfork.test.emu
|
package millfork.test.emu
|
||||||
|
|
||||||
import millfork.output.{AfterCodeByteAllocator, CurrentBankFragmentOutput, UpwardByteAllocator, VariableAllocator}
|
import millfork.output.{AfterCodeByteAllocator, CurrentBankFragmentOutput, UpwardByteAllocator, VariableAllocator}
|
||||||
import millfork.{Cpu, Platform}
|
import millfork.{Cpu, OutputStyle, Platform}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author Karol Stasiak
|
* @author Karol Stasiak
|
||||||
@ -12,8 +12,11 @@ object EmuPlatform {
|
|||||||
Map(),
|
Map(),
|
||||||
Nil,
|
Nil,
|
||||||
CurrentBankFragmentOutput(0, 0xffff),
|
CurrentBankFragmentOutput(0, 0xffff),
|
||||||
new UpwardByteAllocator(0x200, 0xb000),
|
Map("default" -> new UpwardByteAllocator(0x200, 0xb000)),
|
||||||
new VariableAllocator((0 until 256 by 2).toList, new AfterCodeByteAllocator(0xff00)),
|
Map("default" -> new VariableAllocator((0 until 256 by 2).toList, new AfterCodeByteAllocator(0xff00))),
|
||||||
".bin"
|
".bin",
|
||||||
|
Map("default" -> 0),
|
||||||
|
"default",
|
||||||
|
OutputStyle.Single
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -147,14 +147,14 @@ class EmuRun(cpu: millfork.Cpu.Value, nodeOptimizations: List[NodeOptimization],
|
|||||||
// compile
|
// compile
|
||||||
val env2 = new Environment(None, "")
|
val env2 = new Environment(None, "")
|
||||||
env2.collectDeclarations(program, options)
|
env2.collectDeclarations(program, options)
|
||||||
val assembler = new Assembler(program, env2)
|
val assembler = new Assembler(program, env2, platform)
|
||||||
val output = assembler.assemble(callGraph, assemblyOptimizations, options)
|
val output = assembler.assemble(callGraph, assemblyOptimizations, options)
|
||||||
println(";;; compiled: -----------------")
|
println(";;; compiled: -----------------")
|
||||||
output.asm.takeWhile(s => !(s.startsWith(".") && s.contains("= $"))).filterNot(_.contains("; DISCARD_")).foreach(println)
|
output.asm.takeWhile(s => !(s.startsWith(".") && s.contains("= $"))).filterNot(_.contains("; DISCARD_")).foreach(println)
|
||||||
println(";;; ---------------------------")
|
println(";;; ---------------------------")
|
||||||
assembler.labelMap.foreach { case (l, addr) => println(f"$l%-15s $$$addr%04x") }
|
assembler.labelMap.foreach { case (l, addr) => println(f"$l%-15s $$$addr%04x") }
|
||||||
|
|
||||||
val optimizedSize = assembler.mem.banks(0).initialized.count(identity).toLong
|
val optimizedSize = assembler.mem.banks("default").initialized.count(identity).toLong
|
||||||
if (unoptimizedSize == optimizedSize) {
|
if (unoptimizedSize == optimizedSize) {
|
||||||
println(f"Size: $unoptimizedSize%5d B")
|
println(f"Size: $unoptimizedSize%5d B")
|
||||||
} else {
|
} else {
|
||||||
@ -165,7 +165,7 @@ class EmuRun(cpu: millfork.Cpu.Value, nodeOptimizations: List[NodeOptimization],
|
|||||||
|
|
||||||
ErrorReporting.assertNoErrors("Code generation failed")
|
ErrorReporting.assertNoErrors("Code generation failed")
|
||||||
|
|
||||||
val memoryBank = assembler.mem.banks(0)
|
val memoryBank = assembler.mem.banks("default")
|
||||||
if (source.contains("return [")) {
|
if (source.contains("return [")) {
|
||||||
for (_ <- 0 until 10; i <- 0xfffe.to(0, -1)) {
|
for (_ <- 0 until 10; i <- 0xfffe.to(0, -1)) {
|
||||||
if (memoryBank.readable(i)) memoryBank.readable(i + 1) = true
|
if (memoryBank.readable(i)) memoryBank.readable(i + 1) = true
|
||||||
@ -173,14 +173,14 @@ class EmuRun(cpu: millfork.Cpu.Value, nodeOptimizations: List[NodeOptimization],
|
|||||||
}
|
}
|
||||||
platform.cpu match {
|
platform.cpu match {
|
||||||
case millfork.Cpu.Cmos =>
|
case millfork.Cpu.Cmos =>
|
||||||
runViaSymon(memoryBank, platform.codeAllocator.startAt, CpuBehavior.CMOS_6502)
|
runViaSymon(memoryBank, platform.codeAllocators("default").startAt, CpuBehavior.CMOS_6502)
|
||||||
case millfork.Cpu.Ricoh =>
|
case millfork.Cpu.Ricoh =>
|
||||||
runViaHalfnes(memoryBank, platform.codeAllocator.startAt)
|
runViaHalfnes(memoryBank, platform.codeAllocators("default").startAt)
|
||||||
case millfork.Cpu.Mos =>
|
case millfork.Cpu.Mos =>
|
||||||
ErrorReporting.fatal("There's no NMOS emulator with decimal mode support")
|
ErrorReporting.fatal("There's no NMOS emulator with decimal mode support")
|
||||||
Timings(-1, -1) -> memoryBank
|
Timings(-1, -1) -> memoryBank
|
||||||
case millfork.Cpu.StrictMos | millfork.Cpu.StrictRicoh =>
|
case millfork.Cpu.StrictMos | millfork.Cpu.StrictRicoh =>
|
||||||
runViaSymon(memoryBank, platform.codeAllocator.startAt, CpuBehavior.NMOS_6502)
|
runViaSymon(memoryBank, platform.codeAllocators("default").startAt, CpuBehavior.NMOS_6502)
|
||||||
case _ =>
|
case _ =>
|
||||||
ErrorReporting.trace("No emulation support for " + platform.cpu)
|
ErrorReporting.trace("No emulation support for " + platform.cpu)
|
||||||
Timings(-1, -1) -> memoryBank
|
Timings(-1, -1) -> memoryBank
|
||||||
|
Loading…
x
Reference in New Issue
Block a user