mirror of
https://github.com/irmen/prog8.git
synced 2025-01-12 04:30:03 +00:00
slight optimization of repeat loop (0-256 iters) code generation on 65c02 cpu
This commit is contained in:
parent
583e208c1e
commit
1e8d8e40a2
@ -1128,13 +1128,13 @@ class AsmGen(private val program: Program,
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun translate(stmt: RepeatLoop) {
|
private fun translate(stmt: RepeatLoop) {
|
||||||
val repeatLabel = makeLabel("repeat")
|
|
||||||
val endLabel = makeLabel("repeatend")
|
val endLabel = makeLabel("repeatend")
|
||||||
loopEndLabels.push(endLabel)
|
loopEndLabels.push(endLabel)
|
||||||
|
|
||||||
when (stmt.iterations) {
|
when (stmt.iterations) {
|
||||||
null -> {
|
null -> {
|
||||||
// endless loop
|
// endless loop
|
||||||
|
val repeatLabel = makeLabel("repeat")
|
||||||
out(repeatLabel)
|
out(repeatLabel)
|
||||||
translate(stmt.body)
|
translate(stmt.body)
|
||||||
jmp(repeatLabel)
|
jmp(repeatLabel)
|
||||||
@ -1146,13 +1146,11 @@ class AsmGen(private val program: Program,
|
|||||||
throw AssemblyError("invalid number of iterations")
|
throw AssemblyError("invalid number of iterations")
|
||||||
when {
|
when {
|
||||||
iterations == 0 -> {}
|
iterations == 0 -> {}
|
||||||
iterations <= 256 -> {
|
iterations == 1 -> translate(stmt.body)
|
||||||
out(" lda #${iterations and 255}")
|
iterations <= 256 -> repeatByteCount(iterations, stmt)
|
||||||
repeatByteCountInA(iterations, repeatLabel, endLabel, stmt)
|
|
||||||
}
|
|
||||||
else -> {
|
else -> {
|
||||||
out(" lda #<${iterations} | ldy #>${iterations}")
|
out(" lda #<${iterations} | ldy #>${iterations}")
|
||||||
repeatWordCountInAY(iterations, repeatLabel, endLabel, stmt)
|
repeatWordCountInAY(iterations, endLabel, stmt)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1161,12 +1159,12 @@ class AsmGen(private val program: Program,
|
|||||||
val name = asmVariableName(stmt.iterations as IdentifierReference)
|
val name = asmVariableName(stmt.iterations as IdentifierReference)
|
||||||
when(vardecl.datatype) {
|
when(vardecl.datatype) {
|
||||||
DataType.UBYTE, DataType.BYTE -> {
|
DataType.UBYTE, DataType.BYTE -> {
|
||||||
assignVariableToRegister(name, RegisterOrPair.A)
|
assignVariableToRegister(name, RegisterOrPair.Y)
|
||||||
repeatByteCountInA(null, repeatLabel, endLabel, stmt)
|
repeatCountInY(stmt, endLabel)
|
||||||
}
|
}
|
||||||
DataType.UWORD, DataType.WORD -> {
|
DataType.UWORD, DataType.WORD -> {
|
||||||
assignVariableToRegister(name, RegisterOrPair.AY)
|
assignVariableToRegister(name, RegisterOrPair.AY)
|
||||||
repeatWordCountInAY(null, repeatLabel, endLabel, stmt)
|
repeatWordCountInAY(null, endLabel, stmt)
|
||||||
}
|
}
|
||||||
else -> throw AssemblyError("invalid loop variable datatype $vardecl")
|
else -> throw AssemblyError("invalid loop variable datatype $vardecl")
|
||||||
}
|
}
|
||||||
@ -1177,12 +1175,12 @@ class AsmGen(private val program: Program,
|
|||||||
throw AssemblyError("unknown dt")
|
throw AssemblyError("unknown dt")
|
||||||
when (dt.getOr(DataType.UNDEFINED)) {
|
when (dt.getOr(DataType.UNDEFINED)) {
|
||||||
in ByteDatatypes -> {
|
in ByteDatatypes -> {
|
||||||
assignExpressionToRegister(stmt.iterations!!, RegisterOrPair.A)
|
assignExpressionToRegister(stmt.iterations!!, RegisterOrPair.Y)
|
||||||
repeatByteCountInA(null, repeatLabel, endLabel, stmt)
|
repeatCountInY(stmt, endLabel)
|
||||||
}
|
}
|
||||||
in WordDatatypes -> {
|
in WordDatatypes -> {
|
||||||
assignExpressionToRegister(stmt.iterations!!, RegisterOrPair.AY)
|
assignExpressionToRegister(stmt.iterations!!, RegisterOrPair.AY)
|
||||||
repeatWordCountInAY(null, repeatLabel, endLabel, stmt)
|
repeatWordCountInAY(null, endLabel, stmt)
|
||||||
}
|
}
|
||||||
else -> throw AssemblyError("invalid loop expression datatype $dt")
|
else -> throw AssemblyError("invalid loop expression datatype $dt")
|
||||||
}
|
}
|
||||||
@ -1192,13 +1190,14 @@ class AsmGen(private val program: Program,
|
|||||||
loopEndLabels.pop()
|
loopEndLabels.pop()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun repeatWordCountInAY(constIterations: Int?, repeatLabel: String, endLabel: String, stmt: RepeatLoop) {
|
private fun repeatWordCountInAY(constIterations: Int?, endLabel: String, stmt: RepeatLoop) {
|
||||||
// note: A/Y must have been loaded with the number of iterations!
|
// note: A/Y must have been loaded with the number of iterations!
|
||||||
if(constIterations==0)
|
if(constIterations==0)
|
||||||
return
|
return
|
||||||
// no need to explicitly test for 0 iterations as this is done in the countdown logic below
|
// no need to explicitly test for 0 iterations as this is done in the countdown logic below
|
||||||
|
|
||||||
val counterVar: String = createRepeatCounterVar(DataType.UWORD, constIterations, stmt)
|
val repeatLabel = makeLabel("repeat")
|
||||||
|
val counterVar = createRepeatCounterVar(DataType.UWORD, false, stmt)!!
|
||||||
out("""
|
out("""
|
||||||
sta $counterVar
|
sta $counterVar
|
||||||
sty $counterVar+1
|
sty $counterVar+1
|
||||||
@ -1216,23 +1215,58 @@ $repeatLabel lda $counterVar
|
|||||||
out(endLabel)
|
out(endLabel)
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun repeatByteCountInA(constIterations: Int?, repeatLabel: String, endLabel: String, stmt: RepeatLoop) {
|
private fun repeatByteCount(count: Int, stmt: RepeatLoop) {
|
||||||
// note: A must be loaded with the number of iterations!
|
require(count in 2..256)
|
||||||
if(constIterations==0)
|
val repeatLabel = makeLabel("repeat")
|
||||||
return
|
if(isTargetCpu(CpuType.CPU65c02)) {
|
||||||
|
val counterVar = createRepeatCounterVar(DataType.UBYTE, true, stmt)
|
||||||
if(constIterations==null)
|
if(counterVar!=null) {
|
||||||
out(" beq $endLabel ; skip loop if zero iters")
|
out(" lda #${count and 255} | sta $counterVar")
|
||||||
val counterVar = createRepeatCounterVar(DataType.UBYTE, constIterations, stmt)
|
|
||||||
out(" sta $counterVar")
|
|
||||||
out(repeatLabel)
|
out(repeatLabel)
|
||||||
translate(stmt.body)
|
translate(stmt.body)
|
||||||
out(" dec $counterVar | bne $repeatLabel")
|
out(" dec $counterVar | bne $repeatLabel")
|
||||||
if(constIterations==null)
|
} else {
|
||||||
|
out(" ldy #${count and 255}")
|
||||||
|
out("$repeatLabel phy")
|
||||||
|
translate(stmt.body)
|
||||||
|
out(" ply | dey | bne $repeatLabel")
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
val counterVar = createRepeatCounterVar(DataType.UBYTE, false, stmt)
|
||||||
|
out(" lda #${count and 255} | sta $counterVar")
|
||||||
|
out(repeatLabel)
|
||||||
|
translate(stmt.body)
|
||||||
|
out(" dec $counterVar | bne $repeatLabel")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun repeatCountInY(stmt: RepeatLoop, endLabel: String) {
|
||||||
|
// note: Y must just have been loaded with the (variable) number of loops to be performed!
|
||||||
|
val repeatLabel = makeLabel("repeat")
|
||||||
|
if(isTargetCpu(CpuType.CPU65c02)) {
|
||||||
|
val counterVar = createRepeatCounterVar(DataType.UBYTE, true, stmt)
|
||||||
|
if(counterVar!=null) {
|
||||||
|
out(" beq $endLabel | sty $counterVar")
|
||||||
|
out(repeatLabel)
|
||||||
|
translate(stmt.body)
|
||||||
|
out(" dec $counterVar | bne $repeatLabel")
|
||||||
|
} else {
|
||||||
|
out(" beq $endLabel")
|
||||||
|
out("$repeatLabel phy")
|
||||||
|
translate(stmt.body)
|
||||||
|
out(" ply | dey | bne $repeatLabel")
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
val counterVar = createRepeatCounterVar(DataType.UBYTE, false, stmt)!!
|
||||||
|
out(" beq $endLabel | sty $counterVar")
|
||||||
|
out(repeatLabel)
|
||||||
|
translate(stmt.body)
|
||||||
|
out(" dec $counterVar | bne $repeatLabel")
|
||||||
|
}
|
||||||
out(endLabel)
|
out(endLabel)
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun createRepeatCounterVar(dt: DataType, constIterations: Int?, stmt: RepeatLoop): String {
|
private fun createRepeatCounterVar(dt: DataType, mustBeInZeropage: Boolean, stmt: RepeatLoop): String? {
|
||||||
val asmInfo = stmt.definingSubroutine!!.asmGenInfo
|
val asmInfo = stmt.definingSubroutine!!.asmGenInfo
|
||||||
var parent = stmt.parent
|
var parent = stmt.parent
|
||||||
while(parent !is ParentSentinel) {
|
while(parent !is ParentSentinel) {
|
||||||
@ -1245,27 +1279,33 @@ $repeatLabel lda $counterVar
|
|||||||
if(!isNested) {
|
if(!isNested) {
|
||||||
// we can re-use a counter var from the subroutine if it already has one for that datatype
|
// we can re-use a counter var from the subroutine if it already has one for that datatype
|
||||||
val existingVar = asmInfo.extraVars.firstOrNull { it.first==dt }
|
val existingVar = asmInfo.extraVars.firstOrNull { it.first==dt }
|
||||||
if(existingVar!=null)
|
if(existingVar!=null) {
|
||||||
|
if(!mustBeInZeropage || existingVar.third!=null)
|
||||||
return existingVar.second
|
return existingVar.second
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
val counterVar = makeLabel("repeatcounter")
|
val counterVar = makeLabel("counter")
|
||||||
when(dt) {
|
when(dt) {
|
||||||
DataType.UBYTE -> {
|
DataType.UBYTE -> {
|
||||||
if(constIterations!=null && constIterations>=16 && zeropage.hasByteAvailable()) {
|
if(zeropage.hasByteAvailable()) {
|
||||||
// allocate count var on ZP
|
// allocate count var on ZP
|
||||||
val zpAddr = zeropage.allocate(counterVar, DataType.UBYTE, stmt.position, errors)
|
val zpAddr = zeropage.allocate(counterVar, DataType.UBYTE, stmt.position, errors)
|
||||||
asmInfo.extraVars.add(Triple(DataType.UBYTE, counterVar, zpAddr))
|
asmInfo.extraVars.add(Triple(DataType.UBYTE, counterVar, zpAddr))
|
||||||
} else {
|
} else {
|
||||||
|
if(mustBeInZeropage)
|
||||||
|
return null
|
||||||
asmInfo.extraVars.add(Triple(DataType.UBYTE, counterVar, null))
|
asmInfo.extraVars.add(Triple(DataType.UBYTE, counterVar, null))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DataType.UWORD -> {
|
DataType.UWORD -> {
|
||||||
if(constIterations!=null && constIterations>=16 && zeropage.hasWordAvailable()) {
|
if(zeropage.hasWordAvailable()) {
|
||||||
// allocate count var on ZP
|
// allocate count var on ZP
|
||||||
val zpAddr = zeropage.allocate(counterVar, DataType.UWORD, stmt.position, errors)
|
val zpAddr = zeropage.allocate(counterVar, DataType.UWORD, stmt.position, errors)
|
||||||
asmInfo.extraVars.add(Triple(DataType.UWORD, counterVar, zpAddr))
|
asmInfo.extraVars.add(Triple(DataType.UWORD, counterVar, zpAddr))
|
||||||
} else {
|
} else {
|
||||||
|
if(mustBeInZeropage)
|
||||||
|
return null
|
||||||
asmInfo.extraVars.add(Triple(DataType.UWORD, counterVar, null))
|
asmInfo.extraVars.add(Triple(DataType.UWORD, counterVar, null))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,12 +3,14 @@ TODO
|
|||||||
|
|
||||||
For next compiler release (7.6)
|
For next compiler release (7.6)
|
||||||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
...
|
see if the 16 bit repeat code can be optimized on 65c02 a bit more as well, like the 8 bit
|
||||||
|
|
||||||
|
|
||||||
Need help with
|
Need help with
|
||||||
^^^^^^^^^^^^^^
|
^^^^^^^^^^^^^^
|
||||||
- c128 target: various machine specific things (free zp locations, how banking works, getting the floating point routines working, ...)
|
- c128 target: various machine specific things (free zp locations, how banking works, getting the floating point routines working, ...)
|
||||||
- other targets such as Atari 800XL: all required details about the machine, I have no clue whatsoever
|
- other targets such as Atari 800XL: all required details about the machine, I have no clue whatsoever
|
||||||
|
- see the Porting Guide in the documentation for this.
|
||||||
|
|
||||||
|
|
||||||
Blocked by an official Commander-x16 r39 release
|
Blocked by an official Commander-x16 r39 release
|
||||||
@ -45,6 +47,7 @@ Future
|
|||||||
|
|
||||||
More code optimization ideas
|
More code optimization ideas
|
||||||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
- if a for loop's loopvariable isn't referenced in the body, replace by a repeatloop
|
||||||
- automatically convert if statements that test for multiple values (if X==1 or X==2..) to if X in [1,2,..] statements, instead of just a warning
|
- automatically convert if statements that test for multiple values (if X==1 or X==2..) to if X in [1,2,..] statements, instead of just a warning
|
||||||
- byte typed expressions should be evaluated in the accumulator where possible, without (temp)var
|
- byte typed expressions should be evaluated in the accumulator where possible, without (temp)var
|
||||||
for instance value = otherbyte >> 1 --> lda otherbite ; lsr a; sta value
|
for instance value = otherbyte >> 1 --> lda otherbite ; lsr a; sta value
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
#!/usr/bin/env sh
|
#!/usr/bin/env sh
|
||||||
|
|
||||||
rm -f *.jar *.asm *.prg *.vm.txt *.vice-mon-list a.out
|
rm -f *.jar *.asm *.prg *.vm.txt *.vice-mon-list *.list a.out imgui.ini
|
||||||
rm -rf build out
|
rm -rf build out
|
||||||
rm -rf compiler/build codeGeneration/build codeOptimizers/build compilerInterfaces/build compilerAst/build dbusCompilerService/build httpCompilerService/build parser/build
|
rm -rf compiler/build codeGeneration/build codeOptimizers/build compilerInterfaces/build compilerAst/build dbusCompilerService/build httpCompilerService/build parser/build
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user