mirror of
https://github.com/edmccard/twoapple-reboot.git
synced 2025-01-28 10:31:37 +00:00
3337 lines
83 KiB
D
3337 lines
83 KiB
D
module test.base;
|
|
|
|
|
|
import std.algorithm, std.array, std.conv, std.exception, std.stdio,
|
|
std.string;
|
|
|
|
import test.cpu, test.opcodes;
|
|
|
|
|
|
/*
|
|
* Emulates zero page, stack, and 3 additional pages of "main memory"
|
|
* starting at a user-defined address. Accesses outside the defined
|
|
* address space raise an exception.
|
|
*
|
|
* The contents are initialized to 0xFF. Individual locations can be
|
|
* read and written using array index syntax.
|
|
*/
|
|
struct TestMemory
|
|
{
|
|
private:
|
|
ubyte[0x200] data1 = 0xFF;
|
|
ubyte[0x300] data2 = 0xFF;
|
|
ushort data2_base;
|
|
size_t data2_max;
|
|
|
|
public:
|
|
/*
|
|
* Constructs a TestMemory with data filled in from an array of
|
|
* Blocks.
|
|
*
|
|
* The blocks do not need to be contiguous, or ordered by their
|
|
* base address, but note that the base of the 3-page "main
|
|
* memory" will be the start of the page that contains the first
|
|
* block with a base address greater than 0x01FF (there must be at
|
|
* least one such block).
|
|
*/
|
|
this(const(Block[]) blocks ...)
|
|
{
|
|
foreach (block; blocks)
|
|
{
|
|
auto base = block.base;
|
|
auto data = block.data;
|
|
if (base < 0x200)
|
|
{
|
|
enforce(base + data.length <= 0x200,
|
|
format("Address out of bounds %0.4x", base));
|
|
data1[base..base + data.length] = data[];
|
|
}
|
|
else
|
|
{
|
|
if (!data2_base)
|
|
{
|
|
if (base > 0xFD00)
|
|
data2_base = 0xFD00;
|
|
else
|
|
data2_base = base & 0xFF00;
|
|
data2_max = data2_base + 0x300;
|
|
}
|
|
enforce(base + data.length <= data2_max,
|
|
format("Address out of bounds %0.4x", base));
|
|
auto last = base + data.length;
|
|
data2[base-data2_base..last-data2_base] = data[];
|
|
}
|
|
}
|
|
enforce(data2_base, "Missing memory > 0x0200");
|
|
}
|
|
|
|
ubyte read(ushort addr) const
|
|
{
|
|
if (addr < 0x0200)
|
|
return data1[addr];
|
|
else if (addr >= data2_base && addr < data2_max)
|
|
return data2[addr - data2_base];
|
|
else
|
|
throw new TestException(format("read %0.4x", addr));
|
|
}
|
|
|
|
void write(ushort addr, ubyte val)
|
|
{
|
|
if (addr < 0x0200)
|
|
data1[addr] = val;
|
|
else if (addr >= data2_base && addr < data2_max)
|
|
data2[addr - data2_base] = val;
|
|
else
|
|
throw new TestException(format("write %0.4x", addr));
|
|
}
|
|
|
|
ubyte opIndex(size_t i1) const
|
|
{
|
|
auto addr = cast(ushort)i1;
|
|
enforce(addr < 0x0200 || (addr >= data2_base && addr < data2_max),
|
|
"Read out of bounds");
|
|
return read(addr);
|
|
}
|
|
|
|
ubyte opIndexAssign(ubyte val, size_t i1)
|
|
{
|
|
auto addr = cast(ushort)i1;
|
|
enforce(addr < 0x0200 || (addr >= data2_base && addr < data2_max),
|
|
"Write out of bounds");
|
|
write(addr, val);
|
|
return val;
|
|
}
|
|
}
|
|
|
|
|
|
// A block of memory with a given base address.
|
|
struct Block
|
|
{
|
|
const ushort base;
|
|
const(ubyte[]) data;
|
|
|
|
string toString() const
|
|
{
|
|
return format("Block(%0.4X, [%s])", base, formatMemory(data));
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Formats data as a string of 2-digit hex bytes, separated by spaces.
|
|
*
|
|
* If data is longer than max, the string will end with an indication
|
|
* of the number of extra bytes.
|
|
*/
|
|
string formatMemory(const(ubyte[]) data, size_t max = 3)
|
|
{
|
|
if (max > data.length) max = data.length;
|
|
auto hexbytes = map!(`format("%0.2X", a)`)(data[0..max]);
|
|
auto ret = join(array(hexbytes), " ");
|
|
if (data.length > max)
|
|
ret ~= format(" (%d more bytes)", data.length - max);
|
|
return ret;
|
|
}
|
|
|
|
|
|
struct OpInfo
|
|
{
|
|
ushort addr;
|
|
ubyte data;
|
|
int len;
|
|
bool write;
|
|
}
|
|
|
|
alias void delegate(ubyte, CpuInfo, Block[], OpInfo, string, TestSetup*)
|
|
testsetup;
|
|
|
|
class TestSetup
|
|
{
|
|
testsetup setup;
|
|
TestSetup next;
|
|
|
|
auto static opCall(testsetup d)
|
|
{
|
|
auto obj = new TestSetup();
|
|
obj.setup = d;
|
|
return obj;
|
|
}
|
|
|
|
void run(ubyte opcode, CpuInfo cpu = CpuInfo(), Block[] data = [],
|
|
OpInfo info = OpInfo(), string msg = "")
|
|
{
|
|
setup(opcode, cpu, data, info, msg, &next);
|
|
}
|
|
}
|
|
|
|
TestSetup connect(TestSetup first, TestSetup[] rest...)
|
|
{
|
|
if (!(rest.empty))
|
|
{
|
|
auto x = first;
|
|
while (x.next !is null) x = x.next;
|
|
x.next = connect(rest[0], rest[1..$]);
|
|
}
|
|
return first;
|
|
}
|
|
|
|
template testCallNext()
|
|
{
|
|
void callNext(string newMsg = "", Block[] newData = [])
|
|
{
|
|
if (*next !is null)
|
|
next.run(opcode, cpu, data ~ newData, info, msg ~ newMsg);
|
|
}
|
|
}
|
|
|
|
|
|
// Does nothing.
|
|
auto setup_none()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
if (*next !is null) next.run(opcode, cpu, data, info, msg);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// Aborts a test.
|
|
auto setup_op_abort()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// Prints the current message.
|
|
auto setup_debug()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
import std.stdio;
|
|
writeln(format("%s\n data %s\n addr %0.4X op %0.2X %s",
|
|
msg, data, info.addr, opcode, cpu));
|
|
if (*next !is null) next.run(opcode, cpu, data, info, msg);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// Splits with the given flag set, then cleared.
|
|
auto setup_flag(Flag f)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
clearFlag(cpu, f);
|
|
callNext(format("%s0 ", flagToString(f)));
|
|
|
|
setFlag(cpu, f);
|
|
callNext(format("%s1 ", flagToString(f)));
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// Splits with all flags set, then all flags cleared.
|
|
auto setup_mask_flags()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
clearFlag(cpu, Flag.C, Flag.Z, Flag.I, Flag.D, Flag.V, Flag.N);
|
|
callNext("F0 ");
|
|
|
|
setFlag(cpu, Flag.C, Flag.Z, Flag.I, Flag.D, Flag.V, Flag.N);
|
|
callNext("F1 ");
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
/*
|
|
* Used after setup_mask_flags, guarantees coverage of all
|
|
* combinations of N and Z.
|
|
*/
|
|
auto setup_nz()
|
|
{
|
|
return setup_flag(Flag.N);
|
|
}
|
|
|
|
|
|
/*
|
|
* Can be used after setup_mask_flags to cover all combinations of
|
|
* flags.
|
|
*/
|
|
auto setup_nvdzc()
|
|
{
|
|
return connect(setup_flag(Flag.N), setup_flag(Flag.Z),
|
|
setup_flag(Flag.V), setup_flag(Flag.D),
|
|
setup_flag(Flag.C));
|
|
}
|
|
|
|
|
|
// Splits with info.data 0x00, 0x40, and 0x80.
|
|
auto setup_data()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.data = 0x00;
|
|
callNext("zero ");
|
|
|
|
info.data = 0x40;
|
|
callNext("positive ");
|
|
|
|
info.data = 0x80;
|
|
callNext("negative ");
|
|
}
|
|
return connect(TestSetup(&setup));
|
|
}
|
|
|
|
|
|
/*
|
|
* For register transfer opcodes.
|
|
*
|
|
* Splits on 0x00, 0x40, and 0x80 in the source register.
|
|
*/
|
|
auto setup_op_transfer(Reg source, Reg dest)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
ubyte other = (info.data > 0x00 && info.data < 0x80) ? 0x00 : 0x01;
|
|
setX(cpu, other); setY(cpu, other); setA(cpu, other);
|
|
setReg(cpu, source, info.data);
|
|
callNext("xfer ");
|
|
}
|
|
return connect(setup_nz(), setup_data(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For TXS and TSX.
|
|
auto setup_op_stack_xfer()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setSP(cpu, 0x00);
|
|
setX(cpu, 0xFF);
|
|
callNext("TXS ");
|
|
}
|
|
return connect(setup_nz(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
/*
|
|
* For implied address mode.
|
|
*
|
|
* Sets PC to 0x1000 and puts a (1-byte) opcode in memory.
|
|
*/
|
|
auto setup_addr_implied()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setPC(cpu, 0x1000);
|
|
info.len = 1;
|
|
callNext("", [Block(0x1000, [opcode])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
/*
|
|
* For immediate address mode.
|
|
*
|
|
* Sets PC to 0x1000 and info.addr to 0x1001.
|
|
*/
|
|
auto setup_addr_imm()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setPC(cpu, 0x1000);
|
|
info.len = 2;
|
|
info.addr = 0x1001;
|
|
callNext("imm ", [Block(0x1000, [opcode])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
/*
|
|
* For branch opcodes.
|
|
*/
|
|
auto setup_addr_branch(bool cmos)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
if (!cmos || opcode != 0x80)
|
|
{
|
|
expectNoBranch(cpu, opcode);
|
|
setPC(cpu, 0x1000);
|
|
info.addr = 0x1002;
|
|
callNext("no-branch ", [Block(0x1000, [opcode, 0x10])]);
|
|
}
|
|
|
|
expectBranch(cpu, opcode);
|
|
|
|
setPC(cpu, 0x1000);
|
|
info.addr = 0x1012;
|
|
callNext("branch-fwd ", [Block(0x1000, [opcode, 0x10])]);
|
|
|
|
setPC(cpu, 0x1081);
|
|
info.addr = 0x1102;
|
|
callNext("branch-fwd-px ", [Block(0x1081, [opcode, 0x7F])]);
|
|
|
|
setPC(cpu, 0x1000);
|
|
info.addr = 0x1000;
|
|
callNext("branch-bkwd ", [Block(0x1000, [opcode, 0xFE])]);
|
|
|
|
setPC(cpu, 0x1100);
|
|
info.addr = 0x10F7;
|
|
callNext("branch-bkwd-px ", [Block(0x1000, []),
|
|
Block(0x1100, [opcode, 0xF5])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
/*
|
|
* For zeropage adddress mode.
|
|
*
|
|
* Sets PC to 0x1000 and puts [opcode, 0x70] in memory.
|
|
*/
|
|
auto setup_addr_zpg()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setPC(cpu, 0x1000);
|
|
info.len = 2;
|
|
info.addr = 0x0070;
|
|
callNext("zpg ", [Block(0x1000, [opcode, 0x70])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
/*
|
|
* For absolute address mode.
|
|
*
|
|
* Sets PC to 0x1000 and puts [opcode, 0xC5, 0x10] in memory.
|
|
*/
|
|
auto setup_addr_abs()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setPC(cpu, 0x1000);
|
|
info.len = 3;
|
|
info.addr = 0x10C5;
|
|
callNext("abs ", [Block(0x1000, [opcode, 0xC5, 0x10])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
/*
|
|
* For zeropage,x/y address modes.
|
|
*
|
|
* Sets PC to 0x1000 and puts [opcode, 0x70] in memory.
|
|
*/
|
|
auto setup_addr_zpxy(Reg reg)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
string name = (reg == Reg.X ? "zpx " : "zpy ");
|
|
info.len = 2;
|
|
setPC(cpu, 0x1000);
|
|
|
|
setX(cpu, 0x10); setY(cpu, 0x10); setA(cpu, 0x10);
|
|
setReg(cpu, reg, 0x00);
|
|
info.addr = pageWrapAdd(0x0070, 0x00);
|
|
callNext(name ~ "no-idx ", [Block(0x1000, [opcode, 0x70])]);
|
|
|
|
setX(cpu, 0x10); setY(cpu, 0x10); setA(cpu, 0x10);
|
|
setReg(cpu, reg, 0x20);
|
|
info.addr = pageWrapAdd(0x0070, 0x20);
|
|
callNext(name ~ "no-wrap ", [Block(0x1000, [opcode, 0x70])]);
|
|
|
|
setX(cpu, 0x10); setY(cpu, 0x10); setA(cpu, 0x10);
|
|
setReg(cpu, reg, 0xA0);
|
|
info.addr = pageWrapAdd(0x0070, 0xA0);
|
|
callNext(name ~ "wrap ", [Block(0x1000, [opcode, 0x70])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For absolute,x/y address modes.
|
|
auto setup_addr_abxy(Reg reg)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
string name = (reg == Reg.X ? "abx " : "aby ");
|
|
info.len = 3;
|
|
setPC(cpu, 0x1000);
|
|
|
|
setX(cpu, 0x10); setY(cpu, 0x10); setA(cpu, 0x10);
|
|
setReg(cpu, reg, 0x00);
|
|
info.addr = pageCrossAdd(0x10C5, 0x00);
|
|
callNext(name ~ "no-idx ", [Block(0x1000, [opcode, 0xC5, 0x10])]);
|
|
|
|
setX(cpu, 0x10); setY(cpu, 0x10); setA(cpu, 0x10);
|
|
setReg(cpu, reg, 0x20);
|
|
info.addr = pageCrossAdd(0x10C5, 0x20);
|
|
callNext(name ~ "no-px ", [Block(0x1000, [opcode, 0xC5, 0x10])]);
|
|
|
|
setX(cpu, 0x10); setY(cpu, 0x10); setA(cpu, 0x10);
|
|
setReg(cpu, reg, 0x50);
|
|
info.addr = pageCrossAdd(0x10C5, 0x50);
|
|
callNext(name ~ "px ", [Block(0x1000, [opcode, 0xC5, 0x10])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For zeropage indirect,x address mode.
|
|
auto setup_addr_izx()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.addr = 0x10C5;
|
|
info.len = 2;
|
|
setPC(cpu, 0x1000);
|
|
setA(cpu, 0x01); setY(cpu, 0x01);
|
|
|
|
setX(cpu, 0x20);
|
|
callNext("izx no-wrap ", [Block(0x1000, [opcode, 0x70]),
|
|
Block(0x0090, [0xC5, 0x10])]);
|
|
|
|
setX(cpu, 0xA0);
|
|
callNext("izx wrap ", [Block(0x1000, [opcode, 0x70]),
|
|
Block(0x0010, [0xC5, 0x10])]);
|
|
|
|
setX(cpu, 0x8F);
|
|
callNext("izx px ", [Block(0x1000, [opcode, 0x70]),
|
|
Block(0x00FF, [0xC5]),
|
|
Block(0x0000, [0x10])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For zeropage indirect,y address mode.
|
|
auto setup_addr_izy()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.len = 2;
|
|
setPC(cpu, 0x1000);
|
|
setA(cpu, 0x01); setX(cpu, 0x01);
|
|
|
|
info.addr = pageCrossAdd(0x10C5, 0x20);
|
|
setY(cpu, 0x20);
|
|
|
|
callNext("izy no-wrap ", [Block(0x1000, [opcode, 0x70]),
|
|
Block(0x0070, [0xC5, 0x10])]);
|
|
|
|
callNext("izy wrap ", [Block(0x1000, [opcode, 0xFF]),
|
|
Block(0x00FF, [0xC5]),
|
|
Block(0x0000, [0x10])]);
|
|
|
|
info.addr = pageCrossAdd(0x10C5, 0x50);
|
|
setY(cpu, 0x50);
|
|
callNext("izy px ", [Block(0x1000, [opcode, 0x70]),
|
|
Block(0x0070, [0xC5, 0x10])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For zeropage indirect address mode.
|
|
auto setup_addr_zpi()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.len = 2;
|
|
info.addr = 0x10C5;
|
|
setPC(cpu, 0x1000);
|
|
|
|
callNext("zpi no-wrap ", [Block(0x1000, [opcode, 0x70]),
|
|
Block(0x0070, [0xC5, 0x10])]);
|
|
|
|
callNext("zpi wrap ", [Block(0x1000, [opcode, 0xFF]),
|
|
Block(0x00FF, [0xC5]),
|
|
Block(0x0000, [0x10])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
/*
|
|
* Splits with the SP at 0xFF, and then at 0x00 (to cover the case of
|
|
* pull operations when the stack is "empty").
|
|
*/
|
|
auto setup_pull_wrap()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setSP(cpu, 0xFF);
|
|
callNext("wrap ");
|
|
|
|
setSP(cpu, 0x00);
|
|
callNext("no-wrap ");
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
/*
|
|
* For pull opcodes.
|
|
*
|
|
* Splits with 0x00, 0x40, and 0x80 at the top of the stack (to
|
|
* exercise different potential combinations of N and Z).
|
|
*/
|
|
auto setup_op_pull(Reg reg)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
auto sp = pageWrapAdd(getSP(cpu), 1);
|
|
ubyte other = (info.data > 0x00 && info.data < 0x80) ? 0x00 : 0x01;
|
|
setX(cpu, other); setY(cpu, other); setA(cpu, other);
|
|
callNext("pull ", [Block(sp, [info.data])]);
|
|
}
|
|
return connect(setup_pull_wrap(), setup_nz(), setup_data(),
|
|
TestSetup(&setup));
|
|
}
|
|
|
|
|
|
/*
|
|
* Splits with each possible meaningful status byte value on the
|
|
* stack, and with values with break/reserved bits set.
|
|
*/
|
|
auto setup_status()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
for (int i = 0; i < 4; i++)
|
|
{
|
|
for (int j = 0; j < 16; j++)
|
|
{
|
|
info.data = cast(ubyte)(i * 0x40 + j);
|
|
callNext(format("S %0.2X ", info.data | 0x30));
|
|
}
|
|
}
|
|
|
|
string[] msgs = ["B ", "R ", "BR "];
|
|
foreach (val; [0x10, 0x20, 0x30])
|
|
{
|
|
info.data = cast(ubyte)val;
|
|
callNext("S " ~ msgs[(info.data >> 4) - 1]);
|
|
}
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For PLP.
|
|
auto setup_op_PLP()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
auto sp = pageWrapAdd(getSP(cpu), 1);
|
|
setStatus(cpu, ~info.data);
|
|
callNext("PLP ", [Block(sp, [info.data])]);
|
|
}
|
|
return connect(setup_pull_wrap(), setup_status(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
/*
|
|
* Splits with the SP at 0x00, and then at 0xFF (to cover the case of
|
|
* push operations when the stack is "full").
|
|
*/
|
|
auto setup_push_wrap()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setSP(cpu, 0x00);
|
|
callNext("wrap ");
|
|
|
|
setSP(cpu, 0xFF);
|
|
callNext("no-wrap ");
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
/*
|
|
* For push opcodes.
|
|
*
|
|
* Splits with 0x00, 0x40, and 0x80 in the appropriate register.
|
|
*/
|
|
auto setup_op_push(Reg reg)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setX(cpu, 0x01); setY(cpu, 0x01); setA(cpu, 0x01);
|
|
setReg(cpu, reg, info.data);
|
|
callNext("push ", [Block(getSP(cpu), [~info.data])]);
|
|
}
|
|
return connect(setup_push_wrap(), setup_data(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
/*
|
|
* For PHP.
|
|
*
|
|
* Puts a garbage value (that does not correspond to the current
|
|
* status) just below the top of the stack.
|
|
*/
|
|
auto setup_op_PHP()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.data = getStatus(cpu);
|
|
callNext("PHP ", [Block(getSP(cpu), [~info.data])]);
|
|
}
|
|
return connect(setup_nvdzc(), setup_push_wrap(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
/*
|
|
* For load opcodes.
|
|
*
|
|
* Splits with 0x00, 0x40, and 0x80 in the source memory location.
|
|
*/
|
|
auto setup_op_load(Reg reg)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
callNext("load ", [Block(info.addr, [info.data])]);
|
|
}
|
|
return connect(setup_nz(), setup_data(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// Splits with 0x00, 0x40, 0x80 in the source register.
|
|
auto setup_op_store(Reg reg)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setReg(cpu, reg, info.data);
|
|
callNext("store ", [Block(info.addr, [~info.data])]);
|
|
}
|
|
return connect(setup_data(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For SAX.
|
|
auto setup_op_SAX()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
if (opcode == 0x83)
|
|
{
|
|
setA(cpu, 0x00);
|
|
callNext("zero SAX ");
|
|
|
|
setA(cpu, 0x2B);
|
|
callNext("positive SAX ");
|
|
|
|
setA(cpu, 0x8B);
|
|
callNext("negative SAX ");
|
|
}
|
|
else
|
|
{
|
|
setA(cpu, 0x00); setX(cpu, 0x00);
|
|
callNext("zero SAX ");
|
|
|
|
setA(cpu, 0x10); setX(cpu, 0x01);
|
|
callNext("zero SAX ");
|
|
|
|
setA(cpu, 0x0B); setX(cpu, 0x0D);
|
|
callNext("positive SAX ");
|
|
|
|
setA(cpu, 0x8B); setX(cpu, 0x8D);
|
|
callNext("negative SAX ");
|
|
}
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For STZ.
|
|
auto setup_op_STZ()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.data = 0;
|
|
callNext("STZ ", [Block(info.addr, [~info.data])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// setup_bit_val, puts in memory;
|
|
auto setup_bit_data()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.data = 0x80;
|
|
callNext("N1 ", [Block(info.addr, [info.data])]);
|
|
|
|
info.data = 0x40;
|
|
callNext("V1 ", [Block(info.addr, [info.data])]);
|
|
|
|
info.data = 0xC0;
|
|
callNext("N1V1 ", [Block(info.addr, [info.data])]);
|
|
|
|
info.data = 0x20;
|
|
callNext("N0V0 ", [Block(info.addr, [info.data])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For BIT.
|
|
auto setup_op_BIT()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, info.data);
|
|
callNext("BIT non-zero ");
|
|
|
|
setA(cpu, ~info.data);
|
|
callNext("BIT zero");
|
|
}
|
|
return connect(setup_nz(), setup_flag(Flag.V), setup_bit_data(),
|
|
TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For TRB.
|
|
auto setup_op_TRB()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0); setX(cpu, 0xFF); setY(cpu, 0xFF);
|
|
info.data = 0x01;
|
|
callNext("TRB Z1 1 ", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x40); setX(cpu, 0x00); setY(cpu, 0x00);
|
|
info.data = 0x40;
|
|
callNext("TRB Z0 0 ", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x40); setX(cpu, 0x00); setY(cpu, 0x00);
|
|
info.data = 0x41;
|
|
callNext("TRB Z0 1 ", [Block(info.addr, [info.data])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For TSB.
|
|
auto setup_op_TSB()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x01); setX(cpu, 0x00); setY(cpu, 0x00);
|
|
info.data = 0x00;
|
|
callNext("TSB Z1 1 ", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x02); setX(cpu, 0x00); setY(cpu, 0x00);
|
|
info.data = 0x02;
|
|
callNext("TSB Z0 2 ", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x30); setX(cpu, 0x00); setY(cpu, 0x00);
|
|
info.data = 0x80;
|
|
callNext("TSB Z1 B0 ", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x00); setX(cpu, 0xFF); setY(cpu, 0xFF);
|
|
info.data = 0x02;
|
|
callNext("TSB Z1 2 ", [Block(info.addr, [info.data])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For BRK.
|
|
auto setup_op_BRK()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.addr = 0xFE55;
|
|
setPC(cpu, 0xFD00);
|
|
ushort ret = 0xFD02;
|
|
auto sp = getSP(cpu);
|
|
auto sp1 = pageWrapAdd(sp, -1);
|
|
auto sp2 = pageWrapAdd(sp, -2);
|
|
info.data = getStatus(cpu);
|
|
callNext("BRK ", [Block(0xFD00, [opcode]),
|
|
Block(sp, [~addrHi(ret)]),
|
|
Block(sp1, [~addrLo(ret)]),
|
|
Block(sp2, [~info.data]),
|
|
Block(0xFFFE, [0x55, 0xFE])]);
|
|
|
|
}
|
|
return connect(setup_nvdzc(), setup_push_wrap(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
auto setup_RTx(bool isRTI)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
string name = (isRTI ? "RTI " : "RTS ");
|
|
info.addr = 0x1211;
|
|
setPC(cpu, 0x1000);
|
|
auto sp = getSP(cpu);
|
|
auto sp1 = pageWrapAdd(sp, (isRTI ? 1 : 0));
|
|
auto sp2 = pageWrapAdd(sp, (isRTI ? 2 : 1));
|
|
auto sp3 = pageWrapAdd(sp, (isRTI ? 3 : 2));
|
|
callNext(name, [Block(0x1000, [opcode])] ~
|
|
(isRTI ? [Block(sp1, [info.data])] : []) ~
|
|
[Block(sp2, [addrLo(info.addr)]),
|
|
Block(sp3, [addrHi(info.addr)])]);
|
|
}
|
|
return connect(setup_pull_wrap(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For RTI.
|
|
auto setup_op_RTI()
|
|
{
|
|
return connect(setup_status(), setup_RTx(true));
|
|
}
|
|
|
|
|
|
// For RTS.
|
|
auto setup_op_RTS()
|
|
{
|
|
return setup_RTx(false);
|
|
}
|
|
|
|
|
|
// For JMP inx.
|
|
auto setup_op_JMP_inx()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.addr = 0x1234;
|
|
info.len = 3;
|
|
setPC(cpu, 0x1000);
|
|
|
|
setX(cpu, 0x20);
|
|
callNext("JMP inx A ", [Block(0x1000, [opcode, 0xC5, 0x10]),
|
|
Block(0x10E5, [0x34, 0x12])]);
|
|
|
|
setX(cpu, 0x40);
|
|
callNext("JMP inx B ", [Block(0x1000, [opcode, 0xC5, 0x10]),
|
|
Block(0x1105, [0x34, 0x12])]);
|
|
|
|
setX(cpu, 0x30);
|
|
callNext("JMP inx C ", [Block(0x1000, [opcode, 0xCF, 0x10]),
|
|
Block(0x10FF, [0x34, 0x12])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For JMP ind.
|
|
auto setup_op_JMP_ind(bool cmos)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.addr = 0x1234;
|
|
info.len = 3;
|
|
setPC(cpu, 0x1000);
|
|
setX(cpu, 0x40); setY(cpu, 0x40); setA(cpu, 0x40);
|
|
|
|
callNext("JMP ind px ", [Block(0x1000, [opcode, 0xFF, 0x11])] ~
|
|
(cmos ? [Block(0x11FF, [0x34, 0x12])]
|
|
: [Block(0x11FF, [0x34]),
|
|
Block(0x1100, [0x12])]));
|
|
|
|
callNext("JMP ind no px ", [Block(0x1000, [opcode, 0xC5, 0x10]),
|
|
Block(0x10C5, [0x34, 0x12])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For JSR.
|
|
auto setup_op_JSR()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.addr = 0x10C5;
|
|
setPC(cpu, 0x1000);
|
|
ushort ret = 0x1002;
|
|
auto sp = getSP(cpu);
|
|
auto sp1 = pageWrapAdd(sp, 1);
|
|
callNext("JSR ", [Block(0x1000, [opcode, 0xC5, 0x10]),
|
|
Block(sp, [~addrHi(ret)]),
|
|
Block(sp, [~addrLo(ret)])]);
|
|
}
|
|
return connect(setup_push_wrap(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For CMOS opcode 5C.
|
|
auto setup_op_5C()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.len = 3;
|
|
setPC(cpu, 0xFD00);
|
|
callNext("NOP8 ", [Block(0xFD00, [0x5C, 0x72])]);
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
auto setup_dec_data()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.data = 0x01;
|
|
callNext("zero ");
|
|
|
|
info.data = 0x00;
|
|
callNext("negative ");
|
|
|
|
info.data = 0x80;
|
|
callNext("positive ");
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For DEX, DEY, DEA.
|
|
auto setup_op_DEC_reg(Reg reg)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setX(cpu, 0x40); setY(cpu, 0x40); setA(cpu, 0x40);
|
|
setReg(cpu, reg, info.data);
|
|
callNext("DEC r ", [Block(0x1000, [opcode])]);
|
|
}
|
|
return connect(setup_nz(), setup_dec_data(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For DEC.
|
|
auto setup_op_DEC()
|
|
{
|
|
return setup_rmw(false, "DEC ", setup_dec_data());
|
|
}
|
|
|
|
|
|
auto setup_inc_data()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.data = 0xFF;
|
|
callNext("zero ");
|
|
|
|
info.data = 0x7F;
|
|
callNext("negative ");
|
|
|
|
info.data = 0x00;
|
|
callNext("positive ");
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For INX, INY, INA
|
|
auto setup_op_INC_reg(Reg reg)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setX(cpu, 0x40); setY(cpu, 0x40); setA(cpu, 0x40);
|
|
setReg(cpu, reg, info.data);
|
|
callNext("INC r ", [Block(0x1000, [opcode])]);
|
|
}
|
|
return connect(setup_nz(), setup_inc_data(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For INC.
|
|
auto setup_op_INC()
|
|
{
|
|
return setup_rmw(false, "INC ", setup_inc_data());
|
|
}
|
|
|
|
|
|
auto setup_rmw(bool isAcc, string name, TestSetup data_setup)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
if (isAcc)
|
|
{
|
|
setA(cpu, info.data); setX(cpu, 0x10); setY(cpu, 0x10);
|
|
callNext(name ~ "a ");
|
|
}
|
|
else
|
|
{
|
|
callNext(name, [Block(info.addr, [info.data])]);
|
|
}
|
|
}
|
|
return connect(setup_nz(), data_setup, TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For ROL.
|
|
auto setup_op_ROL(bool isAcc)
|
|
{
|
|
return setup_rmw(isAcc, "ROL ", setup_data());
|
|
}
|
|
|
|
|
|
auto setup_asl_data()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.data = 0x00;
|
|
callNext("zero ");
|
|
|
|
info.data = 0x80;
|
|
callNext("zero carry ");
|
|
|
|
info.data = 0x40;
|
|
callNext("negative ");
|
|
|
|
info.data = 0x01;
|
|
callNext("positive ");
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For ASL.
|
|
auto setup_op_ASL(bool isAcc)
|
|
{
|
|
return setup_rmw(isAcc, "ASL ", setup_asl_data());
|
|
}
|
|
|
|
|
|
auto setup_right_data()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
info.data = 0x01;
|
|
callNext("0x01 ");
|
|
|
|
info.data = 0x80;
|
|
callNext("0x80 ");
|
|
|
|
info.data = 0x00;
|
|
callNext("0x00 ");
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
// For ROR.
|
|
auto setup_op_ROR(bool isAcc)
|
|
{
|
|
return setup_rmw(isAcc, "ROR ", setup_right_data());
|
|
}
|
|
|
|
|
|
// For LSR.
|
|
auto setup_op_LSR(bool isAcc)
|
|
{
|
|
return setup_rmw(isAcc, "LSR ", setup_right_data());
|
|
}
|
|
|
|
|
|
// For SLO.
|
|
auto setup_op_SLO()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x00);
|
|
callNext("acc 0x00 ");
|
|
|
|
setA(cpu, 0xFF);
|
|
callNext("acc 0xFF ");
|
|
|
|
setA(cpu, 0x20);
|
|
callNext("acc 0x20 ");
|
|
}
|
|
return connect(TestSetup(&setup),
|
|
setup_rmw(false, "SLO ", setup_asl_data()));
|
|
}
|
|
|
|
// For RLA.
|
|
auto setup_op_RLA()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x00);
|
|
callNext("acc 0x00 ");
|
|
|
|
setA(cpu, 0xFF);
|
|
callNext("acc 0xFF ");
|
|
}
|
|
return connect(TestSetup(&setup),
|
|
setup_rmw(false, "RLA ", setup_data()));
|
|
}
|
|
|
|
|
|
// For SRE.
|
|
auto setup_op_SRE()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x00);
|
|
callNext("acc 0x00 ");
|
|
|
|
setA(cpu, 0xFF);
|
|
callNext("acc 0xFF ");
|
|
}
|
|
return connect(TestSetup(&setup),
|
|
setup_rmw(false, "SRE ", setup_right_data()));
|
|
}
|
|
|
|
|
|
/*
|
|
* For ADC.
|
|
*
|
|
* This opcode is extensively tested elsewhere.
|
|
*/
|
|
auto setup_op_ADC(bool cmos)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x99);
|
|
info.data = 0x01;
|
|
callNext("ADC 1", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x5B);
|
|
info.data = 0x46;
|
|
callNext("ADC 2", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x50);
|
|
info.data = 0x60;
|
|
callNext("ADC 3", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x90);
|
|
info.data = 0x90;
|
|
callNext("ADC 4", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x90);
|
|
info.data = 0x0F;
|
|
callNext("ADC 5", [Block(info.addr, [info.data])]);
|
|
}
|
|
return connect(setup_nz(), setup_flag(Flag.C), setup_flag(Flag.D),
|
|
TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For RRA.
|
|
auto setup_op_RRA()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x00);
|
|
callNext("acc 0x00 ");
|
|
|
|
setA(cpu, 0x10);
|
|
callNext("acc 0x10 ");
|
|
|
|
setA(cpu, 0xBF);
|
|
callNext("acc 0xBF ");
|
|
}
|
|
|
|
return connect(TestSetup(&setup), setup_flag(Flag.C), setup_flag(Flag.D),
|
|
setup_rmw(false, "RRA ", setup_right_data()));
|
|
}
|
|
|
|
|
|
/*
|
|
* For SBC.
|
|
*
|
|
* This opcode is extensively tested elsewhere.
|
|
*/
|
|
auto setup_op_SBC(bool cmos)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x00);
|
|
info.data = 0x01;
|
|
callNext("SBC 1", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x46);
|
|
info.data = 0x12;
|
|
callNext("SBC 2", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x50);
|
|
info.data = 0x60;
|
|
callNext("SBC 3", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x32);
|
|
info.data = 0x02;
|
|
callNext("SBC 4", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x90);
|
|
info.data = 0x0F;
|
|
callNext("SBC 5", [Block(info.addr, [info.data])]);
|
|
}
|
|
return connect(setup_nz(), setup_flag(Flag.C), setup_flag(Flag.D),
|
|
TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For ISC.
|
|
auto setup_op_ISC()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x00);
|
|
callNext("acc 0x00 ");
|
|
|
|
setA(cpu, 0x10);
|
|
callNext("acc 0x10 ");
|
|
|
|
setA(cpu, 0xBF);
|
|
callNext("acc 0xBF ");
|
|
}
|
|
|
|
return connect(TestSetup(&setup), setup_flag(Flag.C), setup_flag(Flag.D),
|
|
setup_rmw(false, "ISC ", setup_inc_data()));
|
|
}
|
|
|
|
|
|
// For CMP, CPY, CPX.
|
|
auto setup_op_cmp(Reg reg)
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setReg(cpu, reg, 0x10);
|
|
info.data = 0x20;
|
|
callNext("compare <", [Block(info.addr, [info.data])]);
|
|
|
|
setReg(cpu, reg, 0x10);
|
|
info.data = 0x10;
|
|
callNext("compare =", [Block(info.addr, [info.data])]);
|
|
|
|
setReg(cpu, reg, 0x20);
|
|
info.data = 0x10;
|
|
callNext("compare >", [Block(info.addr, [info.data])]);
|
|
}
|
|
return connect(setup_nz(), TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For DCP.
|
|
auto setup_op_DCP()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x00);
|
|
callNext("acc 0x00 ");
|
|
|
|
setA(cpu, 0x10);
|
|
callNext("acc 0x10 ");
|
|
|
|
setA(cpu, 0xBF);
|
|
callNext("acc 0xBF ");
|
|
}
|
|
|
|
return connect(TestSetup(&setup), setup_flag(Flag.C),
|
|
setup_rmw(false, "DCP ", setup_dec_data()));
|
|
}
|
|
|
|
|
|
// For ORA.
|
|
auto setup_op_ORA()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x00);
|
|
info.data = 0x00;
|
|
callNext("ORA zero", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x10);
|
|
info.data = 0x01;
|
|
callNext("ORA positive", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x80);
|
|
info.data = 0x00;
|
|
callNext("ORA negative", [Block(info.addr, [info.data])]);
|
|
}
|
|
return connect(setup_nz(),TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For AND.
|
|
auto setup_op_AND()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0x10);
|
|
info.data = 0x01;
|
|
callNext("AND zero", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x0B);
|
|
info.data = 0x0D;
|
|
callNext("AND positive", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x80);
|
|
info.data = 0x80;
|
|
callNext("AND negative", [Block(info.addr, [info.data])]);
|
|
}
|
|
return connect(setup_nz(),TestSetup(&setup));
|
|
}
|
|
|
|
|
|
// For EOR.
|
|
auto setup_op_EOR()
|
|
{
|
|
auto setup(ubyte opcode, CpuInfo cpu, Block[] data, OpInfo info,
|
|
string msg, TestSetup* next)
|
|
{
|
|
mixin testCallNext;
|
|
setA(cpu, 0xFF);
|
|
info.data = 0xFF;
|
|
callNext("EOR zero", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x81);
|
|
info.data = 0x82;
|
|
callNext("EOR positive", [Block(info.addr, [info.data])]);
|
|
|
|
setA(cpu, 0x40);
|
|
info.data = 0x80;
|
|
callNext("EOR negative", [Block(info.addr, [info.data])]);
|
|
}
|
|
return connect(setup_nz(),TestSetup(&setup));
|
|
}
|
|
|
|
|
|
/*
|
|
* A range over the differences in two TestMemory structs. Each
|
|
* element is an address with up to eight bytes of subsequent data
|
|
* from the two structs.
|
|
*/
|
|
struct MemDiff
|
|
{
|
|
|
|
static struct DiffBlock
|
|
{
|
|
ushort base;
|
|
const(ubyte[]) a;
|
|
const(ubyte[]) b;
|
|
}
|
|
|
|
const(TestMemory*) a, b;
|
|
int i;
|
|
bool _empty, _emptyChecked;
|
|
|
|
this(const ref TestMemory a, const ref TestMemory b)
|
|
{
|
|
assert(a.data2_base == b.data2_base && a.data2_max == b.data2_max);
|
|
this.a = &a; this.b = &b;
|
|
}
|
|
|
|
auto front()
|
|
{
|
|
assert(_emptyChecked && !_empty);
|
|
if (i < 0x200)
|
|
{
|
|
auto extent = min(8, 0x200 - i);
|
|
return DiffBlock(cast(ushort)i,
|
|
a.data1[i..i+extent].dup,
|
|
b.data1[i..i+extent].dup);
|
|
}
|
|
else
|
|
{
|
|
auto extent = min(8, a.data2_max - i);
|
|
auto di = i - a.data2_base;
|
|
return DiffBlock(cast(ushort)i,
|
|
a.data2[di..di+extent].dup,
|
|
b.data2[di..di+extent].dup);
|
|
}
|
|
assert(0);
|
|
}
|
|
|
|
void popFront()
|
|
{
|
|
assert(_emptyChecked && !_empty);
|
|
if (i < 0x200)
|
|
i += min(8, 0x200 - i);
|
|
else
|
|
i += min(8, a.data2_max - i);
|
|
}
|
|
|
|
bool empty()
|
|
{
|
|
_emptyChecked = true;
|
|
while (i < 0x200)
|
|
{
|
|
if (a.data1[i] != b.data1[i]) { _empty = false; return false; }
|
|
i++;
|
|
}
|
|
if (i < a.data2_base) i = a.data2_base;
|
|
while (i < a.data2_max)
|
|
{
|
|
auto di = i - a.data2_base;
|
|
if (a.data2[di] != b.data2[di]) { _empty = false; return false; }
|
|
i++;
|
|
}
|
|
_empty = true; return true;
|
|
}
|
|
}
|
|
|
|
|
|
struct Expected
|
|
{
|
|
CpuInfo cpu;
|
|
TestMemory mem;
|
|
}
|
|
|
|
alias void delegate(ref Expected, const OpInfo)
|
|
testexpect;
|
|
|
|
|
|
// Expects nothing to have changed.
|
|
auto expect_none()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For opcodes which modify a register and set N/Z.
|
|
void expect_basic(Reg reg, ubyte val, ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
setReg(cpu, reg, val);
|
|
setNZ(cpu, val);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
|
|
|
|
// For flag opcodes.
|
|
auto expect_flag(Flag f, bool val)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
updateFlag(cpu, f, val);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For NOP.
|
|
auto expect_NOP()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected) { incPC(cpu, info.len); }
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For register transfer opcodes.
|
|
auto expect_transfer(Reg source, Reg dest)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = getReg(expected.cpu, source);
|
|
expect_basic(dest, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For push opcodes.
|
|
auto expect_push(Reg reg)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
mem[getSP(cpu)] = getReg(cpu, reg);
|
|
decSP(cpu);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For PHP.
|
|
auto expect_PHP()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
mem[getSP(cpu)] = getStatus(cpu);
|
|
decSP(cpu);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For pull opcodes.
|
|
auto expect_pull(Reg reg)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
incSP(cpu);
|
|
auto val = mem[getSP(cpu)];
|
|
expect_basic(reg, val, expected, info);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For PLP.
|
|
auto expect_PLP()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
incSP(cpu);
|
|
setStatus(cpu, mem[getSP(cpu)]);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For TSX.
|
|
auto expect_TSX()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
ubyte val = cast(ubyte)getSP(expected.cpu);
|
|
expect_basic(Reg.X, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For TXS.
|
|
auto expect_TXS()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
auto val = getX(cpu);
|
|
setSP(cpu, val);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For load opcodes.
|
|
auto expect_load(Reg reg)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = expected.mem[info.addr];
|
|
expect_basic(reg, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For LAX (except immediate).
|
|
auto expect_LAX()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
expect_load(Reg.A)(expected, info);
|
|
with(expected) { setX(cpu, getA(cpu)); }
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
// For store opcodes.
|
|
auto expect_store(Reg reg)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
mem[info.addr] = getReg(cpu, reg);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For SAX.
|
|
auto expect_SAX()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
mem[info.addr] = getA(cpu) & getX(cpu);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For STZ.
|
|
auto expect_STZ()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
mem[info.addr] = 0;
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For branch opcodes.
|
|
auto expect_branch()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected) { setPC(cpu, info.addr); }
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For BIT.
|
|
auto expect_BIT()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
updateFlag(cpu, Flag.N, (info.data & 0x80) != 0);
|
|
updateFlag(cpu, Flag.V, (info.data & 0x40) != 0);
|
|
updateFlag(cpu, Flag.Z, (info.data & getA(cpu)) == 0);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For BIT.
|
|
auto expect_BIT_imm()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
updateFlag(cpu, Flag.Z, (info.data & getA(cpu)) == 0);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For TRB.
|
|
auto expect_TRB()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
updateFlag(cpu, Flag.Z, (info.data & getA(cpu)) == 0);
|
|
mem[info.addr] = ~getA(cpu) & info.data;
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
auto expect_TSB()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
updateFlag(cpu, Flag.Z, (info.data & getA(cpu)) == 0);
|
|
mem[info.addr] = getA(cpu) | info.data;
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
// For BRK.
|
|
auto expect_BRK()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
ushort ret = cast(ushort)(getPC(cpu) + 2);
|
|
mem[getSP(cpu)] = addrHi(ret);
|
|
decSP(cpu);
|
|
mem[getSP(cpu)] = addrLo(ret);
|
|
decSP(cpu);
|
|
mem[getSP(cpu)] = info.data;
|
|
decSP(cpu);
|
|
setPC(cpu, info.addr);
|
|
setFlag(cpu, Flag.I);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For RTI.
|
|
auto expect_RTI()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
setStatus(cpu, info.data);
|
|
setPC(cpu, info.addr);
|
|
incSP(cpu, 3);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For RTS.
|
|
auto expect_RTS()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
setPC(cpu, info.addr);
|
|
incSP(cpu, 2);
|
|
incPC(cpu, 1);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For JMP.
|
|
auto expect_JMP()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
setPC(cpu, info.addr);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For JSR.
|
|
auto expect_JSR()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected)
|
|
{
|
|
auto ret = cast(ushort)(getPC(cpu) + 2);
|
|
setPC(cpu, info.addr);
|
|
mem[getSP(cpu)] = addrHi(ret);
|
|
decSP(cpu);
|
|
mem[getSP(cpu)] = addrLo(ret);
|
|
decSP(cpu);
|
|
}
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
void expect_rmw(bool isAcc, ubyte val, ref Expected expected,
|
|
const OpInfo info)
|
|
{
|
|
if (isAcc)
|
|
expect_basic(Reg.A, val, expected, info);
|
|
else
|
|
{
|
|
with(expected)
|
|
{
|
|
mem[info.addr] = val;
|
|
setNZ(cpu, val);
|
|
incPC(cpu, info.len);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// For DEX, DEY, DEA.
|
|
auto expect_DEC_reg(Reg reg)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data - 1);
|
|
expect_basic(reg, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For DEC.
|
|
auto expect_DEC()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data - 1);
|
|
expect_rmw(false, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For INX, INY, INA.
|
|
auto expect_INC_reg(Reg reg)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data + 1);
|
|
expect_basic(reg, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For INC.
|
|
auto expect_INC()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data + 1);
|
|
expect_rmw(false, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For ROL.
|
|
auto expect_ROL(bool isAcc)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto carry = (getFlag(expected.cpu, Flag.C) ? 1 : 0);
|
|
auto val = cast(ubyte)(info.data << 1 | carry);
|
|
updateFlag(expected.cpu, Flag.C, (info.data > 0x7F));
|
|
|
|
expect_rmw(isAcc, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For ASL.
|
|
auto expect_ASL(bool isAcc)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data << 1);
|
|
updateFlag(expected.cpu, Flag.C, (info.data > 0x7F));
|
|
|
|
expect_rmw(isAcc, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
// For ROR.
|
|
auto expect_ROR(bool isAcc)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto carry = (getFlag(expected.cpu, Flag.C) ? 0x80 : 0);
|
|
auto val = cast(ubyte)(info.data >> 1 | carry);
|
|
updateFlag(expected.cpu, Flag.C, ((info.data & 0x01) != 0));
|
|
|
|
expect_rmw(isAcc, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For LSR.
|
|
auto expect_LSR(bool isAcc)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data >> 1);
|
|
updateFlag(expected.cpu, Flag.C, ((info.data & 0x01) != 0));
|
|
|
|
expect_rmw(isAcc, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For SLO.
|
|
auto expect_SLO()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data << 1);
|
|
updateFlag(expected.cpu, Flag.C, (info.data > 0x7F));
|
|
setA(expected.cpu, getA(expected.cpu) | val);
|
|
|
|
expect_rmw(false, val, expected, info);
|
|
setNZ(expected.cpu, getA(expected.cpu));
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For RLA.
|
|
auto expect_RLA()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto carry = (getFlag(expected.cpu, Flag.C) ? 1 : 0);
|
|
auto val = cast(ubyte)(info.data << 1 | carry);
|
|
updateFlag(expected.cpu, Flag.C, (info.data > 0x7F));
|
|
setA(expected.cpu, getA(expected.cpu) & val);
|
|
|
|
expect_rmw(false, val, expected, info);
|
|
setNZ(expected.cpu, getA(expected.cpu));
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For SRE.
|
|
auto expect_SRE()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data >> 1);
|
|
updateFlag(expected.cpu, Flag.C, ((info.data & 0x01) != 0));
|
|
setA(expected.cpu, getA(expected.cpu) ^ val);
|
|
|
|
expect_rmw(false, val, expected, info);
|
|
setNZ(expected.cpu, getA(expected.cpu));
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
/*
|
|
* Slightly cheating--this is copied from the original implementation,
|
|
* but it was independently tested (see test_decimal.d).
|
|
*/
|
|
void expect_add(bool cmos, ubyte val, ref Expected expected)
|
|
{
|
|
with(expected)
|
|
{
|
|
ubyte acc = getA(cpu);
|
|
auto d = getFlag(cpu, Flag.D);
|
|
auto c = getFlag(cpu, Flag.C);
|
|
|
|
if (!d)
|
|
{
|
|
uint sum = acc + val + c;
|
|
updateFlag(cpu, Flag.V,
|
|
(!((acc ^ val) & 0x80)) && ((val ^ sum) & 0x80));
|
|
updateFlag(cpu, Flag.C, (sum > 0xFF));
|
|
setNZ(cpu, cast(ubyte)sum);
|
|
setA(cpu, cast(ubyte)sum);
|
|
}
|
|
else
|
|
{
|
|
int a = acc;
|
|
int al = (a & 0x0F) + (val & 0x0F) + c;
|
|
if (al >= 0x0A)
|
|
al = ((al + 0x06) & 0x0F) + 0x10;
|
|
a = (a & 0xF0) + (val & 0xF0) + al;
|
|
|
|
if (!cmos)
|
|
{
|
|
updateFlag(cpu, Flag.N, (a & 0xFF) > 0x7F);
|
|
updateFlag(cpu, Flag.Z, (cast(ubyte)(acc + val + c)) == 0);
|
|
}
|
|
updateFlag(cpu, Flag.V,
|
|
(!((acc ^ val) & 0x80)) && ((val ^ a) & 0x80));
|
|
|
|
if (a >= 0xA0)
|
|
a = a + 0x60;
|
|
updateFlag(cpu, Flag.C, (a >= 0x100));
|
|
|
|
setA(cpu, cast(ubyte)a);
|
|
if (cmos)
|
|
setNZ(cpu, cast(ubyte)a);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// For ADC.
|
|
auto expect_ADC(bool cmos)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected) { incPC(cpu, info.len); }
|
|
expect_add(cmos, info.data, expected);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For RRA.
|
|
auto expect_RRA()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto carry = (getFlag(expected.cpu, Flag.C) ? 0x80 : 0);
|
|
auto val = cast(ubyte)(info.data >> 1 | carry);
|
|
updateFlag(expected.cpu, Flag.C, ((info.data & 0x01) != 0));
|
|
|
|
expect_rmw(false, val, expected, info);
|
|
expect_add(false, val, expected);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
/*
|
|
* Slightly cheating--this is copied from the original implementation,
|
|
* but it was independently tested (see test_decimal.d).
|
|
*/
|
|
void expect_sub(bool cmos, ubyte val, ref Expected expected)
|
|
{
|
|
with(expected)
|
|
{
|
|
ubyte acc = getA(cpu);
|
|
auto d = getFlag(cpu, Flag.D);
|
|
auto c = getFlag(cpu, Flag.C);
|
|
|
|
if (!d)
|
|
{
|
|
uint diff = acc - val - (c ? 0 : 1);
|
|
updateFlag(cpu, Flag.V,
|
|
((acc ^ diff) & 0x80) && ((acc ^ val) & 0x80));
|
|
updateFlag(cpu, Flag.C, (diff < 0x100));
|
|
setA(cpu, cast(ubyte)diff);
|
|
setNZ(cpu, cast(ubyte)diff);
|
|
}
|
|
else
|
|
{
|
|
if (!cmos)
|
|
{
|
|
int a = acc;
|
|
int al = (a & 0x0F) - (val & 0x0F) - !c;
|
|
if (al < 0)
|
|
al = ((al - 0x06) & 0x0F) - 0x10;
|
|
a = (a & 0xF0) - (val & 0xF0) + al;
|
|
if (a < 0)
|
|
a = a - 0x60;
|
|
|
|
uint diff = acc - val - !c;
|
|
updateFlag(cpu, Flag.V,
|
|
((acc ^ diff) & 0x80) && ((acc ^ val) & 0x80));
|
|
updateFlag(cpu, Flag.C, (diff < 0x100));
|
|
setNZ(cpu, cast(ubyte)diff);
|
|
|
|
setA(cpu, cast(ubyte)a);
|
|
}
|
|
else
|
|
{
|
|
int a = acc;
|
|
int al = (a & 0x0F) - (val & 0x0F) - !c;
|
|
a = a - val - !c;
|
|
if (a < 0)
|
|
a = a - 0x60;
|
|
if (al < 0)
|
|
a = a - 0x06;
|
|
|
|
uint diff = acc - val - !c;
|
|
updateFlag(cpu, Flag.V,
|
|
((acc ^ diff) & 0x80) && ((acc ^ val) & 0x80));
|
|
updateFlag(cpu, Flag.C, (diff < 0x100));
|
|
setNZ(cpu, cast(ubyte)a);
|
|
setA(cpu, cast(ubyte)a);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// For SBC.
|
|
auto expect_SBC(bool cmos)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected) { incPC(cpu, info.len); }
|
|
expect_sub(cmos, info.data, expected);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For ISC.
|
|
auto expect_ISC()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data + 1);
|
|
expect_rmw(false, val, expected, info);
|
|
expect_sub(false, val, expected);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
void expect_compare(Reg reg, ubyte val, ref Expected expected)
|
|
{
|
|
with(expected)
|
|
{
|
|
auto r = getReg(cpu, reg);
|
|
updateFlag(cpu, Flag.C, (r >= val));
|
|
setNZ(cpu, cast(ubyte)(r - val));
|
|
}
|
|
}
|
|
|
|
|
|
// For CMP, CPX, CPY.
|
|
auto expect_cmp(Reg reg)
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
with(expected) { incPC(cpu, info.len); }
|
|
expect_compare(reg, info.data, expected);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For DCP.
|
|
auto expect_DCP()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data - 1);
|
|
expect_rmw(false, val, expected, info);
|
|
expect_compare(Reg.A, val, expected);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For ORA.
|
|
auto expect_ORA()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data | getA(expected.cpu));
|
|
expect_basic(Reg.A, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For AND.
|
|
auto expect_AND()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data & getA(expected.cpu));
|
|
expect_basic(Reg.A, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// For EOR.
|
|
auto expect_EOR()
|
|
{
|
|
void expect(ref Expected expected, const OpInfo info)
|
|
{
|
|
auto val = cast(ubyte)(info.data ^ getA(expected.cpu));
|
|
expect_basic(Reg.A, val, expected, info);
|
|
}
|
|
return &expect;
|
|
}
|
|
|
|
|
|
// Associates opcodes with test setup functions.
|
|
string getMemSetup(T)()
|
|
if (isCpu!T)
|
|
{
|
|
string[] tmp1 = new string[256], tmp2 = new string[256],
|
|
tmp3 = new string[256];
|
|
tmp1[] = " setup_addr = setup_none();\n";
|
|
tmp2[] = " setup_test = setup_none();\n";
|
|
tmp3[] = " expect = expect_none();\n";
|
|
|
|
void get_addr(const(ubyte[]) list, string name, string arg = "")
|
|
{
|
|
foreach(op; list)
|
|
tmp1[op] = "setup_addr = setup_addr_" ~ name ~ "(" ~ arg ~ ");\n";
|
|
}
|
|
|
|
void get_test(const(ubyte[]) list, string name, string arg = "")
|
|
{
|
|
foreach(op; list)
|
|
tmp2[op] = "setup_test = setup_op_" ~ name ~ "(" ~ arg ~ ");\n";
|
|
}
|
|
|
|
void get_expect(const(ubyte[]) list, string name, string arg = "")
|
|
{
|
|
foreach(op; list)
|
|
tmp3[op] = "expect = expect_" ~ name ~ "(" ~ arg ~ ");\n";
|
|
}
|
|
|
|
void get_both(const(ubyte[]) list, string name, string arg = "")
|
|
{
|
|
get_test(list, name, arg);
|
|
get_expect(list, name, arg);
|
|
}
|
|
|
|
get_addr(REG_OPS!T, "implied");
|
|
get_addr(PUSH_OPS!T, "implied");
|
|
get_addr(PULL_OPS!T, "implied");
|
|
get_addr(BRANCH_OPS!T, "branch", "isCMOS!T");
|
|
get_addr(IMM_OPS!T, "imm");
|
|
get_addr(ZPG_OPS!T, "zpg");
|
|
get_addr(ABS_OPS!T, "abs");
|
|
get_addr([0x4C], "abs");
|
|
get_addr(ZPX_OPS!T, "zpxy", "Reg.X");
|
|
get_addr(ZPY_OPS!T, "zpxy", "Reg.Y");
|
|
get_addr(ABX_OPS!T, "abxy", "Reg.X");
|
|
get_addr(ABY_OPS!T, "abxy", "Reg.Y");
|
|
get_addr(IZX_OPS!T, "izx");
|
|
get_addr(IZY_OPS!T, "izy");
|
|
|
|
get_expect([0x18], "flag", "Flag.C, false");
|
|
get_expect([0x38], "flag", "Flag.C, true");
|
|
get_expect([0xD8], "flag", "Flag.D, false");
|
|
get_expect([0xF8], "flag", "Flag.D, true");
|
|
get_expect([0x58], "flag", "Flag.I, false");
|
|
get_expect([0x78], "flag", "Flag.I, true");
|
|
get_expect([0xB8], "flag", "Flag.V, false");
|
|
get_both([0x08], "PHP");
|
|
get_both([0x28], "PLP");
|
|
get_both([0x48], "push", "Reg.A");
|
|
get_both([0x68], "pull", "Reg.A");
|
|
get_both([0x8A], "transfer", "Reg.X, Reg.A");
|
|
get_both([0x98], "transfer", "Reg.Y, Reg.A");
|
|
get_both([0xA8], "transfer", "Reg.A, Reg.Y");
|
|
get_both([0xAA], "transfer", "Reg.A, Reg.X");
|
|
get_test([0x9A, 0xBA], "stack_xfer");
|
|
get_expect([0x9A], "TXS");
|
|
get_expect([0xBA], "TSX");
|
|
get_both([0xA1, 0xA5, 0xA9, 0xAD, 0xB1, 0xB5, 0xB9, 0xBD],
|
|
"load", "Reg.A");
|
|
get_both([0xA2, 0xA6, 0xAE, 0xB6, 0xBE], "load", "Reg.X");
|
|
get_both([0xA0, 0xA4, 0xAC, 0xB4, 0xBC], "load", "Reg.Y");
|
|
get_both([0x81, 0x85, 0x8D, 0x91, 0x95, 0x99, 0x9D], "store", "Reg.A");
|
|
get_both([0x86, 0x8E, 0x96], "store", "Reg.X");
|
|
get_both([0x84, 0x8C, 0x94], "store", "Reg.Y");
|
|
get_expect(BRANCH_OPS!T, "branch");
|
|
get_both([0x24, 0x2C], "BIT");
|
|
get_both([0x00], "BRK");
|
|
get_both([0x40], "RTI");
|
|
get_both([0x60], "RTS");
|
|
get_test([0x6C], "JMP_ind", "isCMOS!T");
|
|
get_expect([0x4C, 0x6C], "JMP");
|
|
get_both([0x20], "JSR");
|
|
get_both([0x88], "DEC_reg", "Reg.Y");
|
|
get_both([0xCA], "DEC_reg", "Reg.X");
|
|
get_both([0xC8], "INC_reg", "Reg.Y");
|
|
get_both([0xE8], "INC_reg", "Reg.X");
|
|
get_both([0xE6, 0xEE, 0xF6, 0xFE], "INC");
|
|
get_both([0xC6, 0xCE, 0xD6, 0xDE], "DEC");
|
|
get_both([0x2A], "ROL", "true");
|
|
get_both([0x0A], "ASL", "true");
|
|
get_both([0x6A], "ROR", "true");
|
|
get_both([0x4A], "LSR", "true");
|
|
get_both([0x26, 0x2E, 0x36, 0x3E], "ROL", "false");
|
|
get_both([0x06, 0x0E, 0x16, 0x1E], "ASL", "false");
|
|
get_both([0x66, 0x6E, 0x76, 0x7E], "ROR", "false");
|
|
get_both([0x46, 0x4E, 0x56, 0x5E], "LSR", "false");
|
|
get_both([0x61, 0x65, 0x69, 0x6D, 0x71, 0x75, 0x79, 0x7D],
|
|
"ADC", "isCMOS!T");
|
|
get_both([0xE1, 0xE5, 0xE9, 0xED, 0xF1, 0xF5, 0xF9, 0xFD],
|
|
"SBC", "isCMOS!T");
|
|
get_both([0xC1, 0xC5, 0xC9, 0xCD, 0xD1, 0xD5, 0xD9, 0xDD], "cmp", "Reg.A");
|
|
get_both([0x01, 0x05, 0x09, 0x0D, 0x11, 0x15, 0x19, 0x1D], "ORA");
|
|
get_both([0x21, 0x25, 0x29, 0x2D, 0x31, 0x35, 0x39, 0x3D], "AND");
|
|
get_both([0x41, 0x45, 0x49, 0x4D, 0x51, 0x55, 0x59, 0x5D], "EOR");
|
|
get_both([0xE0, 0xE4, 0xEC], "cmp", "Reg.X");
|
|
get_both([0xC0, 0xC4, 0xCC], "cmp", "Reg.Y");
|
|
|
|
static if (isNMOS!T)
|
|
{
|
|
get_addr(HLT_OPS!T, "implied");
|
|
|
|
get_test([0xA3, 0xA7, 0xAF, 0xB3, 0xB7, 0xBF], "load", "Reg.A");
|
|
get_expect([0xA3, 0xA7, 0xAF, 0xB3, 0xB7, 0xBF], "LAX");
|
|
get_expect([0x1A, 0x3A, 0x5A, 0x7A, 0xDA, 0xEA, 0xFA], "NOP");
|
|
get_expect([0x0C, 0x1C, 0x3C, 0x5C, 0x7C, 0xDC, 0xFC], "NOP");
|
|
get_expect([0x80, 0x82, 0x89, 0xC2, 0xE2], "NOP");
|
|
get_expect([0x04, 0x44, 0x64], "NOP");
|
|
get_expect([0x14, 0x34, 0x54, 0x74, 0xD4, 0xF4], "NOP");
|
|
get_both([0x83, 0x87, 0x8F, 0x97], "SAX");
|
|
get_both([0x03, 0x07, 0x0F, 0x13, 0x17, 0x1B, 0x1F], "SLO");
|
|
get_both([0x23, 0x27, 0x2F, 0x33, 0x37, 0x3B, 0x3F], "RLA");
|
|
get_both([0x43, 0x47, 0x4F, 0x53, 0x57, 0x5B, 0x5F], "SRE");
|
|
get_both([0x63, 0x67, 0x6F, 0x73, 0x77, 0x7B, 0x7F], "RRA");
|
|
get_both([0xE3, 0xE7, 0xEF, 0xF3, 0xF7, 0xFB, 0xFF], "ISC");
|
|
get_both([0xC3, 0xC7, 0xCF, 0xD3, 0xD7, 0xDB, 0xDF], "DCP");
|
|
get_both([0xEB], "SBC", "false");
|
|
|
|
// TODO: implement these opcode tests
|
|
get_test([0x0B, 0x2B, 0x4B, 0x6B, 0x8B, 0x93, 0x9B, 0x9C, 0x9E,
|
|
0x9F, 0xAB, 0xBB, 0xCB, 0xEB], "abort");
|
|
}
|
|
else
|
|
{
|
|
get_addr(NOP1_OPS!T, "implied");
|
|
get_addr(ZPI_OPS!T, "zpi");
|
|
get_both([0x12], "ORA");
|
|
get_both([0x32], "AND");
|
|
get_both([0x52], "EOR");
|
|
get_both([0x72], "ADC", "true");
|
|
get_both([0x92], "store", "Reg.A");
|
|
get_both([0xB2], "load", "Reg.A");
|
|
get_both([0xD2], "cmp", "Reg.A");
|
|
get_both([0xF2], "SBC", "true");
|
|
get_both([0x5A], "push", "Reg.Y");
|
|
get_both([0xDA], "push", "Reg.X");
|
|
get_both([0x7A], "pull", "Reg.Y");
|
|
get_both([0xFA], "pull", "Reg.X");
|
|
get_expect([0xEA], "NOP");
|
|
get_expect(NOP1_OPS!T, "NOP");
|
|
get_expect([0x02, 0x22, 0x42, 0x62, 0x82, 0xC2, 0xE2], "NOP");
|
|
get_expect([0x44, 0x54, 0xD4, 0xF4, 0xDC, 0xFC], "NOP");
|
|
get_both([0x64, 0x74, 0x9C, 0x9E], "STZ");
|
|
get_both([0x34, 0x3C], "BIT");
|
|
get_test([0x89], "BIT"); get_expect([0x89], "BIT_imm");
|
|
get_both([0x14, 0x1C], "TRB");
|
|
get_both([0x04, 0x0C], "TSB");
|
|
get_test([0x7C], "JMP_inx");
|
|
get_expect([0x7C], "JMP");
|
|
get_both([0xB2], "load", "Reg.A");
|
|
get_both([0x92], "store", "Reg.A");
|
|
get_test([0x5C], "5C");
|
|
get_expect([0x5C], "NOP");
|
|
get_both([0x3A], "DEC_reg", "Reg.A");
|
|
get_both([0x1A], "INC_reg", "Reg.A");
|
|
}
|
|
|
|
auto ret = "final switch (opcode)\n{\n";
|
|
for (auto i = 0; i < 256; i++)
|
|
{
|
|
ret ~= " case 0x" ~ to!string(i, 16) ~ ":\n" ~
|
|
" " ~ tmp1[i] ~ " " ~ tmp2[i] ~
|
|
" " ~ tmp3[i] ~ " break;\n";
|
|
}
|
|
return ret ~ "\n}";
|
|
}
|
|
|
|
|
|
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);
|
|
}
|
|
|
|
|
|
T[] If(alias cond, T)(T[] actions)
|
|
{
|
|
if (cond)
|
|
return actions;
|
|
else
|
|
return [];
|
|
}
|
|
|
|
|
|
/// Bus access pattern for register opcodes.
|
|
auto accesses_reg(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(REG_OPS!T.canFind(opcode));
|
|
|
|
cycles = 2;
|
|
return [Bus(Action.READ, pc)] ~
|
|
If!(isStrict!T)(
|
|
[Bus(Action.READ, pc+1)]);
|
|
}
|
|
|
|
|
|
/// Bus access pattern for push opcodes.
|
|
auto accesses_push(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(PUSH_OPS!T.canFind(opcode));
|
|
|
|
auto sp = getSP(cpu);
|
|
|
|
cycles = 3;
|
|
return [Bus(Action.READ, pc)] ~
|
|
If!(isStrict!T)(
|
|
[Bus(Action.READ, pc+1)]) ~
|
|
[Bus(Action.WRITE, sp)];
|
|
}
|
|
|
|
|
|
/// Bus access pattern for pull opcodes.
|
|
auto accesses_pull(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(PULL_OPS!T.canFind(opcode));
|
|
|
|
auto sp = getSP(cpu);
|
|
auto sp1 = pageWrapAdd(sp, 1);
|
|
|
|
cycles = 4;
|
|
return [Bus(Action.READ, pc)] ~
|
|
If!(isStrict!T)(
|
|
[Bus(Action.READ, pc+1),
|
|
Bus(Action.READ, sp)]) ~
|
|
[Bus(Action.READ, sp1)];
|
|
}
|
|
|
|
|
|
/// Bus access pattern for immediate mode opcodes.
|
|
auto accesses_imm(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(IMM_OPS!T.canFind(opcode));
|
|
|
|
bool decimal = isCMOS!T && getFlag(cpu, Flag.D) &&
|
|
BCD_OPS!T.canFind(opcode);
|
|
|
|
cycles = 2 + decimal;
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1)] ~
|
|
If!decimal(If!(isStrict!T)(
|
|
[Bus(Action.READ, pc+2)]));
|
|
}
|
|
|
|
|
|
/// Bus access pattern for branch opcodes.
|
|
auto accesses_rel(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc], op1 = mem[pc+1];
|
|
assert(BRANCH_OPS!T.canFind(opcode));
|
|
|
|
auto base = cast(ushort)(pc + 2);
|
|
bool branch = wouldBranch(cpu, opcode);
|
|
ushort wrongPage = pageWrapAdd(base, cast(byte)op1);
|
|
bool px = wrongPage != pageCrossAdd(base, cast(byte)op1);
|
|
ushort wrongAddr = isNMOS!T ? wrongPage : base;
|
|
|
|
cycles = 2 + branch + px;
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1)] ~
|
|
If!branch(If!(isStrict!T)(
|
|
[Bus(Action.READ, pc+2)] ~
|
|
If!px(
|
|
[Bus(Action.READ, wrongAddr)])));
|
|
}
|
|
|
|
|
|
/// Bus access pattern for zeropage mode opcodes.
|
|
auto accesses_zpg(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc], op1 = mem[pc+1];
|
|
assert(ZPG_OPS!T.canFind(opcode));
|
|
|
|
cycles = 2; // + accesses_end
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1)] ~
|
|
accesses_end(cpu, opcode, 2, op1, cycles);
|
|
}
|
|
|
|
|
|
/// Bus access pattern for absolute mode opcodes.
|
|
auto accesses_abs(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc], op1 = mem[pc+1], op2 = mem[pc+2];
|
|
assert(ABS_OPS!T.canFind(opcode));
|
|
|
|
auto addr = address(op1, op2);
|
|
|
|
cycles = 3; // + accesses_end
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1),
|
|
Bus(Action.READ, pc+2)] ~
|
|
accesses_end(cpu, opcode, 3, addr, cycles);
|
|
}
|
|
|
|
|
|
/// Bus access pattern for zeropage,x/y mode opcodes.
|
|
auto accesses_zpxy(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc], op1 = mem[pc+1];
|
|
bool useX = ZPX_OPS!T.canFind(opcode);
|
|
assert(useX || ZPY_OPS!T.canFind(opcode));
|
|
|
|
auto idx = (useX ? getX(cpu) : getY(cpu));
|
|
auto addr = pageWrapAdd(op1, idx);
|
|
|
|
cycles = 3; // + accesses_end
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1)] ~
|
|
If!(isStrict!T)(
|
|
If!(isNMOS!T)(
|
|
[Bus(Action.READ, op1)]) ~
|
|
If!(isCMOS!T)(
|
|
[Bus(Action.READ, pc+2)])) ~ // XXX
|
|
accesses_end(cpu, opcode, 2, addr, cycles);
|
|
|
|
/*
|
|
* According to "Understanding the Apple IIe", the extra read on
|
|
* the 65C02 (marked XXX above) is the address of the last operand
|
|
* byte (pc + 1).
|
|
*/
|
|
}
|
|
|
|
|
|
/// Bus access pattern for absolute,x/y mode opcodes.
|
|
auto accesses_abxy(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
auto op1 = mem[pc+1], op2 = mem[pc+2];
|
|
bool useX = ABX_OPS!T.canFind(opcode);
|
|
assert(useX || ABY_OPS!T.canFind(opcode));
|
|
|
|
|
|
auto idx = useX ? getX(cpu) : getY(cpu);
|
|
auto base = address(op1, op2);
|
|
auto guess = pageWrapAdd(base, idx);
|
|
auto addr = pageCrossAdd(base, idx);
|
|
|
|
cycles = 3; // + accesses_px + accesses_end
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1),
|
|
Bus(Action.READ, pc+2)] ~
|
|
accesses_px(cpu, opcode, 3, guess, addr, cycles) ~
|
|
accesses_end(cpu, opcode, 3, addr, cycles);
|
|
}
|
|
|
|
|
|
/// Bus access pattern for indirect zeropage,x mode opcodes.
|
|
auto accesses_izx(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc], op1 = mem[pc+1];
|
|
assert(IZX_OPS!T.canFind(opcode));
|
|
|
|
auto idx = getX(cpu);
|
|
auto ial = pageWrapAdd(op1, idx);
|
|
auto iah = pageWrapAdd(ial, 1);
|
|
auto addr = address(mem[ial], mem[iah]);
|
|
|
|
cycles = 5; // + accesses_end
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1)] ~
|
|
If!(isStrict!T)(
|
|
If!(isNMOS!T)(
|
|
[Bus(Action.READ, op1)]) ~
|
|
If!(isCMOS!T)(
|
|
[Bus(Action.READ, pc+2)])) ~ // XXX
|
|
[Bus(Action.READ, ial),
|
|
Bus(Action.READ, iah)] ~
|
|
accesses_end(cpu, opcode, 2, addr, cycles);
|
|
|
|
/*
|
|
* According to "Understanding the Apple IIe", the extra read on
|
|
* the 65C02 (marked XXX above) is the address of the last operand
|
|
* byte (pc + 1).
|
|
*/
|
|
}
|
|
|
|
|
|
/// Bus access pattern for indirect zeropage,y mode opcodes.
|
|
auto accesses_izy(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc], op1 = mem[pc+1];
|
|
assert(IZY_OPS!T.canFind(opcode));
|
|
|
|
auto idx = getY(cpu);
|
|
auto ial = op1;
|
|
auto iah = pageWrapAdd(ial, 1);
|
|
auto base = address(mem[ial], mem[iah]);
|
|
auto guess = pageWrapAdd(base, idx);
|
|
auto addr = pageCrossAdd(base, idx);
|
|
|
|
cycles = 4; // + accesses_px + accesses_end
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1),
|
|
Bus(Action.READ, ial),
|
|
Bus(Action.READ, iah)] ~
|
|
accesses_px(cpu, opcode, 2, guess, addr, cycles) ~
|
|
accesses_end(cpu, opcode, 2, addr, cycles);
|
|
}
|
|
|
|
|
|
/// Bus access pattern for indirect zeropage mode opcodes.
|
|
auto accesses_zpi(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T && isCMOS!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc], op1 = mem[pc+1];
|
|
assert(ZPI_OPS!T.canFind(opcode));
|
|
|
|
auto ial = op1;
|
|
auto iah = pageWrapAdd(ial, 1);
|
|
auto addr = address(mem[ial], mem[iah]);
|
|
|
|
cycles = 4; // + accesses_end
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1),
|
|
Bus(Action.READ, ial),
|
|
Bus(Action.READ, iah)] ~
|
|
accesses_end(cpu, opcode, 2, addr, cycles);
|
|
}
|
|
|
|
|
|
/// Bus access pattern for NMOS HLT opcodes.
|
|
auto accesses_hlt(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T && isNMOS!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(HLT_OPS!T.canFind(opcode));
|
|
|
|
cycles = 1;
|
|
return [Bus(Action.READ, pc)];
|
|
}
|
|
|
|
|
|
/// Bus access pattern for 1-cycle NOPs.
|
|
auto accesses_nop1(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T && isCMOS!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(NOP1_OPS!T.canFind(opcode));
|
|
|
|
cycles = 1;
|
|
return [Bus(Action.READ, pc)];
|
|
}
|
|
|
|
|
|
auto accesses_px(T)(T cpu, ubyte opcode, int opLen, ushort guess, ushort right,
|
|
ref int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
bool noShortcut = WRITE_OPS!T.canFind(opcode) ||
|
|
(isNMOS!T ? (RMW_OPS!T.canFind(opcode))
|
|
: (opcode == 0xDE || opcode == 0xFE));
|
|
|
|
if (guess != right)
|
|
{
|
|
cycles += 1;
|
|
return If!(isStrict!T)(
|
|
If!(isNMOS!T)([Bus(Action.READ, guess)]) ~
|
|
If!(isCMOS!T)([Bus(Action.READ, pc + opLen)])); // XXX
|
|
}
|
|
else if (noShortcut)
|
|
{
|
|
cycles += 1;
|
|
return If!(isStrict!T)([Bus(Action.READ, guess)]);
|
|
}
|
|
else
|
|
{
|
|
return cast(Bus[])[];
|
|
}
|
|
|
|
/*
|
|
* According to "Understanding the Apple IIe", the extra read on
|
|
* the 65C02 (marked XXX above) is the address of the last operand
|
|
* byte (pc + opLen - 1) for abx/aby, or the address of the high
|
|
* byte of the indirect address (op1 + 1) for izy.
|
|
*/
|
|
}
|
|
|
|
auto accesses_end(T)(T cpu, ubyte opcode, int opLen, ushort addr,
|
|
ref int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
bool rmw = RMW_OPS!T.canFind(opcode);
|
|
bool write = !rmw && WRITE_OPS!T.canFind(opcode);
|
|
bool read = !rmw && !write;
|
|
bool decimal = isCMOS!T && getFlag(cpu, Flag.D) &&
|
|
BCD_OPS!T.canFind(opcode);
|
|
|
|
cycles += (rmw ? 3 : (write ? 1 : (1 + decimal)));
|
|
return If!read(
|
|
[Bus(Action.READ, addr)] ~
|
|
If!decimal(If!(isStrict!T)(
|
|
[Bus(Action.READ, pc + opLen)]))) ~
|
|
If!write(
|
|
[Bus(Action.WRITE, addr)]) ~
|
|
If!rmw(
|
|
[Bus(Action.READ, addr)] ~
|
|
If!(isStrict!T)(
|
|
If!(isNMOS!T)(
|
|
[Bus(Action.WRITE, addr)]) ~
|
|
If!(isCMOS!T)(
|
|
[Bus(Action.READ, addr)])) ~
|
|
[Bus(Action.WRITE, addr)]);
|
|
}
|
|
|
|
|
|
/// Bus access pattern for RTS.
|
|
auto accesses_op_RTS(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(opcode == 0x60);
|
|
|
|
auto sp = getSP(cpu);
|
|
auto sp1 = pageWrapAdd(sp, 1);
|
|
auto sp2 = pageWrapAdd(sp, 2);
|
|
auto ret = address(mem[sp1], mem[sp2]);
|
|
|
|
cycles = 6;
|
|
return [Bus(Action.READ, pc)] ~
|
|
If!(isStrict!T)(
|
|
[Bus(Action.READ, pc+1),
|
|
Bus(Action.READ, sp)]) ~
|
|
[Bus(Action.READ, sp1),
|
|
Bus(Action.READ, sp2)] ~
|
|
If!(isStrict!T)(
|
|
[Bus(Action.READ, ret)]);
|
|
}
|
|
|
|
|
|
/// Bus access pattern for RTI.
|
|
auto accesses_op_RTI(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(opcode == 0x40);
|
|
|
|
auto sp = getSP(cpu);
|
|
auto sp1 = pageWrapAdd(sp, 1);
|
|
auto sp2 = pageWrapAdd(sp, 2);
|
|
auto sp3 = pageWrapAdd(sp, 3);
|
|
|
|
cycles = 6;
|
|
return [Bus(Action.READ, pc)] ~
|
|
If!(isStrict!T)(
|
|
[Bus(Action.READ, pc+1),
|
|
Bus(Action.READ, sp)]) ~
|
|
[Bus(Action.READ, sp1),
|
|
Bus(Action.READ, sp2),
|
|
Bus(Action.READ, sp3)];
|
|
}
|
|
|
|
|
|
/// Bus access pattern for BRK.
|
|
auto accesses_op_BRK(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(opcode == 0x00);
|
|
|
|
auto sp = getSP(cpu);
|
|
auto sp1 = pageWrapAdd(sp, -1);
|
|
auto sp2 = pageWrapAdd(sp, -2);
|
|
|
|
cycles = 7;
|
|
return [Bus(Action.READ, pc)] ~
|
|
If!(isStrict!T)(
|
|
[Bus(Action.READ, pc+1)]) ~
|
|
[Bus(Action.WRITE, sp),
|
|
Bus(Action.WRITE, sp1),
|
|
Bus(Action.WRITE, sp2),
|
|
Bus(Action.READ, 0xFFFE),
|
|
Bus(Action.READ, 0xFFFF)];
|
|
}
|
|
|
|
|
|
/// Bus access pattern for JSR
|
|
auto accesses_op_JSR(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(opcode == 0x20);
|
|
|
|
auto sp = getSP(cpu);
|
|
auto sp1 = pageWrapAdd(sp, -1);
|
|
|
|
cycles = 6;
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1)] ~
|
|
If!(isStrict!T)(
|
|
[Bus(Action.READ, sp)]) ~
|
|
[Bus(Action.WRITE, sp),
|
|
Bus(Action.WRITE, sp1),
|
|
Bus(Action.READ, pc+2)];
|
|
}
|
|
|
|
|
|
/// Bus access pattern for JMP absolute
|
|
auto accesses_op_JMP_abs(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(opcode == 0x4C);
|
|
|
|
cycles = 3;
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1),
|
|
Bus(Action.READ, pc+2)];
|
|
}
|
|
|
|
|
|
/// Bus access pattern for JMP indirect
|
|
auto accesses_op_JMP_ind(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc], op1 = mem[pc+1], op2 = mem[pc+2];
|
|
assert(opcode == 0x6C);
|
|
|
|
auto ial = address(op1, op2);
|
|
auto iah = (isNMOS!T ? pageWrapAdd(ial, 1)
|
|
: pageCrossAdd(ial, 1));
|
|
|
|
cycles = 5 + isCMOS!T;
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1),
|
|
Bus(Action.READ, pc+2)] ~
|
|
If!(isStrict!T)(If!(isCMOS!T)(
|
|
[Bus(Action.READ, pc+3)])) ~ // XXX
|
|
[Bus(Action.READ, ial),
|
|
Bus(Action.READ, iah)];
|
|
|
|
/*
|
|
* According to "Understanding the Apple IIe", the extra read on
|
|
* the 65C02 (marked XXX above) is the address of the last operand
|
|
* byte (pc + 2).
|
|
*/
|
|
}
|
|
|
|
|
|
/// Bus access pattern for JMP indirect,x
|
|
auto accesses_op_JMP_inx(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T && isCMOS!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(opcode == 0x7C);
|
|
|
|
auto idx = getX(cpu);
|
|
auto base = address(mem[pc+1], mem[pc+2]);
|
|
auto ial = pageCrossAdd(base, idx);
|
|
auto iah = pageCrossAdd(ial, 1);
|
|
|
|
cycles = 6;
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1),
|
|
Bus(Action.READ, pc+2)] ~
|
|
If!(isStrict!T)(
|
|
[Bus(Action.READ, pc+3)]) ~ // XXX
|
|
[Bus(Action.READ, ial),
|
|
Bus(Action.READ, iah)];
|
|
|
|
/*
|
|
* According to "Understanding the Apple IIe", the extra read on
|
|
* the 65C02 (marked XXX above) is the address of the last operand
|
|
* byte (pc + 2).
|
|
*/
|
|
}
|
|
|
|
|
|
/// Bus access pattern for CMOS opcode 5C
|
|
auto accesses_op_5C(T)(T cpu, ref TestMemory mem, out int cycles)
|
|
if (isCpu!T && isCMOS!T)
|
|
{
|
|
auto pc = getPC(cpu);
|
|
auto opcode = mem[pc];
|
|
assert(opcode == 0x5C);
|
|
|
|
auto weird = address(mem[pc+1], 0xFF);
|
|
|
|
cycles = 8;
|
|
return [Bus(Action.READ, pc),
|
|
Bus(Action.READ, pc+1)] ~
|
|
If!(isStrict!T)(
|
|
[Bus(Action.READ, pc+2),
|
|
Bus(Action.READ, weird),
|
|
Bus(Action.READ, 0xFFFF),
|
|
Bus(Action.READ, 0xFFFF),
|
|
Bus(Action.READ, 0xFFFF),
|
|
Bus(Action.READ, 0xFFFF)]);
|
|
}
|
|
|
|
|
|
// Associates opcodes with expected access patterns.
|
|
string getExpected(T)()
|
|
{
|
|
string[] tmp = new string[256];
|
|
|
|
void add_op(const(ubyte[]) list, string fname)
|
|
{
|
|
foreach(op; list)
|
|
{
|
|
tmp[op] = " case 0x" ~ to!string(op, 16) ~ ": " ~
|
|
"expected = &" ~ fname ~ "!T; break;";
|
|
}
|
|
}
|
|
|
|
add_op(REG_OPS!T, "accesses_reg");
|
|
add_op(PUSH_OPS!T, "accesses_push");
|
|
add_op(PULL_OPS!T, "accesses_pull");
|
|
add_op(BRANCH_OPS!T, "accesses_rel");
|
|
add_op(IMM_OPS!T, "accesses_imm");
|
|
add_op(ZPG_OPS!T, "accesses_zpg");
|
|
add_op(ZPX_OPS!T, "accesses_zpxy");
|
|
add_op(ZPY_OPS!T, "accesses_zpxy");
|
|
add_op(ABS_OPS!T, "accesses_abs");
|
|
add_op(ABX_OPS!T, "accesses_abxy");
|
|
add_op(ABY_OPS!T, "accesses_abxy");
|
|
add_op(IZX_OPS!T, "accesses_izx");
|
|
add_op(IZY_OPS!T, "accesses_izy");
|
|
add_op([0x00], "accesses_op_BRK");
|
|
add_op([0x20], "accesses_op_JSR");
|
|
add_op([0x40], "accesses_op_RTI");
|
|
add_op([0x4C], "accesses_op_JMP_abs");
|
|
add_op([0x60], "accesses_op_RTS");
|
|
add_op([0x6C], "accesses_op_JMP_ind");
|
|
static if (isNMOS!T)
|
|
add_op(HLT_OPS!T, "accesses_hlt");
|
|
else
|
|
{
|
|
add_op(ZPI_OPS!T, "accesses_zpi");
|
|
add_op(NOP1_OPS!T, "accesses_nop1");
|
|
add_op([0x7C], "accesses_op_JMP_inx");
|
|
add_op([0x5C], "accesses_op_5C");
|
|
}
|
|
|
|
return "final switch (opcode)\n{\n" ~ join(tmp, "\n") ~ "\n}";
|
|
}
|
|
|
|
|
|
template timesetup_t(T)
|
|
{
|
|
alias Bus[] function(T, ref TestMemory, out int) timesetup_t;
|
|
}
|
|
|
|
|
|
alias void delegate(int, const Bus[], int, const Bus[], ubyte, string)
|
|
busreport;
|
|
|
|
|
|
auto run_timing_test(T)(timesetup_t!T expect, busreport 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);
|
|
|
|
int expCycles;
|
|
auto expBus = expect(testcpu, mem, expCycles);
|
|
expBus = expBus ~ new Bus[8 - expBus.length];
|
|
|
|
connectMem(testcpu, mem);
|
|
auto actualBus = recordBus(testcpu);
|
|
auto actualCycles = recordCycles(testcpu);
|
|
|
|
runOneOpcode(testcpu);
|
|
|
|
report(actualCycles, actualBus, expCycles, expBus,
|
|
opcode, T.stringof ~ " | " ~ msg);
|
|
callNext();
|
|
}
|
|
return TestSetup(&setup);
|
|
}
|
|
|
|
|
|
|
|
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
|
|
}
|
|
if (actualCycles != expectCycles)
|
|
{
|
|
// XXX make error message, throw
|
|
}
|
|
if (actualBus == expectBus && actualCycles == expectCycles) {}
|
|
else
|
|
{
|
|
write(format("[%0.2X] %s", opcode, msg));
|
|
writeln();
|
|
writeln(expectCycles, " ", actualCycles);
|
|
writeln(expectBus);
|
|
writeln(actualBus);
|
|
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);
|
|
}
|