mirror of https://github.com/bsnes-emu/bsnes.git
125 lines
3.7 KiB
C++
125 lines
3.7 KiB
C++
#include <md/md.hpp>
|
|
|
|
namespace MegaDrive {
|
|
|
|
Cartridge cartridge;
|
|
#include "serialization.cpp"
|
|
|
|
auto Cartridge::load() -> bool {
|
|
information = {};
|
|
|
|
if(auto loaded = platform->load(ID::MegaDrive, "Mega Drive", "md", {"Auto", "NTSC-J", "NTSC-U", "PAL"})) {
|
|
information.pathID = loaded.pathID();
|
|
information.region = loaded.option();
|
|
} else return false;
|
|
|
|
if(auto fp = platform->open(pathID(), "manifest.bml", File::Read, File::Required)) {
|
|
information.manifest = fp->reads();
|
|
} else return false;
|
|
|
|
auto document = BML::unserialize(information.manifest);
|
|
information.title = document["game/label"].text();
|
|
|
|
if(information.region == "Auto") {
|
|
if(auto region = document["game/region"].text()) {
|
|
information.region = region.upcase();
|
|
} else {
|
|
information.region = "NTSC-J";
|
|
}
|
|
}
|
|
|
|
if(auto memory = Emulator::Game::Memory{document["game/board/memory(type=ROM,content=Program)"]}) {
|
|
rom.size = memory.size >> 1;
|
|
rom.mask = bit::round(rom.size) - 1;
|
|
rom.data = new uint16[rom.mask + 1]();
|
|
if(auto fp = platform->open(pathID(), memory.name(), File::Read, File::Required)) {
|
|
for(uint n : range(rom.size)) rom.data[n] = fp->readm(2);
|
|
}
|
|
}
|
|
|
|
//todo: handle mode, offset in Emulator::Game::Memory
|
|
if(auto memory = document["game/board/memory(type=RAM,content=Save)"]) {
|
|
if(auto mode = memory["mode"].text()) {
|
|
if(mode == "lo" ) ram.bits = 0x00ff;
|
|
if(mode == "hi" ) ram.bits = 0xff00;
|
|
if(mode == "word") ram.bits = 0xffff;
|
|
}
|
|
ram.size = memory["size"].natural() >> (ram.bits == 0xffff);
|
|
ram.mask = bit::round(ram.size) - 1;
|
|
ram.data = new uint16[ram.mask + 1]();
|
|
if(!(bool)memory["volatile"]) {
|
|
if(auto fp = platform->open(pathID(), "save.ram", File::Read)) {
|
|
for(uint n : range(ram.size)) {
|
|
if(ram.bits != 0xffff) ram.data[n] = fp->readm(1) * 0x0101;
|
|
if(ram.bits == 0xffff) ram.data[n] = fp->readm(2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
auto Cartridge::save() -> void {
|
|
auto document = BML::unserialize(information.manifest);
|
|
|
|
if(auto memory = Emulator::Game::Memory{document["game/board/memory(type=RAM,content=Save)"]}) {
|
|
if(memory.nonVolatile) {
|
|
if(auto fp = platform->open(pathID(), memory.name(), File::Write)) {
|
|
for(uint n : range(ram.size)) {
|
|
if(ram.bits != 0xffff) fp->writem(ram.data[n], 1);
|
|
if(ram.bits == 0xffff) fp->writem(ram.data[n], 2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
auto Cartridge::unload() -> void {
|
|
delete[] rom.data;
|
|
delete[] ram.data;
|
|
rom = {};
|
|
ram = {};
|
|
}
|
|
|
|
auto Cartridge::power() -> void {
|
|
ramEnable = 1;
|
|
ramWritable = 1;
|
|
for(auto n : range(8)) bank[n] = n;
|
|
}
|
|
|
|
auto Cartridge::read(uint24 address) -> uint16 {
|
|
if(address.bit(21) && ram.size && ramEnable) {
|
|
return ram.data[address >> 1 & ram.mask];
|
|
} else {
|
|
address = bank[address.bits(19,21)] << 19 | address.bits(0,18);
|
|
return rom.data[address >> 1 & rom.mask];
|
|
}
|
|
}
|
|
|
|
auto Cartridge::write(uint24 address, uint16 data) -> void {
|
|
//emulating RAM write protect bit breaks some commercial software
|
|
if(address.bit(21) && ram.size && ramEnable /* && ramWritable */) {
|
|
if(ram.bits == 0x00ff) data = data.byte(0) * 0x0101;
|
|
if(ram.bits == 0xff00) data = data.byte(1) * 0x0101;
|
|
ram.data[address >> 1 & ram.mask] = data;
|
|
}
|
|
}
|
|
|
|
auto Cartridge::readIO(uint24 addr) -> uint16 {
|
|
return 0x0000;
|
|
}
|
|
|
|
auto Cartridge::writeIO(uint24 addr, uint16 data) -> void {
|
|
if(addr == 0xa130f1) ramEnable = data.bit(0), ramWritable = data.bit(1);
|
|
if(addr == 0xa130f3) bank[1] = data;
|
|
if(addr == 0xa130f5) bank[2] = data;
|
|
if(addr == 0xa130f7) bank[3] = data;
|
|
if(addr == 0xa130f9) bank[4] = data;
|
|
if(addr == 0xa130fb) bank[5] = data;
|
|
if(addr == 0xa130fd) bank[6] = data;
|
|
if(addr == 0xa130ff) bank[7] = data;
|
|
}
|
|
|
|
}
|