Update to v106r3 release.

byuu says:

Changelog:

  - Super Famicom: update to newer board markup syntax
  - Super Famicom: update all mapped ROMs to be write-protected
      - errata: SPC7110 set ram.writeProtect(true), I'll fix it in the
        next WIP
  - icarus: rewrote the Super Famicom heuristics module from scratch

Instead of icarus heuristics generating higan-specific mappings, it now
generates generic board IDs that can be used by any emulator. I had
originally planned to print out real PCB ID codes here, but these board
mappings are meant to be more generic, and I don't want them to look
real. The pseudo-codes are easy to parse, for example: `DSP-LOROM-NVRAM`
for Super Mario Kart, `SUPERFX-RAM` for Doom.

I'm going to make a `Boards (Generic).bml` file that will contain mapping
definitions for every board. Until this is done, any games not in the SNES
preservation database will fail to play because the mapping information is
now missing.
This commit is contained in:
Tim Allen 2018-02-05 20:58:02 +11:00
parent 38fbcd5277
commit 3d8be92550
17 changed files with 1583 additions and 1173 deletions

View File

@ -12,7 +12,7 @@ using namespace nall;
namespace Emulator { namespace Emulator {
static const string Name = "higan"; static const string Name = "higan";
static const string Version = "106.02"; static const string Version = "106.03";
static const string Author = "byuu"; static const string Author = "byuu";
static const string License = "GPLv3"; static const string License = "GPLv3";
static const string Website = "https://byuu.org/"; static const string Website = "https://byuu.org/";

View File

@ -12,7 +12,7 @@ auto Cartridge::loadBoard(Markup::Node node) -> Markup::Node {
if(auto fp = platform->open(ID::System, "boards.bml", File::Read, File::Required)) { if(auto fp = platform->open(ID::System, "boards.bml", File::Read, File::Required)) {
auto document = BML::unserialize(fp->reads()); auto document = BML::unserialize(fp->reads());
for(auto leaf : document.find("board")) { for(auto leaf : document.find("board")) {
auto id = leaf["id"].text(); auto id = leaf.text();
bool matched = id == board; bool matched = id == board;
if(!matched && id.match("*(*)*")) { if(!matched && id.match("*(*)*")) {
auto part = id.transform("()", "||").split("|"); auto part = id.transform("()", "||").split("|");
@ -31,7 +31,8 @@ auto Cartridge::loadBoard(Markup::Node node) -> Markup::Node {
output.append("region=pal\n"); output.append("region=pal\n");
} }
uint counter = 0; uint counter = 0;
for(auto& line : leaf.text().split("\n")) { for(auto& line : BML::serialize(leaf).split("\n")) {
line.trimLeft(" ", 1L);
if(line.endsWith("rom") || line.endsWith("ram")) { if(line.endsWith("rom") || line.endsWith("ram")) {
auto memory = node.find("game/memory"); auto memory = node.find("game/memory");
if(counter < memory.size()) { if(counter < memory.size()) {

View File

@ -39,6 +39,12 @@ auto Event::unload() -> void {
auto Event::power() -> void { auto Event::power() -> void {
create(Event::Enter, 1); create(Event::Enter, 1);
rom[0].writeProtect(true);
rom[1].writeProtect(true);
rom[2].writeProtect(true);
rom[3].writeProtect(true);
ram.writeProtect(false);
for(auto n : range(ram.size())) ram.write(n, 0x00); for(auto n : range(ram.size())) ram.write(n, 0x00);
status = 0x00; status = 0x00;
select = 0x00; select = 0x00;

View File

@ -34,6 +34,9 @@ auto HitachiDSP::power() -> void {
HG51B::power(); HG51B::power();
create(HitachiDSP::Enter, Frequency); create(HitachiDSP::Enter, Frequency);
rom.writeProtect(true);
ram.writeProtect(false);
mmio.dma = false; mmio.dma = false;
mmio.dmaSource = 0x000000; mmio.dmaSource = 0x000000;

View File

@ -11,6 +11,9 @@ auto MCC::unload() -> void {
} }
auto MCC::power() -> void { auto MCC::power() -> void {
rom.writeProtect(true);
ram.writeProtect(false);
for(auto n : range(16)) r[n] = 0x00; for(auto n : range(16)) r[n] = 0x00;
r[0x07] = 0x80; r[0x07] = 0x80;
r[0x08] = 0x80; r[0x08] = 0x80;

View File

@ -10,6 +10,8 @@ auto OBC1::unload() -> void {
} }
auto OBC1::power() -> void { auto OBC1::power() -> void {
ram.writeProtect(false);
status.baseptr = (ramRead(0x1ff5) & 1) ? 0x1800 : 0x1c00; status.baseptr = (ramRead(0x1ff5) & 1) ? 0x1800 : 0x1c00;
status.address = (ramRead(0x1ff6) & 0x7f); status.address = (ramRead(0x1ff6) & 0x7f);
status.shift = (ramRead(0x1ff6) & 3) << 1; status.shift = (ramRead(0x1ff6) & 3) << 1;

View File

@ -127,6 +127,10 @@ auto SA1::power() -> void {
WDC65816::power(); WDC65816::power();
create(SA1::Enter, system.cpuFrequency()); create(SA1::Enter, system.cpuFrequency());
rom.writeProtect(true);
bwram.writeProtect(false);
iram.writeProtect(false);
cpubwram.dma = false; cpubwram.dma = false;
for(auto addr : range(iram.size())) { for(auto addr : range(iram.size())) {
iram.write(addr, 0x00); iram.write(addr, 0x00);

View File

@ -13,6 +13,9 @@ auto SDD1::unload() -> void {
} }
auto SDD1::power() -> void { auto SDD1::power() -> void {
rom.writeProtect(true);
ram.writeProtect(false);
//hook S-CPU DMA MMIO registers to gather information for struct dma[]; //hook S-CPU DMA MMIO registers to gather information for struct dma[];
//buffer address and transfer size information for use in SDD1::mcu_read() //buffer address and transfer size information for use in SDD1::mcu_read()
bus.map({&SDD1::dmaRead, &sdd1}, {&SDD1::dmaWrite, &sdd1}, "00-3f,80-bf:4300-437f"); bus.map({&SDD1::dmaRead, &sdd1}, {&SDD1::dmaWrite, &sdd1}, "00-3f,80-bf:4300-437f");

View File

@ -41,6 +41,10 @@ auto SPC7110::unload() -> void {
auto SPC7110::power() -> void { auto SPC7110::power() -> void {
create(SPC7110::Enter, 21'477'272); create(SPC7110::Enter, 21'477'272);
prom.writeProtect(true);
drom.writeProtect(true);
ram.writeProtect(true);
r4801 = 0x00; r4801 = 0x00;
r4802 = 0x00; r4802 = 0x00;
r4803 = 0x00; r4803 = 0x00;

View File

@ -40,6 +40,9 @@ auto SuperFX::power() -> void {
GSU::power(); GSU::power();
create(SuperFX::Enter, system.cpuFrequency()); create(SuperFX::Enter, system.cpuFrequency());
rom.writeProtect(true);
ram.writeProtect(false);
romMask = rom.size() - 1; romMask = rom.size() - 1;
ramMask = ram.size() - 1; ramMask = ram.size() - 1;

View File

@ -11,4 +11,9 @@ auto SufamiTurboCartridge::unload() -> void {
ram.reset(); ram.reset();
} }
auto SufamiTurboCartridge::power() -> void {
rom.writeProtect(true);
ram.writeProtect(false);
}
} }

View File

@ -1,5 +1,6 @@
struct SufamiTurboCartridge { struct SufamiTurboCartridge {
auto unload() -> void; auto unload() -> void;
auto power() -> void;
auto serialize(serializer&) -> void; auto serialize(serializer&) -> void;
uint pathID = 0; uint pathID = 0;

View File

@ -120,6 +120,7 @@ auto System::power(bool reset) -> void {
if(cartridge.has.OBC1) obc1.power(); if(cartridge.has.OBC1) obc1.power();
if(cartridge.has.MSU1) msu1.power(); if(cartridge.has.MSU1) msu1.power();
if(cartridge.has.BSMemorySlot) bsmemory.power(); if(cartridge.has.BSMemorySlot) bsmemory.power();
if(cartridge.has.SufamiTurboSlots) sufamiturboA.power(), sufamiturboB.power();
if(cartridge.has.ICD2) cpu.coprocessors.append(&icd2); if(cartridge.has.ICD2) cpu.coprocessors.append(&icd2);
if(cartridge.has.Event) cpu.coprocessors.append(&event); if(cartridge.has.Event) cpu.coprocessors.append(&event);

View File

@ -1,407 +1,408 @@
database type=sfc-boards revision=2017-12-29 database
revision: 2017-12-29
board id:1A0N-(01,02,10,20,30) board: 1A0N-(01,02,10,20,30)
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
: map address=40-7d,c0-ff:0000-7fff mask=0x8000 map address=40-7d,c0-ff:0000-7fff mask=0x8000
board id:1A1B-(04,05,06) board: 1A1B-(04,05,06)
:rom rom
: map address=00-1f,80-9f:8000-ffff mask=0x8000 map address=00-1f,80-9f:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-ffff map address=70-7d,f0-ff:0000-ffff
board id:1A1M-(01,10,11,20) board: 1A1M-(01,10,11,20)
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:1A3B-(11,12,13) board: 1A3B-(11,12,13)
:rom rom
: map address=00-1f,80-9f:8000-ffff mask=0x8000 map address=00-1f,80-9f:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-ffff map address=70-7d,f0-ff:0000-ffff
board id:1A3B-20 board: 1A3B-20
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:1A3M-(10,20,21,30) board: 1A3M-(10,20,21,30)
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:1A5B-(02,04) board: 1A5B-(02,04)
:rom rom
: map address=00-1f,80-9f:8000-ffff mask=0x8000 map address=00-1f,80-9f:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-ffff map address=70-7d,f0-ff:0000-ffff
board id:1A5M-(01,11,20) board: 1A5M-(01,11,20)
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:1B0N-(02,03,10) board: 1B0N-(02,03,10)
:rom rom
: map address=00-1f,80-9f:8000-ffff mask=0x8000 map address=00-1f,80-9f:8000-ffff mask=0x8000
:necdsp model=uPD7725 frequency=8000000 necdsp model=uPD7725 frequency=8000000
: map address=30-3f,b0-bf:8000-ffff mask=0x3fff map address=30-3f,b0-bf:8000-ffff mask=0x3fff
: prom prom
: drom drom
: dram dram
board id:1B5B-02 board: 1B5B-02
:rom rom
: map address=00-1f,80-9f:8000-ffff mask=0x8000 map address=00-1f,80-9f:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-ffff map address=70-7d,f0-ff:0000-ffff
:necdsp model=uPD7725 frequency=8000000 necdsp model=uPD7725 frequency=8000000
: map address=20-3f,a0-bf:8000-ffff mask=0x3fff map address=20-3f,a0-bf:8000-ffff mask=0x3fff
: prom prom
: drom drom
: dram dram
board id:1C0N board: 1C0N
:superfx superfx
: map address=00-3f,80-bf:3000-34ff map address=00-3f,80-bf:3000-34ff
: rom rom
: map address=00-1f,80-9f:8000-ffff mask=0x8000 map address=00-1f,80-9f:8000-ffff mask=0x8000
: ram ram
: map address=60-7d,e0-ff:0000-ffff map address=60-7d,e0-ff:0000-ffff
board id:1C0N5S-01 board: 1C0N5S-01
:superfx superfx
: map address=00-3f,80-bf:3000-34ff map address=00-3f,80-bf:3000-34ff
: rom rom
: map address=00-1f,80-9f:8000-ffff mask=0x8000 map address=00-1f,80-9f:8000-ffff mask=0x8000
: ram ram
: map address=60-7d,e0-ff:0000-ffff map address=60-7d,e0-ff:0000-ffff
board id:1CA0N5S-01 board: 1CA0N5S-01
:superfx superfx
: map address=00-3f,80-bf:3000-34ff map address=00-3f,80-bf:3000-34ff
: rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x8000 map address=00-3f,80-bf:8000-ffff mask=0x8000
: map address=40-5f,c0-df:0000-ffff map address=40-5f,c0-df:0000-ffff
: ram ram
: map address=00-3f,80-bf:6000-7fff size=0x2000 map address=00-3f,80-bf:6000-7fff size=0x2000
: map address=70-71,f0-f1:0000-ffff map address=70-71,f0-f1:0000-ffff
board id:1CA0N6S-01 board: 1CA0N6S-01
:superfx superfx
: map address=00-3f,80-bf:3000-34ff map address=00-3f,80-bf:3000-34ff
: rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x8000 map address=00-3f,80-bf:8000-ffff mask=0x8000
: map address=40-5f,c0-df:0000-ffff map address=40-5f,c0-df:0000-ffff
: ram ram
: map address=00-3f,80-bf:6000-7fff size=0x2000 map address=00-3f,80-bf:6000-7fff size=0x2000
: map address=70-71,f0-f1:0000-ffff map address=70-71,f0-f1:0000-ffff
board id:1CA6B-01 board: 1CA6B-01
:superfx superfx
: map address=00-3f,80-bf:3000-34ff map address=00-3f,80-bf:3000-34ff
: rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x8000 map address=00-3f,80-bf:8000-ffff mask=0x8000
: map address=40-5f,c0-df:0000-ffff map address=40-5f,c0-df:0000-ffff
: ram ram
: map address=00-3f,80-bf:6000-7fff size=0x2000 map address=00-3f,80-bf:6000-7fff size=0x2000
: map address=70-71,f0-f1:0000-ffff map address=70-71,f0-f1:0000-ffff
board id:1CB0N7S-01 board: 1CB0N7S-01
:superfx superfx
: map address=00-3f,80-bf:3000-34ff map address=00-3f,80-bf:3000-34ff
: rom rom
: map address=00-3f:8000-ffff mask=0x8000 map address=00-3f:8000-ffff mask=0x8000
: map address=40-5f:0000-ffff map address=40-5f:0000-ffff
: ram ram
: map address=00-3f,80-bf:6000-7fff size=0x2000 map address=00-3f,80-bf:6000-7fff size=0x2000
: map address=70-71:0000-ffff map address=70-71:0000-ffff
board id:1CB5B-20 board: 1CB5B-20
:superfx superfx
: map address=00-3f,80-bf:3000-34ff map address=00-3f,80-bf:3000-34ff
: rom rom
: map address=00-3f:8000-ffff mask=0x8000 map address=00-3f:8000-ffff mask=0x8000
: map address=40-5f:0000-ffff map address=40-5f:0000-ffff
: ram ram
: map address=00-3f,80-bf:6000-7fff size=0x2000 map address=00-3f,80-bf:6000-7fff size=0x2000
: map address=70-71:0000-ffff map address=70-71:0000-ffff
board id:1CB7B-01 board: 1CB7B-01
:superfx superfx
: map address=00-3f,80-bf:3000-34ff map address=00-3f,80-bf:3000-34ff
: rom rom
: map address=00-3f:8000-ffff mask=0x8000 map address=00-3f:8000-ffff mask=0x8000
: map address=40-5f:0000-ffff map address=40-5f:0000-ffff
: ram ram
: map address=00-3f,80-bf:6000-7fff size=0x2000 map address=00-3f,80-bf:6000-7fff size=0x2000
: map address=70-71:0000-ffff map address=70-71:0000-ffff
board id:1DC0N-01 board: 1DC0N-01
:hitachidsp model=HG51B169 frequency=20000000 hitachidsp model=HG51B169 frequency=20000000
: map address=00-3f,80-bf:6c00-6fff,7c00-7fff map address=00-3f,80-bf:6c00-6fff,7c00-7fff
: rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x8000 map address=00-3f,80-bf:8000-ffff mask=0x8000
: ram ram
: map address=70-77:0000-7fff mask=0x8000 map address=70-77:0000-7fff mask=0x8000
: drom drom
: dram dram
: map address=00-3f,80-bf:6000-6bff,7000-7bff mask=0xf000 map address=00-3f,80-bf:6000-6bff,7000-7bff mask=0xf000
board id:1DS0B-20 board: 1DS0B-20
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
:necdsp model=uPD96050 frequency=11000000 necdsp model=uPD96050 frequency=11000000
: map address=60-67,e0-e7:0000-3fff map address=60-67,e0-e7:0000-3fff
: prom prom
: drom drom
: dram dram
: map address=68-6f,e8-ef:0000-7fff mask=0x8000 map address=68-6f,e8-ef:0000-7fff mask=0x8000
board id:1J0N-(01,10,20) board: 1J0N-(01,10,20)
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
board id:1J1M-(11,20) board: 1J1M-(11,20)
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
:ram ram
: map address=20-3f,a0-bf:6000-7fff mask=0xe000 map address=20-3f,a0-bf:6000-7fff mask=0xe000
board id:1J3B-01 board: 1J3B-01
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
:ram ram
: map address=20-3f,a0-bf:6000-7fff mask=0xe000 map address=20-3f,a0-bf:6000-7fff mask=0xe000
board id:1J3M-(01,11,20) board: 1J3M-(01,11,20)
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
:ram ram
: map address=20-3f,a0-bf:6000-7fff mask=0xe000 map address=20-3f,a0-bf:6000-7fff mask=0xe000
board id:1J5M-(11,20) board: 1J5M-(11,20)
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
:ram ram
: map address=20-3f,a0-bf:6000-7fff mask=0xe000 map address=20-3f,a0-bf:6000-7fff mask=0xe000
board id:1K0N-01 board: 1K0N-01
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
:necdsp model=uPD7725 frequency=8000000 necdsp model=uPD7725 frequency=8000000
: map address=00-1f,80-9f:6000-7fff mask=0xfff map address=00-1f,80-9f:6000-7fff mask=0xfff
: prom prom
: drom drom
: dram dram
board id:1K1B-01 board: 1K1B-01
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
:ram ram
: map address=20-3f,a0-bf:6000-7fff mask=0xe000 map address=20-3f,a0-bf:6000-7fff mask=0xe000
:necdsp model=uPD7725 frequency=8000000 necdsp model=uPD7725 frequency=8000000
: map address=00-1f,80-9f:6000-7fff mask=0xfff map address=00-1f,80-9f:6000-7fff mask=0xfff
: prom prom
: drom drom
: dram dram
board id:1L0N3S-01 board: 1L0N3S-01
:sa1 sa1
: map address=00-3f,80-bf:2200-23ff map address=00-3f,80-bf:2200-23ff
: rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x408000 map address=00-3f,80-bf:8000-ffff mask=0x408000
: map address=c0-ff:0000-ffff map address=c0-ff:0000-ffff
: bwram bwram
: map address=00-3f,80-bf:6000-7fff size=0x2000 map address=00-3f,80-bf:6000-7fff size=0x2000
: map address=40-4f:0000-ffff map address=40-4f:0000-ffff
: iram iram
: map address=00-3f,80-bf:3000-37ff size=0x800 map address=00-3f,80-bf:3000-37ff size=0x800
board id:1L3B-(02,11) board: 1L3B-(02,11)
:sa1 sa1
: map address=00-3f,80-bf:2200-23ff map address=00-3f,80-bf:2200-23ff
: rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x408000 map address=00-3f,80-bf:8000-ffff mask=0x408000
: map address=c0-ff:0000-ffff map address=c0-ff:0000-ffff
: bwram bwram
: map address=00-3f,80-bf:6000-7fff size=0x2000 map address=00-3f,80-bf:6000-7fff size=0x2000
: map address=40-4f:0000-ffff map address=40-4f:0000-ffff
: iram iram
: map address=00-3f,80-bf:3000-37ff size=0x800 map address=00-3f,80-bf:3000-37ff size=0x800
board id:1L5B-(11,20) board: 1L5B-(11,20)
:sa1 sa1
: map address=00-3f,80-bf:2200-23ff map address=00-3f,80-bf:2200-23ff
: rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x408000 map address=00-3f,80-bf:8000-ffff mask=0x408000
: map address=c0-ff:0000-ffff map address=c0-ff:0000-ffff
: bwram bwram
: map address=00-3f,80-bf:6000-7fff size=0x2000 map address=00-3f,80-bf:6000-7fff size=0x2000
: map address=40-4f:0000-ffff map address=40-4f:0000-ffff
: iram iram
: map address=00-3f,80-bf:3000-37ff size=0x800 map address=00-3f,80-bf:3000-37ff size=0x800
board id:1N0N-01 board: 1N0N-01
:sdd1 sdd1
: map address=00-3f,80-bf:4800-480f map address=00-3f,80-bf:4800-480f
: rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=c0-ff:0000-ffff map address=c0-ff:0000-ffff
board id:2A0N-(01,10,11,20) board: 2A0N-(01,10,11,20)
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
: map address=40-7d,c0-ff:0000-7fff mask=0x8000 map address=40-7d,c0-ff:0000-7fff mask=0x8000
board id:2A1M-01 board: 2A1M-01
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:2A3B-01 board: 2A3B-01
:rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x8000 map address=00-3f,80-bf:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:2A3M-01#R board: 2A3M-01#R
:rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x8000 map address=00-3f,80-bf:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:2A3M-(01,11,20) board: 2A3M-(01,11,20)
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:2A5M-01 board: 2A5M-01
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:2B3B-01 board: 2B3B-01
:rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x8000 map address=00-3f,80-bf:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
:necdsp model=uPD7725 frequency=8000000 necdsp model=uPD7725 frequency=8000000
: map address=60-6f,e0-ef:0000-7fff mask=0x3fff map address=60-6f,e0-ef:0000-7fff mask=0x3fff
: prom prom
: drom drom
: dram dram
board id:2DC0N-01 board: 2DC0N-01
:hitachidsp model=HG51B169 frequency=20000000 hitachidsp model=HG51B169 frequency=20000000
: map address=00-3f,80-bf:6c00-6fff,7c00-7fff map address=00-3f,80-bf:6c00-6fff,7c00-7fff
: rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x8000 map address=00-3f,80-bf:8000-ffff mask=0x8000
: ram ram
: map address=70-77:0000-7fff mask=0x8000 map address=70-77:0000-7fff mask=0x8000
: drom drom
: dram dram
: map address=00-3f,80-bf:6000-6bff,7000-7bff mask=0xf000 map address=00-3f,80-bf:6000-6bff,7000-7bff mask=0xf000
board id:2E3M-01 board: 2E3M-01
:rom rom
: map address=00-3f,80-bf:8000-ffff mask=0x8000 map address=00-3f,80-bf:8000-ffff mask=0x8000
:obc1 obc1
: map address=00-3f,80-bf:6000-7fff mask=0xe000 map address=00-3f,80-bf:6000-7fff mask=0xe000
: map address=70-71,f0-f1:6000-7fff,e000-ffff mask=0xe000 map address=70-71,f0-f1:6000-7fff,e000-ffff mask=0xe000
: ram ram
board id:2J0N-(01,10,11,20) board: 2J0N-(01,10,11,20)
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
board id:2J3M-(01,11,20) board: 2J3M-(01,11,20)
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
:ram ram
: map address=10-1f,30-3f,90-9f,b0-bf:6000-7fff mask=0xe000 map address=10-1f,30-3f,90-9f,b0-bf:6000-7fff mask=0xe000
board id:2J5M-01 board: 2J5M-01
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
:ram ram
: map address=10-1f,30-3f,90-9f,b0-bf:6000-7fff mask=0xe000 map address=10-1f,30-3f,90-9f,b0-bf:6000-7fff mask=0xe000
board id:3J0N-01 board: 3J0N-01
:rom rom
: map address=00-2f,80-af:8000-ffff map address=00-2f,80-af:8000-ffff
: map address=40-6f,c0-ef:0000-ffff map address=40-6f,c0-ef:0000-ffff
board id:BA0N-(01,10) board: BA0N-(01,10)
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
: map address=40-7d,c0-ff:0000-7fff mask=0x8000 map address=40-7d,c0-ff:0000-7fff mask=0x8000
board id:BA1M-01 board: BA1M-01
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:BA3M-(01,10) board: BA3M-(01,10)
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
:ram ram
: map address=70-7d,f0-ff:0000-7fff mask=0x8000 map address=70-7d,f0-ff:0000-7fff mask=0x8000
board id:BJ0N-(01,20) board: BJ0N-(01,20)
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
board id:BJ1M-10 board: BJ1M-10
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
:ram ram
: map address=20-3f,a0-bf:6000-7fff mask=0xe000 map address=20-3f,a0-bf:6000-7fff mask=0xe000
board id:BJ3M-10 board: BJ3M-10
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff
:ram ram
: map address=20-3f,a0-bf:6000-7fff mask=0xe000 map address=20-3f,a0-bf:6000-7fff mask=0xe000
board id:SGB-R-10 board: SGB-R-10
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
: map address=40-7d,c0-ff:0000-7fff mask=0x8000 map address=40-7d,c0-ff:0000-7fff mask=0x8000
:icd2 revision=1 icd2 revision=1
: map address=00-3f,80-bf:6000-67ff,7000-7fff map address=00-3f,80-bf:6000-67ff,7000-7fff
: rom rom
board id:YA0N-01 board: YA0N-01
:rom rom
: map address=00-7d,80-ff:8000-ffff mask=0x8000 map address=00-7d,80-ff:8000-ffff mask=0x8000
: map address=40-7d,c0-ff:0000-7fff mask=0x8000 map address=40-7d,c0-ff:0000-7fff mask=0x8000
board id:YJ0N-01 board: YJ0N-01
:rom rom
: map address=00-3f,80-bf:8000-ffff map address=00-3f,80-bf:8000-ffff
: map address=40-7d,c0-ff:0000-ffff map address=40-7d,c0-ff:0000-ffff

View File

@ -11,10 +11,9 @@ auto Icarus::superFamicomManifest(string location) -> string {
} }
auto Icarus::superFamicomManifest(vector<uint8_t>& buffer, string location) -> string { auto Icarus::superFamicomManifest(vector<uint8_t>& buffer, string location) -> string {
string markup;
string digest = Hash::SHA256(buffer).digest(); string digest = Hash::SHA256(buffer).digest();
if(settings["icarus/UseDatabase"].boolean() && !markup) { if(settings["icarus/UseDatabase"].boolean()) {
for(auto game : database.superFamicom.find("game")) { for(auto game : database.superFamicom.find("game")) {
if(game["sha256"].text() == digest) { if(game["sha256"].text() == digest) {
return BML::serialize(game); return BML::serialize(game);
@ -22,20 +21,21 @@ auto Icarus::superFamicomManifest(vector<uint8_t>& buffer, string location) -> s
} }
} }
if(settings["icarus/UseHeuristics"].boolean() && !markup) { if(settings["icarus/UseHeuristics"].boolean()) {
bool hasMSU1 = exists({location, "msu1.rom"}); bool hasMSU1 = exists({location, "msu1.rom"});
SuperFamicomCartridge cartridge{buffer.data(), buffer.size(), hasMSU1}; Heuristics::SuperFamicom game{buffer.data(), buffer.size(), hasMSU1};
if(markup = cartridge.markup) { if(string manifest = game.manifest()) {
markup.append("\n"); // markup.append("\n");
markup.append("game\n"); // markup.append("game\n");
markup.append(" sha256: ", digest, "\n"); // markup.append(" sha256: ", digest, "\n");
markup.append(" region: ", cartridge.region == SuperFamicomCartridge::Region::NTSC ? "NTSC" : "PAL", "\n"); // markup.append(" region: ", cartridge.region == SuperFamicomCartridge::Region::NTSC ? "NTSC" : "PAL", "\n");
markup.append(" label: ", Location::prefix(location), "\n"); // markup.append(" label: ", Location::prefix(location), "\n");
markup.append(" note: ", "heuristically generated by icarus\n"); // markup.append(" note: ", "heuristically generated by icarus\n");
return manifest;
} }
} }
return markup; return "";
} }
auto Icarus::superFamicomManifestScan(vector<Markup::Node>& roms, Markup::Node node) -> void { auto Icarus::superFamicomManifestScan(vector<Markup::Node>& roms, Markup::Node node) -> void {

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,856 @@
struct SuperFamicomCartridge {
SuperFamicomCartridge(const uint8_t* data, uint size, bool has_msu1 = false);
string markup;
//private:
auto readHeader(const uint8_t* data, uint size) -> void;
auto findHeader(const uint8_t* data, uint size) -> uint;
auto scoreHeader(const uint8_t* data, uint size, uint addr) -> uint;
enum HeaderField : uint {
CartName = 0x00,
Mapper = 0x15,
RomType = 0x16,
RomSize = 0x17,
RamSize = 0x18,
CartRegion = 0x19,
Company = 0x1a,
Version = 0x1b,
Complement = 0x1c, //inverse checksum
Checksum = 0x1e,
ResetVector = 0x3c,
};
enum class Type : uint {
SatellaviewBIOS,
SufamiTurboBIOS,
SuperGameBoy1BIOS,
SuperGameBoy2BIOS,
LoROM,
HiROM,
ExLoROM,
ExHiROM,
SuperFX,
SA1,
LoROMSatellaview,
HiROMSatellaview,
CampusChallenge92,
Powerfest94,
//invalid types
Unknown,
GameBoy,
Satellaview,
SufamiTurbo,
};
enum class Region : uint {
NTSC,
PAL,
};
enum class DSP1Type : uint {
None,
LoROM1MB,
LoROM2MB,
HiROM,
};
bool loaded = false; //is a base cartridge inserted?
uint crc32 = 0; //crc32 of all cartridges (base+slot(s))
uint rom_size = 0;
uint ram_size = 0;
uint firmware_size = 0;
string firmware_missing;
Type type = Type::Unknown;
Region region = Region::NTSC;
DSP1Type dsp1_type = DSP1Type::None;
bool has_bsx_slot = false;
bool has_superfx = false;
bool has_sa1 = false;
bool has_sharprtc = false;
bool has_epsonrtc = false;
bool has_sdd1 = false;
bool has_spc7110 = false;
bool has_cx4 = false;
bool has_dsp1 = false;
bool has_dsp2 = false;
bool has_dsp3 = false;
bool has_dsp4 = false;
bool has_obc1 = false;
bool has_st010 = false;
bool has_st011 = false;
bool has_st018 = false;
};
SuperFamicomCartridge::SuperFamicomCartridge(const uint8_t* data, uint size, bool has_msu1) {
//skip copier header
if((size & 0x7fff) == 512) data += 512, size -= 512;
//ignore images too small to be valid
if(size < 0x8000) return;
readHeader(data, size);
if(type == Type::Unknown) return;
if(type == Type::GameBoy) return;
if(type == Type::Satellaview) return;
if(type == Type::SufamiTurbo) return;
const char* range = (rom_size > 0x200000) || (ram_size > 32 * 1024) ? "0000-7fff" : "0000-ffff";
markup.append("board region=", region == Region::NTSC ? "ntsc" : "pal", "\n");
//detect firmware
if(has_dsp1) {
if((size & 0x7fff) == 0x2000) {
firmware_size = 0x2000;
} else {
firmware_missing = "DSP1";
}
}
if(has_dsp2) {
if((size & 0x7fff) == 0x2000) {
firmware_size = 0x2000;
} else {
firmware_missing = "DSP2";
}
}
if(has_dsp3) {
if((size & 0x7fff) == 0x2000) {
firmware_size = 0x2000;
} else {
firmware_missing = "DSP3";
}
}
if(has_dsp4) {
if((size & 0x7fff) == 0x2000) {
firmware_size = 0x2000;
} else {
firmware_missing = "DSP4";
}
}
if(has_st010) {
if((size & 0xffff) == 0xd000) {
firmware_size = 0xd000;
} else {
firmware_missing = "ST010";
}
}
if(has_st011) {
if((size & 0xffff) == 0xd000) {
firmware_size = 0xd000;
} else {
firmware_missing = "ST011";
}
}
if(has_st018) {
if((size & 0x3ffff) == 0x28000) {
firmware_size = 0x28000;
} else {
firmware_missing = "ST018";
}
}
if(has_cx4) {
if((rom_size & 0x7fff) == 0xc00) {
firmware_size = 0xc00;
} else {
firmware_missing = "CX4";
}
}
if(type == Type::SuperGameBoy1BIOS) {
if((rom_size & 0x7fff) == 0x100) {
firmware_size = 0x100;
} else {
firmware_missing = "SGB1";
}
}
if(type == Type::SuperGameBoy2BIOS) {
if((rom_size & 0x7fff) == 0x100) {
firmware_size = 0x100;
} else {
firmware_missing = "SGB2";
}
}
rom_size -= firmware_size;
//end firmware detection
if(type == Type::SatellaviewBIOS) {
markup.append(
" ram name=save.ram size=0x", hex(ram_size), "\n"
" map address=10-1f:5000-5fff mask=0xf000\n"
" mcc\n"
" map address=00-0f:5000\n"
" map=mcu address=00-3f,80-bf:8000-ffff mask=0x408000\n"
" map=mcu address=40-7d,c0-ff:0000-ffff\n"
" rom name=program.rom size=0x", hex(rom_size), "\n"
" ram name=download.ram size=0x80000\n"
" map address=00-3f,80-bf:6000-7fff mask=0xe000\n"
" bsmemory\n"
);
}
else if(type == Type::SufamiTurboBIOS) {
markup.append(
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-1f,80-9f:8000-ffff mask=0x8000\n"
" sufamiturbo\n"
" rom\n"
" map address=20-3f,a0-bf:8000-ffff mask=0x8000\n"
" ram\n"
" map address=60-6f,e0-ef:0000-ffff\n"
" sufamiturbo\n"
" rom\n"
" map address=40-5f,c0-df:0000-7fff mask=0x8000\n"
" map address=40-5f,c0-df:8000-ffff mask=0x8000\n"
" ram\n"
" map address=70-7d,f0-ff:0000-ffff\n"
);
}
else if(type == Type::SuperGameBoy1BIOS) {
markup.append(
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-7d,80-ff:8000-ffff mask=0x8000\n"
" map address=40-7d,c0-ff:0000-7fff mask=0x8000\n"
" icd2 revision=1\n"
" map address=00-3f,80-bf:6000-67ff,7000-7fff\n"
" rom name=sgb1.boot.rom size=0x100\n"
);
}
else if(type == Type::SuperGameBoy2BIOS) {
markup.append(
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-7d,80-ff:8000-ffff mask=0x8000\n"
" map address=40-7d,c0-ff:0000-7fff mask=0x8000\n"
" icd2 revision=2\n"
" map address=00-3f,80-bf:6000-67ff,7000-7fff\n"
" rom name=sgb2.boot.rom size=0x100\n"
);
}
else if(has_cx4) {
markup.append(
" hitachidsp model=HG51B169 frequency=20000000\n"
" map address=00-3f,80-bf:6c00-6fff,7c00-7fff\n"
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-3f,80-bf:8000-ffff mask=0x8000\n"
" ram name=save.ram size=0\n"
" map address=70-77:0000-7fff mask=0x8000\n"
" drom name=cx4.data.rom size=0xc00\n"
" dram name=cx4.data.ram size=0xc00 volatile\n"
" map address=00-3f,80-bf:6000-6bff,7000-7bff mask=0xf000\n"
);
}
else if(has_spc7110) {
markup.append(
" spc7110\n"
" map address=00-3f,80-bf:4800-483f\n"
" map address=50,58:0000-ffff\n"
" map=mcu address=00-3f,80-bf:8000-ffff mask=0x800000\n"
" map=mcu address=c0-ff:0000-ffff mask=0xc00000\n"
" prom name=program.rom size=0x100000\n"
" drom name=data.rom size=0x", hex(rom_size - 0x100000), "\n"
" ram name=save.ram size=0x", hex(ram_size), "\n"
" map address=00-3f,80-bf:6000-7fff mask=0xe000\n"
);
}
else if(has_sdd1) {
markup.append(
" sdd1\n"
" map address=00-3f,80-bf:4800-480f\n"
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-3f,80-bf:8000-ffff\n"
" map address=c0-ff:0000-ffff\n"
);
if(ram_size > 0) markup.append(
" ram name=save.ram size=0x", hex(ram_size), "\n"
" map address=00-3f,80-bf:6000-7fff mask=0xe000\n"
" map address=70-73:0000-ffff mask=0x8000\n"
);
}
else if(type == Type::LoROM) {
markup.append(
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-7d,80-ff:8000-ffff mask=0x8000\n"
" map address=40-6f,c0-ef:0000-7fff mask=0x8000\n"
);
if(ram_size > 0) markup.append(
" ram name=save.ram size=0x", hex(ram_size), "\n"
" map address=70-7d,f0-ff:", range, "\n"
);
}
else if(type == Type::HiROM) {
markup.append(
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-3f,80-bf:8000-ffff\n"
" map address=40-7f,c0-ff:0000-ffff\n"
);
if(ram_size > 0) markup.append(
" ram name=save.ram size=0x", hex(ram_size), "\n"
" map address=10-3f,90-bf:6000-7fff mask=0xe000\n"
);
}
else if(type == Type::ExLoROM) {
markup.append(
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-3f,80-bf:8000-ffff mask=0x8000\n"
" map address=40-7d:0000-ffff\n"
);
if(ram_size > 0) markup.append(
" ram name=save.ram size=0x", hex(ram_size), "\n"
" map address=20-3f,a0-bf:6000-7fff mask=0xe000\n"
" map address=70-7d:0000-7fff mask=0x8000\n"
);
}
else if(type == Type::ExHiROM) {
markup.append(
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-3f:8000-ffff base=0x400000\n"
" map address=40-7d:0000-ffff base=0x400000\n"
" map address=80-bf:8000-ffff mask=0xc00000\n"
" map address=c0-ff:0000-ffff mask=0xc00000\n"
);
if(ram_size > 0) markup.append(
" ram name=save.ram size=0x", hex(ram_size), "\n"
" map address=20-3f,a0-bf:6000-7fff mask=0xe000\n"
" map address=70-7d:", range, "\n"
);
}
else if(type == Type::SuperFX) {
markup.append(
" superfx\n"
" map address=00-3f,80-bf:3000-34ff\n"
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-3f,80-bf:8000-ffff mask=0x8000\n"
" map address=40-5f,c0-df:0000-ffff\n"
);
if(ram_size > 0) markup.append(
" ram name=save.ram size=0x", hex(ram_size), "\n"
" map address=00-3f,80-bf:6000-7fff size=0x2000\n"
" map address=70-71,f0-f1:0000-ffff\n"
);
}
else if(type == Type::SA1) {
markup.append(
" sa1\n"
" map address=00-3f,80-bf:2200-23ff\n"
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-3f,80-bf:8000-ffff mask=0x408000\n"
" map address=c0-ff:0000-ffff\n"
);
if(ram_size > 0) markup.append(
" bwram name=save.ram size=0x", hex(ram_size), "\n"
" map address=00-3f,80-bf:6000-7fff size=0x2000\n"
" map address=40-4f:0000-ffff\n"
);
markup.append(
" iram id=internal size=0x800 volatile\n"
" map address=00-3f,80-bf:3000-37ff size=0x800\n"
);
}
else if(type == Type::LoROMSatellaview) {
markup.append(
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-1f:8000-ffff base=0x000000 mask=0x8000\n"
" map address=20-3f:8000-ffff base=0x100000 mask=0x8000\n"
" map address=80-9f:8000-ffff base=0x200000 mask=0x8000\n"
" map address=a0-bf:8000-ffff base=0x100000 mask=0x8000\n"
" ram name=save.ram size=0x", hex(ram_size), "\n"
" map address=70-7d,f0-ff:0000-7fff mask=0x8000\n"
" bsmemory\n"
" map address=c0-ef:0000-ffff\n"
);
}
else if(type == Type::HiROMSatellaview) {
markup.append(
" rom name=program.rom size=0x", hex(rom_size), "\n"
" map address=00-1f,80-9f:8000-ffff\n"
" map address=40-5f,c0-df:0000-ffff\n"
" ram name=save.ram size=0x", hex(ram_size), "\n"
" map address=20-3f,a0-bf:6000-7fff\n"
" bsmemory\n"
" map address=20-3f,a0-bf:8000-ffff\n"
" map address=60-7f,e0-ff:0000-ffff\n"
);
}
else if(type == Type::CampusChallenge92) {
markup.append(
" event=CC92 timer=360\n"
" map address=c0,e0:0000\n"
" map=mcu address=00-1f,80-9f:8000-ffff\n"
" rom name=program.rom size=0x40000\n"
" rom name=slot-1.rom size=0x80000\n"
" rom name=slot-2.rom size=0x80000\n"
" rom name=slot-3.rom size=0x80000\n"
" ram name=save.ram size=0x2000 volatile\n"
" map address=70-7d,f0-ff:0000-7fff mask=0x8000\n"
" necdsp model=uPD7725 frequency=8000000\n"
" map address=20-3f,a0-bf:8000-ffff mask=0x7fff\n"
" prom name=dsp1.program.rom size=0x1800\n"
" drom name=dsp1.data.rom size=0x800\n"
" dram name=dsp1.data.ram size=0x200 volatile\n"
);
return;
}
else if(type == Type::Powerfest94) {
markup.append(
" event=PF94 timer=360\n"
" map address=10,20:6000\n"
" map=mcu address=00-3f,80-bf:8000-ffff\n"
" map=mcu address=c0-ff:0000-ffff\n"
" rom name=program.rom size=0x40000\n"
" rom name=slot-1.rom size=0x80000\n"
" rom name=slot-2.rom size=0x80000\n"
" rom name=slot-3.rom size=0x100000\n"
" ram name=save.ram size=0x2000 volatile\n"
" map address=30-3f,b0-bf:6000-7fff mask=0xe000\n"
" necdsp model=uPD7725 frequency=8000000\n"
" map address=00-0f,80-8f:6000-7fff mask=0xfff\n"
" prom name=dsp1.program.rom size=0x1800\n"
" drom name=dsp1.data.rom size=0x800\n"
" dram name=dsp1.data.ram size=0x200 volatile\n"
);
return;
}
if(has_sharprtc) {
markup.append(
" sharprtc\n"
" map address=00-3f,80-bf:2800-2801\n"
" ram name=rtc.ram size=0x10\n"
);
}
if(has_epsonrtc) {
markup.append(
" epsonrtc\n"
" map address=00-3f,80-bf:4840-4842\n"
" ram name=rtc.ram size=0x10\n"
);
}
if(has_obc1) {
markup.append(
" obc1\n"
" map address=00-3f,80-bf:6000-7fff mask=0xe000\n"
" ram name=save.ram size=0x2000\n"
);
}
if(has_dsp1) {
markup.append(
" necdsp model=uPD7725 frequency=8000000\n"
);
if(dsp1_type == DSP1Type::LoROM1MB) markup.append(
" map address=20-3f,a0-bf:8000-ffff mask=0x3fff\n"
);
if(dsp1_type == DSP1Type::LoROM2MB) markup.append(
" map address=60-6f,e0-ef:0000-7fff mask=0x3fff\n"
);
if(dsp1_type == DSP1Type::HiROM) markup.append(
" map address=00-1f,80-9f:6000-7fff mask=0xfff\n"
);
markup.append(
" prom name=dsp1b.program.rom size=0x1800\n"
" drom name=dsp1b.data.rom size=0x800\n"
" dram name=dsp1b.data.ram size=0x200 volatile\n"
);
}
if(has_dsp2) {
markup.append(
" necdsp model=uPD7725 frequency=8000000\n"
" map address=20-3f,a0-bf:8000-ffff mask=0x3fff\n"
" prom name=dsp2.program.rom size=0x1800\n"
" drom name=dsp2.data.rom size=0x800\n"
" dram name=dsp2.data.ram size=0x200 volatile\n"
);
}
if(has_dsp3) {
markup.append(
" necdsp model=uPD7725 frequency=8000000\n"
" map address=20-3f,a0-bf:8000-ffff mask=0x3fff\n"
" prom name=dsp3.program.rom size=0x1800\n"
" drom name=dsp3.data.rom size=0x800\n"
" dram name=dsp3.data.ram size=0x200 volatile\n"
);
}
if(has_dsp4) {
markup.append(
" necdsp model=uPD7725 frequency=8000000\n"
" map address=30-3f,b0-bf:8000-ffff mask=0x3fff\n"
" prom name=dsp4.program.rom size=0x1800\n"
" drom name=dsp4.data.rom size=0x800\n"
" dram name=dsp4.data.ram size=0x200 volatile\n"
);
}
if(has_st010) {
markup.append(
" necdsp model=uPD96050 frequency=11000000\n"
" map address=60-67,e0-e7:0000-3fff\n"
" prom name=st010.program.rom size=0xc000\n"
" drom name=st010.data.rom size=0x1000\n"
" dram name=save.ram size=0x1000\n"
" map address=68-6f,e8-ef:0000-7fff mask=0x8000\n"
);
}
if(has_st011) {
markup.append(
" necdsp model=uPD96050 frequency=15000000\n"
" map address=60-67,e0-e7:0000-3fff\n"
" prom name=st011.program.rom size=0xc000\n"
" drom name=st011.data.rom size=0x1000\n"
" dram name=save.ram size=0x1000\n"
" map address=68-6f,e8-ef:0000-7fff mask=0x8000\n"
);
}
if(has_st018) {
markup.append(
" armdsp frequency=21477272\n"
" map address=00-3f,80-bf:3800-38ff\n"
" prom name=st018.program.rom size=0x20000\n"
" drom name=st018.data.rom size=0x8000\n"
" ram name=save.ram size=0x4000\n"
);
}
if(has_msu1) {
markup.append(
" msu1\n"
" map address=00-3f,80-bf:2000-2007\n"
" rom name=msu1.rom\n"
);
}
}
auto SuperFamicomCartridge::readHeader(const uint8_t* data, uint size) -> void {
//detect Game Boy carts
if(size >= 0x0140) {
if(data[0x0104] == 0xce && data[0x0105] == 0xed && data[0x0106] == 0x66 && data[0x0107] == 0x66
&& data[0x0108] == 0xcc && data[0x0109] == 0x0d && data[0x010a] == 0x00 && data[0x010b] == 0x0b) {
type = Type::GameBoy;
return;
}
}
const uint index = findHeader(data, size);
const uint8_t mapperid = data[index + Mapper];
const uint8_t rom_type = data[index + RomType];
const uint8_t rom_size = data[index + RomSize];
const uint8_t company = data[index + Company];
const uint8_t regionid = data[index + CartRegion] & 0x7f;
const uint16_t complement = data[index + Complement] | data[index + Complement + 1] << 8;
const uint16_t checksum = data[index + Checksum] | data[index + Checksum + 1] << 8;
this->rom_size = size;
ram_size = 1024 << (data[index + RamSize] & 7);
if(ram_size == 1024) ram_size = 0; //no RAM present
if(rom_size == 0 && ram_size) ram_size = 0; //fix for Bazooka Blitzkrieg's malformed header (swapped ROM and RAM sizes)
//0, 1, 13 = NTSC; 2 - 12 = PAL
region = (regionid <= 1 || regionid >= 13) ? Region::NTSC : Region::PAL;
//detect BS-X flash carts
if(data[index + 0x13] == 0x00 || data[index + 0x13] == 0xff) {
if(data[index + 0x14] == 0x00) {
const uint8_t n15 = data[index + 0x15];
if(n15 == 0x00 || n15 == 0x80 || n15 == 0x84 || n15 == 0x9c || n15 == 0xbc || n15 == 0xfc) {
if(data[index + 0x1a] == 0x33 || data[index + 0x1a] == 0xff) {
type = Type::Satellaview;
region = Region::NTSC; //BS-X only released in Japan
return;
}
}
}
}
//detect Sufami Turbo carts
if(!memcmp(data, "BANDAI SFC-ADX", 14)) {
if(!memcmp(data + 16, "SFC-ADX BACKUP", 14)) {
type = Type::SufamiTurboBIOS;
} else {
type = Type::SufamiTurbo;
}
region = Region::NTSC; //Sufami Turbo only released in Japan
return; //RAM size handled outside this routine
}
//detect Super Game Boy BIOS
if(!memcmp(data + index, "Super GAMEBOY2", 14)) {
type = Type::SuperGameBoy2BIOS;
return;
}
if(!memcmp(data + index, "Super GAMEBOY", 13)) {
type = Type::SuperGameBoy1BIOS;
return;
}
//detect competition carts
if(!memcmp(data + index, "\x00\x08\x22\x02\x1c\x00\x10\x00\x08\x65\x80\x84\x20\x00\x22\x25\x00\x83\x0c\x80\x10", 21)
&& complement == 0x0100 && checksum == 0x2d02 && (size == 0x1c0000 || size == 0x1c2000)) {
type = Type::CampusChallenge92; //dark title screen version
return;
}
if(!memcmp(data + index, "\xc9\x80\x80\x44\x15\x00\x62\x09\x29\xa0\x52\x70\x50\x12\x05\x35\x31\x63\xc0\x22\x01", 21)
&& complement == 0x2011 && checksum == 0xf8c0 && (size == 0x240000 || size == 0x242000)) {
type = Type::Powerfest94; //10,000 points version
return;
}
if(!memcmp(data + index, "PREHISTORIK MAN ", 21)
&& complement == 0xffff && checksum == 0x0000 && (size == 0x240000 || size == 0x242000)) {
type = Type::Powerfest94; //1,000,000 points version
return;
}
//detect presence of BS-X flash cartridge connector (reads extended header information)
if(data[index - 14] == 'Z') {
if(data[index - 11] == 'J') {
uint8_t n13 = data[index - 13];
if((n13 >= 'A' && n13 <= 'Z') || (n13 >= '0' && n13 <= '9')) {
if(company == 0x33 || (data[index - 10] == 0x00 && data[index - 4] == 0x00)) {
has_bsx_slot = true;
}
}
}
}
if(has_bsx_slot) {
if(!memcmp(data + index, "Satellaview BS-X ", 21)) {
//BS-X base cart
type = Type::SatellaviewBIOS;
region = Region::NTSC; //BS-X only released in Japan
return; //RAM size handled internally by load_cart_bsx() -> BSXCart class
} else {
type = (index == 0x7fc0 ? Type::LoROMSatellaview : Type::HiROMSatellaview);
region = Region::NTSC; //BS-X slotted cartridges only released in Japan
}
} else {
//standard cart
if(index == 0x7fc0 && size >= 0x401000) {
type = Type::ExLoROM;
} else if(index == 0x7fc0 && mapperid == 0x32) {
type = Type::ExLoROM;
} else if(index == 0x7fc0) {
type = Type::LoROM;
} else if(index == 0xffc0) {
type = Type::HiROM;
} else if(index == 0x40ffc0) {
type = Type::ExHiROM;
} else {
type = Type::Unknown;
return;
}
}
if(mapperid == 0x20 && (rom_type == 0x13 || rom_type == 0x14 || rom_type == 0x15 || rom_type == 0x1a)) {
has_superfx = true;
type = Type::SuperFX;
ram_size = 1024 << (data[index - 3] & 7);
if(ram_size == 1024) ram_size = 0;
}
if(mapperid == 0x23 && (rom_type == 0x32 || rom_type == 0x34 || rom_type == 0x35)) {
has_sa1 = true;
type = Type::SA1;
}
if(mapperid == 0x35 && rom_type == 0x55) {
has_sharprtc = true;
}
if(mapperid == 0x32 && (rom_type == 0x43 || rom_type == 0x45)) {
has_sdd1 = true;
}
if(mapperid == 0x3a && (rom_type == 0xf5 || rom_type == 0xf9)) {
has_spc7110 = true;
has_epsonrtc = (rom_type == 0xf9);
}
if(mapperid == 0x20 && rom_type == 0xf3) {
has_cx4 = true;
}
if((mapperid == 0x20 || mapperid == 0x21) && rom_type == 0x03) {
has_dsp1 = true;
}
if(mapperid == 0x30 && rom_type == 0x05 && company != 0xb2) {
has_dsp1 = true;
}
if(mapperid == 0x31 && (rom_type == 0x03 || rom_type == 0x05)) {
has_dsp1 = true;
}
if(has_dsp1 == true) {
if((mapperid & 0x2f) == 0x20 && size <= 0x100000) {
dsp1_type = DSP1Type::LoROM1MB;
} else if((mapperid & 0x2f) == 0x20) {
dsp1_type = DSP1Type::LoROM2MB;
} else if((mapperid & 0x2f) == 0x21) {
dsp1_type = DSP1Type::HiROM;
}
}
if(mapperid == 0x20 && rom_type == 0x05) {
has_dsp2 = true;
}
if(mapperid == 0x30 && rom_type == 0x05 && company == 0xb2) {
has_dsp3 = true;
}
if(mapperid == 0x30 && rom_type == 0x03) {
has_dsp4 = true;
}
if(mapperid == 0x30 && rom_type == 0x25) {
has_obc1 = true;
}
if(mapperid == 0x30 && rom_type == 0xf6 && rom_size >= 10) {
has_st010 = true;
}
if(mapperid == 0x30 && rom_type == 0xf6 && rom_size < 10) {
has_st011 = true;
}
if(mapperid == 0x30 && rom_type == 0xf5) {
has_st018 = true;
}
}
auto SuperFamicomCartridge::findHeader(const uint8_t* data, uint size) -> uint {
uint score_lo = scoreHeader(data, size, 0x007fc0);
uint score_hi = scoreHeader(data, size, 0x00ffc0);
uint score_ex = scoreHeader(data, size, 0x40ffc0);
if(score_ex) score_ex += 4; //favor ExHiROM on images > 32mbits
if(score_lo >= score_hi && score_lo >= score_ex) {
return 0x007fc0;
} else if(score_hi >= score_ex) {
return 0x00ffc0;
} else {
return 0x40ffc0;
}
}
auto SuperFamicomCartridge::scoreHeader(const uint8_t* data, uint size, uint addr) -> uint {
if(size < addr + 64) return 0; //image too small to contain header at this location?
int score = 0;
uint16_t resetvector = data[addr + ResetVector] | (data[addr + ResetVector + 1] << 8);
uint16_t checksum = data[addr + Checksum ] | (data[addr + Checksum + 1] << 8);
uint16_t complement = data[addr + Complement ] | (data[addr + Complement + 1] << 8);
uint8_t resetop = data[(addr & ~0x7fff) | (resetvector & 0x7fff)]; //first opcode executed upon reset
uint8_t mapper = data[addr + Mapper] & ~0x10; //mask off irrelevent FastROM-capable bit
//$00:[000-7fff] contains uninitialized RAM and MMIO.
//reset vector must point to ROM at $00:[8000-ffff] to be considered valid.
if(resetvector < 0x8000) return 0;
//some images duplicate the header in multiple locations, and others have completely
//invalid header information that cannot be relied upon.
//below code will analyze the first opcode executed at the specified reset vector to
//determine the probability that this is the correct header.
//most likely opcodes
if(resetop == 0x78 //sei
|| resetop == 0x18 //clc (clc; xce)
|| resetop == 0x38 //sec (sec; xce)
|| resetop == 0x9c //stz $nnnn (stz $4200)
|| resetop == 0x4c //jmp $nnnn
|| resetop == 0x5c //jml $nnnnnn
) score += 8;
//plausible opcodes
if(resetop == 0xc2 //rep #$nn
|| resetop == 0xe2 //sep #$nn
|| resetop == 0xad //lda $nnnn
|| resetop == 0xae //ldx $nnnn
|| resetop == 0xac //ldy $nnnn
|| resetop == 0xaf //lda $nnnnnn
|| resetop == 0xa9 //lda #$nn
|| resetop == 0xa2 //ldx #$nn
|| resetop == 0xa0 //ldy #$nn
|| resetop == 0x20 //jsr $nnnn
|| resetop == 0x22 //jsl $nnnnnn
) score += 4;
//implausible opcodes
if(resetop == 0x40 //rti
|| resetop == 0x60 //rts
|| resetop == 0x6b //rtl
|| resetop == 0xcd //cmp $nnnn
|| resetop == 0xec //cpx $nnnn
|| resetop == 0xcc //cpy $nnnn
) score -= 4;
//least likely opcodes
if(resetop == 0x00 //brk #$nn
|| resetop == 0x02 //cop #$nn
|| resetop == 0xdb //stp
|| resetop == 0x42 //wdm
|| resetop == 0xff //sbc $nnnnnn,x
) score -= 8;
//at times, both the header and reset vector's first opcode will match ...
//fallback and rely on info validity in these cases to determine more likely header.
//a valid checksum is the biggest indicator of a valid header.
if((checksum + complement) == 0xffff && (checksum != 0) && (complement != 0)) score += 4;
if(addr == 0x007fc0 && mapper == 0x20) score += 2; //0x20 is usually LoROM
if(addr == 0x00ffc0 && mapper == 0x21) score += 2; //0x21 is usually HiROM
if(addr == 0x007fc0 && mapper == 0x22) score += 2; //0x22 is usually ExLoROM
if(addr == 0x40ffc0 && mapper == 0x25) score += 2; //0x25 is usually ExHiROM
if(data[addr + Company] == 0x33) score += 2; //0x33 indicates extended header
if(data[addr + RomType] < 0x08) score++;
if(data[addr + RomSize] < 0x10) score++;
if(data[addr + RamSize] < 0x08) score++;
if(data[addr + CartRegion] < 14) score++;
if(score < 0) score = 0;
return score;
}