mirror of
https://github.com/dborth/fceugx.git
synced 2025-01-10 15:49:26 +01:00
298 lines
8.4 KiB
C++
298 lines
8.4 KiB
C++
/* FCE Ultra - NES/Famicom Emulator
|
||
*
|
||
* Copyright notice for this file:
|
||
* Copyright (C) 2007-2010 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
|
||
*
|
||
* VR02/VT03 Console and OneBus System
|
||
*
|
||
* Street Dance (Dance pad) (Unl)
|
||
* 101-in-1 Arcade Action II
|
||
* DreamGEAR 75-in-1, etc.
|
||
*
|
||
*/
|
||
|
||
#include "mapinc.h"
|
||
|
||
// General Purpose Registers
|
||
static uint8 cpu410x[16], ppu201x[16], apu40xx[64];
|
||
|
||
// IRQ Registers
|
||
static uint8 IRQCount, IRQa, IRQReload;
|
||
#define IRQLatch cpu410x[0x1] // accc cccc, a = 0, AD12 switching, a = 1, HSYNC switching
|
||
|
||
// MMC3 Registers
|
||
static uint8 inv_hack = 0; // some OneBus Systems have swapped PRG reg commans in MMC3 inplementation,
|
||
// trying to autodetect unusual behavior, due not to add a new mapper.
|
||
#define mmc3cmd cpu410x[0x5] // pcv- ----, p - program swap, c - video swap, v - internal VRAM enable
|
||
#define mirror cpu410x[0x6] // ---- ---m, m = 0 - H, m = 1 - V
|
||
|
||
// APU Registers
|
||
static uint8 pcm_enable = 0, pcm_irq = 0;
|
||
static int16 pcm_addr, pcm_size, pcm_latch, pcm_clock = 0xE1;
|
||
|
||
static writefunc defapuwrite[64];
|
||
static readfunc defapuread[64];
|
||
|
||
static SFORMAT StateRegs[] =
|
||
{
|
||
{ cpu410x, 16, "REGC" },
|
||
{ ppu201x, 16, "REGS" },
|
||
{ apu40xx, 64, "REGA" },
|
||
{ &IRQReload, 1, "IRQR" },
|
||
{ &IRQCount, 1, "IRQC" },
|
||
{ &IRQa, 1, "IRQA" },
|
||
{ &pcm_enable, 1, "PCME" },
|
||
{ &pcm_irq, 1, "PCMI" },
|
||
{ &pcm_addr, 2, "PCMA" },
|
||
{ &pcm_size, 2, "PCMS" },
|
||
{ &pcm_latch, 2, "PCML" },
|
||
{ &pcm_clock, 2, "PCMC" },
|
||
{ 0 }
|
||
};
|
||
|
||
static void PSync(void) {
|
||
uint8 bankmode = cpu410x[0xb] & 7;
|
||
uint8 mask = (bankmode == 0x7) ? (0xff) : (0x3f >> bankmode);
|
||
uint32 block = ((cpu410x[0x0] & 0xf0) << 4) + (cpu410x[0xa] & (~mask));
|
||
uint32 pswap = (mmc3cmd & 0x40) << 8;
|
||
|
||
// uint8 bank0 = (cpu410x[0xb] & 0x40)?(~1):(cpu410x[0x7]);
|
||
// uint8 bank1 = cpu410x[0x8];
|
||
// uint8 bank2 = (cpu410x[0xb] & 0x40)?(cpu410x[0x9]):(~1);
|
||
// uint8 bank3 = ~0;
|
||
uint8 bank0 = cpu410x[0x7 ^ inv_hack];
|
||
uint8 bank1 = cpu410x[0x8 ^ inv_hack];
|
||
uint8 bank2 = (cpu410x[0xb] & 0x40) ? (cpu410x[0x9]) : (~1);
|
||
uint8 bank3 = ~0;
|
||
|
||
// FCEU_printf(" PRG: %04x [%02x]",0x8000^pswap,block | (bank0 & mask));
|
||
setprg8(0x8000 ^ pswap, block | (bank0 & mask));
|
||
// FCEU_printf(" %04x [%02x]",0xa000^pswap,block | (bank1 & mask));
|
||
setprg8(0xa000, block | (bank1 & mask));
|
||
// FCEU_printf(" %04x [%02x]",0xc000^pswap,block | (bank2 & mask));
|
||
setprg8(0xc000 ^ pswap, block | (bank2 & mask));
|
||
// FCEU_printf(" %04x [%02x]\n",0xe000^pswap,block | (bank3 & mask));
|
||
setprg8(0xe000, block | (bank3 & mask));
|
||
}
|
||
|
||
static void CSync(void) {
|
||
static const uint8 midx[8] = { 0, 1, 2, 0, 3, 4, 5, 0 };
|
||
uint8 mask = 0xff >> midx[ppu201x[0xa] & 7];
|
||
uint32 block = ((cpu410x[0x0] & 0x0f) << 11) + ((ppu201x[0x8] & 0x70) << 4) + (ppu201x[0xa] & (~mask));
|
||
uint32 cswap = (mmc3cmd & 0x80) << 5;
|
||
|
||
uint8 bank0 = ppu201x[0x6] & (~1);
|
||
uint8 bank1 = ppu201x[0x6] | 1;
|
||
uint8 bank2 = ppu201x[0x7] & (~1);
|
||
uint8 bank3 = ppu201x[0x7] | 1;
|
||
uint8 bank4 = ppu201x[0x2];
|
||
uint8 bank5 = ppu201x[0x3];
|
||
uint8 bank6 = ppu201x[0x4];
|
||
uint8 bank7 = ppu201x[0x5];
|
||
|
||
setchr1(0x0000 ^ cswap, block | (bank0 & mask));
|
||
setchr1(0x0400 ^ cswap, block | (bank1 & mask));
|
||
setchr1(0x0800 ^ cswap, block | (bank2 & mask));
|
||
setchr1(0x0c00 ^ cswap, block | (bank3 & mask));
|
||
setchr1(0x1000 ^ cswap, block | (bank4 & mask));
|
||
setchr1(0x1400 ^ cswap, block | (bank5 & mask));
|
||
setchr1(0x1800 ^ cswap, block | (bank6 & mask));
|
||
setchr1(0x1c00 ^ cswap, block | (bank7 & mask));
|
||
|
||
setmirror(mirror & 1);
|
||
}
|
||
|
||
static void Sync(void) {
|
||
PSync();
|
||
CSync();
|
||
}
|
||
|
||
static DECLFW(UNLOneBusWriteCPU410X) {
|
||
// FCEU_printf("CPU %04x:%04x\n",A,V);
|
||
switch (A & 0xf) {
|
||
case 0x1: IRQLatch = V & 0xfe; break; // <20><> <20><> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
||
case 0x2: IRQReload = 1; break;
|
||
case 0x3: X6502_IRQEnd(FCEU_IQEXT); IRQa = 0; break;
|
||
case 0x4: IRQa = 1; break;
|
||
default:
|
||
cpu410x[A & 0xf] = V;
|
||
Sync();
|
||
}
|
||
}
|
||
|
||
static DECLFW(UNLOneBusWritePPU201X) {
|
||
// FCEU_printf("PPU %04x:%04x\n",A,V);
|
||
ppu201x[A & 0x0f] = V;
|
||
Sync();
|
||
}
|
||
|
||
static DECLFW(UNLOneBusWriteMMC3) {
|
||
// FCEU_printf("MMC %04x:%04x\n",A,V);
|
||
switch (A & 0xe001) {
|
||
case 0x8000: mmc3cmd = (mmc3cmd & 0x38) | (V & 0xc7); Sync(); break;
|
||
case 0x8001:
|
||
{
|
||
switch (mmc3cmd & 7) {
|
||
case 0: ppu201x[0x6] = V; CSync(); break;
|
||
case 1: ppu201x[0x7] = V; CSync(); break;
|
||
case 2: ppu201x[0x2] = V; CSync(); break;
|
||
case 3: ppu201x[0x3] = V; CSync(); break;
|
||
case 4: ppu201x[0x4] = V; CSync(); break;
|
||
case 5: ppu201x[0x5] = V; CSync(); break;
|
||
case 6: cpu410x[0x7] = V; PSync(); break;
|
||
case 7: cpu410x[0x8] = V; PSync(); break;
|
||
}
|
||
break;
|
||
}
|
||
case 0xa000: mirror = V ^ 1; CSync(); break;
|
||
case 0xc000: IRQLatch = V & 0xfe; break;
|
||
case 0xc001: IRQReload = 1; break;
|
||
case 0xe000: X6502_IRQEnd(FCEU_IQEXT); IRQa = 0; break;
|
||
case 0xe001: IRQa = 1; break;
|
||
}
|
||
}
|
||
|
||
static void UNLOneBusIRQHook(void) {
|
||
uint32 count = IRQCount;
|
||
if (!count || IRQReload) {
|
||
IRQCount = IRQLatch;
|
||
IRQReload = 0;
|
||
} else
|
||
IRQCount--;
|
||
if (count && !IRQCount) {
|
||
if (IRQa)
|
||
X6502_IRQBegin(FCEU_IQEXT);
|
||
}
|
||
}
|
||
|
||
static DECLFW(UNLOneBusWriteAPU40XX) {
|
||
// if(((A & 0x3f)!=0x16) && ((apu40xx[0x30] & 0x10) || ((A & 0x3f)>0x17)))FCEU_printf("APU %04x:%04x\n",A,V);
|
||
apu40xx[A & 0x3f] = V;
|
||
switch (A & 0x3f) {
|
||
case 0x12:
|
||
if (apu40xx[0x30] & 0x10) {
|
||
pcm_addr = V << 6;
|
||
}
|
||
break;
|
||
case 0x13:
|
||
if (apu40xx[0x30] & 0x10) {
|
||
pcm_size = (V << 4) + 1;
|
||
}
|
||
break;
|
||
case 0x15:
|
||
if (apu40xx[0x30] & 0x10) {
|
||
pcm_enable = V & 0x10;
|
||
if (pcm_irq) {
|
||
X6502_IRQEnd(FCEU_IQEXT);
|
||
pcm_irq = 0;
|
||
}
|
||
if (pcm_enable)
|
||
pcm_latch = pcm_clock;
|
||
V &= 0xef;
|
||
}
|
||
break;
|
||
}
|
||
defapuwrite[A & 0x3f](A, V);
|
||
}
|
||
|
||
static DECLFR(UNLOneBusReadAPU40XX) {
|
||
uint8 result = defapuread[A & 0x3f](A);
|
||
// FCEU_printf("read %04x, %02x\n",A,result);
|
||
switch (A & 0x3f) {
|
||
case 0x15:
|
||
if (apu40xx[0x30] & 0x10) {
|
||
result = (result & 0x7f) | pcm_irq;
|
||
}
|
||
break;
|
||
}
|
||
return result;
|
||
}
|
||
|
||
static void UNLOneBusCpuHook(int a) {
|
||
if (pcm_enable) {
|
||
pcm_latch -= a;
|
||
if (pcm_latch <= 0) {
|
||
pcm_latch += pcm_clock;
|
||
pcm_size--;
|
||
if (pcm_size < 0) {
|
||
pcm_irq = 0x80;
|
||
pcm_enable = 0;
|
||
X6502_IRQBegin(FCEU_IQEXT);
|
||
} else {
|
||
uint16 addr = pcm_addr | ((apu40xx[0x30]^3) << 14);
|
||
uint8 raw_pcm = ARead[addr](addr) >> 1;
|
||
defapuwrite[0x11](0x4011, raw_pcm);
|
||
pcm_addr++;
|
||
pcm_addr &= 0x7FFF;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
static void UNLOneBusPower(void) {
|
||
uint32 i;
|
||
IRQReload = IRQCount = IRQa = 0;
|
||
|
||
memset(cpu410x, 0x00, sizeof(cpu410x));
|
||
memset(ppu201x, 0x00, sizeof(ppu201x));
|
||
memset(apu40xx, 0x00, sizeof(apu40xx));
|
||
|
||
SetupCartCHRMapping(0, PRGptr[0], PRGsize[0], 0);
|
||
|
||
for (i = 0; i < 64; i++) {
|
||
defapuread[i] = GetReadHandler(0x4000 | i);
|
||
defapuwrite[i] = GetWriteHandler(0x4000 | i);
|
||
}
|
||
SetReadHandler(0x4000, 0x403f, UNLOneBusReadAPU40XX);
|
||
SetWriteHandler(0x4000, 0x403f, UNLOneBusWriteAPU40XX);
|
||
|
||
SetReadHandler(0x8000, 0xFFFF, CartBR);
|
||
SetWriteHandler(0x2010, 0x201f, UNLOneBusWritePPU201X);
|
||
SetWriteHandler(0x4100, 0x410f, UNLOneBusWriteCPU410X);
|
||
SetWriteHandler(0x8000, 0xffff, UNLOneBusWriteMMC3);
|
||
|
||
Sync();
|
||
}
|
||
|
||
static void UNLOneBusReset(void) {
|
||
IRQReload = IRQCount = IRQa = 0;
|
||
|
||
memset(cpu410x, 0x00, sizeof(cpu410x));
|
||
memset(ppu201x, 0x00, sizeof(ppu201x));
|
||
memset(apu40xx, 0x00, sizeof(apu40xx));
|
||
|
||
Sync();
|
||
}
|
||
|
||
static void StateRestore(int version) {
|
||
Sync();
|
||
}
|
||
|
||
void UNLOneBus_Init(CartInfo *info) {
|
||
info->Power = UNLOneBusPower;
|
||
info->Reset = UNLOneBusReset;
|
||
|
||
if (((*(uint32*)&(info->MD5)) == 0x305fcdc3) || // PowerJoy Supermax Carts
|
||
((*(uint32*)&(info->MD5)) == 0x6abfce8e))
|
||
inv_hack = 0xf;
|
||
|
||
GameHBIRQHook = UNLOneBusIRQHook;
|
||
MapIRQHook = UNLOneBusCpuHook;
|
||
GameStateRestore = StateRestore;
|
||
AddExState(&StateRegs, ~0, 0, 0);
|
||
}
|