auto HuC6280::disassemble(uint16 pc) -> string { uint8 bank = r.mpr[pc.bits(13,15)]; uint13 addr = pc.bits(0,12); string s{hex(bank, 2L), ":", hex(addr, 4L), " "}; auto readByte = [&]() -> uint8 { return read(bank, addr++); }; auto readWord = [&]() -> uint16 { uint16 data = read(bank, addr++) << 0; return data | read(bank, addr++) << 8; }; auto absolute = [&]() -> string { return {"$", hex(readWord(), 4L)}; }; auto absoluteX = [&]() -> string { return {"$", hex(readWord(), 4L), ",x"}; }; auto absoluteY = [&]() -> string { return {"$", hex(readWord(), 4L), ",y"}; }; auto immediate = [&]() -> string { return {"#$", hex(readByte(), 2L)}; }; auto indirect = [&]() -> string { return {"($", hex(readByte(), 2L), ")"}; }; auto indirectX = [&]() -> string { return {"($", hex(readByte(), 2L), ",x)"}; }; auto indirectY = [&]() -> string { return {"($", hex(readByte(), 2L), "),y"}; }; auto indirectLong = [&]() -> string { return {"($", hex(readWord(), 4L), ")"}; }; auto indirectLongX = [&]() -> string { return {"($", hex(readWord(), 4L), ",x)"}; }; auto memory = [&]() -> string { return {"(x),#$", hex(readByte(), 2L)}; }; auto relative = [&]() -> string { auto displacement = readByte(); return {"$", hex(pc + (int8)displacement, 4L)}; }; auto zeropage = [&]() -> string { return {"$", hex(readByte(), 2L)}; }; auto zeropageX = [&]() -> string { return {"$", hex(readByte(), 2L), ",x"}; }; auto zeropageY = [&]() -> string { return {"$", hex(readByte(), 2L), ",y"}; }; auto immediateAbsolute = [&](string index = "") -> string { auto immediate = readByte(); auto absolute = readWord(); if(index) index.prepend(","); return {"#$", hex(immediate, 2L), ",$", hex(absolute, 4L), index}; }; auto immediateZeropage = [&](string index = "") -> string { auto immediate = readByte(); auto zeropage = readByte(); if(index) index.prepend(","); return {"#$", hex(immediate, 2L), ",$", hex(zeropage, 2L), index}; }; auto zeropageBit = [&](uint3 bit) -> string { auto zeropage = readByte(); return {"$", hex(zeropage, 2L), ":", bit}; }; auto zeropageBitRelative = [&](uint3 bit) -> string { auto zeropage = readByte(); auto displacement = readByte(); return {"$", hex(zeropage, 2L), ":", bit, ",$", hex(pc + (int8)displacement, 4L)}; }; auto blockMove = [&]() -> string { auto source = readWord(); auto target = readWord(); auto length = readWord(); return {"$", hex(source, 4L), ",$", hex(target, 4L), ",$", hex(length, 4L)}; }; uint8 opcode = readByte(); #define op(id, name, ...) case id: o = {name, " ", string_vector{__VA_ARGS__}.merge(",")}; break; string o; if(T == 1) switch(opcode) { op(0x09, "ora", memory()) op(0x29, "and", memory()) op(0x49, "eor", memory()) op(0x69, "adc", memory()) } #define U if(T == 0) switch(opcode) { op(0x00, "brk") op(0x01, "ora", indirectX()) op(0x02, "sxy") op(0x03, "st0", immediate()) op(0x04, "tsb", zeropage()) op(0x05, "ora", zeropage()) op(0x06, "asl", zeropage()) op(0x07, "rmb", zeropageBit(0)) op(0x08, "php") op(0x09, "ora", immediate()) op(0x0a, "asl") U op(0x0b, "nop", "$0b") op(0x0c, "tsb", absolute()) op(0x0d, "ora", absolute()) op(0x0e, "asl", absolute()) op(0x0f, "bbr", zeropageBitRelative(0)) op(0x10, "bpl", relative()) op(0x11, "ora", indirectY()) op(0x12, "ora", indirect()) op(0x13, "st1", immediate()) op(0x14, "trb", zeropage()) op(0x15, "ora", zeropageX()) op(0x16, "asl", zeropageX()) op(0x17, "rmb", zeropageBit(1)) op(0x18, "clc") op(0x19, "ora", absoluteY()) op(0x1a, "inc") U op(0x1b, "nop", "$1b") op(0x1c, "trb", absolute()) op(0x1d, "ora", absoluteX()) op(0x1e, "asl", absoluteX()) op(0x1f, "bbr", zeropageBitRelative(1)) op(0x20, "jsr", absolute()) op(0x21, "and", indirectX()) op(0x22, "sax") op(0x23, "st2", immediate()) op(0x24, "bit", zeropage()) op(0x25, "and", zeropage()) op(0x26, "rol", zeropage()) op(0x27, "rmb", zeropageBit(2)) op(0x28, "plp") op(0x29, "and", immediate()) op(0x2a, "rol") U op(0x2b, "nop", "$2b") op(0x2c, "bit", absolute()) op(0x2d, "and", absolute()) op(0x2e, "rol", absolute()) op(0x2f, "bbr", zeropageBitRelative(2)) op(0x30, "bmi", relative()) op(0x31, "and", indirectY()) op(0x32, "and", indirect()) op(0x34, "bit", zeropageX()) op(0x35, "and", zeropageX()) op(0x36, "rol", zeropageX()) op(0x37, "rmb", zeropageBit(3)) op(0x38, "sec") op(0x39, "and", absoluteY()) op(0x3a, "dec") U op(0x3b, "nop", "$3b") op(0x3c, "bit", absoluteX()) op(0x3d, "and", absoluteX()) op(0x3e, "rol", absoluteX()) op(0x3f, "bbr", zeropageBitRelative(3)) op(0x40, "rti") op(0x41, "eor", indirectX()) op(0x42, "say") op(0x43, "tma", immediate()) op(0x44, "bsr", relative()) op(0x45, "eor", zeropage()) op(0x46, "lsr", zeropage()) op(0x47, "rmb", zeropageBit(4)) op(0x48, "pha") op(0x49, "eor", immediate()) op(0x4a, "lsr") U op(0x4b, "nop", "$4b") op(0x4c, "jmp", absolute()) op(0x4d, "eor", absolute()) op(0x4e, "lsr", absolute()) op(0x4f, "bbr", zeropageBitRelative(4)) op(0x50, "bvc", relative()) op(0x51, "eor", indirectY()) op(0x52, "eor", indirect()) op(0x53, "tam", immediate()) op(0x54, "csl") op(0x55, "eor", zeropageX()) op(0x56, "lsr", zeropageX()) op(0x57, "rmb", zeropageBit(5)) op(0x58, "cli") op(0x59, "eor", absoluteY()) op(0x5a, "phy") U op(0x5b, "nop", "$5b") U op(0x5c, "nop", "$5c") op(0x5d, "eor", absoluteX()) op(0x5e, "lsr", absoluteX()) op(0x5f, "bbr", zeropageBitRelative(5)) op(0x60, "rts") op(0x61, "adc", indirectX()) op(0x62, "cla") U op(0x63, "nop", "$63") op(0x64, "stz", zeropage()) op(0x65, "adc", zeropage()) op(0x66, "ror", zeropage()) op(0x67, "rmb", zeropageBit(6)) op(0x68, "pla") op(0x69, "adc", immediate()) op(0x6a, "ror") U op(0x6b, "nop", "$6b") op(0x6c, "jmp", indirectLong()) op(0x6d, "adc", absolute()) op(0x6e, "ror", absolute()) op(0x6f, "bbr", zeropageBitRelative(6)) op(0x70, "bvs", relative()) op(0x71, "adc", indirectY()) op(0x72, "adc", indirect()) op(0x73, "tii", blockMove()) op(0x74, "stz", zeropageX()) op(0x75, "adc", zeropageX()) op(0x76, "ror", zeropageX()) op(0x77, "rmb", zeropageBit(7)) op(0x78, "sei") op(0x79, "adc", absoluteY()) op(0x7a, "ply") U op(0x7b, "nop", "$7b") op(0x7c, "jmp", indirectLongX()) op(0x7d, "adc", absoluteX()) op(0x7e, "ror", absoluteX()) op(0x7f, "bbr", zeropageBitRelative(7)) op(0x80, "bra", relative()) op(0x81, "sta", indirectX()) op(0x82, "clx") op(0x83, "tst", immediateZeropage()) op(0x84, "sty", zeropage()) op(0x85, "sta", zeropage()) op(0x86, "stx", zeropage()) op(0x87, "smb", zeropageBit(0)) op(0x88, "dey") op(0x89, "bit", immediate()) op(0x8a, "txa") U op(0x8b, "nop", "$8b") op(0x8c, "sty", absolute()) op(0x8d, "sta", absolute()) op(0x8e, "stx", absolute()) op(0x8f, "bbs", zeropageBitRelative(0)) op(0x90, "bcc", relative()) op(0x91, "sta", indirectY()) op(0x92, "sta", indirect()) op(0x93, "tst", immediateAbsolute()) op(0x94, "sty", zeropageX()) op(0x95, "sta", zeropageX()) op(0x96, "stx", zeropageY()) op(0x97, "smb", zeropageBit(1)) op(0x98, "tya") op(0x99, "sta", absoluteY()) op(0x9a, "txs") U op(0x9b, "nop", "$9b") op(0x9c, "stz", absolute()) op(0x9d, "sta", absoluteX()) op(0x9e, "stz", absoluteX()) op(0x9f, "bbs", zeropageBitRelative(1)) op(0xa0, "ldy", immediate()) op(0xa1, "lda", indirectX()) op(0xa2, "ldx", immediate()) op(0xa3, "tst", immediateZeropage("x")) op(0xa4, "ldy", zeropage()) op(0xa5, "lda", zeropage()) op(0xa6, "ldx", zeropage()) op(0xa7, "smb", zeropageBit(2)) op(0xa8, "tay") op(0xa9, "lda", immediate()) op(0xaa, "tax") U op(0xab, "nop", "$ab") op(0xac, "ldy", absolute()) op(0xad, "lda", absolute()) op(0xae, "ldx", absolute()) op(0xaf, "bbs", zeropageBitRelative(2)) op(0xb0, "bcs", relative()) op(0xb1, "lda", indirectY()) op(0xb2, "lda", indirect()) op(0xb3, "tst", immediateAbsolute("x")) op(0xb4, "ldy", zeropageX()) op(0xb5, "lda", zeropageX()) op(0xb6, "ldx", zeropageY()) op(0xb7, "smb", zeropageBit(3)) op(0xb8, "clv") op(0xb9, "lda", absoluteY()) op(0xba, "tsx") U op(0xbb, "nop", "$bb") op(0xbc, "ldy", absoluteX()) op(0xbd, "lda", absoluteX()) op(0xbe, "ldx", absoluteY()) op(0xbf, "bbs", zeropageBitRelative(3)) op(0xc0, "cpy", immediate()) op(0xc1, "cmp", indirectX()) op(0xc2, "cly") op(0xc3, "tdd", blockMove()) op(0xc4, "cpy", zeropage()) op(0xc5, "cmp", zeropage()) op(0xc6, "dec", zeropage()) op(0xc7, "smb", zeropageBit(4)) op(0xc8, "iny") op(0xc9, "cmp", immediate()) op(0xca, "dex") U op(0xcb, "nop", "$cb") op(0xcc, "cpy", absolute()) op(0xcd, "cmp", absolute()) op(0xce, "dec", absolute()) op(0xcf, "bbs", zeropageBitRelative(4)) op(0xd0, "bne", relative()) op(0xd1, "cmp", indirectY()) op(0xd2, "cmp", indirect()) op(0xd3, "tin", blockMove()) op(0xd4, "csh") op(0xd5, "cmp", zeropageX()) op(0xd6, "dec", zeropageX()) op(0xd7, "smb", zeropageBit(5)) op(0xd8, "cld") op(0xd9, "cmp", absoluteY()) op(0xda, "phx") U op(0xdb, "nop", "$db") U op(0xdc, "nop", "$dc") op(0xdd, "cmp", absoluteX()) op(0xde, "dec", absoluteX()) op(0xdf, "bbs", zeropageBitRelative(5)) op(0xe0, "cpx", immediate()) op(0xe1, "sbc", indirectX()) U op(0xe2, "nop", "$e2") op(0xe3, "tia", blockMove()) op(0xe4, "cpx", zeropage()) op(0xe5, "sbc", zeropage()) op(0xe6, "inc", zeropage()) op(0xe7, "smb", zeropageBit(6)) op(0xe8, "inx") op(0xe9, "sbc", immediate()) op(0xea, "nop") U op(0xeb, "nop", "$eb") op(0xec, "cpx", absolute()) op(0xed, "sbc", absolute()) op(0xee, "inc", absolute()) op(0xef, "bbs", zeropageBitRelative(6)) op(0xf0, "beq", relative()) op(0xf1, "sbc", indirectY()) op(0xf2, "sbc", indirect()) op(0xf3, "tai", blockMove()) op(0xf4, "set") op(0xf5, "sbc", zeropageX()) op(0xf6, "inc", zeropageX()) op(0xf7, "smb", zeropageBit(7)) op(0xf8, "sed") op(0xf9, "sbc", absoluteY()) op(0xfa, "plx") U op(0xfb, "nop", "$fb") U op(0xfc, "nop", "$fc") op(0xfd, "sbc", absoluteX()) op(0xfe, "inc", absoluteX()) op(0xff, "bbs", zeropageBitRelative(7)) } #undef U if(!o) o = {"??? (", hex(opcode, 2L), ")"}; s.append(pad(o, -22L, ' ')); #undef op s.append(" A:", hex(r.a, 2L)); s.append(" X:", hex(r.x, 2L)); s.append(" Y:", hex(r.y, 2L)); s.append(" S:", hex(r.s, 2L)); s.append(" PC:", hex(pc, 4L)); s.append(" "); s.append(r.p.n ? "N" : "n"); s.append(r.p.v ? "V" : "v"); s.append(r.p.t ? "T" : "t"); s.append(r.p.b ? "B" : "b"); s.append(r.p.d ? "D" : "d"); s.append(r.p.i ? "I" : "i"); s.append(r.p.z ? "Z" : "z"); s.append(r.p.c ? "C" : "c"); s.append(r.cs == 3 ? "+" : "-"); return s; }