BizHawk/BizHawk.Emulation/Consoles/Nintendo/NES/Boards/Taito_X1_017.cs

218 lines
4.7 KiB
C#

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace BizHawk.Emulation.Consoles.Nintendo
{
class Taito_X1_017 : NES.NESBoardBase
{
/*
ines Mapper 82
http://wiki.nesdev.com/w/index.php/INES_Mapper_082
* Example Games:
--------------------------
SD Keiji - Blader
Notes:
---------------------------
Regs appear at $7EFx, I'm unsure whether or not PRG-RAM can exist at $6000-7FFF
Registers:
---------------------------
$7EF0-7EF5: CHR Regs
$7EF6: [.... ..CM] CHR Mode/Mirroring
C = CHR Mode select
M = Mirroring:
0 = Horz
1 = Vert
$7EFA: [PPPP PP..] PRG Reg 0 (8k @ $8000)
$7EFB: [PPPP PP..] PRG Reg 1 (8k @ $A000)
$7EFC: [PPPP PP..] PRG Reg 2 (8k @ $C000)
CHR Setup:
---------------------------
$0000 $0400 $0800 $0C00 $1000 $1400 $1800 $1C00
+---------------+---------------+-------+-------+-------+-------+
CHR Mode 0: | <$7EF0> | <$7EF1> | $7EF2 | $7EF3 | $7EF4 | $7EF5 |
+---------------+---------------+---------------+---------------+
CHR Mode 1: | $7EF2 | $7EF3 | $7EF4 | $7EF5 | <$7EF0> | <$7EF1> |
+-------+-------+-------+-------+---------------+---------------+
PRG Setup:
---------------------------
$8000 $A000 $C000 $E000
+-------+-------+-------+-------+
| $7EFA | $7EFB | $7EFC | { -1} |
+-------+-------+-------+-------+
Note: remember that the low 2 bits are not used (right-shift written values by 2)
*/
int prg_bank_mask, chr_bank_mask;
ByteBuffer prg_regs_8k = new ByteBuffer(4);
ByteBuffer chr_regs_1k = new ByteBuffer(8);
bool ChrMode;
public override void Dispose()
{
base.Dispose();
chr_regs_1k.Dispose();
prg_regs_8k.Dispose();
}
public override void SyncState(Serializer ser)
{
base.SyncState(ser);
ser.Sync("prg_regs_8k", ref prg_regs_8k);
ser.Sync("chr_regs_1k", ref chr_regs_1k);
ser.Sync("ChrMode", ref ChrMode);
}
public override bool Configure(NES.EDetectionOrigin origin)
{
//configure
switch (Cart.board_type)
{
case "MAPPER082":
break;
case "TAITO-X1-017":
break;
default:
return false;
}
SetMirrorType(EMirrorType.Vertical);
chr_bank_mask = Cart.chr_size / 1 - 1;
prg_bank_mask = Cart.prg_size / 8 - 1;
prg_regs_8k[3] = 0xFF;
return true;
}
public override void WriteWRAM(int addr, byte value)
{
switch (addr)
{
case 0x1EF0:
if (ChrMode)
{
chr_regs_1k[4] = (byte)(value / 2 * 2);
chr_regs_1k[5] = (byte)(value / 2 * 2 + 1);
}
else
{
chr_regs_1k[0] = (byte)(value / 2 * 2);
chr_regs_1k[1] = (byte)(value / 2 * 2 + 1);
}
break;
case 0x1EF1:
if (ChrMode)
{
chr_regs_1k[6] = (byte)(value / 2 * 2);
chr_regs_1k[7] = (byte)(value / 2 * 2 + 1);
}
else
{
chr_regs_1k[2] = (byte)(value / 2 * 2);
chr_regs_1k[3] = (byte)(value / 2 * 2 + 1);
}
break;
case 0x1EF2:
if (ChrMode)
{
chr_regs_1k[0] = value;
}
else
{
chr_regs_1k[4] = value;
}
break;
case 0x1EF3:
if (ChrMode)
{
chr_regs_1k[1] = value;
}
else
{
chr_regs_1k[5] = value;
}
break;
case 0x1EF4:
if (ChrMode)
{
chr_regs_1k[2] = value;
}
else
{
chr_regs_1k[6] = value;
}
break;
case 0X1EF5:
if (ChrMode)
{
chr_regs_1k[3] = value;
}
else
{
chr_regs_1k[7] = value;
}
break;
case 0x1EF6:
ChrMode = value.Bit(1);
if (value.Bit(0))
SetMirrorType(EMirrorType.Vertical);
else
SetMirrorType(EMirrorType.Horizontal);
break;
case 0x1EFA:
prg_regs_8k[0] = (byte)(value >> 2);
break;
case 0x1EFB:
prg_regs_8k[1] = (byte)(value >> 2);
break;
case 0x1EFC:
prg_regs_8k[2] = (byte)(value >> 2);
break;
}
}
public override byte ReadPRG(int addr)
{
int bank_8k = addr >> 13;
int ofs = addr & ((1 << 13) - 1);
bank_8k = prg_regs_8k[bank_8k];
bank_8k &= prg_bank_mask;
addr = (bank_8k << 13) | ofs;
return ROM[addr];
}
public override byte ReadPPU(int addr)
{
if (addr < 0x2000)
{
int bank_1k = addr >> 10;
int ofs = addr & ((1 << 10) - 1);
bank_1k = chr_regs_1k[bank_1k];
bank_1k &= chr_bank_mask;
addr = (bank_1k << 10) | ofs;
return VROM[addr];
}
else return base.ReadPPU(addr);
}
}
}