BizHawk/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Boards/Mapper236.cs

118 lines
2.4 KiB
C#
Raw Normal View History

2016-09-16 21:47:52 +00:00
using System;
using BizHawk.Common;
namespace BizHawk.Emulation.Cores.Nintendo.NES
{
public sealed class Mapper236 : NES.NESBoardBase
{
[MapperProp]
public byte CartSwitch_70in1 = 0xd;
[MapperProp]
public byte CartSwitch_800in1 = 06;
private bool isLargeBanks = false;
private byte large_bank;
private byte prg_bank;
private byte chr_bank;
private byte bank_mode;
public override bool Configure(NES.EDetectionOrigin origin)
{
switch (Cart.board_type)
{
2016-10-28 21:52:29 +00:00
2016-09-16 21:47:52 +00:00
case "UNIF_BMC-70in1":
isLargeBanks = false;
break;
2016-10-28 21:52:29 +00:00
case "MAPPER236":
2016-09-16 21:47:52 +00:00
case "UNIF_BMC-70in1B":
isLargeBanks = true;
break;
default:
return false;
}
AutoMapperProps.Apply(this);
return true;
}
public override void SyncState(Serializer ser)
{
base.SyncState(ser);
ser.Sync("isLargeBanks", ref isLargeBanks);
ser.Sync("large_bank", ref large_bank);
ser.Sync("prg_bank", ref prg_bank);
ser.Sync("chr_bank", ref chr_bank);
ser.Sync("bank_mode", ref bank_mode);
ser.Sync("Cart_Switch", ref CartSwitch_70in1);
ser.Sync("Cart_Switch", ref CartSwitch_800in1);
}
public override void WritePRG(int addr, byte value)
{
addr += 0x8000;
if ((addr & 0x4000) > 0)
{
bank_mode = (byte)(addr & 0x30);
prg_bank = (byte)(addr & 7);
}
else
{
var mirroring = ((addr & 0x20) >> 5) ^ 1;
SetMirrorType(mirroring > 0 ? EMirrorType.Vertical : EMirrorType.Horizontal);
if (isLargeBanks)
{
large_bank = (byte)((addr & 3) << 3);
}
else
{
chr_bank = (byte)(addr & 7);
}
}
}
public override byte ReadPRG(int addr)
{
switch (bank_mode)
{
default:
throw new InvalidOperationException("Unexpected bank_mode value");
case 0x00:
case 0x10:
int bank;
if (addr < 0x4000)
{
bank = (large_bank | prg_bank);
}
else
{
bank = (large_bank | 7);
}
if (bank_mode == 0x10)
{
addr = (addr & 0x7FF0) | ((isLargeBanks ? CartSwitch_800in1 : CartSwitch_70in1) & 0xf);
}
return ROM[(bank << 14) + (addr & 0x3FFF)];
case 0x20:
return ROM[(((large_bank | prg_bank) >> 1) << 15) + addr];
case 0x30:
return ROM[((large_bank | prg_bank) << 14) + (addr & 0x3FFF)];
}
}
public override byte ReadPPU(int addr)
{
if (addr < 0x2000 && VROM != null)
{
return VROM[(chr_bank << 13) + (addr & 0x1FFF)];
}
return base.ReadPPU(addr);
}
}
}