bsnes/higan/ms/cartridge/mapper.cpp

93 lines
1.4 KiB
C++
Raw Normal View History

Update to v101r27 release. byuu says: Changelog: - SMS: emulated the generic Sega memory mapper (none of the more limited forms of it yet) - (missing ROM shift, ROM write enable emulation -- no commercial games use either, though) - SMS: bus I/O returns 0xff instead of 0x00 so games don't think every key is being pressed at once - (this is a hack until I implement proper controller pad reading) - SMS: very limited protection against reading/writing past the end of ROM/RAM (todo: should mirror) - SMS: VDP background HSCROLL subtracts, rather than adds, to the offset (unlike VSCROLL) - SMS: VDP VSCROLL is 9-bit, modulates voffset+vscroll to 224 in 192-line mode (32x28 tilemap) - SMS: VDP tiledata for backgrounds and sprites use `7-(x&7)` rather than `(x&7)` - SMS: fix output color to be 6-bit rather than 5-bit - SMS: left clip uses register `#7`, not palette color `#7` - (todo: do we want `color[reg7]` or `color[16 + reg7]`?) - SMS: refined handling of 0xcb, 0xed prefixes in the Z80 core and its disassembler - SMS: emulated (0xfd, 0xdd) 0xcb opcodes 0x00-0x0f (still missing 0x10-0xff) - SMS: fixed 0xcb 0b-----110 opcodes to use direct HL and never allow (IX,IY)+d - SMS: fixed major logic bug in (IX,IY)+d displacement - (was using `read(x)` instead of `operand()` for the displacement byte fetch before) - icarus: fake there always being 32KiB of RAM in all SMS cartridges for the time being - (not sure how to detect this stuff yet; although I've read it's not even really possible `>_>`) TODO: remove processor/z80/dissassembler.cpp code block at line 396 (as it's unnecessary.) Lots of commercial games are starting to show trashed graphical output now.
2017-01-06 08:11:38 +00:00
auto Cartridge::read(uint16 addr) -> maybe<uint8> {
uint2 page = addr >> 14;
addr &= 0x3fff;
switch(page) {
case 0: {
if(addr <= 0x03ff) return rom.read(addr);
return rom.read(mapper.romPage0 << 14 | addr);
}
case 1: {
return rom.read(mapper.romPage1 << 14 | addr);
}
case 2: {
if(mapper.ramEnablePage2) {
return ram.read(mapper.ramPage2 << 14 | addr);
}
return rom.read(mapper.romPage2 << 14 | addr);
}
case 3: {
if(mapper.ramEnablePage3) {
return ram.read(addr);
}
return nothing;
}
}
unreachable;
}
auto Cartridge::write(uint16 addr, uint8 data) -> bool {
if(addr == 0xfffc) {
mapper.shift = data.bits(0,1);
mapper.ramPage2 = data.bit(2);
mapper.ramEnablePage2 = data.bit(3);
mapper.ramEnablePage3 = data.bit(4);
mapper.romWriteEnable = data.bit(7);
}
if(addr == 0xfffd) {
mapper.romPage0 = data;
}
if(addr == 0xfffe) {
mapper.romPage1 = data;
}
if(addr == 0xffff) {
mapper.romPage2 = data;
}
uint2 page = addr >> 14;
addr &= 0x3fff;
switch(page) {
case 0: {
return false;
}
case 1: {
return false;
}
case 2: {
if(mapper.ramEnablePage2) {
ram.write(mapper.ramPage2 << 14 | addr, data);
return true;
}
return false;
}
case 3: {
if(mapper.ramEnablePage3) {
ram.write(addr, data);
return true;
}
return false;
}
}
unreachable;
}