slight optimization of repeat loop (0-256 iters) code generation on 65c02 cpu

This commit is contained in:
Irmen de Jong 2021-12-31 14:01:45 +01:00
parent 583e208c1e
commit 1e8d8e40a2
3 changed files with 79 additions and 36 deletions

View File

@ -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(repeatLabel)
out(" sta $counterVar") translate(stmt.body)
out(repeatLabel) out(" dec $counterVar | bne $repeatLabel")
translate(stmt.body) } else {
out(" dec $counterVar | bne $repeatLabel") out(" ldy #${count and 255}")
if(constIterations==null) out("$repeatLabel phy")
out(endLabel) 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 createRepeatCounterVar(dt: DataType, constIterations: Int?, stmt: RepeatLoop): String { 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)
}
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) {
return existingVar.second if(!mustBeInZeropage || existingVar.third!=null)
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))
} }
} }

View File

@ -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

View File

@ -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