2015-11-21 07:36:48 +00:00
|
|
|
auto Cartridge::MBC3::second() -> void {
|
2017-07-26 12:42:06 +00:00
|
|
|
if(io.rtc.halt) return;
|
|
|
|
if(++io.rtc.second >= 60) {
|
|
|
|
io.rtc.second = 0;
|
|
|
|
if(++io.rtc.minute >= 60) {
|
|
|
|
io.rtc.minute = 0;
|
|
|
|
if(++io.rtc.hour >= 24) {
|
|
|
|
io.rtc.hour = 0;
|
|
|
|
if(++io.rtc.day == 0) {
|
|
|
|
io.rtc.dayCarry = true;
|
2011-01-04 10:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-26 12:42:06 +00:00
|
|
|
auto Cartridge::MBC3::read(uint16 address) -> uint8 {
|
|
|
|
if((address & 0xc000) == 0x0000) { //$0000-3fff
|
|
|
|
return cartridge.rom.read(address.bits(0,13));
|
2011-01-04 10:42:27 +00:00
|
|
|
}
|
|
|
|
|
2017-07-26 12:42:06 +00:00
|
|
|
if((address & 0xc000) == 0x4000) { //$4000-7fff
|
|
|
|
return cartridge.rom.read(io.rom.bank << 14 | address.bits(0,13));
|
2011-01-04 10:42:27 +00:00
|
|
|
}
|
|
|
|
|
2017-07-26 12:42:06 +00:00
|
|
|
if((address & 0xe000) == 0xa000) { //$a000-bfff
|
|
|
|
if(!io.ram.enable) return 0xff;
|
|
|
|
if(io.ram.bank <= 0x03) return cartridge.ram.read(io.ram.bank << 13 | address.bits(0,12));
|
|
|
|
if(io.ram.bank == 0x08) return io.rtc.latchSecond;
|
|
|
|
if(io.ram.bank == 0x09) return io.rtc.latchMinute;
|
|
|
|
if(io.ram.bank == 0x0a) return io.rtc.latchHour;
|
|
|
|
if(io.ram.bank == 0x0b) return io.rtc.latchDay;
|
|
|
|
if(io.ram.bank == 0x0c) return io.rtc.latchDayCarry << 7 | io.rtc.latchDay >> 8;
|
2016-01-08 09:23:46 +00:00
|
|
|
return 0xff;
|
2011-01-04 10:42:27 +00:00
|
|
|
}
|
|
|
|
|
2016-01-08 09:23:46 +00:00
|
|
|
return 0xff;
|
2011-01-04 10:42:27 +00:00
|
|
|
}
|
|
|
|
|
2017-07-26 12:42:06 +00:00
|
|
|
auto Cartridge::MBC3::write(uint16 address, uint8 data) -> void {
|
|
|
|
if((address & 0xe000) == 0x0000) { //$0000-1fff
|
|
|
|
io.ram.enable = data.bits(0,3) == 0x0a;
|
2011-01-04 10:42:27 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-07-26 12:42:06 +00:00
|
|
|
if((address & 0xe000) == 0x2000) { //$2000-3fff
|
|
|
|
io.rom.bank = data.bits(0,6);
|
|
|
|
if(!io.rom.bank) io.rom.bank = 0x01;
|
2011-01-04 10:42:27 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-07-26 12:42:06 +00:00
|
|
|
if((address & 0xe000) == 0x4000) { //$4000-5fff
|
|
|
|
io.ram.bank = data;
|
2011-01-04 10:42:27 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-07-26 12:42:06 +00:00
|
|
|
if((address & 0xe000) == 0x6000) { //$6000-7fff
|
|
|
|
if(io.rtc.latch == 0 && data == 1) {
|
|
|
|
io.rtc.latchSecond = io.rtc.second;
|
|
|
|
io.rtc.latchMinute = io.rtc.minute;
|
|
|
|
io.rtc.latchHour = io.rtc.hour;
|
|
|
|
io.rtc.latchDay = io.rtc.day;
|
|
|
|
io.rtc.latchDayCarry = io.rtc.dayCarry;
|
2011-01-04 10:42:27 +00:00
|
|
|
}
|
2017-07-26 12:42:06 +00:00
|
|
|
io.rtc.latch = data;
|
2011-01-04 10:42:27 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-07-26 12:42:06 +00:00
|
|
|
if((address & 0xe000) == 0xa000) { //$a000-bfff
|
|
|
|
if(!io.ram.enable) return;
|
|
|
|
if(io.ram.bank <= 0x03) {
|
|
|
|
cartridge.ram.write(io.ram.bank << 13 | address.bits(0,12), data);
|
|
|
|
} else if(io.ram.bank == 0x08) {
|
|
|
|
if(data >= 60) data = 0;
|
|
|
|
io.rtc.second = data;
|
|
|
|
} else if(io.ram.bank == 0x09) {
|
|
|
|
if(data >= 60) data = 0;
|
|
|
|
io.rtc.minute = data;
|
|
|
|
} else if(io.ram.bank == 0x0a) {
|
|
|
|
if(data >= 24) data = 0;
|
|
|
|
io.rtc.hour = data;
|
|
|
|
} else if(io.ram.bank == 0x0b) {
|
|
|
|
io.rtc.day.bits(0,7) = data.bits(0,7);
|
|
|
|
} else if(io.ram.bank == 0x0c) {
|
|
|
|
io.rtc.day.bit(8) = data.bit(0);
|
|
|
|
io.rtc.halt = data.bit(6);
|
|
|
|
io.rtc.dayCarry = data.bit(7);
|
2011-01-04 10:42:27 +00:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-21 07:36:48 +00:00
|
|
|
auto Cartridge::MBC3::power() -> void {
|
2017-07-26 12:42:06 +00:00
|
|
|
io = {};
|
|
|
|
}
|
2016-06-28 10:43:47 +00:00
|
|
|
|
2017-07-26 12:42:06 +00:00
|
|
|
auto Cartridge::MBC3::serialize(serializer& s) -> void {
|
|
|
|
s.integer(io.rom.bank);
|
|
|
|
s.integer(io.ram.enable);
|
|
|
|
s.integer(io.ram.bank);
|
|
|
|
s.integer(io.rtc.halt);
|
|
|
|
s.integer(io.rtc.latch);
|
|
|
|
s.integer(io.rtc.second);
|
|
|
|
s.integer(io.rtc.minute);
|
|
|
|
s.integer(io.rtc.hour);
|
|
|
|
s.integer(io.rtc.day);
|
|
|
|
s.integer(io.rtc.dayCarry);
|
|
|
|
s.integer(io.rtc.latchSecond);
|
|
|
|
s.integer(io.rtc.latchMinute);
|
|
|
|
s.integer(io.rtc.latchHour);
|
|
|
|
s.integer(io.rtc.latchDay);
|
|
|
|
s.integer(io.rtc.latchDayCarry);
|
2011-01-04 10:42:27 +00:00
|
|
|
}
|