1
0
mirror of https://github.com/KarolS/millfork.git synced 2025-01-11 12:29:46 +00:00

65816: Various fixes regarding (d,S),Y addressing mode

This commit is contained in:
Karol Stasiak 2019-04-15 00:16:50 +02:00
parent 7635fc256f
commit 194f79f907
4 changed files with 17 additions and 32 deletions

View File

@ -494,12 +494,18 @@ object AssemblyLine {
def absoluteX(opcode: Opcode.Value, thing: ThingInMemory, offset: Int = 0) = def absoluteX(opcode: Opcode.Value, thing: ThingInMemory, offset: Int = 0) =
AssemblyLine(opcode, AddrMode.AbsoluteX, thing.toAddress + offset) AssemblyLine(opcode, AddrMode.AbsoluteX, thing.toAddress + offset)
def indexedY(opcode: Opcode.Value, addr: Int) =
AssemblyLine(opcode, AddrMode.IndexedY, NumericConstant(addr & 0xff, 1))
def indexedY(opcode: Opcode.Value, addr: Constant) = def indexedY(opcode: Opcode.Value, addr: Constant) =
AssemblyLine(opcode, AddrMode.IndexedY, addr) AssemblyLine(opcode, AddrMode.IndexedY, addr)
def indexedY(opcode: Opcode.Value, thing: ThingInMemory, offset: Int = 0) = def indexedY(opcode: Opcode.Value, thing: ThingInMemory, offset: Int = 0) =
AssemblyLine(opcode, AddrMode.IndexedY, thing.toAddress + offset) AssemblyLine(opcode, AddrMode.IndexedY, thing.toAddress + offset)
def indexedSY(opcode: Opcode.Value, offset: Int = 0) =
AssemblyLine(opcode, AddrMode.IndexedSY, NumericConstant(offset & 0xff, 1))
def stackRelative(opcode: Opcode.Value, addr: Int) = def stackRelative(opcode: Opcode.Value, addr: Int) =
AssemblyLine(opcode, AddrMode.Stack, NumericConstant(addr & 0xff, 1)) AssemblyLine(opcode, AddrMode.Stack, NumericConstant(addr & 0xff, 1))
} }

View File

