2015-08-21 11:29:53 +00:00
|
|
|
struct GameBoyCartridge {
|
|
|
|
GameBoyCartridge(uint8_t* data, unsigned size);
|
|
|
|
|
|
|
|
string markup;
|
|
|
|
|
|
|
|
//private:
|
|
|
|
struct Information {
|
|
|
|
string mapper;
|
|
|
|
bool ram;
|
|
|
|
bool battery;
|
|
|
|
bool rtc;
|
|
|
|
bool rumble;
|
|
|
|
|
|
|
|
unsigned romsize;
|
|
|
|
unsigned ramsize;
|
|
|
|
|
|
|
|
bool cgb;
|
|
|
|
bool cgbonly;
|
|
|
|
} info;
|
|
|
|
};
|
|
|
|
|
|
|
|
GameBoyCartridge::GameBoyCartridge(uint8_t *romdata, unsigned romsize) {
|
|
|
|
if(romsize < 0x4000) return;
|
|
|
|
|
|
|
|
info.mapper = "unknown";
|
|
|
|
info.ram = false;
|
|
|
|
info.battery = false;
|
|
|
|
info.rtc = false;
|
|
|
|
info.rumble = false;
|
|
|
|
|
|
|
|
info.romsize = 0;
|
|
|
|
info.ramsize = 0;
|
|
|
|
|
|
|
|
unsigned base = romsize - 0x8000;
|
|
|
|
if(romdata[base + 0x0104] == 0xce && romdata[base + 0x0105] == 0xed
|
|
|
|
&& romdata[base + 0x0106] == 0x66 && romdata[base + 0x0107] == 0x66
|
|
|
|
&& romdata[base + 0x0108] == 0xcc && romdata[base + 0x0109] == 0x0d
|
|
|
|
&& romdata[base + 0x0147] >= 0x0b && romdata[base + 0x0147] <= 0x0d
|
|
|
|
) {
|
|
|
|
//MMM01 stores header at bottom of image
|
|
|
|
//flip this around for consistency with all other mappers
|
|
|
|
uint8_t header[0x8000];
|
|
|
|
memcpy(header, romdata + base, 0x8000);
|
|
|
|
memmove(romdata + 0x8000, romdata, romsize - 0x8000);
|
|
|
|
memcpy(romdata, header, 0x8000);
|
|
|
|
}
|
|
|
|
|
|
|
|
info.cgb = (romdata[0x0143] & 0x80) == 0x80;
|
|
|
|
info.cgbonly = (romdata[0x0143] & 0xc0) == 0xc0;
|
|
|
|
|
|
|
|
switch(romdata[0x0147]) {
|
|
|
|
case 0x00: info.mapper = "none"; break;
|
|
|
|
case 0x01: info.mapper = "MBC1"; break;
|
|
|
|
case 0x02: info.mapper = "MBC1"; info.ram = true; break;
|
|
|
|
case 0x03: info.mapper = "MBC1"; info.ram = true; info.battery = true; break;
|
|
|
|
case 0x05: info.mapper = "MBC2"; info.ram = true; break;
|
|
|
|
case 0x06: info.mapper = "MBC2"; info.ram = true; info.battery = true; break;
|
|
|
|
case 0x08: info.mapper = "none"; info.ram = true; break;
|
|
|
|
case 0x09: info.mapper = "MBC0"; info.ram = true; info.battery = true; break;
|
|
|
|
case 0x0b: info.mapper = "MMM01"; break;
|
|
|
|
case 0x0c: info.mapper = "MMM01"; info.ram = true; break;
|
|
|
|
case 0x0d: info.mapper = "MMM01"; info.ram = true; info.battery = true; break;
|
|
|
|
case 0x0f: info.mapper = "MBC3"; info.rtc = true; info.battery = true; break;
|
|
|
|
case 0x10: info.mapper = "MBC3"; info.rtc = true; info.ram = true; info.battery = true; break;
|
|
|
|
case 0x11: info.mapper = "MBC3"; break;
|
|
|
|
case 0x12: info.mapper = "MBC3"; info.ram = true; break;
|
|
|
|
case 0x13: info.mapper = "MBC3"; info.ram = true; info.battery = true; break;
|
|
|
|
case 0x19: info.mapper = "MBC5"; break;
|
|
|
|
case 0x1a: info.mapper = "MBC5"; info.ram = true; break;
|
|
|
|
case 0x1b: info.mapper = "MBC5"; info.ram = true; info.battery = true; break;
|
|
|
|
case 0x1c: info.mapper = "MBC5"; info.rumble = true; break;
|
|
|
|
case 0x1d: info.mapper = "MBC5"; info.rumble = true; info.ram = true; break;
|
|
|
|
case 0x1e: info.mapper = "MBC5"; info.rumble = true; info.ram = true; info.battery = true; break;
|
|
|
|
case 0xfc: break; //Pocket Camera
|
|
|
|
case 0xfd: break; //Bandai TAMA5
|
|
|
|
case 0xfe: info.mapper = "HuC3"; break;
|
|
|
|
case 0xff: info.mapper = "HuC1"; info.ram = true; info.battery = true; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(romdata[0x0148]) { default:
|
|
|
|
case 0x00: info.romsize = 2 * 16 * 1024; break;
|
|
|
|
case 0x01: info.romsize = 4 * 16 * 1024; break;
|
|
|
|
case 0x02: info.romsize = 8 * 16 * 1024; break;
|
|
|
|
case 0x03: info.romsize = 16 * 16 * 1024; break;
|
|
|
|
case 0x04: info.romsize = 32 * 16 * 1024; break;
|
|
|
|
case 0x05: info.romsize = 64 * 16 * 1024; break;
|
|
|
|
case 0x06: info.romsize = 128 * 16 * 1024; break;
|
|
|
|
case 0x07: info.romsize = 256 * 16 * 1024; break;
|
|
|
|
case 0x52: info.romsize = 72 * 16 * 1024; break;
|
|
|
|
case 0x53: info.romsize = 80 * 16 * 1024; break;
|
|
|
|
case 0x54: info.romsize = 96 * 16 * 1024; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(romdata[0x0149]) { default:
|
|
|
|
case 0x00: info.ramsize = 0 * 1024; break;
|
|
|
|
case 0x01: info.ramsize = 2 * 1024; break;
|
|
|
|
case 0x02: info.ramsize = 8 * 1024; break;
|
|
|
|
case 0x03: info.ramsize = 32 * 1024; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(info.mapper == "MBC2") info.ramsize = 512; //512 x 4-bit
|
|
|
|
|
Update to higan and icarus v095r15 release.
r13 and r14 weren't posted as individual releases, but their changelogs
were posted.
byuu says about r13:
I'm not going to be posting WIPs for r13 and above for a while.
The reason is that I'm working on the major manifest overhaul I've
discussed previously on the icarus subforum.
I'm recreating my boards database from scratch using the map files
and the new map analyzer. The only games that will load are ones
I've created board definitions for, and updated
sfc/cartridge/markup.cpp to parse. Once I've finished all the
boards, then I'll update the heuristics.
Then finally, I'll sync the syntax changes over to the fc, gb, gba
cores.
Once that's done, I'll start posting WIPs again, along with a new
build of icarus.
But I'll still post changelogs as I work through things.
Changelog (r13):
- preservation: created new database-builder tool (merges
region-specific databases with boards)
- icarus: support new, external database format
(~/.config/icarus/Database/(Super Famicom.bml, ...)
- added 1A3B-(10,11,12); 1A3B-20
byuu says about r14:
r14 work:
I successfully created mappings for every board used in the US set.
I also updated icarus' heuristics to use the new mappings, and
created ones there for the boards that are only in the JP set.
Then I patched icarus to support pulling games out of the database
when it's used on a game folder to generate a manifest file.
Then I updated a lot of code in higan/sfc to support the new mapping
syntax. sfc/cartridge/markup.cpp is about half the size it used to
be with the new mappings, and I was able to kill off both map/id and
map/select entirely.
Then I updated all four emulated systems (and both subsystems) to
use "board" as the root node, and harmonized their syntax (made them
all more consistent with each other.)
Then I added a manifest viewer to the tools window+menu. It's kind
of an advanced user feature, but oh well. No reason to coddle people
when the feature is very useful for developers. The viewer will show
all manifests in order when you load multi-cart games as well.
Still not going to call any syntax 100% done right now, but
thankfully with the new manifest-free folders, nobody will have to
do anything to use the new format. Just download the new version and
go.
The Super Famicom Event stuff is currently broken (CC92/PF94
boards). That's gonna be fun to support.
byuu says about r15:
EDIT: small bug in icarus with heuristics. Edit
core/super-famicom.cpp line 27:
if(/*auto*/ markup = cartridge.markup) {
Gotta remove that "auto" so that it returns valid markup.
Resolved the final concerns I had with the new manifest format.
Right now there are two things that are definitely broken: MCC (BS-X
Town cart) and Event (CC '92 and PF'94).
And there are a few things that are untested: SPC7110, EpsonRTC,
SharpRTC, SDD1+RAM, SufamiTurbo, BS-X slotted carts.
2015-12-19 08:52:34 +00:00
|
|
|
markup.append("board mapper=", info.mapper, "\n");
|
2015-08-21 11:29:53 +00:00
|
|
|
markup.append(" rom name=program.rom size=0x", hex(romsize), "\n");
|
|
|
|
if(info.ramsize > 0) markup.append(" ram name=save.ram size=0x", hex(info.ramsize), "\n");
|
|
|
|
}
|