fceugx/source/fceultra/boards/vrc3.cpp
2016-09-17 20:43:24 -07:00

136 lines
3.3 KiB
C++

/* FCE Ultra - NES/Famicom Emulator
*
* Copyright notice for this file:
* Copyright (C) 2012 CaH4e3
*
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*
* VRC-3
*
*/
#include "mapinc.h"
static uint8 preg;
static uint8 IRQx; //autoenable
static uint8 IRQm; //mode
static uint8 IRQa;
static uint16 IRQReload, IRQCount;
static uint8 *WRAM = NULL;
static uint32 WRAMSIZE;
static SFORMAT StateRegs[] =
{
{ &preg, 1, "PREG" },
{ &IRQa, 1, "IRQA" },
{ &IRQx, 1, "IRQX" },
{ &IRQm, 1, "IRQM" },
{ &IRQReload, 2, "IRQR" },
{ &IRQCount, 2, "IRQC" },
{ 0 }
};
static void Sync(void) {
setprg8r(0x10, 0x6000, 0);
setprg16(0x8000, preg);
setprg16(0xC000, ~0);
setchr8(0);
}
static DECLFW(M73Write) {
switch (A & 0xF000) {
case 0x8000: IRQReload &= 0xFFF0; IRQReload |= (V & 0xF) << 0; break;
case 0x9000: IRQReload &= 0xFF0F; IRQReload |= (V & 0xF) << 4; break;
case 0xA000: IRQReload &= 0xF0FF; IRQReload |= (V & 0xF) << 8; break;
case 0xB000: IRQReload &= 0x0FFF; IRQReload |= (V & 0xF) << 12; break;
case 0xC000:
IRQm = V & 4;
IRQx = V & 1;
IRQa = V & 2;
if (IRQa) {
if (IRQm) {
IRQCount &= 0xFFFF;
IRQCount |= (IRQReload & 0xFF);
} else
IRQCount = IRQReload;
}
X6502_IRQEnd(FCEU_IQEXT);
break;
case 0xD000: X6502_IRQEnd(FCEU_IQEXT); IRQa = IRQx; break;
case 0xF000: preg = V; Sync(); break;
}
}
static void M73IRQHook(int a) {
int32 i;
if (!IRQa) return;
for (i = 0; i < a; i++) {
if (IRQm) {
uint16 temp = IRQCount;
temp &= 0xFF;
IRQCount &= 0xFF00;
if (temp == 0xFF) {
IRQCount = IRQReload;
IRQCount |= (uint16)(IRQReload & 0xFF);
X6502_IRQBegin(FCEU_IQEXT);
} else {
temp++;
IRQCount |= temp;
}
} else {
//16 bit mode
if (IRQCount == 0xFFFF) {
IRQCount = IRQReload;
X6502_IRQBegin(FCEU_IQEXT);
} else
IRQCount++;
}
}
}
static void M73Power(void) {
IRQReload = IRQm = IRQx = 0;
Sync();
SetReadHandler(0x6000, 0xFFFF, CartBR);
SetWriteHandler(0x6000, 0x7FFF, CartBW);
SetWriteHandler(0x8000, 0xFFFF, M73Write);
FCEU_CheatAddRAM(WRAMSIZE >> 10, 0x6000, WRAM);
}
static void M73Close(void)
{
if (WRAM)
FCEU_gfree(WRAM);
WRAM = NULL;
}
static void StateRestore(int version) {
Sync();
}
void Mapper73_Init(CartInfo *info) {
info->Power = M73Power;
info->Close = M73Close;
MapIRQHook = M73IRQHook;
WRAMSIZE = 8192;
WRAM = (uint8*)FCEU_gmalloc(WRAMSIZE);
SetupCartPRGMapping(0x10, WRAM, WRAMSIZE, 1);
AddExState(WRAM, WRAMSIZE, 0, "WRAM");
AddExState(&StateRegs, ~0, 0, 0);
GameStateRestore = StateRestore;
}