Test refactoring; complete timing and nearly complete functional

This commit is contained in:
edmccard 2012-04-02 20:58:51 -04:00
parent afc38b6775
commit f79021fc14
5 changed files with 2713 additions and 699 deletions

File diff suppressed because it is too large Load Diff

View File

@ -58,6 +58,19 @@ template CPU(string arch, bool strict, bool cumulative)
} }
auto makeCpu(T)(CpuInfo info)
if (isCpu!T)
{
auto cpu = new T();
cpu.programCounter = info.PC;
cpu.stackPointer = info.SP;
cpu.flag.fromByte(info.S);
cpu.accumulator = info.A;
cpu.xIndex = info.X;
cpu.yIndex = info.Y;
return cpu;
}
// Connects test memory to a cpu. // Connects test memory to a cpu.
void connectMem(T, S)(T cpu, ref S mem) void connectMem(T, S)(T cpu, ref S mem)
if (isCpu!T) if (isCpu!T)
@ -222,6 +235,33 @@ if (isCpu!T)
} }
struct CpuInfo
{
ushort PC;
ubyte SP = 0xFF;
ubyte A, X, Y;
ubyte S = 0x30;
string toString() const
{
return format("PC %0.4X SP %0.2X S %0.2X A %0.2X X %0.2X Y %0.2X",
PC, SP, S, A, X, Y);
}
static CpuInfo fromCpu(T)(T cpu)
{
CpuInfo info;
info.PC = cpu.programCounter;
info.SP = cpu.stackPointer;
info.A = cpu.accumulator;
info.X = cpu.xIndex;
info.Y = cpu.yIndex;
info.S = cpu.flag.toByte();
return info;
}
}
// Sets the program counter. // Sets the program counter.
void setPC(T)(T cpu, int addr) void setPC(T)(T cpu, int addr)
if (isCpu!T) if (isCpu!T)
@ -229,6 +269,16 @@ if (isCpu!T)
cpu.programCounter = cast(ushort)addr; cpu.programCounter = cast(ushort)addr;
} }
void setPC(T : CpuInfo)(ref T cpu, int addr)
{
cpu.PC = cast(ushort)addr;
}
void incPC(T : CpuInfo)(ref T cpu, int amt = 1)
{
cpu.PC = pageCrossAdd(cpu.PC, amt);
}
// Returns the program counter. // Returns the program counter.
ushort getPC(T)(T cpu) ushort getPC(T)(T cpu)
if (isCpu!T) if (isCpu!T)
@ -236,6 +286,11 @@ if (isCpu!T)
return cpu.programCounter; return cpu.programCounter;
} }
ushort getPC(T : CpuInfo)(ref T cpu)
{
return cpu.PC;
}
/* /*
* Sets the stack pointer. * Sets the stack pointer.
@ -250,6 +305,23 @@ if (isCpu!T)
cpu.stackPointer = cast(ubyte)val; cpu.stackPointer = cast(ubyte)val;
} }
void setSP(T : CpuInfo)(ref T cpu, int val)
{
assert(val < 0x0200);
cpu.SP = cast(ubyte)val;
}
void incSP(T : CpuInfo)(ref T cpu, int amt = 1)
{
cpu.SP = cast(ubyte)pageWrapAdd(cpu.SP, amt);
}
void decSP(T : CpuInfo)(ref T cpu, int amt = -1)
{
cpu.SP = cast(ubyte)pageWrapAdd(cpu.SP, amt);
}
/* /*
* Returns the stack address (in the range 0x0100-0x01FF) represented * Returns the stack address (in the range 0x0100-0x01FF) represented
* by the stack pointer. * by the stack pointer.
@ -260,6 +332,66 @@ if (isCpu!T)
return 0x100 | cpu.stackPointer; return 0x100 | cpu.stackPointer;
} }
ushort getSP(T : CpuInfo)(ref T cpu)
{
return 0x0100 | cpu.SP;
}
// The names of the registers.
enum Reg
{
A, X, Y
}
// Sets a register.
void setReg(T)(ref T cpu, Reg reg, int val)
if (isCpu!T || is(T == CpuInfo))
{
final switch (reg)
{
case Reg.A: setA(cpu, val); break;
case Reg.X: setX(cpu, val); break;
case Reg.Y: setY(cpu, val); break;
}
}
// Returns a register
ubyte getReg(T)(ref T cpu, Reg reg)
if (isCpu!T || is(T == CpuInfo))
{
final switch (reg)
{
case Reg.A: return getA(cpu);
case Reg.X: return getX(cpu);
case Reg.Y: return getY(cpu);
}
}
// Sets the A register.
void setA(T)(T cpu, int val)
if (isCpu!T)
{
cpu.accumulator = cast(ubyte)val;
}
void setA(T : CpuInfo)(ref T cpu, int val)
{
cpu.A = cast(ubyte)val;
}
// Returns the A register.
ubyte getA(T)(T cpu, int val)
if (isCpu!T)
{
return cpu.accumulator;
}
ubyte getA(T : CpuInfo)(ref T cpu)
{
return cpu.A;
}
// Sets the X register. // Sets the X register.
void setX(T)(T cpu, int val) void setX(T)(T cpu, int val)
@ -268,6 +400,11 @@ if (isCpu!T)
cpu.xIndex = cast(ubyte)val; cpu.xIndex = cast(ubyte)val;
} }
void setX(T : CpuInfo)(ref T cpu, int val)
{
cpu.X = cast(ubyte)val;
}
// Returns the X register. // Returns the X register.
ubyte getX(T)(T cpu) ubyte getX(T)(T cpu)
if (isCpu!T) if (isCpu!T)
@ -275,6 +412,11 @@ if (isCpu!T)
return cpu.xIndex; return cpu.xIndex;
} }
ubyte getX(T : CpuInfo)(ref T cpu)
{
return cpu.X;
}
// Sets the Y register. // Sets the Y register.
void setY(T)(T cpu, int val) void setY(T)(T cpu, int val)
@ -283,6 +425,11 @@ if (isCpu!T)
cpu.yIndex = cast(ubyte)val; cpu.yIndex = cast(ubyte)val;
} }
void setY(T : CpuInfo)(ref T cpu, int val)
{
cpu.Y = cast(ubyte)val;
}
// Returns the Y register. // Returns the Y register.
ubyte getY(T)(T cpu) ubyte getY(T)(T cpu)
if (isCpu!T) if (isCpu!T)
@ -290,6 +437,11 @@ if (isCpu!T)
return cpu.yIndex; return cpu.yIndex;
} }
ubyte getY(T : CpuInfo)(ref T cpu)
{
return cpu.Y;
}
// The names of the status flags. // The names of the status flags.
enum Flag : ubyte enum Flag : ubyte
@ -302,6 +454,20 @@ enum Flag : ubyte
N = 0x80 N = 0x80
} }
string flagToString(Flag f)
{
switch (f)
{
case Flag.C: return "C";
case Flag.Z: return "Z";
case Flag.I: return "I";
case Flag.D: return "D";
case Flag.V: return "V";
case Flag.N: return "N";
default: return "?";
}
}
// Sets one or more status flags. // Sets one or more status flags.
void setFlag(T)(T cpu, Flag[] flags...) void setFlag(T)(T cpu, Flag[] flags...)
if (isCpu!T) if (isCpu!T)
@ -311,6 +477,11 @@ if (isCpu!T)
cpu.flag.fromByte(reg); cpu.flag.fromByte(reg);
} }
void setFlag(T : CpuInfo)(ref T cpu, Flag[] flags...)
{
foreach (flag; flags) cpu.S |= flag;
}
// Clears one or more status flags. // Clears one or more status flags.
void clearFlag(T)(T cpu, Flag[] flags...) void clearFlag(T)(T cpu, Flag[] flags...)
if (isCpu!T) if (isCpu!T)
@ -320,6 +491,11 @@ if (isCpu!T)
cpu.flag.fromByte(reg); cpu.flag.fromByte(reg);
} }
void clearFlag(T : CpuInfo)(ref T cpu, Flag[] flags...)
{
foreach (flag; flags) cpu.S &= ~flag;
}
// Returns a status flag. // Returns a status flag.
bool getFlag(T)(T cpu, Flag f) bool getFlag(T)(T cpu, Flag f)
if (isCpu!T) if (isCpu!T)
@ -327,9 +503,39 @@ if (isCpu!T)
return (cpu.flag.toByte() & f) != 0; return (cpu.flag.toByte() & f) != 0;
} }
// Sets or clears a single status flag. bool getFlag(T : CpuInfo)(ref T cpu, Flag f)
void updateFlag(T)(T cpu, Flag f, bool val) {
return (cpu.S & f) != 0;
}
// Sets the status register from a byte.
void setStatus(T)(T cpu, int val)
if (isCpu!T) if (isCpu!T)
{
cpu.flag.fromByte(cast(ubyte)val);
}
void setStatus(T : CpuInfo)(ref T cpu, int val)
{
cpu.S = cast(ubyte)val | 0x30;
}
// Returns the status register as a byte.
ubyte getStatus(T)(T cpu)
if (isCpu!T)
{
return cpu.flag.toByte();
}
ubyte getStatus(T : CpuInfo)(ref T cpu)
{
return cpu.S | 0x30;
}
// Sets or clears a single status flag.
void updateFlag(T)(ref T cpu, Flag f, bool val)
if (isCpu!T || is(T == CpuInfo))
{ {
if (val) if (val)
setFlag(cpu, f); setFlag(cpu, f);
@ -337,10 +543,15 @@ if (isCpu!T)
clearFlag(cpu, f); clearFlag(cpu, f);
} }
void setNZ(T : CpuInfo)(ref T cpu, ubyte val)
{
updateFlag(cpu, Flag.Z, (val == 0));
updateFlag(cpu, Flag.N, (val >= 0x80));
}
// Sets or clears the flag required for a given opcode to branch. // Sets or clears the flag required for a given opcode to branch.
void expectBranch(T)(T cpu, ubyte opcode) void expectBranch(T)(ref T cpu, ubyte opcode)
if (isCpu!T) if (isCpu!T || is(T == CpuInfo))
{ {
switch (opcode) switch (opcode)
{ {
@ -348,19 +559,19 @@ if (isCpu!T)
case /*BMI*/ 0x30: setFlag(cpu, Flag.N); break; case /*BMI*/ 0x30: setFlag(cpu, Flag.N); break;
case /*BVC*/ 0x50: clearFlag(cpu, Flag.V); break; case /*BVC*/ 0x50: clearFlag(cpu, Flag.V); break;
case /*BVS*/ 0x70: setFlag(cpu, Flag.V); break; case /*BVS*/ 0x70: setFlag(cpu, Flag.V); break;
case /*BRA*/ 0x80: break;
case /*BCC*/ 0x90: clearFlag(cpu, Flag.C); break; case /*BCC*/ 0x90: clearFlag(cpu, Flag.C); break;
case /*BCS*/ 0xB0: setFlag(cpu, Flag.C); break; case /*BCS*/ 0xB0: setFlag(cpu, Flag.C); break;
case /*BNE*/ 0xD0: clearFlag(cpu, Flag.Z); break; case /*BNE*/ 0xD0: clearFlag(cpu, Flag.Z); break;
case /*BEQ*/ 0xF0: setFlag(cpu, Flag.Z); break; case /*BEQ*/ 0xF0: setFlag(cpu, Flag.Z); break;
default: default:
if (isCMOS!T) { if (opcode == /*BRA*/ 0x80) break; }
enforce(0, format("not a branching opcpde %0.2X", opcode)); enforce(0, format("not a branching opcpde %0.2X", opcode));
} }
} }
// Returns whether an opcode would branch if executed. // Returns whether an opcode would branch if executed.
bool wouldBranch(T)(T cpu, ubyte opcode) bool wouldBranch(T)(ref T cpu, ubyte opcode)
if (isCpu!T) if (isCpu!T || is(T == CpuInfo))
{ {
switch (opcode) switch (opcode)
{ {
@ -368,18 +579,19 @@ if (isCpu!T)
case /*BMI*/ 0x30: return getFlag(cpu, Flag.N); case /*BMI*/ 0x30: return getFlag(cpu, Flag.N);
case /*BVC*/ 0x50: return !getFlag(cpu, Flag.V); case /*BVC*/ 0x50: return !getFlag(cpu, Flag.V);
case /*BVS*/ 0x70: return getFlag(cpu, Flag.V); case /*BVS*/ 0x70: return getFlag(cpu, Flag.V);
case /*BRA*/ 0x80: return true;
case /*BCC*/ 0x90: return !getFlag(cpu, Flag.C); case /*BCC*/ 0x90: return !getFlag(cpu, Flag.C);
case /*BCS*/ 0xB0: return getFlag(cpu, Flag.C); case /*BCS*/ 0xB0: return getFlag(cpu, Flag.C);
case /*BNE*/ 0xD0: return !getFlag(cpu, Flag.Z); case /*BNE*/ 0xD0: return !getFlag(cpu, Flag.Z);
case /*BEQ*/ 0xF0: return getFlag(cpu, Flag.Z); case /*BEQ*/ 0xF0: return getFlag(cpu, Flag.Z);
default: default:
if (isCMOS!T) { if (opcode == /*BRA*/ 0x80) return true; }
assert(0, format("not a branching opcpde %0.2X", opcode)); assert(0, format("not a branching opcpde %0.2X", opcode));
} }
} }
// Sets or clears the flag required for a given opcode to not branch. // Sets or clears the flag required for a given opcode to not branch.
void expectNoBranch(T)(T cpu, ubyte opcode) void expectNoBranch(T)(ref T cpu, ubyte opcode)
if (isCpu!T || is(T == CpuInfo))
{ {
switch (opcode) switch (opcode)
{ {
@ -406,6 +618,17 @@ ushort address(ubyte l, ubyte h)
} }
ubyte addrLo(ushort addr)
{
return cast(ubyte)(addr & 0xFF);
}
ubyte addrHi(ushort addr)
{
return cast(ubyte)(addr >> 8);
}
/* /*
* Adds an offset to an address, resulting in an address in the same * Adds an offset to an address, resulting in an address in the same
* page. * page.
@ -435,7 +658,7 @@ ubyte XX()
// A number different from some other number. // A number different from some other number.
ubyte notXX(ubyte val) ubyte notXX(ubyte val)
{ {
return cast(ubyte)(val ^ 0xAA); return cast(ubyte)~val;
} }

View File

@ -10,12 +10,12 @@ if (isCpu!T)
{ {
static if (isNMOS!T) static if (isNMOS!T)
enum REG_OPS = cast(ubyte[]) enum REG_OPS = cast(ubyte[])
x"0A 18 1A 2A 38 3A 4A 58 5A 6A 78 7A 8A 88 98 9A x"0A 18 1A 2A 38 3A 4A 58 5A 6A 78 7A 88 8A 98 9A
A8 AA B8 BA C8 CA D8 DA E8 EA F8 FA"; A8 AA B8 BA C8 CA D8 DA E8 EA F8 FA";
else else
enum REG_OPS = cast(ubyte[]) enum REG_OPS = cast(ubyte[])
x"0A 18 1A 2A 38 3A 4A 58 6A 78 8A 88 98 9A x"0A 18 1A 2A 38 3A 4A 58 6A 78 88 8A 98 9A A8 AA
A8 AA B8 BA C8 CA D8 E8 EA F8"; B8 BA C8 CA D8 E8 EA F8";
} }

View File

@ -77,14 +77,14 @@ if (isCpu!T)
auto opcode = mem[pc]; auto opcode = mem[pc];
assert(IMM_OPS!T.canFind(opcode)); assert(IMM_OPS!T.canFind(opcode));
bool decimal = isCMOS!T && isStrict!T && getFlag(cpu, Flag.D) && bool decimal = isCMOS!T && getFlag(cpu, Flag.D) &&
BCD_OPS!T.canFind(opcode); BCD_OPS!T.canFind(opcode);
cycles = 2 + decimal; cycles = 2 + decimal;
return [Bus(Action.READ, pc), return [Bus(Action.READ, pc),
Bus(Action.READ, pc+1)] ~ Bus(Action.READ, pc+1)] ~
If!decimal( If!decimal(If!(isStrict!T)(
[Bus(Action.READ, pc+2)]); [Bus(Action.READ, pc+2)]));
} }
@ -287,6 +287,7 @@ if (isCpu!T && isNMOS!T)
auto opcode = mem[pc]; auto opcode = mem[pc];
assert(HLT_OPS!T.canFind(opcode)); assert(HLT_OPS!T.canFind(opcode));
cycles = 1;
return [Bus(Action.READ, pc)]; return [Bus(Action.READ, pc)];
} }
@ -346,14 +347,14 @@ if (isCpu!T)
bool rmw = RMW_OPS!T.canFind(opcode); bool rmw = RMW_OPS!T.canFind(opcode);
bool write = !rmw && WRITE_OPS!T.canFind(opcode); bool write = !rmw && WRITE_OPS!T.canFind(opcode);
bool read = !rmw && !write; bool read = !rmw && !write;
bool decimal = isCMOS!T && isStrict!T && getFlag(cpu, Flag.D) && bool decimal = isCMOS!T && getFlag(cpu, Flag.D) &&
BCD_OPS!T.canFind(opcode); BCD_OPS!T.canFind(opcode);
cycles += (rmw ? 3 : (write ? 1 : (1 + decimal))); cycles += (rmw ? 3 : (write ? 1 : (1 + decimal)));
return If!read( return If!read(
[Bus(Action.READ, addr)] ~ [Bus(Action.READ, addr)] ~
If!decimal( If!decimal(If!(isStrict!T)(
[Bus(Action.READ, pc + opLen)])) ~ [Bus(Action.READ, pc + opLen)]))) ~
If!write( If!write(
[Bus(Action.WRITE, addr)]) ~ [Bus(Action.WRITE, addr)]) ~
If!rmw( If!rmw(
@ -612,93 +613,116 @@ template timesetup_t(T)
} }
// Tests the bus access patterns and cycles taken for a given opcode. alias void delegate(int, const Bus[], int, const Bus[], ubyte, string)
void test_opcode_timing(T)(ubyte opcode) busreport;
auto run_timing_test(T)(timesetup_t!T expect, busreport report)
{ {
addrsetup_t!T[] function(ubyte) setups1; auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
datasetup_t!T[] function(ubyte) setups2; string msg, TestSetup* next)
mixin(getMemSetup!T());
timesetup_t!T expected;
mixin(getExpected!T());
auto funcs1 = setups1(opcode);
string name1;
foreach(func1; funcs1)
{ {
ushort addr; mixin testCallNext;
int cycles; auto testcpu = makeCpu!T(cpu);
auto cpu = new T(); auto mem = TestMemory(data);
auto block1 = func1(cpu, addr, name1);
auto mem = TestMemory(block1); int expCycles;
connectMem(cpu, mem); auto expBus = expect(testcpu, mem, expCycles);
auto exp = expected(cpu, mem, cycles); expBus = expBus ~ new Bus[8 - expBus.length];
exp = exp ~ new Bus[8 - exp.length];
auto actual = recordBus(cpu); connectMem(testcpu, mem);
auto actualCycles = recordCycles(cpu); auto actualBus = recordBus(testcpu);
// XXX debug auto actualCycles = recordCycles(testcpu);
write(format("Testing %s (%0.2X) -- ", name1, opcode));
try runOneOpcode(testcpu);
{
runOneOpcode(cpu); report(actualCycles, actualBus, expCycles, expBus,
} opcode, T.stringof ~ " | " ~ msg);
catch (TestException e) // possibly not related to timing callNext();
{ }
// XXX wrap return TestSetup(&setup);
throw e; }
}
if (actual != exp)
auto report_timing_debug()
{
void report(int actualCycles, const Bus[] actualBus,
int expectCycles, const Bus[] expectBus,
ubyte opcode, string msg)
{
if (actualBus != expectBus)
{ {
// XXX make error message, throw // XXX make error message, throw
} }
if (actualCycles != cycles) if (actualCycles != expectCycles)
{ {
// XXX make error message, throw // XXX make error message, throw
} }
if (actual == exp && actualCycles == cycles) if (actualBus == expectBus && actualCycles == expectCycles) {}
writeln("OK");
else else
{ {
write(format("[%0.2X] %s", opcode, msg));
writeln(); writeln();
writeln(actualCycles, " ", cycles); writeln(expectCycles, " ", actualCycles);
writeln(actual); writeln(expectBus);
writeln(exp); writeln(actualBus);
throw new TestException("timing"); throw new TestException("timing");
} }
} }
return &report;
}
// Tests the bus access patterns and cycles taken for a given opcode.
void test_opcode_timing(T)(ubyte opcode, busreport report)
{
TestSetup setup_addr;
TestSetup setup_test;
testexpect expect;
timesetup_t!T expected;
mixin(getMemSetup!T());
mixin(getExpected!T());
auto setup = connect(setup_mask_flags(), setup_addr, setup_test);
auto run = connect(setup, run_timing_test!T(expected, report));
run.run(opcode);
} }
unittest unittest
{ {
auto report = report_timing_debug();
alias CPU!("65C02", false, false) T1; alias CPU!("65C02", false, false) T1;
for (int op = 0x00; op < 0x100; op++) for (int op = 0x00; op < 0x100; op++)
test_opcode_timing!T1(cast(ubyte)op); test_opcode_timing!T1(cast(ubyte)op, report);
alias CPU!("65C02", true, false) T2; alias CPU!("65C02", true, false) T2;
for (int op = 0x00; op < 0x100; op++) for (int op = 0x00; op < 0x100; op++)
test_opcode_timing!T2(cast(ubyte)op); test_opcode_timing!T2(cast(ubyte)op, report);
alias CPU!("6502", false, false) T3; alias CPU!("6502", false, false) T3;
for (int op = 0x00; op < 0x100; op++) for (int op = 0x00; op < 0x100; op++)
test_opcode_timing!T3(cast(ubyte)op); test_opcode_timing!T3(cast(ubyte)op, report);
alias CPU!("6502", true, false) T4; alias CPU!("6502", true, false) T4;
for (int op = 0x00; op < 0x100; op++) for (int op = 0x00; op < 0x100; op++)
test_opcode_timing!T4(cast(ubyte)op); test_opcode_timing!T4(cast(ubyte)op, report);
alias CPU!("65C02", false, true) T5; alias CPU!("65C02", false, true) T5;
for (int op = 0x00; op < 0x100; op++) for (int op = 0x00; op < 0x100; op++)
test_opcode_timing!T5(cast(ubyte)op); test_opcode_timing!T5(cast(ubyte)op, report);
alias CPU!("65C02", true, true) T6; alias CPU!("65C02", true, true) T6;
for (int op = 0x00; op < 0x100; op++) for (int op = 0x00; op < 0x100; op++)
test_opcode_timing!T6(cast(ubyte)op); test_opcode_timing!T6(cast(ubyte)op, report);
alias CPU!("6502", false, true) T7; alias CPU!("6502", false, true) T7;
for (int op = 0x00; op < 0x100; op++) for (int op = 0x00; op < 0x100; op++)
test_opcode_timing!T7(cast(ubyte)op); test_opcode_timing!T7(cast(ubyte)op, report);
alias CPU!("6502", true, true) T8; alias CPU!("6502", true, true) T8;
for (int op = 0x00; op < 0x100; op++) for (int op = 0x00; op < 0x100; op++)
test_opcode_timing!T8(cast(ubyte)op); test_opcode_timing!T8(cast(ubyte)op, report);
} }

104
test/test_func.d Normal file
View File

@ -0,0 +1,104 @@
module test.test_func;
import std.string;
import test.base, test.cpu, test.opcodes;
alias void delegate(ubyte, ref Expected, CpuInfo, const ref TestMemory, string)
testreport;
/*
* Runs one opcode. Calls expect for the expected values of the cpu
* registers and memory. Calls report with the expected and actual
* values.
*/
auto run_opcode_test(T)(testexpect expect, testreport report)
{
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
string msg, TestSetup* next)
{
mixin testCallNext;
auto testcpu = makeCpu!T(cpu);
auto mem = TestMemory(data);
auto expected = Expected(cpu, mem);
expect(expected, info);
connectMem(testcpu, mem);
runOneOpcode(testcpu);
auto cpuResult = CpuInfo.fromCpu(testcpu);
report(opcode, expected, cpuResult, mem, T.stringof ~ " | " ~ msg);
callNext();
}
return TestSetup(&setup);
}
// Dummy function. Reports nothing.
auto report_none()
{
void report(ubyte opcode, ref Expected expected, CpuInfo cpu,
const ref TestMemory mem, string msg)
{
}
return &report;
}
// Prints the differences between expected and actual cpu/memory.
auto report_debug()
{
void report(ubyte opcode, ref Expected expected, CpuInfo cpu,
const ref TestMemory mem, string msg)
{
import std.stdio;
bool badCpu = (expected.cpu != cpu);
bool badMem = (expected.mem != mem);
if (badCpu || badMem)
writeln(format("[%0.2X] %s", opcode, msg));
if (badCpu)
{
writeln(" expect ", expected.cpu);
writeln(" actual ", cpu);
}
if (badMem)
{
foreach (h; MemDiff(expected.mem, mem))
{
writeln(format(" %0.4X | %s", h.base, formatMemory(h.a, 8)));
writeln(format(" | %s", formatMemory(h.b, 8)));
}
}
if (badCpu || badMem) throw new Exception("BAD");
}
return &report;
}
void test_one_opcode(T)(ubyte opcode, testreport report)
{
TestSetup setup_addr;
TestSetup setup_test;
testexpect expect;
mixin(getMemSetup!T());
auto setup = connect(setup_mask_flags(), setup_addr, setup_test);
auto run = connect(setup, run_opcode_test!T(expect, report));
run.run(opcode);
}
unittest
{
auto report = report_debug();
alias CPU!("65C02", false, false) T1;
foreach (opcode; 0..255)
test_one_opcode!T1(cast(ubyte)opcode, report);
alias CPU!("6502", false, false) T2;
foreach (opcode; 0..255)
test_one_opcode!T2(cast(ubyte)opcode, report);
}