1
0
mirror of https://github.com/KarolS/millfork.git synced 2025-01-12 03:30:09 +00:00

Allocating arbitrary variables in the zeropage and using zeropage addressing when appropriate

This commit is contained in:
Karol Stasiak 2018-06-18 17:59:47 +02:00
parent ca14e417dd
commit daf8461c07
10 changed files with 231 additions and 96 deletions

View File

@ -23,6 +23,7 @@ class Platform(
val outputPackager: OutputPackager, val outputPackager: OutputPackager,
val codeAllocators: Map[String, UpwardByteAllocator], val codeAllocators: Map[String, UpwardByteAllocator],
val variableAllocators: Map[String, VariableAllocator], val variableAllocators: Map[String, VariableAllocator],
val freeZpPointers: List[Int],
val fileExtension: String, val fileExtension: String,
val generateBbcMicroInfFile: Boolean, val generateBbcMicroInfFile: Boolean,
val bankNumbers: Map[String, Int], val bankNumbers: Map[String, Int],
@ -171,6 +172,7 @@ object Platform {
new Platform(cpu, flagOverrides, startingModules, outputPackager, new Platform(cpu, flagOverrides, startingModules, outputPackager,
codeAllocators.toMap, codeAllocators.toMap,
variableAllocators.toMap, variableAllocators.toMap,
freePointers,
if (fileExtension == "" || fileExtension.startsWith(".")) fileExtension else "." + fileExtension, if (fileExtension == "" || fileExtension.startsWith(".")) fileExtension else "." + fileExtension,
generateBbcMicroInfFile, generateBbcMicroInfFile,
bankNumbers, bankNumbers,

View File

@ -9,5 +9,7 @@ import millfork.env.NormalFunction
trait AssemblyOptimization[T <: AbstractCode] { trait AssemblyOptimization[T <: AbstractCode] {
def name: String def name: String
def optimize(f: NormalFunction, code: List[T], options: CompilationOptions): List[T] def optimize(f: NormalFunction, code: List[T], options: CompilationOptions, labelMap: Map[String, Int]): List[T] = optimize(f, code, options)
def optimize(f: NormalFunction, code: List[T], options: CompilationOptions): List[T] = optimize(f, code, options, Map())
} }

View File

@ -32,18 +32,18 @@ class RuleBasedAssemblyOptimization(val name: String, val needsFlowInfo: FlowInf
rules.foreach(_.pattern.validate(needsFlowInfo)) rules.foreach(_.pattern.validate(needsFlowInfo))
override def optimize(f: NormalFunction, code: List[AssemblyLine], options: CompilationOptions): List[AssemblyLine] = { override def optimize(f: NormalFunction, code: List[AssemblyLine], options: CompilationOptions, labelMap: Map[String, Int]): List[AssemblyLine] = {
val effectiveCode = code.map(a => a.copy(parameter = a.parameter.quickSimplify)) val effectiveCode = code.map(a => a.copy(parameter = a.parameter.quickSimplify))
val taggedCode = FlowAnalyzer.analyze(f, effectiveCode, options, needsFlowInfo) val taggedCode = FlowAnalyzer.analyze(f, effectiveCode, options, needsFlowInfo)
optimizeImpl(f, taggedCode, options) optimizeImpl(f, taggedCode, options, labelMap)
} }
def optimizeImpl(f: NormalFunction, code: List[(FlowInfo, AssemblyLine)], options: CompilationOptions): List[AssemblyLine] = { def optimizeImpl(f: NormalFunction, code: List[(FlowInfo, AssemblyLine)], options: CompilationOptions, labelMap: Map[String, Int]): List[AssemblyLine] = {
code match { code match {
case Nil => Nil case Nil => Nil
case head :: tail => case head :: tail =>
for ((rule, index) <- rules.zipWithIndex) { for ((rule, index) <- rules.zipWithIndex) {
val ctx = new AssemblyMatchingContext(options) val ctx = new AssemblyMatchingContext(options, labelMap)
rule.pattern.matchTo(ctx, code) match { rule.pattern.matchTo(ctx, code) match {
case Some(rest: List[(FlowInfo, AssemblyLine)]) => case Some(rest: List[(FlowInfo, AssemblyLine)]) =>
val matchedChunkToOptimize: List[AssemblyLine] = code.take(code.length - rest.length).map(_._2) val matchedChunkToOptimize: List[AssemblyLine] = code.take(code.length - rest.length).map(_._2)
@ -59,19 +59,19 @@ class RuleBasedAssemblyOptimization(val name: String, val needsFlowInfo: FlowInf
ErrorReporting.trace(" ↓") ErrorReporting.trace(" ↓")
optimizedChunk.filter(_.isPrintable).foreach(l => ErrorReporting.trace(l.toString)) optimizedChunk.filter(_.isPrintable).foreach(l => ErrorReporting.trace(l.toString))
if (needsFlowInfo != FlowInfoRequirement.NoRequirement) { if (needsFlowInfo != FlowInfoRequirement.NoRequirement) {
return optimizedChunk ++ optimizeImpl(f, rest, options) return optimizedChunk ++ optimizeImpl(f, rest, options, labelMap)
} else { } else {
return optimize(f, optimizedChunk ++ rest.map(_._2), options) return optimize(f, optimizedChunk ++ rest.map(_._2), options)
} }
case None => () case None => ()
} }
} }
head._2 :: optimizeImpl(f, tail, options) head._2 :: optimizeImpl(f, tail, options, labelMap)
} }
} }
} }
class AssemblyMatchingContext(val compilationOptions: CompilationOptions) { class AssemblyMatchingContext(val compilationOptions: CompilationOptions, val labelMap: Map[String, Int]) {
private val map = mutable.Map[Int, Any]() private val map = mutable.Map[Int, Any]()
override def toString: String = map.mkString(", ") override def toString: String = map.mkString(", ")
@ -1020,3 +1020,29 @@ case class MatchElidableCopyOf(i: Int, firstLinePattern: AssemblyLinePattern, la
Some(after) Some(after)
} }
} }
case object IsZeroPage extends AssemblyLinePattern {
override def matchLineTo(ctx: AssemblyMatchingContext, flowInfo: FlowInfo, line: AssemblyLine): Boolean = {
import Opcode._
line match {
case AssemblyLine(_, AddrMode.ZeroPage, _, _) => true
case l@AssemblyLine(LDA | STA | CMP |
LDX | STX | CPX |
LDY | STY | CPY |
LDZ | STZ | CPZ |
BIT |
ADC | SBC | AND | ORA | EOR |
INC | DEC | ROL | ROR | ASL | LSR |
ISC | DCP | LAX | SAX | RLA | RRA | SLO | SRE, AddrMode.Absolute, p, true) =>
p match {
case NumericConstant(n, _) => n <= 255
case MemoryAddressConstant(th) => ctx.labelMap.getOrElse(th.name, 0x800) < 0x100
case CompoundConstant(MathOperator.Plus,
MemoryAddressConstant(th),
NumericConstant(n, _)) => ctx.labelMap.getOrElse(th.name, 0x800) + n < 0x100
case _ => false
}
case _ => false
}
}
}

View File

@ -13,7 +13,7 @@ import scala.collection.mutable
*/ */
object SuperOptimizer extends AssemblyOptimization[AssemblyLine] { object SuperOptimizer extends AssemblyOptimization[AssemblyLine] {
def optimize(m: NormalFunction, code: List[AssemblyLine], options: CompilationOptions): List[AssemblyLine] = { override def optimize(m: NormalFunction, code: List[AssemblyLine], options: CompilationOptions): List[AssemblyLine] = {
val oldVerbosity = ErrorReporting.verbosity val oldVerbosity = ErrorReporting.verbosity
ErrorReporting.verbosity = -1 ErrorReporting.verbosity = -1
var allOptimizers = OptimizationPresets.Good ++ LaterOptimizations.All var allOptimizers = OptimizationPresets.Good ++ LaterOptimizations.All

View File

@ -8,7 +8,7 @@ import millfork.assembly.mos.Opcode
import millfork.assembly.z80.{IfFlagClear, IfFlagSet, ZFlag} import millfork.assembly.z80.{IfFlagClear, IfFlagSet, ZFlag}
import millfork.error.ErrorReporting import millfork.error.ErrorReporting
import millfork.node._ import millfork.node._
import millfork.output.{CompiledMemory, VariableAllocator} import millfork.output.{AllocationLocation, CompiledMemory, VariableAllocator}
import scala.collection.mutable import scala.collection.mutable
@ -89,7 +89,14 @@ class Environment(val parent: Option[Environment], val prefix: String) {
} }
} }
def allocateVariables(nf: Option[NormalFunction], mem: CompiledMemory, callGraph: CallGraph, allocators: Map[String, 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,
pass: Int,
forZpOnly: Boolean): 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 =>
@ -100,8 +107,13 @@ class Environment(val parent: Option[Environment], val prefix: String) {
Nil Nil
} }
}.toSet }.toSet
def passForAlloc(m: VariableAllocationMethod.Value) = m match {
case VariableAllocationMethod.Zeropage => 1
case VariableAllocationMethod.Register => 2
case _ => 3
}
val toAdd = things.values.flatMap { val toAdd = things.values.flatMap {
case m: UninitializedMemory if nf.isDefined == isLocalVariableName(m.name) && !m.name.endsWith(".addr") && maybeGet[Thing](m.name + ".array").isEmpty => case m: UninitializedMemory if passForAlloc(m.alloc) == pass && nf.isDefined == isLocalVariableName(m.name) && !m.name.endsWith(".addr") && maybeGet[Thing](m.name + ".array").isEmpty =>
val vertex = if (options.flag(CompilationFlag.VariableOverlap)) { val vertex = if (options.flag(CompilationFlag.VariableOverlap)) {
nf.fold[VariableVertex](GlobalVertex) { f => nf.fold[VariableVertex](GlobalVertex) { f =>
if (m.alloc == VariableAllocationMethod.Static) { if (m.alloc == VariableAllocationMethod.Static) {
@ -118,38 +130,44 @@ class Environment(val parent: Option[Environment], val prefix: String) {
case VariableAllocationMethod.None => case VariableAllocationMethod.None =>
Nil Nil
case VariableAllocationMethod.Zeropage => case VariableAllocationMethod.Zeropage =>
m.sizeInBytes match { if (forZpOnly) {
case 2 =>
val addr = val addr =
allocators(bank).allocatePointer(mem.banks(bank), callGraph, vertex) allocators(bank).allocateBytes(mem.banks(bank), callGraph, vertex, options, m.sizeInBytes, initialized = false, writeable = true, location = AllocationLocation.Zeropage)
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)
) )
} } else Nil
case VariableAllocationMethod.Auto | VariableAllocationMethod.Register | VariableAllocationMethod.Static => case VariableAllocationMethod.Auto | VariableAllocationMethod.Register | VariableAllocationMethod.Static =>
if (m.alloc == VariableAllocationMethod.Register) { if (m.alloc == VariableAllocationMethod.Register) {
ErrorReporting.warn(s"Failed to inline variable `${m.name}` into a register", options, None) ErrorReporting.warn(s"Failed to inline variable `${m.name}` into a register", options, None)
} }
m.sizeInBytes match { if (m.sizeInBytes == 0) Nil else {
case 0 => Nil val graveName = m.name.stripPrefix(prefix) + "`"
case 2 => if (forZpOnly) {
val addr = if (bank == "default") {
allocators(bank).allocateBytes(mem.banks(bank), callGraph, vertex, options, 2, initialized = false, writeable = true) allocators(bank).tryAllocateZeropageBytes(mem.banks(bank), callGraph, vertex, options, m.sizeInBytes) match {
case None => Nil
case Some(addr) =>
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 => }
val addr = allocators(bank).allocateBytes(mem.banks(bank), callGraph, vertex, options, count, initialized = false, writeable = true) } else Nil
} else if (things.contains(graveName)) {
Nil
} else {
val addr = allocators(bank).allocateBytes(mem.banks(bank), callGraph, vertex, options, m.sizeInBytes, initialized = false, writeable = true, location = AllocationLocation.Either)
onEachVariable(m.name, addr) onEachVariable(m.name, addr)
List( List(
ConstantThing(m.name.stripPrefix(prefix) + "`", NumericConstant(addr, 2), p) ConstantThing(graveName, NumericConstant(addr, 2), p)
) )
} }
} }
}
case f: NormalFunction => case f: NormalFunction =>
f.environment.allocateVariables(Some(f), mem, callGraph, allocators, options, onEachVariable) f.environment.allocateVariables(Some(f), mem, callGraph, allocators, options, onEachVariable, pass, forZpOnly)
Nil Nil
case _ => Nil case _ => Nil
}.toList }.toList

View File

@ -170,6 +170,13 @@ abstract class AbstractAssembler[T <: AbstractCode](private val program: Program
def assemble(callGraph: CallGraph, optimizations: Seq[AssemblyOptimization[T]], options: CompilationOptions): AssemblerOutput = { def assemble(callGraph: CallGraph, optimizations: Seq[AssemblyOptimization[T]], options: CompilationOptions): AssemblerOutput = {
val platform = options.platform val platform = options.platform
val variableAllocators = platform.variableAllocators
val zpOccupied = mem.banks("default").occupied
(0 until 0x100).foreach(i => zpOccupied(i) = true)
platform.freeZpPointers.foreach { i =>
zpOccupied(i) = false
zpOccupied(i + 1) = false
}
val assembly = mutable.ArrayBuffer[String]() val assembly = mutable.ArrayBuffer[String]()
@ -186,12 +193,16 @@ abstract class AbstractAssembler[T <: AbstractCode](private val program: Program
val potentiallyInlineable: Map[String, Int] = inliningResult.potentiallyInlineableFunctions val potentiallyInlineable: Map[String, Int] = inliningResult.potentiallyInlineableFunctions
var nonInlineableFunctions: Set[String] = inliningResult.nonInlineableFunctions var nonInlineableFunctions: Set[String] = inliningResult.nonInlineableFunctions
env.allocateVariables(None, mem, callGraph, variableAllocators, options, labelMap.put, 1, forZpOnly = true)
env.allocateVariables(None, mem, callGraph, variableAllocators, options, labelMap.put, 2, forZpOnly = true)
env.allocateVariables(None, mem, callGraph, variableAllocators, options, labelMap.put, 3, forZpOnly = true)
var inlinedFunctions = Map[String, List[T]]() var inlinedFunctions = Map[String, List[T]]()
val compiledFunctions = mutable.Map[String, List[T]]() val compiledFunctions = mutable.Map[String, List[T]]()
val recommendedCompilationOrder = callGraph.recommendedCompilationOrder val recommendedCompilationOrder = callGraph.recommendedCompilationOrder
recommendedCompilationOrder.foreach { f => recommendedCompilationOrder.foreach { f =>
env.maybeGet[NormalFunction](f).foreach { function => env.maybeGet[NormalFunction](f).foreach { function =>
val code = compileFunction(function, optimizations, options, inlinedFunctions) val code = compileFunction(function, optimizations, options, inlinedFunctions, labelMap.toMap)
val strippedCodeForInlining = for { val strippedCodeForInlining = for {
limit <- potentiallyInlineable.get(f) limit <- potentiallyInlineable.get(f)
if code.map(_.sizeInBytes).sum <= limit if code.map(_.sizeInBytes).sum <= limit
@ -269,7 +280,7 @@ abstract class AbstractAssembler[T <: AbstractCode](private val program: Program
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 = codeAllocators(bank).allocateBytes(mem.banks(bank), options, size, initialized = true, writeable = false) val index = codeAllocators(bank).allocateBytes(mem.banks(bank), options, size, initialized = true, writeable = false, location = AllocationLocation.High)
labelMap(f.name) = index labelMap(f.name) = index
justAfterCode += bank -> outputFunction(bank, code, index, assembly, options) justAfterCode += bank -> outputFunction(bank, code, index, assembly, options)
} }
@ -282,7 +293,7 @@ abstract class AbstractAssembler[T <: AbstractCode](private val program: Program
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 = codeAllocators(bank).allocateBytes(bank0, options, size, initialized = true, writeable = false) val index = codeAllocators(bank).allocateBytes(bank0, options, size, initialized = true, writeable = false, location = AllocationLocation.High)
labelMap(f.name) = index labelMap(f.name) = index
justAfterCode += bank -> outputFunction(bank, code, index, assembly, options) justAfterCode += bank -> outputFunction(bank, code, index, assembly, options)
} }
@ -297,7 +308,7 @@ abstract class AbstractAssembler[T <: AbstractCode](private val program: Program
val bank = m.bank(options) val bank = m.bank(options)
if (bank != "default") ??? if (bank != "default") ???
val bank0 = mem.banks(bank) val bank0 = mem.banks(bank)
var index = codeAllocators(bank).allocateBytes(bank0, options, typ.size + 1, initialized = true, writeable = false) var index = codeAllocators(bank).allocateBytes(bank0, options, typ.size + 1, initialized = true, writeable = false, location = AllocationLocation.High)
labelMap(name) = index + 1 labelMap(name) = index + 1
val altName = m.name.stripPrefix(env.prefix) + "`" val altName = m.name.stripPrefix(env.prefix) + "`"
val thing = if (name.endsWith(".addr")) env.get[ThingInMemory](name.stripSuffix(".addr")) else env.get[ThingInMemory](name + ".array") val thing = if (name.endsWith(".addr")) env.get[ThingInMemory](name.stripSuffix(".addr")) else env.get[ThingInMemory](name + ".array")
@ -318,7 +329,7 @@ abstract class AbstractAssembler[T <: AbstractCode](private val program: Program
} }
case _ => () case _ => ()
} }
val index = codeAllocators("default").allocateBytes(mem.banks("default"), options, 1, initialized = true, writeable = false) val index = codeAllocators("default").allocateBytes(mem.banks("default"), options, 1, initialized = true, writeable = false, location = AllocationLocation.High)
writeByte("default", index, 2.toByte) // BIT abs writeByte("default", index, 2.toByte) // BIT abs
assembly.append("* = $" + index.toHexString) assembly.append("* = $" + index.toHexString)
assembly.append(" !byte 2 ;; end of LUnix relocatable segment") assembly.append(" !byte 2 ;; end of LUnix relocatable segment")
@ -328,7 +339,7 @@ abstract class AbstractAssembler[T <: AbstractCode](private val program: Program
case thing@InitializedArray(name, None, items, _) => case thing@InitializedArray(name, None, items, _) =>
val bank = thing.bank(options) val bank = thing.bank(options)
val bank0 = mem.banks(bank) val bank0 = mem.banks(bank)
var index = codeAllocators(bank).allocateBytes(bank0, options, items.size, initialized = true, writeable = true) var index = codeAllocators(bank).allocateBytes(bank0, options, items.size, initialized = true, writeable = true, location = AllocationLocation.High)
labelMap(name) = index labelMap(name) = index
assembly.append("* = $" + index.toHexString) assembly.append("* = $" + index.toHexString)
assembly.append(name) assembly.append(name)
@ -350,7 +361,7 @@ abstract class AbstractAssembler[T <: AbstractCode](private val program: Program
case m@InitializedMemoryVariable(name, None, typ, value, _) => case m@InitializedMemoryVariable(name, None, typ, value, _) =>
val bank = m.bank(options) val bank = m.bank(options)
val bank0 = mem.banks(bank) val bank0 = mem.banks(bank)
var index = codeAllocators(bank).allocateBytes(bank0, options, typ.size, initialized = true, writeable = true) var index = codeAllocators(bank).allocateBytes(bank0, options, typ.size, initialized = true, writeable = true, location = AllocationLocation.High)
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"))
@ -376,15 +387,23 @@ abstract class AbstractAssembler[T <: AbstractCode](private val program: Program
val bank0 = mem.banks(bank) 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 variableAllocators = platform.variableAllocators
variableAllocators.foreach { case (b, a) => a.notifyAboutEndOfCode(justAfterCode(b)) } variableAllocators.foreach { case (b, a) => a.notifyAboutEndOfCode(justAfterCode(b)) }
env.allocateVariables(None, mem, callGraph, variableAllocators, options, labelMap.put) env.allocateVariables(None, mem, callGraph, variableAllocators, options, labelMap.put, 2, forZpOnly = false)
env.allocateVariables(None, mem, callGraph, variableAllocators, options, labelMap.put, 3, forZpOnly = false)
val zeropageOccupation = mem.banks("default").occupied.slice(variableAllocators("default").pointers.head, variableAllocators("default").pointers.last + 2) if (platform.freeZpPointers.nonEmpty) {
val zpUsageOffset = platform.freeZpPointers.min
val zeropageOccupation = zpOccupied.slice(zpUsageOffset, platform.freeZpPointers.max + 2)
labelMap += "__zeropage_usage" -> (zeropageOccupation.lastIndexOf(true) - zeropageOccupation.indexOf(true) + 1) labelMap += "__zeropage_usage" -> (zeropageOccupation.lastIndexOf(true) - zeropageOccupation.indexOf(true) + 1)
labelMap += "__zeropage_first" -> (zeropageOccupation.indexOf(true) max 0) labelMap += "__zeropage_first" -> (zpUsageOffset + (zeropageOccupation.indexOf(true) max 0))
labelMap += "__zeropage_last" -> (zeropageOccupation.lastIndexOf(true) max 0) labelMap += "__zeropage_last" -> (zpUsageOffset + (zeropageOccupation.lastIndexOf(true) max 0))
labelMap += "__zeropage_end" -> (zeropageOccupation.lastIndexOf(true) + 1) labelMap += "__zeropage_end" -> (zpUsageOffset + zeropageOccupation.lastIndexOf(true) + 1)
} else {
labelMap += "__zeropage_usage" -> 0
labelMap += "__zeropage_first" -> 3
labelMap += "__zeropage_last" -> 2
labelMap += "__zeropage_end" -> 3
}
env = rootEnv.allThings env = rootEnv.allThings
@ -421,16 +440,18 @@ abstract class AbstractAssembler[T <: AbstractCode](private val program: Program
AssemblerOutput(code, assembly.toArray, labelMap.toList) AssemblerOutput(code, assembly.toArray, labelMap.toList)
} }
private def compileFunction(f: NormalFunction, optimizations: Seq[AssemblyOptimization[T]], options: CompilationOptions, inlinedFunctions: Map[String, List[T]]): List[T] = { def injectLabels(labelMap: Map[String, Int], code: List[T]): List[T]
private def compileFunction(f: NormalFunction, optimizations: Seq[AssemblyOptimization[T]], options: CompilationOptions, inlinedFunctions: Map[String, List[T]], labelMap: Map[String, Int]): List[T] = {
ErrorReporting.debug("Compiling: " + f.name, f.position) ErrorReporting.debug("Compiling: " + f.name, f.position)
val unoptimized: List[T] = val unoptimized: List[T] =
inliningCalculator.inline( injectLabels(labelMap, inliningCalculator.inline(
compiler.compile(CompilationContext(env = f.environment, function = f, extraStackOffset = 0, options = options)), compiler.compile(CompilationContext(env = f.environment, function = f, extraStackOffset = 0, options = options)),
inlinedFunctions, inlinedFunctions,
compiler) compiler))
unoptimizedCodeSize += unoptimized.map(_.sizeInBytes).sum unoptimizedCodeSize += unoptimized.map(_.sizeInBytes).sum
val code = optimizations.foldLeft(unoptimized) { (c, opt) => val code = optimizations.foldLeft(unoptimized) { (c, opt) =>
opt.optimize(f, c, options) opt.optimize(f, c, options, labelMap)
} }
performFinalOptimizationPass(f, optimizations.nonEmpty, options, code) performFinalOptimizationPass(f, optimizations.nonEmpty, options, code)
} }

View File

@ -63,6 +63,30 @@ class MosAssembler(program: Program,
index + 4 index + 4
} }
} }
override def injectLabels(labelMap: Map[String, Int], code: List[AssemblyLine]): List[AssemblyLine] = {
import Opcode._
code.map {
case l@AssemblyLine(LDA | STA | CMP |
LDX | STX | CPX |
LDY | STY | CPY |
LDZ | STZ | CPZ |
BIT |
ADC | SBC | AND | ORA | EOR |
INC | DEC | ROL | ROR | ASL | LSR |
ISC | DCP | LAX | SAX | RLA | RRA | SLO | SRE, AddrMode.Absolute, p, true) =>
p match {
case NumericConstant(n, _) => if (n <= 255) l.copy(addrMode = AddrMode.ZeroPage) else l
case MemoryAddressConstant(th) => if (labelMap.getOrElse(th.name, 0x800) < 0x100) l.copy(addrMode = AddrMode.ZeroPage) else l
case CompoundConstant(MathOperator.Plus,
MemoryAddressConstant(th),
NumericConstant(n, _)) => if (labelMap.getOrElse(th.name, 0x800) + n < 0x100) l.copy(addrMode = AddrMode.ZeroPage) else l
case _ => l
}
case l => l
}
}
} }

View File

@ -11,9 +11,20 @@ import scala.collection.mutable
* @author Karol Stasiak * @author Karol Stasiak
*/ */
object AllocationLocation extends Enumeration {
val Zeropage, High, Either = Value
def matches(addr:Int, location: AllocationLocation.Value): Boolean = location match {
case Zeropage => addr < 0x100
case High => addr >= 0x100
case Either => true
}
}
sealed trait ByteAllocator { sealed trait ByteAllocator {
protected def startAt: Int def startAt: Int
protected def endBefore: Int def endBefore: Int
def preferredOrder: Option[List[Int]]
def notifyAboutEndOfCode(org: Int): Unit def notifyAboutEndOfCode(org: Int): Unit
@ -21,7 +32,7 @@ sealed trait ByteAllocator {
var lastFree = startAt var lastFree = startAt
var counter = 0 var counter = 0
val occupied = mem.occupied val occupied = mem.occupied
for(i <- startAt until endBefore) { for(i <- preferredOrder.getOrElse(startAt until endBefore)) {
if (occupied(i) || counter == 0 && count == 2 && i.&(0xff) == 0xff && options.flags(CompilationFlag.PreventJmpIndirectBug)) { if (occupied(i) || counter == 0 && count == 2 && i.&(0xff) == 0xff && options.flags(CompilationFlag.PreventJmpIndirectBug)) {
counter = 0 counter = 0
} else { } else {
@ -37,72 +48,99 @@ sealed trait ByteAllocator {
} }
} }
} }
ErrorReporting.fatal("Out of high memory") -1
} }
} }
class UpwardByteAllocator(val startAt: Int, val endBefore: Int) extends ByteAllocator { class UpwardByteAllocator(val startAt: Int, val endBefore: Int) extends ByteAllocator {
def notifyAboutEndOfCode(org: Int): Unit = () def notifyAboutEndOfCode(org: Int): Unit = ()
override def preferredOrder: Option[List[Int]] = None
}
class ZeropageAllocator(val freeZpPointers: List[Int]) extends ByteAllocator {
def notifyAboutEndOfCode(org: Int): Unit = ()
override def preferredOrder: Option[List[Int]] = if (freeZpPointers.isEmpty) None else Some(freeZpPointers.flatMap(i => Seq(i,i+1)))
override def startAt: Int = if (freeZpPointers.isEmpty) 2 else freeZpPointers.min
override def endBefore: Int = if (freeZpPointers.isEmpty) 2 else freeZpPointers.max + 2
} }
class AfterCodeByteAllocator(val endBefore: Int) extends ByteAllocator { class AfterCodeByteAllocator(val endBefore: Int) extends ByteAllocator {
var startAt = 0x200 var startAt = 0x200
def notifyAboutEndOfCode(org: Int): Unit = startAt = org def notifyAboutEndOfCode(org: Int): Unit = startAt = org
override def preferredOrder: Option[List[Int]] = None
} }
class VariableAllocator(val pointers: List[Int], private val bytes: ByteAllocator) { class VariableAllocator(pointers: List[Int], private val bytes: ByteAllocator) {
val zeropage: ByteAllocator = new ZeropageAllocator(pointers)
private val pointerMap = mutable.Map[Int, Set[VariableVertex]]()
private val variableMap = mutable.Map[Int, mutable.Map[Int, Set[VariableVertex]]]() private val variableMap = mutable.Map[Int, mutable.Map[Int, Set[VariableVertex]]]()
def allocatePointer(mem: MemoryBank, callGraph: CallGraph, p: VariableVertex): Int = { def allocateBytes(mem: MemoryBank, callGraph: CallGraph, p: VariableVertex, options: CompilationOptions, count: Int, initialized: Boolean, writeable: Boolean, location: AllocationLocation.Value): Int = {
// TODO: search for free zeropage locations
pointerMap.foreach { case (addr, alreadyThere) =>
if (alreadyThere.forall(q => callGraph.canOverlap(p, q))) {
pointerMap(addr) += p
return addr
}
}
@tailrec
def pickFreePointer(ps: List[Int]): Int =
ps match {
case Nil =>
ErrorReporting.trace(pointerMap.mkString(", "))
ErrorReporting.fatal("Out of zero-page memory")
case next :: rest =>
if (mem.occupied(next) || mem.occupied(next + 1)) {
pickFreePointer(rest)
} else {
mem.readable(next) = true
mem.readable(next + 1) = true
mem.occupied(next) = true
mem.occupied(next + 1) = true
mem.writeable(next) = true
mem.writeable(next + 1) = true
pointerMap(next) = Set(p)
next
}
}
pickFreePointer(pointers)
}
def allocateBytes(mem: MemoryBank, callGraph: CallGraph, p: VariableVertex, options: CompilationOptions, count: Int, initialized: Boolean, writeable: Boolean): Int = {
if (!variableMap.contains(count)) { if (!variableMap.contains(count)) {
variableMap(count) = mutable.Map() variableMap(count) = mutable.Map()
} }
variableMap(count).foreach { case (a, alreadyThere) => variableMap(count).foreach { case (a, alreadyThere) =>
if (alreadyThere.forall(q => callGraph.canOverlap(p, q))) { if (AllocationLocation.matches(a, location) && alreadyThere.forall(q => callGraph.canOverlap(p, q))) {
variableMap(count)(a) += p variableMap(count)(a) += p
return a return a
} }
} }
val addr = allocateBytes(mem, options, count, initialized, writeable) val addr = allocateBytes(mem, options, count, initialized, writeable, location)
variableMap(count)(addr) = Set(p) variableMap(count)(addr) = Set(p)
addr addr
} }
def allocateBytes(mem: MemoryBank, options: CompilationOptions, count: Int, initialized: Boolean, writeable: Boolean): Int = { def tryAllocateZeropageBytes(mem: MemoryBank, callGraph: CallGraph, p: VariableVertex, options: CompilationOptions, count: Int): Option[Int]={
val addr = bytes.findFreeBytes(mem, count, options) if (!variableMap.contains(count)) {
variableMap(count) = mutable.Map()
}
variableMap(count).foreach { case (a, alreadyThere) =>
if (a < 0x100 && alreadyThere.forall(q => callGraph.canOverlap(p, q))) {
variableMap(count)(a) += p
return Some(a)
}
}
val addr = zeropage.findFreeBytes(mem, count, options)
if (addr < 0) None else {
markBytes(mem, addr, count, initialized = false, writeable = true)
Some(addr)
}
}
def allocateBytes(mem: MemoryBank, options: CompilationOptions, count: Int, initialized: Boolean, writeable: Boolean, location: AllocationLocation.Value): Int = {
val addr = location match {
case AllocationLocation.Zeropage =>
val a = zeropage.findFreeBytes(mem, count, options)
if (a < 0) {
ErrorReporting.fatal("Out of zeropage memory")
}
a
case AllocationLocation.High =>
val a = bytes.findFreeBytes(mem, count, options)
if (a < 0) {
ErrorReporting.fatal("Out of high memory")
}
a
case AllocationLocation.Either =>
var a = zeropage.findFreeBytes(mem, count, options)
if (a < 0) {
a = bytes.findFreeBytes(mem, count, options)
if (a < 0) {
ErrorReporting.fatal("Out of high memory")
}
}
a
}
markBytes(mem, addr, count, initialized, writeable)
addr
}
private def markBytes(mem: MemoryBank, addr: Int, count: Int, initialized: Boolean, writeable: Boolean): Unit = {
ErrorReporting.trace(s"allocating $count bytes at $$${addr.toHexString}") ErrorReporting.trace(s"allocating $count bytes at $$${addr.toHexString}")
(addr until (addr + count)).foreach { i => (addr until (addr + count)).foreach { i =>
if (mem.occupied(i)) ErrorReporting.fatal("Overlapping objects") if (mem.occupied(i)) ErrorReporting.fatal("Overlapping objects")
@ -111,7 +149,6 @@ class VariableAllocator(val pointers: List[Int], private val bytes: ByteAllocato
mem.initialized(i) = initialized mem.initialized(i) = initialized
mem.writeable(i) = writeable mem.writeable(i) = writeable
} }
addr
} }
def notifyAboutEndOfCode(org: Int): Unit = bytes.notifyAboutEndOfCode(org) def notifyAboutEndOfCode(org: Int): Unit = bytes.notifyAboutEndOfCode(org)

View File

@ -18,6 +18,8 @@ class Z80Assembler(program: Program,
// TODO // TODO
index index
} }
override def injectLabels(labelMap: Map[String, Int], code: List[ZLine]): List[ZLine] = code // TODO
} }
object Z80Assembler { object Z80Assembler {

View File

@ -7,13 +7,16 @@ import millfork.{Cpu, OutputStyle, Platform}
* @author Karol Stasiak * @author Karol Stasiak
*/ */
object EmuPlatform { object EmuPlatform {
private val pointers: List[Int] = (0 until 256 by 2).toList
def get(cpu: Cpu.Value) = new Platform( def get(cpu: Cpu.Value) = new Platform(
cpu, cpu,
Map(), Map(),
Nil, Nil,
CurrentBankFragmentOutput(0, 0xffff), CurrentBankFragmentOutput(0, 0xffff),
Map("default" -> new UpwardByteAllocator(0x200, 0xb000)), Map("default" -> new UpwardByteAllocator(0x200, 0xb000)),
Map("default" -> new VariableAllocator((0 until 256 by 2).toList, new AfterCodeByteAllocator(0xff00))), Map("default" -> new VariableAllocator(pointers, new AfterCodeByteAllocator(0xff00))),
pointers,
".bin", ".bin",
false, false,
Map("default" -> 0), Map("default" -> 0),