bsnes/higan/processor/huc6280/disassembler.cpp

357 lines
11 KiB
C++
Raw Normal View History

Update to v102r03 release. byuu says: Changelog: - PCE: split VCE from VDC - HuC6280: changed bus from (uint21 addr) to (uint8 bank, uint13 addr) - added SuperGrafx emulation (adds secondary VDC, plus new VPC) The VDC now has no concept of the actual display raster timing, and instead is driven by Vpulse (start of frame) and Hpulse (start of scanline) signals from the VCE. One still can't render the start of the next scanline onto the current scanline through overly aggressive timings, but it shouldn't be too much more difficult to allow that to occur now. This process incurs quite a major speed hit, so low-end systems with Atom CPUs can't run things at 60fps anymore. The timing needs a lot of work. The pixels end up very jagged if the VCE doesn't output batches of 2-4 pixels at a time. But this should not be a requirement at all, so I'm not sure what's going wrong there. Yo, Bro and the 512-width mode of TV Sports Basketball is now broken as a result of these changes, and I'm not sure why. To load SuperGrafx games, you're going to have to change the .pce extensions to .sg or .sgx. Or you can manually move the games from the PC Engine folder to the SuperGrafx folder and change the game folder extensions. I have no way to tell the games apart. Mednafen uses CRC32 comparisons, and I may consider that since there's only five games, but I'm not sure yet. The only SuperGrafx game that's playable right now is Aldynes. And the priorities are all screwed up. I don't understand how the windows or the priorities work at all from sgxtech.txt, so ... yeah. It's pretty broken, but it's a start. I could really use some help with this, as I'm very lost right now with rendering :/ ----- Note that the SuperGrafx is technically its own system, it's not an add-on. As such, I'm giving it a separate .sys folder, and a separate library. There's debate over how to name this thing. "SuperGrafx" appears more popular than "Super Grafx". And you might also call it the "PC Engine SuperGrafx", but I decided to leave off the prefix so it appears more distinct.
2017-01-23 21:18:54 +00:00
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 {
Update to v102r03 release. byuu says: Changelog: - PCE: split VCE from VDC - HuC6280: changed bus from (uint21 addr) to (uint8 bank, uint13 addr) - added SuperGrafx emulation (adds secondary VDC, plus new VPC) The VDC now has no concept of the actual display raster timing, and instead is driven by Vpulse (start of frame) and Hpulse (start of scanline) signals from the VCE. One still can't render the start of the next scanline onto the current scanline through overly aggressive timings, but it shouldn't be too much more difficult to allow that to occur now. This process incurs quite a major speed hit, so low-end systems with Atom CPUs can't run things at 60fps anymore. The timing needs a lot of work. The pixels end up very jagged if the VCE doesn't output batches of 2-4 pixels at a time. But this should not be a requirement at all, so I'm not sure what's going wrong there. Yo, Bro and the 512-width mode of TV Sports Basketball is now broken as a result of these changes, and I'm not sure why. To load SuperGrafx games, you're going to have to change the .pce extensions to .sg or .sgx. Or you can manually move the games from the PC Engine folder to the SuperGrafx folder and change the game folder extensions. I have no way to tell the games apart. Mednafen uses CRC32 comparisons, and I may consider that since there's only five games, but I'm not sure yet. The only SuperGrafx game that's playable right now is Aldynes. And the priorities are all screwed up. I don't understand how the windows or the priorities work at all from sgxtech.txt, so ... yeah. It's pretty broken, but it's a start. I could really use some help with this, as I'm very lost right now with rendering :/ ----- Note that the SuperGrafx is technically its own system, it's not an add-on. As such, I'm giving it a separate .sys folder, and a separate library. There's debate over how to name this thing. "SuperGrafx" appears more popular than "Super Grafx". And you might also call it the "PC Engine SuperGrafx", but I decided to leave off the prefix so it appears more distinct.
2017-01-23 21:18:54 +00:00
return read(bank, addr++);
};
auto readWord = [&]() -> uint16 {
Update to v102r03 release. byuu says: Changelog: - PCE: split VCE from VDC - HuC6280: changed bus from (uint21 addr) to (uint8 bank, uint13 addr) - added SuperGrafx emulation (adds secondary VDC, plus new VPC) The VDC now has no concept of the actual display raster timing, and instead is driven by Vpulse (start of frame) and Hpulse (start of scanline) signals from the VCE. One still can't render the start of the next scanline onto the current scanline through overly aggressive timings, but it shouldn't be too much more difficult to allow that to occur now. This process incurs quite a major speed hit, so low-end systems with Atom CPUs can't run things at 60fps anymore. The timing needs a lot of work. The pixels end up very jagged if the VCE doesn't output batches of 2-4 pixels at a time. But this should not be a requirement at all, so I'm not sure what's going wrong there. Yo, Bro and the 512-width mode of TV Sports Basketball is now broken as a result of these changes, and I'm not sure why. To load SuperGrafx games, you're going to have to change the .pce extensions to .sg or .sgx. Or you can manually move the games from the PC Engine folder to the SuperGrafx folder and change the game folder extensions. I have no way to tell the games apart. Mednafen uses CRC32 comparisons, and I may consider that since there's only five games, but I'm not sure yet. The only SuperGrafx game that's playable right now is Aldynes. And the priorities are all screwed up. I don't understand how the windows or the priorities work at all from sgxtech.txt, so ... yeah. It's pretty broken, but it's a start. I could really use some help with this, as I'm very lost right now with rendering :/ ----- Note that the SuperGrafx is technically its own system, it's not an add-on. As such, I'm giving it a separate .sys folder, and a separate library. There's debate over how to name this thing. "SuperGrafx" appears more popular than "Super Grafx". And you might also call it the "PC Engine SuperGrafx", but I decided to leave off the prefix so it appears more distinct.
2017-01-23 21:18:54 +00:00
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));
Update to v102r03 release. byuu says: Changelog: - PCE: split VCE from VDC - HuC6280: changed bus from (uint21 addr) to (uint8 bank, uint13 addr) - added SuperGrafx emulation (adds secondary VDC, plus new VPC) The VDC now has no concept of the actual display raster timing, and instead is driven by Vpulse (start of frame) and Hpulse (start of scanline) signals from the VCE. One still can't render the start of the next scanline onto the current scanline through overly aggressive timings, but it shouldn't be too much more difficult to allow that to occur now. This process incurs quite a major speed hit, so low-end systems with Atom CPUs can't run things at 60fps anymore. The timing needs a lot of work. The pixels end up very jagged if the VCE doesn't output batches of 2-4 pixels at a time. But this should not be a requirement at all, so I'm not sure what's going wrong there. Yo, Bro and the 512-width mode of TV Sports Basketball is now broken as a result of these changes, and I'm not sure why. To load SuperGrafx games, you're going to have to change the .pce extensions to .sg or .sgx. Or you can manually move the games from the PC Engine folder to the SuperGrafx folder and change the game folder extensions. I have no way to tell the games apart. Mednafen uses CRC32 comparisons, and I may consider that since there's only five games, but I'm not sure yet. The only SuperGrafx game that's playable right now is Aldynes. And the priorities are all screwed up. I don't understand how the windows or the priorities work at all from sgxtech.txt, so ... yeah. It's pretty broken, but it's a start. I could really use some help with this, as I'm very lost right now with rendering :/ ----- Note that the SuperGrafx is technically its own system, it's not an add-on. As such, I'm giving it a separate .sys folder, and a separate library. There's debate over how to name this thing. "SuperGrafx" appears more popular than "Super Grafx". And you might also call it the "PC Engine SuperGrafx", but I decided to leave off the prefix so it appears more distinct.
2017-01-23 21:18:54 +00:00
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;
}