125 lines
2.7 KiB
C#
125 lines
2.7 KiB
C#
![]() |
using BizHawk.Common;
|
|||
|
|
|||
|
namespace BizHawk.Emulation.Cores.Nintendo.NES
|
|||
|
{
|
|||
|
// Adpated from FCEUX src
|
|||
|
public sealed class Mapper183 : NES.NESBoardBase
|
|||
|
{
|
|||
|
private ByteBuffer prg = new ByteBuffer(4);
|
|||
|
private ByteBuffer chr = new ByteBuffer(8);
|
|||
|
|
|||
|
private int prg_bank_mask_8k, chr_bank_mask_1k;
|
|||
|
|
|||
|
public override bool Configure(NES.EDetectionOrigin origin)
|
|||
|
{
|
|||
|
switch (Cart.board_type)
|
|||
|
{
|
|||
|
case "MAPPER183":
|
|||
|
break;
|
|||
|
default:
|
|||
|
return false;
|
|||
|
}
|
|||
|
|
|||
|
prg_bank_mask_8k = Cart.prg_size / 8 - 1;
|
|||
|
chr_bank_mask_1k = Cart.chr_size - 1;
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
public override void SyncState(Serializer ser)
|
|||
|
{
|
|||
|
base.SyncState(ser);
|
|||
|
ser.Sync("prg", ref prg);
|
|||
|
ser.Sync("chr", ref chr);
|
|||
|
}
|
|||
|
|
|||
|
private void SetMirroring(int mirr)
|
|||
|
{
|
|||
|
switch (mirr & 3)
|
|||
|
{
|
|||
|
case 0: SetMirrorType(EMirrorType.Vertical); break;
|
|||
|
case 1: SetMirrorType(EMirrorType.Horizontal); break;
|
|||
|
case 2: SetMirrorType(EMirrorType.OneScreenA); break;
|
|||
|
case 3: SetMirrorType(EMirrorType.Vertical); break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public override void WriteWRAM(int addr, byte value)
|
|||
|
{
|
|||
|
WriteReg(addr + 0x6000, value);
|
|||
|
}
|
|||
|
|
|||
|
public override void WritePRG(int addr, byte value)
|
|||
|
{
|
|||
|
WriteReg(addr + 0x8000, value);
|
|||
|
}
|
|||
|
|
|||
|
private void WriteReg(int addr, byte value)
|
|||
|
{
|
|||
|
if ((addr & 0xF800) == 0x6800)
|
|||
|
{
|
|||
|
prg[3] = (byte)(addr & 0x3F);
|
|||
|
}
|
|||
|
else if (((addr & 0xF80C) >= 0xB000) && ((addr & 0xF80C) <= 0xE00C))
|
|||
|
{
|
|||
|
int index = (((addr >> 11) - 6) | (addr >> 3)) & 7;
|
|||
|
chr[index] = (byte)((chr[index] & (0xF0 >> (addr & 4))) | ((value & 0x0F) << (addr & 4)));
|
|||
|
}
|
|||
|
else switch (addr & 0xF80C)
|
|||
|
{
|
|||
|
case 0x8800: prg[0] = value; break;
|
|||
|
case 0xA800: prg[1] = value; break;
|
|||
|
case 0xA000: prg[2] = value; break;
|
|||
|
case 0x9800: SetMirroring(value & 3); break;
|
|||
|
|
|||
|
// TODO: IRQ
|
|||
|
case 0xF000: break;
|
|||
|
case 0xF004: break;
|
|||
|
case 0xF008: break;
|
|||
|
case 0xF00C: break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public override byte ReadPPU(int addr)
|
|||
|
{
|
|||
|
if (addr < 0x2000)
|
|||
|
{
|
|||
|
int x = (addr >> 10) & 7;
|
|||
|
int bank = (chr[x] & chr_bank_mask_1k) << 10;
|
|||
|
return VROM[bank + (addr & 0x3FF)]; // TODO
|
|||
|
}
|
|||
|
|
|||
|
return base.ReadPPU(addr);
|
|||
|
}
|
|||
|
|
|||
|
public override byte ReadWRAM(int addr)
|
|||
|
{
|
|||
|
return ROM[((prg[3] & prg_bank_mask_8k)) << 13 + (addr & 0x1FFF)];
|
|||
|
}
|
|||
|
|
|||
|
public override byte ReadPRG(int addr)
|
|||
|
{
|
|||
|
int bank_8k;
|
|||
|
if (addr < 0x2000) // 0x8000
|
|||
|
{
|
|||
|
bank_8k = prg[0] & prg_bank_mask_8k;
|
|||
|
|
|||
|
}
|
|||
|
else if (addr < 0x4000) // 0xA000
|
|||
|
{
|
|||
|
bank_8k = prg[1] & prg_bank_mask_8k;
|
|||
|
}
|
|||
|
else if (addr < 0x6000) // 0xC000
|
|||
|
{
|
|||
|
bank_8k = prg[2] & prg_bank_mask_8k;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
bank_8k = prg_bank_mask_8k;
|
|||
|
}
|
|||
|
|
|||
|
return ROM[(bank_8k << 13) + (addr & 0x1FFF)];
|
|||
|
}
|
|||
|
}
|
|||
|
}
|