mirror of
https://github.com/KarolS/millfork.git
synced 2025-01-10 20:29:35 +00:00
More optimizations, including optimizations with jump counting
This commit is contained in:
parent
f5f4c033f3
commit
e28ff16717
@ -4,7 +4,7 @@ import java.nio.charset.StandardCharsets
|
|||||||
import java.nio.file.{Files, Paths}
|
import java.nio.file.{Files, Paths}
|
||||||
import java.util.Locale
|
import java.util.Locale
|
||||||
|
|
||||||
import millfork.assembly.opt.{CmosOptimizations, DangerousOptimizations, SuperOptimizer, UndocumentedOptimizations}
|
import millfork.assembly.opt._
|
||||||
import millfork.buildinfo.BuildInfo
|
import millfork.buildinfo.BuildInfo
|
||||||
import millfork.cli.{CliParser, CliStatus}
|
import millfork.cli.{CliParser, CliStatus}
|
||||||
import millfork.env.Environment
|
import millfork.env.Environment
|
||||||
@ -87,7 +87,7 @@ object Main {
|
|||||||
env.collectDeclarations(program, options)
|
env.collectDeclarations(program, options)
|
||||||
val extras = List(
|
val extras = List(
|
||||||
if (options.flag(CompilationFlag.EmitIllegals)) UndocumentedOptimizations.All else Nil,
|
if (options.flag(CompilationFlag.EmitIllegals)) UndocumentedOptimizations.All else Nil,
|
||||||
if (options.flag(CompilationFlag.EmitCmosOpcodes)) CmosOptimizations.All else Nil,
|
if (options.flag(CompilationFlag.EmitCmosOpcodes)) CmosOptimizations.All else LaterOptimizations.Nmos,
|
||||||
if (options.flag(CompilationFlag.DangerousOptimizations)) DangerousOptimizations.All else Nil,
|
if (options.flag(CompilationFlag.DangerousOptimizations)) DangerousOptimizations.All else Nil,
|
||||||
).flatten
|
).flatten
|
||||||
val goodCycle = List.fill(optLevel - 1)(OptimizationPresets.Good ++ extras).flatten
|
val goodCycle = List.fill(optLevel - 1)(OptimizationPresets.Good ++ extras).flatten
|
||||||
|
@ -107,6 +107,7 @@ object OptimizationPresets {
|
|||||||
LaterOptimizations.UseXInsteadOfStack,
|
LaterOptimizations.UseXInsteadOfStack,
|
||||||
LaterOptimizations.UseYInsteadOfStack,
|
LaterOptimizations.UseYInsteadOfStack,
|
||||||
LaterOptimizations.IndexSwitchingOptimization,
|
LaterOptimizations.IndexSwitchingOptimization,
|
||||||
|
LaterOptimizations.LoadingBranchesOptimization,
|
||||||
)
|
)
|
||||||
|
|
||||||
val Good: List[AssemblyOptimization] = List[AssemblyOptimization](
|
val Good: List[AssemblyOptimization] = List[AssemblyOptimization](
|
||||||
@ -114,6 +115,7 @@ object OptimizationPresets {
|
|||||||
AlwaysGoodOptimizations.CarryFlagConversion,
|
AlwaysGoodOptimizations.CarryFlagConversion,
|
||||||
DangerousOptimizations.ConstantIndexOffsetPropagation,
|
DangerousOptimizations.ConstantIndexOffsetPropagation,
|
||||||
AlwaysGoodOptimizations.BranchInPlaceRemoval,
|
AlwaysGoodOptimizations.BranchInPlaceRemoval,
|
||||||
|
AlwaysGoodOptimizations.CommonBranchBodyOptimization,
|
||||||
AlwaysGoodOptimizations.ConstantFlowAnalysis,
|
AlwaysGoodOptimizations.ConstantFlowAnalysis,
|
||||||
AlwaysGoodOptimizations.ConstantIndexPropagation,
|
AlwaysGoodOptimizations.ConstantIndexPropagation,
|
||||||
AlwaysGoodOptimizations.FlagFlowAnalysis,
|
AlwaysGoodOptimizations.FlagFlowAnalysis,
|
||||||
@ -146,5 +148,6 @@ object OptimizationPresets {
|
|||||||
UnusedLabelRemoval,
|
UnusedLabelRemoval,
|
||||||
AlwaysGoodOptimizations.TailCallOptimization,
|
AlwaysGoodOptimizations.TailCallOptimization,
|
||||||
AlwaysGoodOptimizations.UnusedCodeRemoval,
|
AlwaysGoodOptimizations.UnusedCodeRemoval,
|
||||||
|
AlwaysGoodOptimizations.UnusedLabelRemoval,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -137,7 +137,8 @@ object OpcodeClasses {
|
|||||||
|
|
||||||
val SupportsZeroPageIndirect = Set(ORA, AND, EOR, ADC, STA, LDA, CMP, SBC)
|
val SupportsZeroPageIndirect = Set(ORA, AND, EOR, ADC, STA, LDA, CMP, SBC)
|
||||||
|
|
||||||
val ShortBranching = Set(BEQ, BNE, BMI, BPL, BVC, BVS, BCC, BCS, BRA)
|
val ShortConditionalBranching = Set(BEQ, BNE, BMI, BPL, BVC, BVS, BCC, BCS)
|
||||||
|
val ShortBranching = ShortConditionalBranching + BRA
|
||||||
val AllDirectJumps = ShortBranching + JMP
|
val AllDirectJumps = ShortBranching + JMP
|
||||||
val AllLinear = Set(
|
val AllLinear = Set(
|
||||||
ORA, AND, EOR,
|
ORA, AND, EOR,
|
||||||
|
@ -14,7 +14,7 @@ case object EmptyChunk extends Chunk {
|
|||||||
}
|
}
|
||||||
|
|
||||||
case class LabelledChunk(label: String, chunk: Chunk) extends Chunk {
|
case class LabelledChunk(label: String, chunk: Chunk) extends Chunk {
|
||||||
override def linearize: List[AssemblyLine] = AssemblyLine.label(Label(label)) :: chunk.linearize
|
override def linearize: List[AssemblyLine] = AssemblyLine.label(Label(label)).copy(elidable=false) :: chunk.linearize
|
||||||
|
|
||||||
override def sizeInBytes: Int = chunk.sizeInBytes
|
override def sizeInBytes: Int = chunk.sizeInBytes
|
||||||
}
|
}
|
||||||
|
@ -846,5 +846,22 @@ object AlwaysGoodOptimizations {
|
|||||||
Where(ctx => ctx.get[Constant](1).quickSimplify.isLowestByteAlwaysEqual(ctx.get[Int](0)-1)) ~~> (_ => List(AssemblyLine.implied(DEX))),
|
Where(ctx => ctx.get[Constant](1).quickSimplify.isLowestByteAlwaysEqual(ctx.get[Int](0)-1)) ~~> (_ => List(AssemblyLine.implied(DEX))),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
val CommonBranchBodyOptimization = new RuleBasedAssemblyOptimization("Common branch body optimization",
|
||||||
|
needsFlowInfo = FlowInfoRequirement.JustLabels,
|
||||||
|
(Elidable & Linear & MatchOpcode(3) & MatchAddrMode(4) & MatchParameter(5)).capture(1) ~
|
||||||
|
(
|
||||||
|
(HasOpcode(JMP) & MatchParameter(2)) ~ Not(MatchOpcode(3)).*
|
||||||
|
).capture(11) ~
|
||||||
|
(Elidable & Linear & MatchOpcode(3) & MatchAddrMode(4) & MatchParameter(5)) ~
|
||||||
|
(HasOpcode(LABEL) & MatchParameter(2) & HasCallerCount(1)).capture(12) ~~> { (code, ctx) =>
|
||||||
|
ctx.get[List[AssemblyLine]](11) ++ ctx.get[List[AssemblyLine]](12) :+ code.head
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
val UnusedLabelRemoval = new RuleBasedAssemblyOptimization("Unused label removal",
|
||||||
|
needsFlowInfo = FlowInfoRequirement.JustLabels,
|
||||||
|
(Elidable & HasOpcode(LABEL) & HasCallerCount(0)) ~~> (_ => Nil)
|
||||||
|
)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,34 +1,53 @@
|
|||||||
package millfork.assembly.opt
|
package millfork.assembly.opt
|
||||||
|
|
||||||
import millfork.{CompilationFlag, CompilationOptions}
|
import millfork.{CompilationFlag, CompilationOptions}
|
||||||
import millfork.assembly.{AssemblyLine, State}
|
import millfork.assembly.{AssemblyLine, Opcode, State}
|
||||||
import millfork.env.NormalFunction
|
import millfork.env.{Label, MemoryAddressConstant, NormalFunction}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author Karol Stasiak
|
* @author Karol Stasiak
|
||||||
*/
|
*/
|
||||||
|
|
||||||
case class FlowInfo(statusBefore: CpuStatus, importanceAfter: CpuImportance) {
|
case class FlowInfo(statusBefore: CpuStatus, importanceAfter: CpuImportance, labelUseCountMap: Option[Map[String, Int]]) {
|
||||||
|
|
||||||
def hasClear(state: State.Value): Boolean = statusBefore.hasClear(state)
|
def hasClear(state: State.Value): Boolean = statusBefore.hasClear(state)
|
||||||
|
|
||||||
def hasSet(state: State.Value): Boolean = statusBefore.hasSet(state)
|
def hasSet(state: State.Value): Boolean = statusBefore.hasSet(state)
|
||||||
|
|
||||||
def isUnimportant(state: State.Value): Boolean = importanceAfter.isUnimportant(state)
|
def isUnimportant(state: State.Value): Boolean = importanceAfter.isUnimportant(state)
|
||||||
|
|
||||||
|
def labelUseCount(label: String): Int = labelUseCountMap.map(_.getOrElse(label, 0)).getOrElse(-1)
|
||||||
}
|
}
|
||||||
|
|
||||||
object FlowInfo {
|
object FlowInfo {
|
||||||
val Default = FlowInfo(CpuStatus(), CpuImportance())
|
val Default = FlowInfo(CpuStatus(), CpuImportance(), None)
|
||||||
}
|
}
|
||||||
|
|
||||||
object FlowAnalyzer {
|
object FlowAnalyzer {
|
||||||
def analyze(f: NormalFunction, code: List[AssemblyLine], options: CompilationOptions): List[(FlowInfo, AssemblyLine)] = {
|
def analyze(f: NormalFunction, code: List[AssemblyLine], options: CompilationOptions, req: FlowInfoRequirement.Value): List[(FlowInfo, AssemblyLine)] = {
|
||||||
val forwardFlow = if (options.flag(CompilationFlag.DetailedFlowAnalysis)) {
|
val forwardFlow = req match {
|
||||||
QuantumFlowAnalyzer.analyze(f, code).map(_.collapse)
|
case FlowInfoRequirement.BothFlows | FlowInfoRequirement.ForwardFlow =>
|
||||||
} else {
|
if (options.flag(CompilationFlag.DetailedFlowAnalysis)) {
|
||||||
CoarseFlowAnalyzer.analyze(f, code)
|
QuantumFlowAnalyzer.analyze(f, code).map(_.collapse)
|
||||||
|
} else {
|
||||||
|
CoarseFlowAnalyzer.analyze(f, code)
|
||||||
|
}
|
||||||
|
case FlowInfoRequirement.BackwardFlow | FlowInfoRequirement.JustLabels | FlowInfoRequirement.NoRequirement =>
|
||||||
|
List.fill(code.size)(CpuStatus())
|
||||||
}
|
}
|
||||||
val reverseFlow = ReverseFlowAnalyzer.analyze(f, code)
|
val reverseFlow = req match {
|
||||||
forwardFlow.zip(reverseFlow).map{case (s,i) => FlowInfo(s,i)}.zip(code)
|
case FlowInfoRequirement.BothFlows | FlowInfoRequirement.BackwardFlow =>
|
||||||
|
ReverseFlowAnalyzer.analyze(f, code)
|
||||||
|
case FlowInfoRequirement.ForwardFlow | FlowInfoRequirement.JustLabels | FlowInfoRequirement.NoRequirement =>
|
||||||
|
List.fill(code.size)(CpuImportance())
|
||||||
|
}
|
||||||
|
val labelMap = req match {
|
||||||
|
case FlowInfoRequirement.NoRequirement => None
|
||||||
|
case _ => Some(code.flatMap {
|
||||||
|
case AssemblyLine(op, _, MemoryAddressConstant(Label(l)), _) if op != Opcode.LABEL => Some(l)
|
||||||
|
case _ => None
|
||||||
|
}.groupBy(identity).mapValues(_.size))
|
||||||
|
}
|
||||||
|
forwardFlow.zip(reverseFlow).map { case (s, i) => FlowInfo(s, i, labelMap) }.zip(code)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
package millfork.assembly.opt
|
package millfork.assembly.opt
|
||||||
|
|
||||||
import millfork.assembly.{AddrMode, AssemblyLine, Opcode, State}
|
import millfork.assembly._
|
||||||
import millfork.assembly.Opcode._
|
import millfork.assembly.Opcode._
|
||||||
import millfork.assembly.AddrMode._
|
import millfork.assembly.AddrMode._
|
||||||
import millfork.assembly.OpcodeClasses._
|
import millfork.assembly.OpcodeClasses._
|
||||||
@ -228,15 +228,140 @@ object LaterOptimizations {
|
|||||||
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
private val LdxAddrModes = Set(Immediate, Absolute, ZeroPage, ZeroPageY, AbsoluteY)
|
||||||
|
private val LdyAddrModes = Set(Immediate, Absolute, ZeroPage, ZeroPageX, AbsoluteX)
|
||||||
|
private val StxAddrModes = Set(Absolute, ZeroPage, ZeroPageY)
|
||||||
|
private val StyAddrModes = Set(Absolute, ZeroPage, ZeroPageX)
|
||||||
|
private val CpxyAddrModes = Set(Immediate, Absolute, ZeroPage)
|
||||||
|
|
||||||
|
private def incDecThroughIndexRegister(amount: Int, dec: Boolean, carrySet: Boolean, useX: Boolean) = {
|
||||||
|
val ldAddrModes = if (useX) LdxAddrModes else LdyAddrModes
|
||||||
|
val stAddrModes = if (useX) StxAddrModes else StyAddrModes
|
||||||
|
val ldOp = if (useX) LDX else LDY
|
||||||
|
val stOp = if (useX) STX else STY
|
||||||
|
val changeOp = if (dec) if (useX) DEX else DEY else if (useX) INX else INY
|
||||||
|
val addOp = if (dec) SBC else ADC
|
||||||
|
val addParam = if (dec ^ carrySet) amount + 1 else amount
|
||||||
|
val indexState = if (useX) State.X else State.Y
|
||||||
|
val cState = if (carrySet) HasSet(State.C) else HasClear(State.C)
|
||||||
|
val carryOp = if (carrySet) SEC else CLC
|
||||||
|
|
||||||
|
(Elidable & HasOpcode(LDA) & HasAddrModeIn(ldAddrModes)).capture(11) ~
|
||||||
|
(Elidable & HasOpcode(carryOp)).? ~
|
||||||
|
(Elidable & HasOpcode(addOp) & HasImmediate(addParam) & cState & HasClear(State.D)) ~
|
||||||
|
(Elidable & HasOpcode(STA) & HasAddrModeIn(stAddrModes) & DoesntMatterWhatItDoesWith(State.A, State.C, State.V, indexState)).capture(12) ~~> { (_, ctx) =>
|
||||||
|
ctx.get[List[AssemblyLine]](11).head.copy(opcode = ldOp) ::
|
||||||
|
(List.fill(amount)(AssemblyLine.implied(changeOp)) :+
|
||||||
|
ctx.get[List[AssemblyLine]](12).head.copy(opcode = stOp))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val IncrementThroughIndexRegisters = new RuleBasedAssemblyOptimization("Increment through index registers",
|
||||||
|
needsFlowInfo = FlowInfoRequirement.BothFlows,
|
||||||
|
incDecThroughIndexRegister(1, dec = false, carrySet = false, useX = true),
|
||||||
|
incDecThroughIndexRegister(1, dec = false, carrySet = false, useX = false),
|
||||||
|
incDecThroughIndexRegister(1, dec = false, carrySet = true, useX = true),
|
||||||
|
incDecThroughIndexRegister(1, dec = false, carrySet = true, useX = false),
|
||||||
|
incDecThroughIndexRegister(1, dec = true, carrySet = true, useX = true),
|
||||||
|
incDecThroughIndexRegister(1, dec = true, carrySet = true, useX = false),
|
||||||
|
incDecThroughIndexRegister(2, dec = false, carrySet = false, useX = true),
|
||||||
|
incDecThroughIndexRegister(2, dec = false, carrySet = false, useX = false),
|
||||||
|
incDecThroughIndexRegister(2, dec = false, carrySet = true, useX = true),
|
||||||
|
incDecThroughIndexRegister(2, dec = false, carrySet = true, useX = false),
|
||||||
|
incDecThroughIndexRegister(2, dec = true, carrySet = true, useX = true),
|
||||||
|
incDecThroughIndexRegister(2, dec = true, carrySet = true, useX = false),
|
||||||
|
)
|
||||||
|
|
||||||
|
val LoadingBranchesOptimization = new RuleBasedAssemblyOptimization("Loading branches optimization",
|
||||||
|
needsFlowInfo = FlowInfoRequirement.BackwardFlow,
|
||||||
|
(Elidable & HasOpcode(LDA) & HasAddrModeIn(LdxAddrModes) & DoesntMatterWhatItDoesWith(State.X)) ~
|
||||||
|
(Linear & Not(ConcernsX) & Not(ChangesA) & Not(HasOpcode(CMP)) & (Not(ReadsA) | Elidable & HasOpcode(STA) & HasAddrModeIn(StxAddrModes)) ).*.capture(39) ~
|
||||||
|
(Elidable & HasOpcode(CMP) & HasAddrModeIn(CpxyAddrModes)).?.capture(40) ~
|
||||||
|
(Elidable & HasOpcodeIn(OpcodeClasses.ShortConditionalBranching) & MatchParameter(22)).capture(41) ~
|
||||||
|
(Elidable & HasOpcode(LDA)).capture(31) ~
|
||||||
|
(Elidable & HasOpcodeIn(Set(JMP, BRA)) & MatchParameter(21)) ~
|
||||||
|
(Elidable & HasOpcode(LABEL) & MatchParameter(22)).capture(42) ~
|
||||||
|
(Elidable & HasOpcode(LDA)).capture(32) ~
|
||||||
|
(Elidable & HasOpcode(LABEL) & MatchParameter(21) & HasCallerCount(1) & DoesntMatterWhatItDoesWith(State.A, State.X, State.N, State.Z)) ~~> { (code, ctx) =>
|
||||||
|
val ldx = List(code.head.copy(opcode = LDX))
|
||||||
|
val stx = ctx.get[List[AssemblyLine]](39).map(l => if (l.opcode == STA) l.copy(opcode = STX) else l )
|
||||||
|
val cpx = ctx.get[List[AssemblyLine]](40).map(_.copy(opcode = CPX))
|
||||||
|
val branch = ctx.get[List[AssemblyLine]](41)
|
||||||
|
val label = ctx.get[List[AssemblyLine]](42)
|
||||||
|
val loadIfJumped = ctx.get[List[AssemblyLine]](32)
|
||||||
|
val loadIfNotJumped = ctx.get[List[AssemblyLine]](31)
|
||||||
|
List(loadIfJumped, ldx, stx, cpx, branch, loadIfNotJumped, label).flatten
|
||||||
|
},
|
||||||
|
(Elidable & HasOpcode(LDA) & HasAddrModeIn(LdyAddrModes) & DoesntMatterWhatItDoesWith(State.Y)) ~
|
||||||
|
(Linear & Not(ConcernsY) & Not(ChangesA) & Not(HasOpcode(CMP)) & (Not(ReadsA) | Elidable & HasOpcode(STA) & HasAddrModeIn(StyAddrModes)) ).*.capture(39) ~
|
||||||
|
(Elidable & HasOpcode(CMP) & HasAddrModeIn(CpxyAddrModes)).?.capture(40) ~
|
||||||
|
(Elidable & HasOpcodeIn(OpcodeClasses.ShortConditionalBranching) & MatchParameter(22)).capture(41) ~
|
||||||
|
(Elidable & HasOpcode(LDA)).capture(31) ~
|
||||||
|
(Elidable & HasOpcodeIn(Set(JMP, BRA)) & MatchParameter(21)) ~
|
||||||
|
(Elidable & HasOpcode(LABEL) & MatchParameter(22)).capture(42) ~
|
||||||
|
(Elidable & HasOpcode(LDA)).capture(32) ~
|
||||||
|
(Elidable & HasOpcode(LABEL) & MatchParameter(21) & HasCallerCount(1) & DoesntMatterWhatItDoesWith(State.A, State.Y, State.N, State.Z)) ~~> { (code, ctx) =>
|
||||||
|
val ldy = List(code.head.copy(opcode = LDY))
|
||||||
|
val sty = ctx.get[List[AssemblyLine]](39).map(l => if (l.opcode == STA) l.copy(opcode = STY) else l )
|
||||||
|
val cpy = ctx.get[List[AssemblyLine]](40).map(_.copy(opcode = CPY))
|
||||||
|
val branch = ctx.get[List[AssemblyLine]](41)
|
||||||
|
val label = ctx.get[List[AssemblyLine]](42)
|
||||||
|
val loadIfJumped = ctx.get[List[AssemblyLine]](32)
|
||||||
|
val loadIfNotJumped = ctx.get[List[AssemblyLine]](31)
|
||||||
|
List(loadIfJumped, ldy, sty, cpy, branch, loadIfNotJumped, label).flatten
|
||||||
|
},
|
||||||
|
(HasOpcode(LDA) & DoesntMatterWhatItDoesWith(State.X)) ~
|
||||||
|
(Linear & Not(ConcernsX)).*.capture(39) ~
|
||||||
|
(Elidable & HasOpcodeIn(OpcodeClasses.ShortConditionalBranching) & MatchParameter(22)).capture(41) ~
|
||||||
|
(Elidable & HasOpcode(LDA) & HasAddrModeIn(LdxAddrModes)).capture(31) ~
|
||||||
|
(Elidable & HasOpcodeIn(Set(JMP, BRA)) & MatchParameter(21)) ~
|
||||||
|
(Elidable & HasOpcode(LABEL) & MatchParameter(22)).capture(42) ~
|
||||||
|
(Elidable & HasOpcode(LDA) & HasAddrModeIn(LdxAddrModes)).capture(32) ~
|
||||||
|
(Elidable & HasOpcode(LABEL) & MatchParameter(21) & HasCallerCount(1)) ~
|
||||||
|
(Elidable & HasOpcode(STA) & HasAddrModeIn(StxAddrModes) & DoesntMatterWhatItDoesWith(State.A, State.X, State.N, State.Z)).capture(33) ~~> { (code, ctx) =>
|
||||||
|
val lda = List(code.head)
|
||||||
|
val cmp = ctx.get[List[AssemblyLine]](39)
|
||||||
|
val branch = ctx.get[List[AssemblyLine]](41)
|
||||||
|
val label = ctx.get[List[AssemblyLine]](42)
|
||||||
|
val loadIfJumped = ctx.get[List[AssemblyLine]](32).map(_.copy(opcode = LDX))
|
||||||
|
val loadIfNotJumped = ctx.get[List[AssemblyLine]](31).map(_.copy(opcode = LDX))
|
||||||
|
val stx = ctx.get[List[AssemblyLine]](33).map(_.copy(opcode = STX))
|
||||||
|
List(loadIfJumped, lda, cmp, branch, loadIfNotJumped, label, stx).flatten
|
||||||
|
},
|
||||||
|
(HasOpcode(LDA) & DoesntMatterWhatItDoesWith(State.Y)) ~
|
||||||
|
(Linear & Not(ConcernsY)).*.capture(39) ~
|
||||||
|
(Elidable & HasOpcodeIn(OpcodeClasses.ShortConditionalBranching) & MatchParameter(22)).capture(41) ~
|
||||||
|
(Elidable & HasOpcode(LDA) & HasAddrModeIn(LdyAddrModes)).capture(31) ~
|
||||||
|
(Elidable & HasOpcodeIn(Set(JMP, BRA)) & MatchParameter(21)) ~
|
||||||
|
(Elidable & HasOpcode(LABEL) & MatchParameter(22)).capture(42) ~
|
||||||
|
(Elidable & HasOpcode(LDA) & HasAddrModeIn(LdyAddrModes)).capture(32) ~
|
||||||
|
(Elidable & HasOpcode(LABEL) & MatchParameter(21) & HasCallerCount(1)) ~
|
||||||
|
(Elidable & HasOpcode(STA) & HasAddrModeIn(StyAddrModes) & DoesntMatterWhatItDoesWith(State.A, State.Y, State.N, State.Z)).capture(33) ~~> { (code, ctx) =>
|
||||||
|
val lda = List(code.head)
|
||||||
|
val cmp = ctx.get[List[AssemblyLine]](39)
|
||||||
|
val branch = ctx.get[List[AssemblyLine]](41)
|
||||||
|
val label = ctx.get[List[AssemblyLine]](42)
|
||||||
|
val loadIfJumped = ctx.get[List[AssemblyLine]](32).map(_.copy(opcode = LDY))
|
||||||
|
val loadIfNotJumped = ctx.get[List[AssemblyLine]](31).map(_.copy(opcode = LDY))
|
||||||
|
val sty = ctx.get[List[AssemblyLine]](33).map(_.copy(opcode = STY))
|
||||||
|
List(loadIfJumped, lda, cmp, branch, loadIfNotJumped, label, sty).flatten
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
val All = List(
|
val All = List(
|
||||||
DoubleLoadToDifferentRegisters,
|
DoubleLoadToDifferentRegisters,
|
||||||
DoubleLoadToTheSameRegister,
|
DoubleLoadToTheSameRegister,
|
||||||
IndexSwitchingOptimization,
|
IndexSwitchingOptimization,
|
||||||
|
LoadingBranchesOptimization,
|
||||||
PointlessLoadAfterStore,
|
PointlessLoadAfterStore,
|
||||||
PointessLoadingForShifting,
|
PointessLoadingForShifting,
|
||||||
LoadingAfterShifting,
|
LoadingAfterShifting,
|
||||||
UseXInsteadOfStack,
|
UseXInsteadOfStack,
|
||||||
UseYInsteadOfStack,
|
UseYInsteadOfStack,
|
||||||
UseZeropageAddressingMode)
|
UseZeropageAddressingMode)
|
||||||
|
|
||||||
|
val Nmos = List(
|
||||||
|
IncrementThroughIndexRegisters
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,16 +13,16 @@ import scala.collection.mutable
|
|||||||
|
|
||||||
object FlowInfoRequirement extends Enumeration {
|
object FlowInfoRequirement extends Enumeration {
|
||||||
|
|
||||||
val NoRequirement, BothFlows, ForwardFlow, BackwardFlow = Value
|
val NoRequirement, JustLabels, BothFlows, ForwardFlow, BackwardFlow = Value
|
||||||
|
|
||||||
def assertForward(x: FlowInfoRequirement.Value): Unit = x match {
|
def assertForward(x: FlowInfoRequirement.Value): Unit = x match {
|
||||||
case BothFlows | ForwardFlow => ()
|
case BothFlows | ForwardFlow => ()
|
||||||
case NoRequirement | BackwardFlow => ErrorReporting.fatal("Forward flow info required")
|
case NoRequirement | JustLabels | BackwardFlow => ErrorReporting.fatal("Forward flow info required")
|
||||||
}
|
}
|
||||||
|
|
||||||
def assertBackward(x: FlowInfoRequirement.Value): Unit = x match {
|
def assertBackward(x: FlowInfoRequirement.Value): Unit = x match {
|
||||||
case BothFlows | BackwardFlow => ()
|
case BothFlows | BackwardFlow => ()
|
||||||
case NoRequirement | ForwardFlow => ErrorReporting.fatal("Backward flow info required")
|
case NoRequirement | JustLabels | ForwardFlow => ErrorReporting.fatal("Backward flow info required")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -32,18 +32,7 @@ class RuleBasedAssemblyOptimization(val name: String, val needsFlowInfo: FlowInf
|
|||||||
|
|
||||||
override def optimize(f: NormalFunction, code: List[AssemblyLine], options: CompilationOptions): List[AssemblyLine] = {
|
override def optimize(f: NormalFunction, code: List[AssemblyLine], options: CompilationOptions): 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 = needsFlowInfo match {
|
val taggedCode = FlowAnalyzer.analyze(f, effectiveCode, options, needsFlowInfo)
|
||||||
case FlowInfoRequirement.NoRequirement => effectiveCode.map(FlowInfo.Default -> _)
|
|
||||||
case FlowInfoRequirement.BothFlows => FlowAnalyzer.analyze(f, effectiveCode, options)
|
|
||||||
case FlowInfoRequirement.ForwardFlow =>
|
|
||||||
if (options.flag(CompilationFlag.DetailedFlowAnalysis)) {
|
|
||||||
QuantumFlowAnalyzer.analyze(f, code).map(s => FlowInfo(s.collapse, CpuImportance())).zip(code)
|
|
||||||
} else {
|
|
||||||
CoarseFlowAnalyzer.analyze(f, code).map(s => FlowInfo(s, CpuImportance())).zip(code)
|
|
||||||
}
|
|
||||||
case FlowInfoRequirement.BackwardFlow =>
|
|
||||||
ReverseFlowAnalyzer.analyze(f, code).map(i => FlowInfo(CpuStatus(), i)).zip(code)
|
|
||||||
}
|
|
||||||
optimizeImpl(f, taggedCode, options)
|
optimizeImpl(f, taggedCode, options)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -755,3 +744,11 @@ case class Before(pattern: AssemblyPattern) extends AssemblyLinePattern {
|
|||||||
|
|
||||||
override def matchLineTo(ctx: AssemblyMatchingContext, flowInfo: FlowInfo, line: AssemblyLine): Boolean = ???
|
override def matchLineTo(ctx: AssemblyMatchingContext, flowInfo: FlowInfo, line: AssemblyLine): Boolean = ???
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case class HasCallerCount(count: Int) extends AssemblyLinePattern {
|
||||||
|
override def matchLineTo(ctx: AssemblyMatchingContext, flowInfo: FlowInfo, line: AssemblyLine): Boolean =
|
||||||
|
line match {
|
||||||
|
case AssemblyLine(Opcode.LABEL, _, MemoryAddressConstant(Label(l)), _) => flowInfo.labelUseCount(l) == count
|
||||||
|
case _ => false
|
||||||
|
}
|
||||||
|
}
|
@ -21,6 +21,8 @@ object SuperOptimizer extends AssemblyOptimization {
|
|||||||
}
|
}
|
||||||
if (options.flag(CompilationFlag.EmitCmosOpcodes)) {
|
if (options.flag(CompilationFlag.EmitCmosOpcodes)) {
|
||||||
allOptimizers ++= CmosOptimizations.All
|
allOptimizers ++= CmosOptimizations.All
|
||||||
|
} else {
|
||||||
|
allOptimizers ++= LaterOptimizations.Nmos
|
||||||
}
|
}
|
||||||
allOptimizers ++= List(
|
allOptimizers ++= List(
|
||||||
VariableToRegisterOptimization,
|
VariableToRegisterOptimization,
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
package millfork.test
|
package millfork.test
|
||||||
|
|
||||||
import millfork.test.emu.EmuBenchmarkRun
|
import millfork.test.emu.{EmuBenchmarkRun, EmuSuperOptimizedRun, EmuUltraBenchmarkRun}
|
||||||
import org.scalatest.{FunSuite, Matchers}
|
import org.scalatest.{FunSuite, Matchers}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -55,7 +55,7 @@ class ComparisonSuite extends FunSuite with Matchers {
|
|||||||
}
|
}
|
||||||
|
|
||||||
test("Carry flag optimization test") {
|
test("Carry flag optimization test") {
|
||||||
EmuBenchmarkRun(
|
EmuUltraBenchmarkRun(
|
||||||
"""
|
"""
|
||||||
| byte output @$c000
|
| byte output @$c000
|
||||||
| void main () {
|
| void main () {
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
package millfork.test.emu
|
package millfork.test.emu
|
||||||
|
|
||||||
|
import millfork.assembly.opt.LaterOptimizations
|
||||||
import millfork.{Cpu, OptimizationPresets}
|
import millfork.{Cpu, OptimizationPresets}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -8,7 +9,10 @@ import millfork.{Cpu, OptimizationPresets}
|
|||||||
object EmuOptimizedInlinedRun extends EmuRun(
|
object EmuOptimizedInlinedRun extends EmuRun(
|
||||||
Cpu.StrictMos,
|
Cpu.StrictMos,
|
||||||
OptimizationPresets.NodeOpt,
|
OptimizationPresets.NodeOpt,
|
||||||
OptimizationPresets.AssOpt ++ OptimizationPresets.Good ++ OptimizationPresets.Good ++ OptimizationPresets.Good,
|
OptimizationPresets.AssOpt ++
|
||||||
|
OptimizationPresets.Good ++ LaterOptimizations.Nmos ++
|
||||||
|
OptimizationPresets.Good ++ LaterOptimizations.Nmos ++
|
||||||
|
OptimizationPresets.Good,
|
||||||
false)
|
false)
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
package millfork.test.emu
|
package millfork.test.emu
|
||||||
|
|
||||||
|
import millfork.assembly.opt.LaterOptimizations
|
||||||
import millfork.{Cpu, OptimizationPresets}
|
import millfork.{Cpu, OptimizationPresets}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -8,7 +9,10 @@ import millfork.{Cpu, OptimizationPresets}
|
|||||||
object EmuOptimizedRun extends EmuRun(
|
object EmuOptimizedRun extends EmuRun(
|
||||||
Cpu.StrictMos,
|
Cpu.StrictMos,
|
||||||
OptimizationPresets.NodeOpt,
|
OptimizationPresets.NodeOpt,
|
||||||
OptimizationPresets.AssOpt ++ OptimizationPresets.Good ++ OptimizationPresets.Good ++ OptimizationPresets.Good,
|
OptimizationPresets.AssOpt ++
|
||||||
|
OptimizationPresets.Good ++ LaterOptimizations.Nmos ++
|
||||||
|
OptimizationPresets.Good ++ LaterOptimizations.Nmos ++
|
||||||
|
OptimizationPresets.Good,
|
||||||
false)
|
false)
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
package millfork.test.emu
|
package millfork.test.emu
|
||||||
|
|
||||||
|
import millfork.assembly.opt.LaterOptimizations
|
||||||
import millfork.{Cpu, OptimizationPresets}
|
import millfork.{Cpu, OptimizationPresets}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -8,7 +9,10 @@ import millfork.{Cpu, OptimizationPresets}
|
|||||||
object EmuQuantumOptimizedRun extends EmuRun(
|
object EmuQuantumOptimizedRun extends EmuRun(
|
||||||
Cpu.StrictMos,
|
Cpu.StrictMos,
|
||||||
OptimizationPresets.NodeOpt,
|
OptimizationPresets.NodeOpt,
|
||||||
OptimizationPresets.AssOpt ++ OptimizationPresets.Good ++ OptimizationPresets.Good ++ OptimizationPresets.Good,
|
OptimizationPresets.AssOpt ++
|
||||||
|
OptimizationPresets.Good ++ LaterOptimizations.Nmos ++
|
||||||
|
OptimizationPresets.Good ++ LaterOptimizations.Nmos ++
|
||||||
|
OptimizationPresets.Good,
|
||||||
true)
|
true)
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
package millfork.test.emu
|
package millfork.test.emu
|
||||||
|
|
||||||
import millfork.assembly.opt.UndocumentedOptimizations
|
import millfork.assembly.opt.{LaterOptimizations, UndocumentedOptimizations}
|
||||||
import millfork.{Cpu, OptimizationPresets}
|
import millfork.{Cpu, OptimizationPresets}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -9,7 +9,11 @@ import millfork.{Cpu, OptimizationPresets}
|
|||||||
object EmuUndocumentedRun extends EmuRun(
|
object EmuUndocumentedRun extends EmuRun(
|
||||||
Cpu.Ricoh, // not Cpu.Mos, because I haven't found an emulator that supports both illegals and decimal mode yet
|
Cpu.Ricoh, // not Cpu.Mos, because I haven't found an emulator that supports both illegals and decimal mode yet
|
||||||
OptimizationPresets.NodeOpt,
|
OptimizationPresets.NodeOpt,
|
||||||
OptimizationPresets.AssOpt ++ UndocumentedOptimizations.All ++ OptimizationPresets.Good ++ UndocumentedOptimizations.All ++ OptimizationPresets.Good,
|
OptimizationPresets.AssOpt ++ LaterOptimizations.Nmos ++
|
||||||
|
UndocumentedOptimizations.All ++
|
||||||
|
OptimizationPresets.Good ++ LaterOptimizations.Nmos ++
|
||||||
|
UndocumentedOptimizations.All ++
|
||||||
|
OptimizationPresets.Good,
|
||||||
false) {
|
false) {
|
||||||
|
|
||||||
override def emitIllegals = true
|
override def emitIllegals = true
|
||||||
|
Loading…
x
Reference in New Issue
Block a user