fceugx/source/fceultra/boards/datalatch.cpp

445 lines
9.1 KiB
C++
Raw Normal View History

2008-04-03 03:58:35 +00:00
/* FCE Ultra - NES/Famicom Emulator
*
* Copyright notice for this file:
* Copyright (C) 2002 Xodnizel
*
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include "mapinc.h"
2012-01-09 01:59:06 +00:00
static uint8 latche, latcheinit, bus_conflict;
2008-09-20 01:02:49 +00:00
static uint16 addrreg0, addrreg1;
2010-12-06 02:46:57 +00:00
static uint8 *WRAM=NULL;
static uint32 WRAMSIZE;
2008-09-20 01:02:49 +00:00
static void(*WSync)(void);
2008-04-03 03:58:35 +00:00
2008-09-20 01:02:49 +00:00
static DECLFW(LatchWrite)
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
// FCEU_printf("bs %04x %02x\n",A,V);
2010-03-11 06:30:34 +00:00
if(bus_conflict)
latche=V&CartBR(A);
else
2008-09-20 01:02:49 +00:00
latche=V;
WSync();
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
static void LatchPower(void)
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
latche=latcheinit;
WSync();
2012-01-09 01:59:06 +00:00
if(WRAM)
{
SetReadHandler(0x6000,0xFFFF,CartBR);
SetWriteHandler(0x6000,0x7FFF,CartBW);
}
else
{
SetReadHandler(0x8000,0xFFFF,CartBR);
}
2008-09-20 01:02:49 +00:00
SetWriteHandler(addrreg0,addrreg1,LatchWrite);
2008-04-03 03:58:35 +00:00
}
2010-12-06 02:46:57 +00:00
static void LatchClose(void)
{
if(WRAM)
FCEU_gfree(WRAM);
WRAM=NULL;
}
2008-09-20 01:02:49 +00:00
static void StateRestore(int version)
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
WSync();
2008-04-03 03:58:35 +00:00
}
2012-01-09 01:59:06 +00:00
static void Latch_Init(CartInfo *info, void (*proc)(void), uint8 init, uint16 adr0, uint16 adr1, uint8 wram, uint8 busc)
2008-04-03 03:58:35 +00:00
{
2012-01-09 01:59:06 +00:00
bus_conflict = busc;
2008-09-20 01:02:49 +00:00
latcheinit=init;
addrreg0=adr0;
addrreg1=adr1;
WSync=proc;
info->Power=LatchPower;
2010-12-06 02:46:57 +00:00
info->Close=LatchClose;
2008-09-20 01:02:49 +00:00
GameStateRestore=StateRestore;
2010-12-06 02:46:57 +00:00
if(wram)
{
WRAMSIZE=8192;
WRAM=(uint8*)FCEU_gmalloc(WRAMSIZE);
SetupCartPRGMapping(0x10,WRAM,WRAMSIZE,1);
if(info->battery)
{
info->SaveGame[0]=WRAM;
info->SaveGameLen[0]=WRAMSIZE;
}
AddExState(WRAM, WRAMSIZE, 0, "WRAM");
}
2008-09-20 01:02:49 +00:00
AddExState(&latche, 1, 0, "LATC");
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
//------------------ CPROM ---------------------------
static void CPROMSync(void)
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
setchr4(0x0000,0);
setchr4(0x1000,latche&3);
setprg16(0x8000,0);
setprg16(0xC000,1);
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
void CPROM_Init(CartInfo *info)
2008-04-03 03:58:35 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, CPROMSync, 0, 0x8000, 0xFFFF, 0, 0);
2008-09-20 01:02:49 +00:00
}
//------------------ Map 184 ---------------------------
static void M184Sync(void)
{
setchr4(0x0000,latche);
setchr4(0x1000,latche>>4);
2008-04-03 03:58:35 +00:00
setprg16(0x8000,0);
setprg16(0xC000,1);
}
2008-09-20 01:02:49 +00:00
void Mapper184_Init(CartInfo *info)
2008-04-03 03:58:35 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M184Sync, 0, 0x6000, 0x7FFF, 0, 0);
2008-09-20 01:02:49 +00:00
}
//------------------ CNROM ---------------------------
static void CNROMSync(void)
{
2009-07-17 17:27:04 +00:00
//mbg 8/10/08 - fixed this so that large homebrew roms would work.
//setchr8(latche&3);
setchr8(latche);
setprg16(0x8000,0);
setprg16(0xC000,1);
2010-12-06 02:46:57 +00:00
setprg8r(0x10,0x6000,0); // Hayauchy IGO uses 2Kb or RAM
2008-04-03 03:58:35 +00:00
}
void CNROM_Init(CartInfo *info)
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, CNROMSync, 0, 0x8000, 0xFFFF, 1, 1);
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
//------------------ ANROM ---------------------------
static void ANROMSync()
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
setprg32(0x8000,latche&0xf);
setmirror(MI_0+((latche>>4)&1));
2008-04-03 03:58:35 +00:00
setchr8(0);
}
2008-09-20 01:02:49 +00:00
void ANROM_Init(CartInfo *info)
2008-04-03 03:58:35 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, ANROMSync, 0, 0x8000, 0xFFFF, 0, 0);
2008-09-20 01:02:49 +00:00
}
//------------------ Map 70 ---------------------------
static void M70Sync()
{
setprg16(0x8000,latche>>4);
setprg16(0xc000,~0);
setchr8(latche&0xf);
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
void Mapper70_Init(CartInfo *info)
2008-04-03 03:58:35 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M70Sync, 0, 0x8000, 0xFFFF, 0, 0);
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
//------------------ Map 152 ---------------------------
static void M152Sync()
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
setprg16(0x8000,(latche>>4)&7);
setprg16(0xc000,~0);
setchr8(latche&0xf);
setmirror(MI_0+((latche>>7)&1)); /* Saint Seiya...hmm. */
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
void Mapper152_Init(CartInfo *info)
2008-04-03 03:58:35 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M152Sync, 0, 0x8000, 0xFFFF, 0, 0);
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
//------------------ Map 78 ---------------------------
/* Should be two separate emulation functions for this "mapper". Sigh. URGE TO KILL RISING. */
static void M78Sync()
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
setprg16(0x8000,(latche&7));
setprg16(0xc000,~0);
setchr8(latche>>4);
setmirror(MI_0+((latche>>3)&1));
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
void Mapper78_Init(CartInfo *info)
2008-04-03 03:58:35 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M78Sync, 0, 0x8000, 0xFFFF, 0, 0);
2008-09-20 01:02:49 +00:00
}
//------------------ MHROM ---------------------------
static void MHROMSync(void)
{
setprg32(0x8000,latche>>4);
setchr8(latche&0xf);
2008-04-03 03:58:35 +00:00
}
2008-09-19 05:33:26 +00:00
void MHROM_Init(CartInfo *info)
2010-03-11 06:30:34 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, MHROMSync, 0, 0x8000, 0xFFFF, 0, 0);
2008-09-20 01:02:49 +00:00
}
void Mapper140_Init(CartInfo *info)
2010-03-11 06:30:34 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, MHROMSync, 0, 0x6000, 0x7FFF, 0, 0);
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
void Mapper240_Init(CartInfo *info)
2010-03-11 06:30:34 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, MHROMSync, 0, 0x4020, 0x5FFF, 0, 0);
2008-09-20 01:02:49 +00:00
// need SRAM.
}
//------------------ Map 87 ---------------------------
static void M87Sync(void)
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
setprg16(0x8000,0);
setprg16(0xC000,1);
2009-07-17 17:27:04 +00:00
setchr8(((latche>>1)&1)|((latche<<1)&2));
// setchr8(latche);
2008-09-20 01:02:49 +00:00
}
void Mapper87_Init(CartInfo *info)
2010-03-11 06:30:34 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M87Sync, ~0, 0x6000, 0xFFFF, 0, 0);
2008-04-03 03:58:35 +00:00
}
2009-07-17 17:27:04 +00:00
//------------------ Map 101 ---------------------------
static void M101Sync(void)
{
setprg16(0x8000,0);
setprg16(0xC000,1);
setchr8(latche);
}
void Mapper101_Init(CartInfo *info)
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M101Sync, ~0, 0x6000, 0x7FFF, 0, 0);
2009-07-17 17:27:04 +00:00
}
2008-09-20 01:02:49 +00:00
//------------------ Map 11 ---------------------------
static void M11Sync(void)
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
setprg32(0x8000,latche&0xf);
setchr8(latche>>4);
}
void Mapper11_Init(CartInfo *info)
2010-03-11 06:30:34 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M11Sync, 0, 0x8000, 0xFFFF, 0, 0);
2008-09-20 01:02:49 +00:00
}
void Mapper144_Init(CartInfo *info)
2010-03-11 06:30:34 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M11Sync, 0, 0x8001, 0xFFFF, 0, 0);
2008-04-03 03:58:35 +00:00
}
2009-07-17 17:27:04 +00:00
//------------------ Map 38 ---------------------------
static void M38Sync(void)
{
setprg32(0x8000,latche&3);
setchr8(latche>>2);
}
void Mapper38_Init(CartInfo *info)
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M38Sync, 0, 0x7000, 0x7FFF, 0, 0);
2009-07-17 17:27:04 +00:00
}
//------------------ Map 36 ---------------------------
static void M36Sync(void)
{
setprg32(0x8000,latche>>4);
setchr8((latche)&0xF);
}
void Mapper36_Init(CartInfo *info)
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M36Sync, 0, 0x8400, 0xfffe, 0, 0);
2009-07-17 17:27:04 +00:00
}
2008-09-20 01:02:49 +00:00
//------------------ UNROM ---------------------------
static void UNROMSync(void)
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
setprg16(0x8000,latche);
setprg16(0xc000,~0);
setchr8(0);
2008-04-03 03:58:35 +00:00
}
2008-09-19 05:33:26 +00:00
void UNROM_Init(CartInfo *info)
2008-04-03 03:58:35 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, UNROMSync, 0, 0x8000, 0xFFFF, 0, 1);
2008-09-20 01:02:49 +00:00
}
//------------------ Map 93 ---------------------------
static void SSUNROMSync(void)
{
setprg16(0x8000,latche>>4);
setprg16(0xc000,~0);
setchr8(0);
}
void SUNSOFT_UNROM_Init(CartInfo *info)
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, SSUNROMSync, 0, 0x8000, 0xFFFF, 0, 0);
2008-09-20 01:02:49 +00:00
}
//------------------ Map 94 ---------------------------
static void M94Sync(void)
{
setprg16(0x8000,latche>>2);
setprg16(0xc000,~0);
setchr8(0);
}
void Mapper94_Init(CartInfo *info)
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M94Sync, 0, 0x8000, 0xFFFF, 0, 0);
2008-09-20 01:02:49 +00:00
}
//------------------ Map 180 ---------------------------
static void M180Sync(void)
{
setprg16(0x8000,0);
setprg16(0xc000,latche);
setchr8(0);
}
void Mapper180_Init(CartInfo *info)
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M180Sync, 0, 0x8000, 0xFFFF, 0, 0);
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
//------------------ Map 107 ---------------------------
static void M107Sync(void)
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
setprg32(0x8000,(latche>>1)&3);
setchr8(latche&7);
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
void Mapper107_Init(CartInfo *info)
2008-04-03 03:58:35 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M107Sync, ~0, 0x8000, 0xFFFF, 0, 0);
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
//------------------ Map 113 ---------------------------
static void M113Sync(void)
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
setprg32(0x8000,(latche>>3)&7);
setchr8(((latche>>3)&8)|(latche&7));
// setmirror(latche>>7); // only for HES 6in1
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
void Mapper113_Init(CartInfo *info)
2008-04-03 03:58:35 +00:00
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, M113Sync, 0, 0x4100, 0x7FFF, 0, 0);
2008-09-20 01:02:49 +00:00
}
//------------------ A65AS ---------------------------
// actually, there is two cart in one... First have extra mirroring
// mode (one screen) and 32K bankswitching, second one have only
2010-03-11 06:30:34 +00:00
// 16 bankswitching mode and normal mirroring... But there is no any
2008-09-20 01:02:49 +00:00
// correlations between modes and they can be used in one mapper code.
static void BMCA65ASSync(void)
{
if(latche&0x40)
setprg32(0x8000,(latche>>1)&0x0F);
else
{
setprg16(0x8000,((latche&0x30)>>1)|(latche&7));
setprg16(0xC000,((latche&0x30)>>1)|7);
}
setchr8(0);
if(latche&0x80)
setmirror(MI_0+(((latche>>5)&1)));
else
setmirror(((latche>>3)&1)^1);
}
void BMCA65AS_Init(CartInfo *info)
{
2012-01-09 01:59:06 +00:00
Latch_Init(info, BMCA65ASSync, 0, 0x8000, 0xFFFF, 0, 0);
2008-09-20 01:02:49 +00:00
}
//------------------ NROM ---------------------------
#ifdef DEBUG_MAPPER
static DECLFW(WriteHandler)
{
FCEU_printf("bs %04x %02x\n",A,V);
2010-12-06 02:46:57 +00:00
CartBW(A,V);
2008-09-20 01:02:49 +00:00
}
#endif
static void NROMPower(void)
{
2010-12-06 02:46:57 +00:00
setprg8r(0x10,0x6000,0); // Famili BASIC (v3.0) need it (uses only 4KB), FP-BASIC uses 8KB
2008-09-20 01:02:49 +00:00
setprg16(0x8000,0);
setprg16(0xC000,~0);
setchr8(0);
2010-12-06 02:46:57 +00:00
2012-01-09 01:59:06 +00:00
SetReadHandler(0x6000,0x7FFF,CartBR);
SetWriteHandler(0x6000,0x7FFF,CartBW);
2008-09-20 01:02:49 +00:00
SetReadHandler(0x8000,0xFFFF,CartBR);
2010-12-06 02:46:57 +00:00
2008-09-20 01:02:49 +00:00
#ifdef DEBUG_MAPPER
SetWriteHandler(0x4020,0xFFFF,WriteHandler);
#endif
2008-04-03 03:58:35 +00:00
}
2008-09-20 01:02:49 +00:00
void NROM_Init(CartInfo *info)
2008-04-03 03:58:35 +00:00
{
2008-09-20 01:02:49 +00:00
info->Power=NROMPower;
2010-12-06 02:46:57 +00:00
info->Close=LatchClose;
WRAMSIZE=8192;
WRAM=(uint8*)FCEU_gmalloc(WRAMSIZE);
SetupCartPRGMapping(0x10,WRAM,WRAMSIZE,1);
if(info->battery)
{
info->SaveGame[0]=WRAM;
info->SaveGameLen[0]=WRAMSIZE;
}
AddExState(WRAM, WRAMSIZE, 0, "WRAM");
2008-04-03 03:58:35 +00:00
}