2016-07-09 14:34:55 +00:00
|
|
|
#include <stdbool.h>
|
2016-07-09 16:25:13 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2016-07-09 14:34:55 +00:00
|
|
|
#include "gb.h"
|
|
|
|
|
|
|
|
const GB_cartridge_t GB_cart_defs[256] = {
|
|
|
|
// From http://gbdev.gg8.se/wiki/articles/The_Cartridge_Header#0147_-_Cartridge_Type
|
|
|
|
/* MBC RAM BAT. RTC RUMB. */
|
|
|
|
{ GB_NO_MBC, false, false, false, false}, // 00h ROM ONLY
|
|
|
|
{ GB_MBC1 , false, false, false, false}, // 01h MBC1
|
|
|
|
{ GB_MBC1 , true , false, false, false}, // 02h MBC1+RAM
|
|
|
|
{ GB_MBC1 , true , true , false, false}, // 03h MBC1+RAM+BATTERY
|
|
|
|
[5] =
|
|
|
|
{ GB_MBC2 , true , false, false, false}, // 05h MBC2
|
|
|
|
{ GB_MBC2 , true , true , false, false}, // 06h MBC2+BATTERY
|
|
|
|
[8] =
|
|
|
|
{ GB_NO_MBC, true , false, false, false}, // 08h ROM+RAM
|
|
|
|
{ GB_NO_MBC, true , true , false, false}, // 09h ROM+RAM+BATTERY
|
|
|
|
[0xB] =
|
|
|
|
// Todo: What are these?
|
|
|
|
{ GB_NO_MBC, false, false, false, false}, // 0Bh MMM01
|
|
|
|
{ GB_NO_MBC, false, false, false, false}, // 0Ch MMM01+RAM
|
|
|
|
{ GB_NO_MBC, false, false, false, false}, // 0Dh MMM01+RAM+BATTERY
|
|
|
|
[0xF] =
|
|
|
|
{ GB_MBC3 , false, true, true , false}, // 0Fh MBC3+TIMER+BATTERY
|
|
|
|
{ GB_MBC3 , true , true, true , false}, // 10h MBC3+TIMER+RAM+BATTERY
|
|
|
|
{ GB_MBC3 , false, false, false, false}, // 11h MBC3
|
|
|
|
{ GB_MBC3 , true , false, false, false}, // 12h MBC3+RAM
|
|
|
|
{ GB_MBC3 , true , true , false, false}, // 13h MBC3+RAM+BATTERY
|
|
|
|
[0x15] =
|
|
|
|
// Todo: Do these exist?
|
|
|
|
{ GB_MBC4 , false, false, false, false}, // 15h MBC4
|
|
|
|
{ GB_MBC4 , true , false, false, false}, // 16h MBC4+RAM
|
|
|
|
{ GB_MBC4 , true , true , false, false}, // 17h MBC4+RAM+BATTERY
|
|
|
|
[0x19] =
|
|
|
|
{ GB_MBC5 , false, false, false, false}, // 19h MBC5
|
|
|
|
{ GB_MBC5 , true , false, false, false}, // 1Ah MBC5+RAM
|
|
|
|
{ GB_MBC5 , true , true , false, false}, // 1Bh MBC5+RAM+BATTERY
|
|
|
|
{ GB_MBC5 , false, false, false, true }, // 1Ch MBC5+RUMBLE
|
|
|
|
{ GB_MBC5 , true , false, false, true }, // 1Dh MBC5+RUMBLE+RAM
|
|
|
|
{ GB_MBC5 , true , true , false, true }, // 1Eh MBC5+RUMBLE+RAM+BATTERY
|
|
|
|
[0xFC] =
|
|
|
|
// Todo: What are these?
|
|
|
|
{ GB_NO_MBC, false, false, false, false}, // FCh POCKET CAMERA
|
|
|
|
{ GB_NO_MBC, false, false, false, false}, // FDh BANDAI TAMA5
|
|
|
|
{ GB_NO_MBC, false, false, false, false}, // FEh HuC3
|
|
|
|
{ GB_NO_MBC, true , true , false, false}, // FFh HuC1+RAM+BATTERY
|
|
|
|
};
|
|
|
|
|
|
|
|
void GB_update_mbc_mappings(GB_gameboy_t *gb)
|
|
|
|
{
|
|
|
|
switch (gb->cartridge_type->mbc_type) {
|
|
|
|
case GB_NO_MBC: case GB_MBC4: return;
|
|
|
|
case GB_MBC1:
|
2016-07-09 16:25:13 +00:00
|
|
|
/* Todo: some obscure behaviors of MBC1 are not supported. See http://forums.nesdev.com/viewtopic.php?f=20&t=14099 */
|
2016-07-09 14:34:55 +00:00
|
|
|
if (gb->mbc1.mode == 0) {
|
2016-07-09 16:25:13 +00:00
|
|
|
switch (gb->mbc1_wiring) {
|
|
|
|
case GB_STANDARD_MBC1_WIRING:
|
|
|
|
gb->mbc_rom_bank = gb->mbc1.bank_low | (gb->mbc1.bank_high << 5);
|
|
|
|
gb->mbc_ram_bank = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GB_MBC1M_WIRING:
|
|
|
|
gb->mbc_rom_bank = (gb->mbc1.bank_low & 0xF) | (gb->mbc1.bank_high << 4);
|
|
|
|
gb->mbc_ram_bank = 0;
|
|
|
|
gb->mbc_rom0_bank = 0;
|
|
|
|
}
|
2016-07-09 14:34:55 +00:00
|
|
|
}
|
|
|
|
else {
|
2016-07-09 16:25:13 +00:00
|
|
|
switch (gb->mbc1_wiring) {
|
|
|
|
case GB_STANDARD_MBC1_WIRING:
|
|
|
|
gb->mbc_rom_bank = gb->mbc1.bank_low;
|
|
|
|
gb->mbc_ram_bank = gb->mbc1.bank_high;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GB_MBC1M_WIRING:
|
|
|
|
gb->mbc_rom_bank = (gb->mbc1.bank_low & 0xF) | (gb->mbc1.bank_high << 4);
|
|
|
|
gb->mbc_rom0_bank = gb->mbc1.bank_high << 4;
|
|
|
|
gb->mbc_ram_bank = 0;
|
|
|
|
}
|
2016-07-09 14:34:55 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case GB_MBC2:
|
|
|
|
gb->mbc_rom_bank = gb->mbc2.rom_bank;
|
|
|
|
break;
|
|
|
|
case GB_MBC3:
|
|
|
|
gb->mbc_rom_bank = gb->mbc3.rom_bank;
|
|
|
|
gb->mbc_ram_bank = gb->mbc3.ram_bank;
|
|
|
|
break;
|
|
|
|
case GB_MBC5:
|
|
|
|
gb->mbc_rom_bank = gb->mbc5.rom_bank_low | (gb->mbc5.rom_bank_high << 8);
|
|
|
|
gb->mbc_ram_bank = gb->mbc5.ram_bank;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (gb->mbc_rom_bank == 0 && gb->cartridge_type->mbc_type != GB_MBC5) {
|
|
|
|
gb->mbc_rom_bank = 1;
|
|
|
|
}
|
2016-07-09 16:25:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GB_configure_cart(GB_gameboy_t *gb)
|
|
|
|
{
|
|
|
|
gb->cartridge_type = &GB_cart_defs[gb->rom[0x147]];
|
|
|
|
|
2016-09-03 01:00:37 +00:00
|
|
|
if (gb->rom[0x147] == 0xFF || (gb->rom[0x147] != 0 && memcmp(gb->cartridge_type, &GB_cart_defs[0], sizeof(GB_cart_defs[0])) == 0)) {
|
|
|
|
GB_log(gb, "Cartridge type %02x is not yet supported.\n", gb->rom[0x147]);
|
|
|
|
}
|
|
|
|
|
2016-07-09 16:25:13 +00:00
|
|
|
if (gb->cartridge_type->has_ram) {
|
2016-09-30 20:34:06 +00:00
|
|
|
if (gb->cartridge_type->mbc_type == GB_MBC2) {
|
|
|
|
gb->mbc_ram_size = 0x200;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
static const int ram_sizes[256] = {0, 0x800, 0x2000, 0x8000, 0x20000, 0x10000};
|
|
|
|
gb->mbc_ram_size = ram_sizes[gb->rom[0x149]];
|
|
|
|
}
|
2016-07-09 16:25:13 +00:00
|
|
|
gb->mbc_ram = malloc(gb->mbc_ram_size);
|
2016-09-30 20:34:06 +00:00
|
|
|
|
|
|
|
/* Todo: Some games assume unintialized MBC RAM is 0xFF. It this true for all cartridges types?*/
|
2016-09-10 10:25:18 +00:00
|
|
|
memset(gb->mbc_ram, 0xFF, gb->mbc_ram_size);
|
2016-07-09 16:25:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* MBC1 has at least 3 types of wiring (We currently support two (Standard and 4bit-MBC1M) of these).
|
|
|
|
See http://forums.nesdev.com/viewtopic.php?f=20&t=14099 */
|
|
|
|
|
|
|
|
/* Attempt to "guess" wiring */
|
|
|
|
if (gb->cartridge_type->mbc_type == GB_MBC1) {
|
|
|
|
if (gb->rom_size >= 0x44000 && memcmp(gb->rom + 0x104, gb->rom + 0x40104, 0x30) == 0) {
|
|
|
|
gb->mbc1_wiring = GB_MBC1M_WIRING;
|
|
|
|
}
|
|
|
|
}
|
2016-07-09 14:34:55 +00:00
|
|
|
}
|