wii7800/src/Cartridge.cpp
Chris Smith 3e4adaee79
Xm and more (#17)
* Added partial XM support
Added POT support to Pokey
Modifed Pokey and TIA Buffers
Fixed issue where pokey counter was not being reset
Added temporary hack to get Pac320 to work (may cause issues)

* Added Pokey @450 support
Added Support for carts about 144k
Improved cart detection (via header and by size)
Fixed issue occurring when Kangaroo and Holey were enabled (caused
backgroud to be displayed causing large black squares)

* Added ability to edit cartridge DB in UI
Added disable bios, left/right switch, swap buttons, pokey@450, and xm
to cartridge database.
Fixed Pokey pops that were occurring
Removed WSYNC, cycle stealing, and swap buttons from global settings

* Fixed bug with cart db UI and dual analog

* Added XM to save states

* Defaults to latest save state when loading cart

* Minor menu fix (able to select spacer)

* Reorganized advanced menu

* Added whether cart was loaded from db in debug

* Temporarily disable BIOS support
Will enable once issue is resolved

* Reduced TIA volume slightly

* Fixed controller for crossbow and alien brigade

* Added Bentley Bear's Crystal Quest to DB

* Bank switching fixes, new cart type
Fixed 64k cart bank switching when specifying out of range bank
Added new cartridge mode "Standard RAM". 64k or less w/ RAM at $4000
Removed global high score cart setting
Added per-cartridge setting for HSC (default to header value)

* Updates to prosystem database
Cartridge size uses file size (vs. header)
Default controllers to joysticks
Fix database load issue (sometimes read into next entry)
Added default cart title (when not in header or DB)
Added another default screen size
Fixed vsync issue (caused stutter, mostly on PAL titles)

* Many homebrew additions to cart database
Fixed issue where some IM hacks were not loading
Updated db editor to always write out pokey450 and XM settings

* More cart database entries

* Updated readme files and images for 0.5 release

* Updated readme

* Updated release date
2020-03-24 19:30:12 -07:00

797 lines
26 KiB
C++

// ----------------------------------------------------------------------------
// ___ ___ ___ ___ ___ ____ ___ _ _
// /__/ /__/ / / /__ /__/ /__ / /_ / |/ /
// / / \ /__/ ___/ ___/ ___/ / /__ / / emulator
//
// ----------------------------------------------------------------------------
// Copyright 2005 Greg Stanton
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; either version 2 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
// ----------------------------------------------------------------------------
// Cartridge.cpp
// ----------------------------------------------------------------------------
#include "Cartridge.h"
#include "Region.h"
#include "wii_app_common.h"
#include "wii_atari.h"
#include "wii_app.h"
#ifdef WII_NETTRACE
#include <network.h>
#include "net_print.h"
#endif
#include <string.h>
#define CARTRIDGE_SOURCE "Cartridge.cpp"
std::string cartridge_title;
std::string cartridge_description;
std::string cartridge_year;
std::string cartridge_maker;
std::string cartridge_digest;
std::string cartridge_filename;
byte cartridge_type;
byte cartridge_region;
bool cartridge_pokey;
bool cartridge_pokey450;
byte cartridge_controller[2] = {1, 1};
byte cartridge_bank;
uint cartridge_flags;
int cartridge_crosshair_x;
int cartridge_crosshair_y;
bool cartridge_dualanalog = false;
bool cartridge_xm = false;
bool cartridge_disable_bios = false;
uint cartridge_hblank = 34;
byte cartridge_left_switch = 1;
byte cartridge_right_switch = 0;
bool cartridge_swap_buttons = false;
bool cartridge_hsc_enabled = false;
// Whether the cartridge has accessed the high score ROM (indicates that the
// SRAM should be persisted when the cartridge is unloaded)
bool high_score_set = false;
// Whether the high score cart has been loaded
static bool high_score_cart_loaded = false;
static byte* cartridge_buffer = NULL;
static uint cartridge_size = 0;
// ----------------------------------------------------------------------------
// HasHeader
// ----------------------------------------------------------------------------
static bool cartridge_HasHeader(const byte* header) {
const char HEADER_ID[ ] = {"ATARI7800"};
for(int index = 0; index < 9; index++) {
if(HEADER_ID[index] != header[index + 1]) {
return false;
}
}
return true;
}
// 1.3
// ----------------------------------------------------------------------------
// Header for CC2 hack
// ----------------------------------------------------------------------------
static bool cartridge_CC2(const byte* header) {
const char HEADER_ID[ ] = {">>"};
for(int index = 0; index < 2; index++) {
if(HEADER_ID[index] != header[index+1]) {
return false;
}
}
return true;
}
// ----------------------------------------------------------------------------
// GetBank
// ----------------------------------------------------------------------------
static uint cartridge_GetBank(byte bank) {
if ((cartridge_type == CARTRIDGE_TYPE_SUPERCART || cartridge_type == CARTRIDGE_TYPE_SUPERCART_ROM || cartridge_type == CARTRIDGE_TYPE_SUPERCART_RAM) && cartridge_size <= 65536) {
// for some of these carts, there are only 4 banks. in this case we ignore bit 3
// previously, games of this type had to be doubled. The first 4 banks needed to be duplicated at the end of the ROM
return (bank & 3);
}
return bank;
}
// ----------------------------------------------------------------------------
// GetBankOffset
// ----------------------------------------------------------------------------
static uint cartridge_GetBankOffset(byte bank) {
return cartridge_GetBank(bank) * 16384;
}
// ----------------------------------------------------------------------------
// WriteBank
// ----------------------------------------------------------------------------
static void cartridge_WriteBank(word address, byte bank) {
#ifdef TRACE_BANK_SWITCH
net_print_string(NULL, 0, "Bank switch: %d, %d, max:%d\n",
address, cartridge_GetBank(bank), cartridge_size / 16384);
#endif
uint offset = cartridge_GetBankOffset(bank);
if(offset < cartridge_size) {
memory_WriteROM(address, 16384, cartridge_buffer + offset);
cartridge_bank = bank;
}
}
static void cartridge_SetTypeBySize(uint size) {
if (size <= 0x10000) {
int old_type = cartridge_type;
cartridge_type = CARTRIDGE_TYPE_NORMAL;
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Update: no bits and <= 64k: %d, %d\n", old_type, cartridge_type);
#endif
} else if (size == 0x24000 ) {
int old_type = cartridge_type;
cartridge_type = CARTRIDGE_TYPE_SUPERCART_LARGE;
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Update: size == 144k: %d, %d\n", old_type, cartridge_type);
#endif
} else if (size == 0x20000 ) {
int old_type = cartridge_type;
cartridge_type = CARTRIDGE_TYPE_SUPERCART_ROM;
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Update: size == 128k: %d, %d\n", old_type, cartridge_type);
#endif
} else {
int old_type = cartridge_type;
cartridge_type = CARTRIDGE_TYPE_SUPERCART;
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Update: default for > 64k: %d, %d\n", old_type, cartridge_type);
#endif
}
}
// ----------------------------------------------------------------------------
// ReadHeader
// ----------------------------------------------------------------------------
static void cartridge_ReadHeader(const byte* header) {
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Reading cartridge header\n");
#endif
char temp[33] = {0};
for(int index = 0; index < 32; index++) {
temp[index] = header[index + 17];
}
cartridge_title = temp;
cartridge_size = header[49] << 24;
cartridge_size |= header[50] << 16;
cartridge_size |= header[51] << 8;
cartridge_size |= header[52];
if(header[53] == 0) {
if(cartridge_size > 131072) {
cartridge_type = CARTRIDGE_TYPE_SUPERCART_LARGE;
}
else if(header[54] == 2 || header[54] == 3) {
cartridge_type = CARTRIDGE_TYPE_SUPERCART;
}
else if(header[54] == 4 || header[54] == 5 || header[54] == 6 || header[54] == 7) {
cartridge_type = CARTRIDGE_TYPE_SUPERCART_RAM;
}
else if(header[54] == 8 || header[54] == 9 || header[54] == 10 || header[54] == 11) {
cartridge_type = CARTRIDGE_TYPE_SUPERCART_ROM;
}
else {
cartridge_type = CARTRIDGE_TYPE_NORMAL;
}
}
else {
if(header[53] == 2 /*1*/) { // Wii: Abs and Act were swapped
cartridge_type = CARTRIDGE_TYPE_ABSOLUTE;
}
else if(header[53] == 1 /*2*/) { // Wii: Abs and Act were swapped
cartridge_type = CARTRIDGE_TYPE_ACTIVISION;
}
else {
cartridge_type = CARTRIDGE_TYPE_NORMAL;
}
}
cartridge_pokey = (header[54]&1)? true: false;
cartridge_pokey450 = (header[54]&0x40)? true : false;
if (cartridge_pokey450) {
cartridge_pokey = true;
}
cartridge_controller[0] = header[55];
cartridge_controller[1] = header[56];
cartridge_region = header[57];
cartridge_flags = 0;
cartridge_xm = (header[63] & 1)? true: false;
cartridge_hsc_enabled = header[58]&0x01;
// Wii: Updates to header interpretation
byte ct1 = header[54];
if(header[53] == 0) {
if ((ct1&0x0a)==0x0a) { // BIT1 and BIT3 (Supercart Large: 2) rom at $4000
int old_type = cartridge_type;
cartridge_type = CARTRIDGE_TYPE_SUPERCART_LARGE;
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Update: (0x10) bit1 & bit3: %d, %d\n", old_type, cartridge_type);
#endif
} else if ((ct1&0x12)==0x12) { // BIT1 and BIT4 (Supercart ROM: 4) bank6 at $4000
int old_type = cartridge_type;
cartridge_type = CARTRIDGE_TYPE_SUPERCART_ROM;
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Update: (0x12) bit1 & bit4: %d, %d\n", old_type, cartridge_type);
#endif
} else if ((ct1&0x06)==0x06) { // BIT1 and BIT2 (Supercart RAM: 3) ram at $4000
int old_type = cartridge_type;
cartridge_type = CARTRIDGE_TYPE_SUPERCART_RAM;
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Update: (0x06) bit1 & bit2: %d, %d\n", old_type, cartridge_type);
#endif
} else if ((ct1&0x02)==0x02) { // BIT1 (Supercart) bank switched
int old_type = cartridge_type;
cartridge_type = CARTRIDGE_TYPE_SUPERCART;
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Update: (0x01) bit1: %d, %d\n", old_type, cartridge_type);
#endif
} else if (cartridge_size <= 0x10000 && ((ct1&0x04)==0x04)) { // Size < 64k && BIT2 (Normal RAM: ?) ram at $4000 )
int old_type = cartridge_type;
cartridge_type = CARTRIDGE_TYPE_NORMAL_RAM;
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Update: (0x04) bit2: %d, %d\n", old_type, cartridge_type);
#endif
} else {
// Attempt to determine the cartridge type based on its size
cartridge_SetTypeBySize(cartridge_size);
}
}
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "Header info:\n");
if (ct1&0x01) {
net_print_string(NULL, 0, " bit0: pokey at $4000\n");
}
if (ct1&0x02) {
net_print_string(NULL, 0, " bit1: supergame bank switched\n");
}
if (ct1&0x04) {
net_print_string(NULL, 0, " bit2: supergame ram at $4000\n");
}
if (ct1&0x08) {
net_print_string(NULL, 0, " bit3: rom at $4000\n");
}
if (ct1&0x10) {
net_print_string(NULL, 0, " bit4: bank 6 at $4000\n");
}
if (ct1&0x20) {
net_print_string(NULL, 0, " bit5: supergame banked ram\n");
}
if (ct1&0x40) {
net_print_string(NULL, 0, " bit6: pokey at $450\n");
}
if (ct1&0x80) {
net_print_string(NULL, 0, " bit7: mirror ram at $4000\n");
}
net_print_string(NULL, 0, " xm: %s\n", (cartridge_xm ? "1" : "0"));
net_print_string(NULL, 0, " pokey: %s\n", (cartridge_pokey ? "1" : "0"));
net_print_string(NULL, 0, " pokey450: %s\n", (cartridge_pokey450 ? "1" : "0"));
net_print_string(NULL, 0, " tv type: %s\n", cartridge_region ? "PAL" : "NTSC");
net_print_string(NULL, 0, " Save device: [%x]%s%s\n", header[58],
((header[58]&0x02) ? " SaveKey/AtariVox": ""),
((header[58]&0x01) ? " HSC": ""));
net_print_string(NULL, 0, " controller1: %d\n", cartridge_controller[0]);
net_print_string(NULL, 0, " controller2: %d\n", cartridge_controller[1]);
net_print_string(NULL, 0, " cartridge_type 53: %d\n", header[53]);
net_print_string(NULL, 0, " cartridge_type 54: %d\n", header[54]);
net_print_string(NULL, 0, " cartridge_size: %d\n", cartridge_size);
net_print_string(NULL, 0, "cartridge_type (from header): %d\n", cartridge_type);
#endif
}
// ----------------------------------------------------------------------------
// Load
// ----------------------------------------------------------------------------
static bool cartridge_Load(const byte* data, uint size) {
if(size <= 128) {
logger_LogError("Cartridge data is invalid.", CARTRIDGE_SOURCE);
return false;
}
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "actual cartridge_size: %d\n", size);
#endif
cartridge_Release( );
byte header[128] = {0};
for(int index = 0; index < 128; index++) {
header[index] = data[index];
}
// 1.3
if (cartridge_CC2(header)) {
logger_LogError("Prosystem doesn't support CC2 hacks.", CARTRIDGE_SOURCE);
return false;
}
uint offset = 0;
if(cartridge_HasHeader(header)) {
cartridge_ReadHeader(header);
size -= 128;
offset = 128;
// Several cartridge headers do not have the proper size. So attempt to use the size
// of the file.
if (cartridge_size != size) {
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "!!! CARTRIDGE SIZE IN HEADER DOES NOT MATCH !!! : %d %d\n",
cartridge_size, size);
#endif
// Necessary for the following roms:
// Impossible Mission hacks w/ C64 style graphics
if (size%1024 == 0) {
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "!!! ROM size is 1k multiple, using ROM size !!! : %d\n",
size);
#endif
cartridge_size = size;
} else {
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "!!! ROM size is not 1k multiple, using header size !!! : %d\n",
cartridge_size);
#endif
}
}
}
else {
cartridge_size = size;
// Attempt to guess the cartridge type based on its size
cartridge_SetTypeBySize(size);
}
#ifdef WII_NETTRACE
net_print_string(NULL, 0, "cartridge_type: %d\n", cartridge_type);
net_print_string(NULL, 0, "cartridge_size: %d\n", cartridge_size);
#endif
cartridge_buffer = new byte[cartridge_size];
for(int index = 0; index < cartridge_size; index++) {
cartridge_buffer[index] = data[index + offset];
}
cartridge_digest = hash_Compute(cartridge_buffer, cartridge_size);
return true;
}
// ----------------------------------------------------------------------------
// Load
// ----------------------------------------------------------------------------
uint cartridge_Read(std::string filename, byte **outData ) {
byte *data = NULL;
uint size = archive_GetUncompressedFileSize(filename);
if(size == 0) {
FILE *file = fopen(filename.c_str( ), "rb");
if(file == NULL) {
logger_LogError("Failed to open the cartridge file " + filename + " for reading.", CARTRIDGE_SOURCE);
return 0;
}
if(fseek(file, 0L, SEEK_END)) {
fclose(file);
logger_LogError("Failed to find the end of the cartridge file.", CARTRIDGE_SOURCE);
return 0;
}
size = ftell(file);
if(fseek(file, 0L, SEEK_SET)) {
fclose(file);
logger_LogError("Failed to find the size of the cartridge file.", CARTRIDGE_SOURCE);
return 0;
}
data = new byte[size];
if(fread(data, 1, size, file) != size && ferror(file)) {
fclose(file);
logger_LogError("Failed to read the cartridge data.", CARTRIDGE_SOURCE);
cartridge_Release( );
delete [ ] data;
return 0;
}
fclose(file);
}
else {
data = new byte[size];
archive_Uncompress(filename, data, size);
}
*outData = data;
return size;
}
bool cartridge_Load(std::string filename) {
if(filename.empty( ) || filename.length( ) == 0) {
logger_LogError("Cartridge filename is invalid.", CARTRIDGE_SOURCE);
return false;
}
cartridge_Release();
logger_LogInfo("Opening cartridge file " + filename + ".");
byte *data = NULL;
uint size = cartridge_Read( filename, &data );
if( data == NULL )
{
return false;
}
if(!cartridge_Load(data, size)) {
logger_LogError("Failed to load the cartridge data into memory.", CARTRIDGE_SOURCE);
delete [ ] data;
return false;
}
if(data != NULL) {
delete [ ] data;
}
cartridge_filename = filename;
return true;
}
bool cartridge_Load_buffer(char* rom_buffer, int rom_size) {
cartridge_Release();
byte* data = (byte *)rom_buffer;
uint size = rom_size;
if(!cartridge_Load(data, size)) {
return false;
}
cartridge_filename = "";
return true;
}
// The memory location of the high score cartridge SRAM
#define HS_SRAM_START 0x1000
// The size of the high score cartridge SRAM
#define HS_SRAM_SIZE 2048
/*
* Saves the high score cartridge SRAM
*
* return Whether the save was successful
*/
bool cartridge_SaveHighScoreSram()
{
if( !high_score_cart_loaded || !high_score_set )
{
// If we didn't load the high score cartridge, or the game didn't
// access the high score ROM, don't save.
return false;
}
char sram_file[WII_MAX_PATH] = "";
snprintf(sram_file, WII_MAX_PATH, "%s%s", wii_get_fs_prefix(),
WII_HIGH_SCORE_CART_SRAM);
std::string filename(sram_file);
FILE* file = fopen(filename.c_str(), "wb");
if( file == NULL )
{
logger_LogError("Failed to open the file " + filename + " for writing.");
return false;
}
if( fwrite( &(memory_ram[HS_SRAM_START]), 1, HS_SRAM_SIZE, file ) != HS_SRAM_SIZE )
{
fclose( file );
logger_LogError("Failed to write highscore sram data to the file " + filename + ".");
return false;
}
fflush(file);
fclose(file);
return true;
}
/*
* Loads the high score cartridge SRAM
*
* return Whether the load was successful
*/
static bool cartridge_LoadHighScoreSram()
{
char sram_file[WII_MAX_PATH] = "";
snprintf(sram_file, WII_MAX_PATH, "%s%s", wii_get_fs_prefix(),
WII_HIGH_SCORE_CART_SRAM);
std::string filename( sram_file );
FILE* file = fopen( filename.c_str(), "rb" );
if( file == NULL )
{
return false;
}
byte sram[HS_SRAM_SIZE];
if( fread( sram, 1, HS_SRAM_SIZE, file ) != HS_SRAM_SIZE )
{
fclose( file );
logger_LogError("Failed to read highscore sram data from the file " + filename + ".");
return false;
}
for( uint i = 0; i < HS_SRAM_SIZE; i++ )
{
memory_Write( HS_SRAM_START + i, sram[i] );
}
fclose(file);
return true;
}
/*
* Loads the high score cartridge
*
* return Whether the load was successful
*/
bool cartridge_LoadHighScoreCart() {
if( !cartridge_hsc_enabled || cartridge_region != REGION_NTSC )
{
// Only load the cart if it is enabled and the region is NTSC
return false;
}
byte* high_score_buffer = NULL;
char high_score_cart[WII_MAX_PATH] = "";
snprintf(high_score_cart, WII_MAX_PATH, "%s%s", wii_get_fs_prefix(),
WII_HIGH_SCORE_CART);
uint hsSize = cartridge_Read( high_score_cart, &high_score_buffer );
if( high_score_buffer != NULL )
{
logger_LogInfo("Found high score cartridge.");
std::string digest = hash_Compute( high_score_buffer, hsSize );
if( digest == std::string("c8a73288ab97226c52602204ab894286") )
{
cartridge_LoadHighScoreSram();
for( uint i = 0; i < hsSize; i++ )
{
//memory_WriteROM( 0x3000, hsSize, high_score_buffer );
memory_Write( 0x3000 + i, high_score_buffer[i] );
}
high_score_cart_loaded = true;
}
else
{
logger_LogError("High score cartridge hash is invalid.");
}
delete [] high_score_buffer;
return high_score_cart_loaded;
}
else
{
logger_LogInfo("Unable to locate high score cartridge.");
}
return false;
}
// ----------------------------------------------------------------------------
// Store
// ----------------------------------------------------------------------------
void cartridge_Store( ) {
switch(cartridge_type) {
case CARTRIDGE_TYPE_NORMAL:
memory_WriteROM(65536 - cartridge_size, cartridge_size, cartridge_buffer);
break;
case CARTRIDGE_TYPE_NORMAL_RAM:
memory_WriteROM(65536 - cartridge_size, cartridge_size, cartridge_buffer);
memory_ClearROM(16384, 16384);
break;
case CARTRIDGE_TYPE_SUPERCART: {
uint offset = cartridge_size - 16384;
if(offset < cartridge_size) {
memory_WriteROM(49152, 16384, cartridge_buffer + offset);
}
} break;
case CARTRIDGE_TYPE_SUPERCART_LARGE: {
uint offset = cartridge_size - 16384;
if(offset < cartridge_size) {
memory_WriteROM(49152, 16384, cartridge_buffer + offset);
memory_WriteROM(16384, 16384, cartridge_buffer + cartridge_GetBankOffset(0));
}
} break;
case CARTRIDGE_TYPE_SUPERCART_RAM: {
uint offset = cartridge_size - 16384;
if(offset < cartridge_size) {
memory_WriteROM(49152, 16384, cartridge_buffer + offset);
memory_ClearROM(16384, 16384);
}
} break;
case CARTRIDGE_TYPE_SUPERCART_ROM: {
uint offset = cartridge_size - 16384;
if(offset < cartridge_size && cartridge_GetBankOffset(6) < cartridge_size) {
memory_WriteROM(49152, 16384, cartridge_buffer + offset);
memory_WriteROM(16384, 16384, cartridge_buffer + cartridge_GetBankOffset(6));
}
} break;
case CARTRIDGE_TYPE_ABSOLUTE:
memory_WriteROM(16384, 16384, cartridge_buffer);
memory_WriteROM(32768, 32768, cartridge_buffer + cartridge_GetBankOffset(2));
break;
case CARTRIDGE_TYPE_ACTIVISION:
if(122880 < cartridge_size) {
memory_WriteROM(40960, 16384, cartridge_buffer);
memory_WriteROM(16384, 8192, cartridge_buffer + 106496);
memory_WriteROM(24576, 8192, cartridge_buffer + 98304);
memory_WriteROM(32768, 8192, cartridge_buffer + 122880);
memory_WriteROM(57344, 8192, cartridge_buffer + 114688);
}
break;
}
}
// ----------------------------------------------------------------------------
// Write
// ----------------------------------------------------------------------------
void cartridge_Write(word address, byte data) {
#if 0
net_print_string(NULL, 0, "Cartridge write: %d, %d\n", address, data);
#endif
switch(cartridge_type) {
case CARTRIDGE_TYPE_SUPERCART:
case CARTRIDGE_TYPE_SUPERCART_RAM:
case CARTRIDGE_TYPE_SUPERCART_ROM: {
uint maxbank = cartridge_size / 16384;
if(address >= 32768 && address < 49152 && cartridge_GetBank(data) < maxbank /*9*/) {
cartridge_StoreBank(data);
}
} break;
case CARTRIDGE_TYPE_SUPERCART_LARGE: {
uint maxbank = cartridge_size / 16384;
if(address >= 32768 && address < 49152 && cartridge_GetBank(data) < maxbank /*9*/) {
cartridge_StoreBank(data + 1);
}
} break;
case CARTRIDGE_TYPE_ABSOLUTE:
if(address == 32768 && (data == 1 || data == 2)) {
cartridge_StoreBank(data - 1);
}
break;
case CARTRIDGE_TYPE_ACTIVISION:
if(address >= 65408) {
cartridge_StoreBank(address & 7);
}
break;
}
#if 0 // WIi: Moved to Memory.cpp
if(cartridge_pokey && address >= 0x4000 && address <= 0x400f) {
switch(address) {
case POKEY_AUDF1:
pokey_SetRegister(POKEY_AUDF1, data);
break;
case POKEY_AUDC1:
pokey_SetRegister(POKEY_AUDC1, data);
break;
case POKEY_AUDF2:
pokey_SetRegister(POKEY_AUDF2, data);
break;
case POKEY_AUDC2:
pokey_SetRegister(POKEY_AUDC2, data);
break;
case POKEY_AUDF3:
pokey_SetRegister(POKEY_AUDF3, data);
break;
case POKEY_AUDC3:
pokey_SetRegister(POKEY_AUDC3, data);
break;
case POKEY_AUDF4:
pokey_SetRegister(POKEY_AUDF4, data);
break;
case POKEY_AUDC4:
pokey_SetRegister(POKEY_AUDC4, data);
break;
case POKEY_AUDCTL:
pokey_SetRegister(POKEY_AUDCTL, data);
break;
case POKEY_SKCTLS:
pokey_SetRegister(POKEY_SKCTLS, data);
break;
}
}
#endif
}
// ----------------------------------------------------------------------------
// StoreBank
// ----------------------------------------------------------------------------
void cartridge_StoreBank(byte bank) {
switch(cartridge_type) {
case CARTRIDGE_TYPE_SUPERCART:
cartridge_WriteBank(32768, bank);
break;
case CARTRIDGE_TYPE_SUPERCART_RAM:
cartridge_WriteBank(32768, bank);
break;
case CARTRIDGE_TYPE_SUPERCART_ROM:
cartridge_WriteBank(32768, bank);
break;
case CARTRIDGE_TYPE_SUPERCART_LARGE:
cartridge_WriteBank(32768, bank);
break;
case CARTRIDGE_TYPE_ABSOLUTE:
cartridge_WriteBank(16384, bank);
break;
case CARTRIDGE_TYPE_ACTIVISION:
cartridge_WriteBank(40960, bank);
break;
}
}
// ----------------------------------------------------------------------------
// IsLoaded
// ----------------------------------------------------------------------------
bool cartridge_IsLoaded( ) {
return (cartridge_buffer != NULL)? true: false;
}
// ----------------------------------------------------------------------------
// Release
// ----------------------------------------------------------------------------
void cartridge_Release( ) {
high_score_cart_loaded = false;
if(cartridge_buffer != NULL) {
delete [ ] cartridge_buffer;
cartridge_size = 0;
cartridge_buffer = NULL;
//
// Wii
//
// These values need to be reset so that moving between carts works
// consistently. This seems to be a ProSystem emulator bug.
//
cartridge_title = "";
cartridge_type = 0;
cartridge_region = 0;
cartridge_pokey = 0;
cartridge_pokey450 = 0;
cartridge_xm = false;
// Default to joysticks
memset( cartridge_controller, 1, sizeof( cartridge_controller ) );
cartridge_bank = 0;
cartridge_flags = 0;
cartridge_disable_bios = false;
cartridge_crosshair_x = 0;
cartridge_crosshair_y = 0;
high_score_set = false;
cartridge_hblank = HBLANK_DEFAULT;
cartridge_dualanalog = false;
cartridge_left_switch = 1;
cartridge_right_switch = 0;
cartridge_swap_buttons = false;
cartridge_hsc_enabled = false;
}
}