Add opcodes to new cpu (BRK)

This commit is contained in:
edmccard 2012-04-08 16:25:35 -04:00
parent d3a95c455d
commit 04f05fbc9a

View File

@ -40,6 +40,17 @@ version(OpFunctions)
{
enum versionCheck = 2;
enum opArray = true;
// With free functions, strict and cumulative need to be set by
// version.
version(Strict)
enum vStrict = true;
else
enum vStrict = false;
version(Cumulative)
enum vCumulative = true;
else
enum vCumulative = false;
}
// OpSwitch: each opcode is inlined in a 256-case switch.
@ -105,7 +116,10 @@ final class Cpu(string chip, bool strict, bool cumulative)
ubyte N, Z;
bool V, D, I, C;
static if (opArray) mixin(OpArrayDef());
static if (opArray)
{
mixin(OpArrayDef());
}
// TODO: other methods for stopping cpu
bool keepRunning;
@ -122,7 +136,7 @@ final class Cpu(string chip, bool strict, bool cumulative)
D = ((p & 0x08) != 0);
I = ((p & 0x04) != 0);
Z = ((p & 0x02) ? 0 : 1);
C = ((val & 0x01) != 0);
C = ((p & 0x01) != 0);
}
final ubyte statusToByte()
@ -141,23 +155,33 @@ final class Cpu(string chip, bool strict, bool cumulative)
keepRunning = continuous;
// TODO debugging info?
ubyte opcode;
static if (!opArray)
{
static if (cumulative) int cycles;
}
do {
// XXX check signals
// XXX figure out cumulative/final cycle stuff
static if (cumulative) {}
else
static if (cumulative && !opArray)
cycles = 1;
// XXX figure out final cycle stuff
static if (!cumulative)
clock.tick();
// XXX check signals, NMI/IRQ delays, etc.
opcode = memory.read(PC++);
mixin(OpExecute(_chip));
mixin(OpExecute(_chip, strict, cumulative));
} while (keepRunning);
}
version(OpDelegates) mixin (OpBodies(_chip));
version(OpDelegates) mixin (OpBodies(_chip, strict, cumulative));
}
version(OpFunctions) mixin(OpBodies("6502"));
version(OpFunctions) mixin(OpBodies("65C02"));
enum ushort IRQ_VECTOR = 0xFFFE;
private:
version(OpFunctions) mixin(OpBodies("6502", vStrict, vCumulative));
version(OpFunctions) mixin(OpBodies("65C02", vStrict, vCumulative));
string OpArrayDef()
@ -189,7 +213,7 @@ string OpArrayInit()
}
}
string OpBodies(string chip)
string OpBodies(string chip, bool strict, bool cumulative)
{
static if (!opArray) return "";
else
@ -199,37 +223,40 @@ string OpBodies(string chip)
{
version(OpDelegates)
ret ~= "final void opcode_" ~ Hex2(op) ~ "()\n{\n" ~
OpBody(op, chip) ~ "}\n";
If!(cumulative)("int cycles = 1;\n") ~
OpBody(op, chip, strict, cumulative) ~ "}\n";
version(OpFunctions)
ret ~= "void opcode_" ~ Hex2(op) ~
"(T)(T cpu) if (is" ~ chip ~ "!T)\n{\n" ~
OpBody(op, chip) ~ "}\n";
If!(cumulative)("int cycles = 1;\n") ~
OpBody(op, chip, strict, cumulative) ~ "}\n";
}
return ret;
}
}
string OpExecute(string chip)
string OpExecute(string chip, bool strict, bool cumulative)
{
version(OpDelegates)
return q{opcodes[opcode]();};
version(OpFunctions)
return q{opcodes[opcode](this);};
version(OpSwitch)
return Switch256(chip);
return Switch256(chip, strict, cumulative);
version(OpNestedSwitch)
return Switch16x16(chip);
return Switch16x16(chip, strict, cumulative);
}
string Switch256(string chip)
string Switch256(string chip, bool strict, bool cumulative)
{
string ret = "final switch (opcode)\n{\n";
foreach (op; 0..256)
ret ~= "case 0x" ~ Hex2(op) ~ ":\n" ~ OpBody(op, chip) ~ "break;\n\n";
ret ~= "case 0x" ~ Hex2(op) ~ ":\n" ~
OpBody(op, chip, strict, cumulative) ~ "break;\n";
return ret ~ "}\n";
}
string Switch16x16(string chip)
string Switch16x16(string chip, bool strict, bool cumulative)
{
string ret = "final switch (opcode & 0xF0)\n{\n";
foreach (opHi; 0..16)
@ -239,8 +266,9 @@ string Switch16x16(string chip)
foreach (opLo; 0..16)
{
int op = opLo | (opHi << 4);
ret ~= "case 0x0" ~ Hex1(opLo) ~ ":\n" ~ OpBody(op, chip) ~
"break;\n\n";
ret ~= "case 0x0" ~ Hex1(opLo) ~ ":\n" ~
OpBody(op, chip, strict, cumulative) ~
"break;\n";
}
ret ~= "}\nbreak;\n";
}
@ -248,12 +276,509 @@ string Switch16x16(string chip)
}
string OpBody(int op, string chip)
string OpBody(int op, string chip, bool strict, bool cumulative)
{
return "";
final switch (opName(op, chip))
{
case "BRK":
return Break(strict, cumulative);
case "RTI":
return "";
case "JSR":
return "";
case "RTS":
return "";
case "JMP":
return ""; // address modes
case "KIL":
return "";
case "BPL":
return "";
case "BMI":
return "";
case "BVC":
return "";
case "BVS":
return "";
case "BRA":
return "";
case "BCC":
return "";
case "BCS":
return "";
case "BNE":
return "";
case "BEQ":
return "";
case "CLC":
return "";
case "SEC":
return "";
case "CLI":
return "";
case "SEI":
return "";
case "CLV":
return "";
case "CLD":
return "";
case "SED":
return "";
case "NOP":
return ""; // address modes
case "TAX":
return "";
case "TXA":
return "";
case "TAY":
return "";
case "TYA":
return "";
case "TSX":
return "";
case "TXS":
return "";
case "DEX":
return "";
case "DEY":
return "";
case "INX":
return "";
case "INY":
return "";
case "PLP":
return "";
case "PLA":
return "";
case "PLX":
return "";
case "PLY":
return "";
case "PHP":
return "";
case "PHA":
return "";
case "PHX":
return "";
case "PHY":
return "";
case "LDA":
return "";
case "LDX":
return "";
case "LDY":
return "";
case "STA":
return "";
case "STX":
return "";
case "STY":
return "";
case "STZ":
return "";
case "BIT":
return ""; // address modes
case "CMP":
return "";
case "CPX":
return "";
case "CPY":
return "";
case "ORA":
return "";
case "AND":
return "";
case "EOR":
return "";
case "ADC":
return ""; // n/c (op, cyc)
case "SBC":
return ""; // n/c (op, cyc)
case "ASL":
return ""; // n/c (op, cyc)
case "ROL":
return ""; // n/c (op, cyc)
case "LSR":
return ""; // n/c (op, cyc)
case "ROR":
return ""; // n/c (op, cyc)
case "INC":
return ""; // n/c (op, +ina)
case "DEC":
return ""; // n/c (op, +dea)
case "TRB":
return "";
case "TSB":
return "";
case "LAS":
return "";
case "LAX":
return ""; // address modes
case "SAX":
return "";
case "ANC":
return "";
case "ALR":
return "";
case "ARR":
return "";
case "AXS":
return "";
case "AHX":
return "";
case "SHY":
return "";
case "SHX":
return "";
case "TAS":
return "";
case "XAA":
return "";
case "SLO":
return "";
case "RLA":
return "";
case "SRE":
return "";
case "RRA":
return "";
case "DCP":
return "";
case "ISC":
return "";
}
}
string Break(bool s, bool c)
{
return Peek(Attr("PC"), s, c) ~
IncPC() ~
PushPC(s, c) ~
Push(Attr("statusToByte()"), s, c) ~
Set(Attr("I")) ~
ReadWord(Attr("PC"), "IRQ_VECTOR", c) ~
Done(c);
}
string PreAccess(bool cumulative)
{
return If!(cumulative)("++cycles;\n", Attr("clock") ~ ".tick();\n");
}
string Peek(string addr, bool strict, bool cumulative)
{
return PreAccess(cumulative) ~
If!(strict)(Attr("memory") ~ ".read(" ~ addr ~");\n");
}
string Read(string var, string addr, bool c)
{
return PreAccess(c) ~
var ~ " = " ~ Attr("memory") ~ ".read(" ~ addr ~");\n";
}
string Write(string addr, string val, bool cumulative)
{
return If!(cumulative)("++cycles;\n", Attr("clock") ~ ".tick();\n") ~
Attr("memory") ~ ".write(" ~ addr ~ ", " ~ val ~ ");\n";
}
string ReadWord(string var, string addr, bool c)
{
return PreAccess(c) ~
var ~ " = " ~ Attr("memory") ~ ".read(" ~ addr ~");\n" ~
PreAccess(c) ~
var ~ " |= (" ~ Attr("memory") ~
".read(cast(ushort)((" ~ addr ~ ") + 1)) << 8);\n";
}
string IncPC()
{
return "++" ~ Attr("PC") ~ ";\n";
}
string IncSP()
{
return "++" ~ Attr("S") ~ ";\n";
}
string DecSP()
{
return "--" ~ Attr("S") ~ ";\n";
}
string Push(string val, bool s, bool c)
{
return Write("0x0100 + " ~ Attr("S"), val, c) ~
DecSP();
}
string PushPC(bool s, bool c)
{
return Push(HiByte(Attr("PC")), s, c) ~
Push(LoByte(Attr("PC")), s, c);
}
string Set(string flag)
{
return flag ~ " = true;\n";
}
string Done(bool cumulative)
{
return If!(cumulative)(Attr("clock") ~ ".tick(cycles);\n");
}
string Attr(string var)
{
version(OpFunctions)
return "cpu." ~ var;
else
return var;
}
string HiByte(string var)
{
return var ~ " >> 8";
}
string LoByte(string var)
{
return var ~ " & 0xff";
}
string If(alias cond)(string yes, string no = "")
{
if (cond)
return yes;
else
return no;
}
string opName(int op, string chip)
{
if (chip == "6502")
return OP_NAMES_6502[op];
else
return OP_NAMES_65C02[op];
}
int opMode(int op, string chip)
{
if (chip == "6502")
return ADDR_MODES_6502[op];
else
return ADDR_MODES_65C02[op];
}
int opExCyc(int op, string chip)
{
if (chip == "6502")
return EXTRA_CYCLES_6502[op];
else
return EXTRA_CYCLES_65C02[op];
}
// Opcode names.
immutable OP_NAMES_6502 = [
"BRK", "ORA", "KIL", "SLO", "NOP", "ORA", "ASL", "SLO",
"PHP", "ORA", "ASL", "ANC", "NOP", "ORA", "ASL", "SLO",
"BPL", "ORA", "KIL", "SLO", "NOP", "ORA", "ASL", "SLO",
"CLC", "ORA", "NOP", "SLO", "NOP", "ORA", "ASL", "SLO",
"JSR", "AND", "KIL", "RLA", "BIT", "AND", "ROL", "RLA",
"PLP", "AND", "ROL", "ANC", "BIT", "AND", "ROL", "RLA",
"BMI", "AND", "KIL", "RLA", "NOP", "AND", "ROL", "RLA",
"SEC", "AND", "NOP", "RLA", "NOP", "AND", "ROL", "RLA",
"RTI", "EOR", "KIL", "SRE", "NOP", "EOR", "LSR", "SRE",
"PHA", "EOR", "LSR", "ALR", "JMP", "EOR", "LSR", "SRE",
"BVC", "EOR", "KIL", "SRE", "NOP", "EOR", "LSR", "SRE",
"CLI", "EOR", "NOP", "SRE", "NOP", "EOR", "LSR", "SRE",
"RTS", "ADC", "KIL", "RRA", "NOP", "ADC", "ROR", "RRA",
"PLA", "ADC", "ROR", "ARR", "JMP", "ADC", "ROR", "RRA",
"BVS", "ADC", "KIL", "RRA", "NOP", "ADC", "ROR", "RRA",
"SEI", "ADC", "NOP", "RRA", "NOP", "ADC", "ROR", "RRA",
"NOP", "STA", "NOP", "SAX", "STY", "STA", "STX", "SAX",
"DEY", "NOP", "TXA", "XAA", "STY", "STA", "STX", "SAX",
"BCC", "STA", "KIL", "AHX", "STY", "STA", "STX", "SAX",
"TYA", "STA", "TXS", "TAS", "SHY", "STA", "SHX", "AHX",
"LDY", "LDA", "LDX", "LAX", "LDY", "LDA", "LDX", "LAX",
"TAY", "LDA", "TAX", "LAX", "LDY", "LDA", "LDX", "LAX",
"BCS", "LDA", "KIL", "LAX", "LDY", "LDA", "LDX", "LAX",
"CLV", "LDA", "TSX", "LAS", "LDY", "LDA", "LDX", "LAX",
"CPY", "CMP", "NOP", "DCP", "CPY", "CMP", "DEC", "DCP",
"INY", "CMP", "DEX", "AXS", "CPY", "CMP", "DEC", "DCP",
"BNE", "CMP", "KIL", "DCP", "NOP", "CMP", "DEC", "DCP",
"CLD", "CMP", "NOP", "DCP", "NOP", "CMP", "DEC", "DCP",
"CPX", "SBC", "NOP", "ISC", "CPX", "SBC", "INC", "ISC",
"INX", "SBC", "NOP", "SBC", "CPX", "SBC", "INC", "ISC",
"BEQ", "SBC", "KIL", "ISC", "NOP", "SBC", "INC", "ISC",
"SED", "SBC", "NOP", "ISC", "NOP", "SBC", "INC", "ISC"
];
immutable OP_NAMES_65C02 = [
"BRK", "ORA", "NOP", "NOP", "TSB", "ORA", "ASL", "NOP",
"PHP", "ORA", "ASL", "NOP", "TSB", "ORA", "ASL", "NOP",
"BPL", "ORA", "ORA", "NOP", "TRB", "ORA", "ASL", "NOP",
"CLC", "ORA", "INC", "NOP", "TRB", "ORA", "ASL", "NOP",
"JSR", "AND", "NOP", "NOP", "BIT", "AND", "ROL", "NOP",
"PLP", "AND", "ROL", "NOP", "BIT", "AND", "ROL", "NOP",
"BMI", "AND", "AND", "NOP", "BIT", "AND", "ROL", "NOP",
"SEC", "AND", "DEC", "NOP", "BIT", "AND", "ROL", "NOP",
"RTI", "EOR", "NOP", "NOP", "NOP", "EOR", "LSR", "NOP",
"PHA", "EOR", "LSR", "NOP", "JMP", "EOR", "LSR", "NOP",
"BVC", "EOR", "EOR", "NOP", "NOP", "EOR", "LSR", "NOP",
"CLI", "EOR", "PHY", "NOP", "NOP", "EOR", "LSR", "NOP",
"RTS", "ADC", "NOP", "NOP", "STZ", "ADC", "ROR", "NOP",
"PLA", "ADC", "ROR", "NOP", "JMP", "ADC", "ROR", "NOP",
"BVS", "ADC", "ADC", "NOP", "STZ", "ADC", "ROR", "NOP",
"SEI", "ADC", "PLY", "NOP", "JMP", "ADC", "ROR", "NOP",
"BRA", "STA", "NOP", "NOP", "STY", "STA", "STX", "NOP",
"DEY", "BIT", "TXA", "NOP", "STY", "STA", "STX", "NOP",
"BCC", "STA", "STA", "NOP", "STY", "STA", "STX", "NOP",
"TYA", "STA", "TXS", "NOP", "STZ", "STA", "STZ", "NOP",
"LDY", "LDA", "LDX", "NOP", "LDY", "LDA", "LDX", "NOP",
"TAY", "LDA", "TAX", "NOP", "LDY", "LDA", "LDX", "NOP",
"BCS", "LDA", "LDA", "NOP", "LDY", "LDA", "LDX", "NOP",
"CLV", "LDA", "TSX", "NOP", "LDY", "LDA", "LDX", "NOP",
"CPY", "CMP", "NOP", "NOP", "CPY", "CMP", "DEC", "NOP",
"INY", "CMP", "DEX", "NOP", "CPY", "CMP", "DEC", "NOP",
"BNE", "CMP", "CMP", "NOP", "NOP", "CMP", "DEC", "NOP",
"CLD", "CMP", "PHX", "NOP", "NOP", "CMP", "DEC", "NOP",
"CPX", "SBC", "NOP", "NOP", "CPX", "SBC", "INC", "NOP",
"INX", "SBC", "NOP", "NOP", "CPX", "SBC", "INC", "NOP",
"BEQ", "SBC", "SBC", "NOP", "NOP", "SBC", "INC", "NOP",
"SED", "SBC", "PLX", "NOP", "NOP", "SBC", "INC", "NOP"
];
// Addressing modes.
enum { IMP, IMM, ZP, ZPX, ZPY, IZX, IZY, ABS, ABX, ABY, IND, REL,
ZPI, ABI, NP1, NP8, KIL }
immutable ADDR_MODES_6502 = [
IMP, IZX, KIL, IZX, ZP, ZP, ZP, ZP,
IMP, IMM, IMP, IMM, ABS, ABS, ABS, ABS,
REL, IZY, KIL, IZY, ZPX, ZPX, ZPX, ZPX,
IMP, ABY, IMP, ABY, ABX, ABX, ABX, ABX,
ABS, IZX, KIL, IZX, ZP, ZP, ZP, ZP,
IMP, IMM, IMP, IMM, ABS, ABS, ABS, ABS,
REL, IZY, KIL, IZY, ZPX, ZPX, ZPX, ZPX,
IMP, ABY, IMP, ABY, ABX, ABX, ABX, ABX,
IMP, IZX, KIL, IZX, ZP, ZP, ZP, ZP,
IMP, IMM, IMP, IMM, ABS, ABS, ABS, ABS,
REL, IZY, KIL, IZY, ZPX, ZPX, ZPX, ZPX,
IMP, ABY, IMP, ABY, ABX, ABX, ABX, ABX,
IMP, IZX, KIL, IZX, ZP, ZP, ZP, ZP,
IMP, IMM, IMP, IMM, IND, ABS, ABS, ABS,
REL, IZY, KIL, IZY, ZPX, ZPX, ZPX, ZPX,
IMP, ABY, IMP, ABY, ABX, ABX, ABX, ABX,
IMM, IZX, IMM, IZX, ZP, ZP, ZP, ZP,
IMP, IMM, IMP, IMM, ABS, ABS, ABS, ABS,
REL, IZY, KIL, IZY, ZPX, ZPX, ZPY, ZPY,
IMP, ABY, IMP, ABY, ABX, ABX, ABY, ABY,
IMM, IZX, IMM, IZX, ZP, ZP, ZP, ZP,
IMP, IMM, IMP, IMM, ABS, ABS, ABS, ABS,
REL, IZY, KIL, IZY, ZPX, ZPX, ZPY, ZPY,
IMP, ABY, IMP, ABY, ABX, ABX, ABY, ABY,
IMM, IZX, IMM, IZX, ZP, ZP, ZP, ZP,
IMP, IMM, IMP, IMM, ABS, ABS, ABS, ABS,
REL, IZY, KIL, IZY, ZPX, ZPX, ZPX, ZPX,
IMP, ABY, IMP, ABY, ABX, ABX, ABX, ABX,
IMM, IZX, IMM, IZX, ZP, ZP, ZP, ZP,
IMP, IMM, IMP, IMM, ABS, ABS, ABS, ABS,
REL, IZY, KIL, IZY, ZPX, ZPX, ZPX, ZPX,
IMP, ABY, IMP, ABY, ABX, ABX, ABX, ABX
];
immutable ADDR_MODES_65C02 = [
IMP, IZX, IMM, NP1, ZP, ZP, ZP, NP1,
IMP, IMM, IMP, NP1, ABS, ABS, ABS, NP1,
REL, IZY, ZPI, NP1, ZP, ZPX, ZPX, NP1,
IMP, ABY, IMP, NP1, ABS, ABX, ABX, NP1,
ABS, IZX, IMM, NP1, ZP, ZP, ZP, NP1,
IMP, IMM, IMP, NP1, ABS, ABS, ABS, NP1,
REL, IZY, ZPI, NP1, ZPX, ZPX, ZPX, NP1,
IMP, ABY, IMP, NP1, ABX, ABX, ABX, NP1,
IMP, IZX, IMM, NP1, ZP, ZP, ZP, NP1,
IMP, IMM, IMP, NP1, ABS, ABS, ABS, NP1,
REL, IZY, ZPI, NP1, ZPX, ZPX, ZPX, NP1,
IMP, ABY, IMP, NP1, NP8, ABX, ABX, NP1,
IMP, IZX, IMM, NP1, ZP, ZP, ZP, NP1,
IMP, IMM, IMP, NP1, IND, ABS, ABS, NP1,
REL, IZY, ZPI, NP1, ZPX, ZPX, ZPX, NP1,
IMP, ABY, IMP, NP1, ABI, ABX, ABX, NP1,
REL, IZX, IMM, NP1, ZP, ZP, ZP, NP1,
IMP, IMM, IMP, NP1, ABS, ABS, ABS, NP1,
REL, IZY, ZPI, NP1, ZPX, ZPX, ZPY, NP1,
IMP, ABY, IMP, NP1, ABX, ABX, ABX, NP1,
IMM, IZX, IMM, NP1, ZP, ZP, ZP, NP1,
IMP, IMM, IMP, NP1, ABS, ABS, ABS, NP1,
REL, IZY, ZPI, NP1, ZPX, ZPX, ZPY, NP1,
IMP, ABY, IMP, NP1, ABX, ABX, ABY, NP1,
IMM, IZX, IMM, NP1, ZP, ZP, ZP, NP1,
IMP, IMM, IMP, NP1, ABS, ABS, ABS, NP1,
REL, IZY, ZPI, NP1, ZPX, ZPX, ZPX, NP1,
IMP, ABY, IMP, NP1, ABX, ABX, ABX, NP1,
IMM, IZX, IMM, NP1, ZP, ZP, ZP, NP1,
IMP, IMM, IMP, NP1, ABS, ABS, ABS, NP1,
REL, IZY, ZPI, NP1, ZPX, ZPX, ZPX, NP1,
IMP, ABY, IMP, NP1, ABX, ABX, ABX, NP1
];
// Page-crossing extra cycles.
immutable EXTRA_CYCLES_6502 = [
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 1, 0, 1, 0, 0, 0, 0, 0, 1, 0, 1, 1, 1, 1, 1,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1,
];
immutable EXTRA_CYCLES_65C02 = [
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 1, 1, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0,
];
// Custom hex printing.
// (to!string(x, 16) uses uppercase, which makes "8" and "B" hard to
// tell apart, and format("%0.2x", x) can't be used in CTFE.)
@ -273,19 +798,3 @@ string Hex2(int dec)
return HEX_DIGITS[highNybble..highNybble+1] ~
HEX_DIGITS[lowNybble..lowNybble+1];
}
alias Cpu!("6502", false, false) T1;
alias Cpu!("6502", false, true) T2;
alias Cpu!("6502", true, false) T3;
alias Cpu!("6502", true, true) T4;
alias Cpu!("65C02", false, false) T5;
alias Cpu!("65C02", false, true) T6;
alias Cpu!("65C02", true, false) T7;
alias Cpu!("65C02", true, true) T8;
void main()
{
import std.stdio;
writeln(Switch16x16("6502"));
}