2011-09-18 02:49:09 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
|
|
|
|
|
namespace BizHawk.Emulation.Consoles.Nintendo
|
|
|
|
|
{
|
|
|
|
|
//AKA mapper 16 & 159
|
|
|
|
|
/*
|
|
|
|
|
Example Games:
|
|
|
|
|
--------------------------
|
|
|
|
|
Dragon Ball - Dai Maou Jukkatsu (016)
|
|
|
|
|
Dragon Ball Z Gaiden (016)
|
|
|
|
|
Dragon Ball Z 2 (016)
|
|
|
|
|
Rokudenashi Blues (016)
|
|
|
|
|
Akuma-kun - Makai no Wana (016)
|
|
|
|
|
Dragon Ball Z - Kyoushuu! Saiya Jin (159)
|
|
|
|
|
SD Gundam Gaiden (159)
|
|
|
|
|
Magical Taruruuto Kun 1, 2 (159)
|
|
|
|
|
|
|
|
|
|
PRG_ROM: 128KB
|
|
|
|
|
PRG_RAM: None
|
|
|
|
|
CHR-ROM: 128KB
|
|
|
|
|
CHR_RAM: None
|
|
|
|
|
No Batter
|
|
|
|
|
Mapper controlled mirroring
|
|
|
|
|
No CIC present
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
class BANDAI_FCG_1 : NES.NESBoardBase
|
|
|
|
|
{
|
|
|
|
|
//configuration
|
2011-09-18 16:14:55 +00:00
|
|
|
|
int prg_bank_mask_8k, chr_bank_mask_1k;
|
|
|
|
|
|
|
|
|
|
//state
|
|
|
|
|
byte prg_bank_8k, eprom;
|
|
|
|
|
ByteBuffer regs = new ByteBuffer(10);
|
|
|
|
|
ByteBuffer prg_banks_8k = new ByteBuffer(2);
|
2011-09-18 02:49:09 +00:00
|
|
|
|
bool irq_countdown, irq_enabled, irq_asserted;
|
|
|
|
|
ushort irq_counter;
|
|
|
|
|
int clock_counter;
|
|
|
|
|
|
2011-09-18 16:14:55 +00:00
|
|
|
|
public override void SyncState(Serializer ser)
|
|
|
|
|
{
|
|
|
|
|
base.SyncState(ser);
|
|
|
|
|
ser.Sync("chr_bank_mask_1k", ref chr_bank_mask_1k);
|
|
|
|
|
ser.Sync("irq_counter", ref irq_counter);
|
|
|
|
|
ser.Sync("irq_countdown", ref irq_countdown);
|
|
|
|
|
ser.Sync("irq_enabled", ref irq_enabled);
|
|
|
|
|
ser.Sync("irq_asserted", ref irq_asserted);
|
|
|
|
|
ser.Sync("clock_counter", ref clock_counter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override void Dispose()
|
|
|
|
|
{
|
|
|
|
|
base.Dispose();
|
|
|
|
|
regs.Dispose();
|
|
|
|
|
prg_banks_8k.Dispose();
|
|
|
|
|
}
|
2011-09-18 02:49:09 +00:00
|
|
|
|
|
|
|
|
|
public override bool Configure(NES.EDetectionOrigin origin)
|
|
|
|
|
{
|
|
|
|
|
switch (Cart.board_type)
|
|
|
|
|
{
|
|
|
|
|
case "BANDAI-FCG-1":
|
|
|
|
|
AssertPrg(128); AssertChr(128); AssertWram(0); AssertVram(0);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2011-09-18 16:14:55 +00:00
|
|
|
|
BaseConfigure();
|
2011-09-18 02:49:09 +00:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected void BaseConfigure()
|
|
|
|
|
{
|
|
|
|
|
chr_bank_mask_1k = Cart.chr_size - 1;
|
2011-09-18 16:14:55 +00:00
|
|
|
|
prg_bank_mask_8k = (Cart.prg_size / 8) - 1;
|
|
|
|
|
prg_banks_8k[1] = 0xFF;
|
2011-09-18 02:49:09 +00:00
|
|
|
|
SetMirrorType(EMirrorType.Vertical);
|
|
|
|
|
}
|
|
|
|
|
|
2011-09-18 16:14:55 +00:00
|
|
|
|
void SyncPRG()
|
2011-09-18 02:49:09 +00:00
|
|
|
|
{
|
2011-09-18 16:14:55 +00:00
|
|
|
|
prg_banks_8k[0] = regs[8];
|
2011-09-18 02:49:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override void WritePRG(int addr, byte value)
|
|
|
|
|
{
|
2011-09-18 16:14:55 +00:00
|
|
|
|
addr &= 0xC000;
|
2011-09-18 02:49:09 +00:00
|
|
|
|
switch (addr)
|
|
|
|
|
{
|
|
|
|
|
case 0: case 1: case 2: case 3:
|
|
|
|
|
case 4: case 5: case 6: case 7:
|
2011-09-18 16:14:55 +00:00
|
|
|
|
regs[addr] = value;
|
2011-09-18 02:49:09 +00:00
|
|
|
|
break;
|
|
|
|
|
case 8:
|
2011-09-18 16:14:55 +00:00
|
|
|
|
regs[8] = value;
|
|
|
|
|
SyncPRG();
|
2011-09-18 02:49:09 +00:00
|
|
|
|
break;
|
|
|
|
|
case 9:
|
2011-09-18 16:14:55 +00:00
|
|
|
|
switch (value & 3)
|
2011-09-18 02:49:09 +00:00
|
|
|
|
{
|
|
|
|
|
case 0: SetMirrorType(NES.NESBoardBase.EMirrorType.Vertical); break;
|
|
|
|
|
case 1: SetMirrorType(NES.NESBoardBase.EMirrorType.Horizontal); break;
|
|
|
|
|
case 2: SetMirrorType(NES.NESBoardBase.EMirrorType.OneScreenA); break;
|
|
|
|
|
case 3: SetMirrorType(NES.NESBoardBase.EMirrorType.OneScreenB); break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0xA:
|
|
|
|
|
irq_enabled = value.Bit(0);
|
2011-09-18 16:14:55 +00:00
|
|
|
|
if (!irq_enabled) irq_asserted = false;
|
|
|
|
|
SyncIrq();
|
2011-09-18 02:49:09 +00:00
|
|
|
|
break;
|
|
|
|
|
case 0xB:
|
|
|
|
|
irq_counter &= 0xFF00;
|
|
|
|
|
irq_counter |= value;
|
|
|
|
|
break;
|
|
|
|
|
case 0xC:
|
|
|
|
|
irq_counter &= 0x00FF;
|
|
|
|
|
irq_counter |= (ushort)(value << 8);
|
|
|
|
|
break;
|
|
|
|
|
case 0xD:
|
2011-09-18 16:14:55 +00:00
|
|
|
|
eprom = value;
|
2011-09-18 02:49:09 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-09-18 16:14:55 +00:00
|
|
|
|
void SyncIrq()
|
2011-09-18 02:49:09 +00:00
|
|
|
|
{
|
2011-09-18 16:14:55 +00:00
|
|
|
|
NES.irq_cart = irq_asserted;
|
2011-09-18 02:49:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-09-18 16:14:55 +00:00
|
|
|
|
void ClockCPU()
|
2011-09-18 02:49:09 +00:00
|
|
|
|
{
|
2011-09-18 16:14:55 +00:00
|
|
|
|
if (!irq_countdown) return;
|
|
|
|
|
irq_counter--;
|
|
|
|
|
if (irq_counter == 0x0000)
|
|
|
|
|
{
|
|
|
|
|
irq_asserted = true;
|
|
|
|
|
SyncIrq();
|
|
|
|
|
}
|
2011-09-18 02:49:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override void ClockPPU()
|
|
|
|
|
{
|
|
|
|
|
clock_counter++;
|
|
|
|
|
if (clock_counter == 3)
|
|
|
|
|
{
|
|
|
|
|
ClockCPU();
|
|
|
|
|
clock_counter = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-09-18 16:14:55 +00:00
|
|
|
|
public override byte ReadPRG(int addr)
|
2011-09-18 02:49:09 +00:00
|
|
|
|
{
|
2011-09-18 16:14:55 +00:00
|
|
|
|
int bank_8k = addr >> 13;
|
|
|
|
|
int ofs = addr & ((1 << 13) - 1);
|
|
|
|
|
bank_8k = prg_banks_8k[bank_8k];
|
|
|
|
|
bank_8k &= prg_bank_mask_8k;
|
|
|
|
|
addr = (prg_bank_8k << 13) | ofs;
|
|
|
|
|
return ROM[addr];
|
2011-09-18 02:49:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-09-18 16:14:55 +00:00
|
|
|
|
int CalcPPUAddress(int addr)
|
2011-09-18 02:49:09 +00:00
|
|
|
|
{
|
2011-09-18 16:14:55 +00:00
|
|
|
|
int bank_1k = addr >> 10;
|
|
|
|
|
int ofs = addr & ((1 << 10) - 1);
|
|
|
|
|
bank_1k = regs[bank_1k];
|
|
|
|
|
bank_1k &= chr_bank_mask_1k;
|
|
|
|
|
return (bank_1k << 10) | ofs;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override byte ReadPPU(int addr)
|
|
|
|
|
{
|
|
|
|
|
if (addr < 0x2000)
|
|
|
|
|
return VROM[CalcPPUAddress(addr)];
|
|
|
|
|
else return base.ReadPPU(addr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override void WritePPU(int addr, byte value)
|
|
|
|
|
{
|
|
|
|
|
if (addr < 0x2000) { }
|
|
|
|
|
base.WritePPU(addr, value);
|
2011-09-18 02:49:09 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|