@ -38,14 +38,14 @@ class ChangeIndexRegisterOptimization(preferX2Y: Boolean) extends AssemblyOptimi
OpcodeClasses.ReadsYAlways(l.opcode) || OpcodeClasses.ReadsYAlways(l.opcode) ||
OpcodeClasses.ChangesX(l.opcode) || OpcodeClasses.ChangesX(l.opcode) ||
OpcodeClasses.ChangesY(l.opcode) || OpcodeClasses.ChangesY(l.opcode) ||
Set(AbsoluteX, AbsoluteY, ZeroPageY, ZeroPageX, IndexedX, IndexedY)(l.addrMode) Set(AbsoluteX, AbsoluteY, ZeroPageY, ZeroPageX, IndexedX, IndexedY, LongIndexedY, IndexedSY)(l.addrMode)
) )
val usesY = code.exists(l => val usesY = code.exists(l =>
OpcodeClasses.ReadsXAlways(l.opcode) || OpcodeClasses.ReadsXAlways(l.opcode) ||
OpcodeClasses.ReadsYAlways(l.opcode) || OpcodeClasses.ReadsYAlways(l.opcode) ||
OpcodeClasses.ChangesX(l.opcode) || OpcodeClasses.ChangesX(l.opcode) ||
OpcodeClasses.ChangesY(l.opcode) || OpcodeClasses.ChangesY(l.opcode) ||
Set(AbsoluteX, AbsoluteY, ZeroPageY, ZeroPageX, IndexedX, IndexedY)(l.addrMode) Set(AbsoluteX, AbsoluteY, ZeroPageY, ZeroPageX, IndexedX, IndexedY, LongIndexedY, IndexedSY)(l.addrMode)
) )
if (!usesX && !usesY) { if (!usesX && !usesY) {
return code return code
@ -109,6 +109,7 @@ class ChangeIndexRegisterOptimization(preferX2Y: Boolean) extends AssemblyOptimi
case AssemblyLine0(_, ZeroPageY, _) :: xs if loaded != Some(Y) => false case AssemblyLine0(_, ZeroPageY, _) :: xs if loaded != Some(Y) => false
case AssemblyLine0(_, IndexedY, _) :: xs if dir == Y2X || loaded != Some(Y) => false case AssemblyLine0(_, IndexedY, _) :: xs if dir == Y2X || loaded != Some(Y) => false
case AssemblyLine0(_, LongIndexedY, _) :: xs if dir == Y2X || loaded != Some(Y) => false case AssemblyLine0(_, LongIndexedY, _) :: xs if dir == Y2X || loaded != Some(Y) => false
case AssemblyLine0(_, IndexedSY, _) :: xs if dir == Y2X || loaded != Some(Y) => false
case AssemblyLine0(_, AbsoluteX, _) :: xs if loaded != Some(X) => false case AssemblyLine0(_, AbsoluteX, _) :: xs if loaded != Some(X) => false
case AssemblyLine0(_, LongAbsoluteX, _) :: xs if loaded != Some(X) => false case AssemblyLine0(_, LongAbsoluteX, _) :: xs if loaded != Some(X) => false
case AssemblyLine0(_, ZeroPageX, _) :: xs if loaded != Some(X) => false case AssemblyLine0(_, ZeroPageX, _) :: xs if loaded != Some(X) => false
@ -225,6 +226,8 @@ class ChangeIndexRegisterOptimization(preferX2Y: Boolean) extends AssemblyOptimi
case (x@AssemblyLine0(_, AbsoluteY, _)) :: xs => x.copy(addrMode = AbsoluteX) :: switchY2X(xs) case (x@AssemblyLine0(_, AbsoluteY, _)) :: xs => x.copy(addrMode = AbsoluteX) :: switchY2X(xs)
case (x@AssemblyLine0(_, ZeroPageY, _)) :: xs => x.copy(addrMode = ZeroPageX) :: switchY2X(xs) case (x@AssemblyLine0(_, ZeroPageY, _)) :: xs => x.copy(addrMode = ZeroPageX) :: switchY2X(xs)
case AssemblyLine0(_, IndexedY, _) :: xs => log.fatal("Unexpected IndexedY") case AssemblyLine0(_, IndexedY, _) :: xs => log.fatal("Unexpected IndexedY")
case AssemblyLine0(_, LongIndexedY, _) :: xs => log.fatal("Unexpected LongIndexedY")
case AssemblyLine0(_, IndexedSY, _) :: xs => log.fatal("Unexpected IndexedSY")
case x::xs => x :: switchY2X(xs) case x::xs => x :: switchY2X(xs)
case Nil => Nil case Nil => Nil

View File

@ -50,7 +50,7 @@ class MosAssembler(program: Program,
writeByte(bank, index, MosAssembler.opcodeFor(op, Relative, options)) writeByte(bank, index, MosAssembler.opcodeFor(op, Relative, options))
writeByte(bank, index + 1, AssertByte(param - (index + 2))) writeByte(bank, index + 1, AssertByte(param - (index + 2)))
index + 2 index + 2
case AssemblyLine0(op, am@(Immediate | ZeroPage | ZeroPageX | ZeroPageY | IndexedY | IndexedX | IndexedZ | LongIndexedY | LongIndexedZ | Stack), param) => case AssemblyLine0(op, am@(Immediate | ZeroPage | ZeroPageX | ZeroPageY | IndexedY | IndexedX | IndexedZ | IndexedSY | LongIndexedY | LongIndexedZ | Stack), param) =>
writeByte(bank, index, MosAssembler.opcodeFor(op, am, options)) writeByte(bank, index, MosAssembler.opcodeFor(op, am, options))
writeByte(bank, index + 1, param) writeByte(bank, index + 1, param)
index + 2 index + 2

View File

@ -2898,6 +2898,7 @@ var STA_stack_relative_indirect_indexed_y= {
execute:function(cpu, bytes) { execute:function(cpu, bytes) {
cpu.cycle_count+=7; cpu.cycle_count+=7;
// TODO: fixes below are not well-tested!
if(cpu.p.e) { if(cpu.p.e) {
var location_loc = 0x100 | ((cpu.r.s + bytes[0]) & 0xff), var location_loc = 0x100 | ((cpu.r.s + bytes[0]) & 0xff),
low_byte = cpu.mmu.read_byte(location_loc), low_byte = cpu.mmu.read_byte(location_loc),
@ -2907,41 +2908,16 @@ var STA_stack_relative_indirect_indexed_y= {
} else { } else {
high_byte = cpu.mmu.read_byte(location_loc+1); high_byte = cpu.mmu.read_byte(location_loc+1);
} }
var absolute_location = ((high_byte<<8)|low_byte)+cpu.r.y, var absolute_location = ((high_byte<<8)|low_byte)+cpu.r.y;
b; cpu.mmu.store_byte(absolute_location, cpu.r.a);
if(absolute_location>=0x10000) {
b = cpu.mmu.read_byte_long(absolute_location, cpu.r.dbr+1);
} else {
b = cpu.mmu.read_byte(absolute_location);
}
cpu.mmu.store_byte(b, cpu.r.a);
} else { } else {
var location_loc = cpu.r.s + bytes[0], var location_loc = cpu.r.s + bytes[0],
absolute_location = cpu.mmu.read_word(location_loc)+cpu.r.y; absolute_location = cpu.mmu.read_word(location_loc)+cpu.r.y;
if(cpu.p.m) { if(cpu.p.m) {
var b; cpu.mmu.store_byte(absolute_location, cpu.r.a);
if(absolute_location>=0x10000) {
b = cpu.mmu.read_byte_long(absolute_location, cpu.r.dbr+1);
} else {
b = cpu.mmu.read_byte(absolute_location);
}
cpu.mmu.store_byte(b, cpu.r.a);
} else { } else {
cpu.cycle_count++; cpu.cycle_count++;
cpu.mmu.store_word(absolute_location, cpu.r.a);
var low_byte, high_byte;
if(absolute_location>=0x10000) {
low_byte = cpu.mmu.read_byte_long(absolute_location, cpu.r.dbr+1);
high_byte = cpu.mmu.read_byte_long(absolute_location+1, cpu.r.dbr+1);
} else {
low_byte = cpu.mmu.read_byte(absolute_location);
if(absolute_location===0xffff) {
high_byte = cpu.mmu.read_byte_long(0, cpu.r.dbr+1);
} else {
high_byte = cpu.mmu.read_byte(absolute_location);
}
}
cpu.mmu.store_word((high_byte<<8)|low_byte, cpu.r.a);
} }
} }
} }