BizHawk/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Odyssey2/O2Hawk.cs

189 lines
4.3 KiB
C#

using BizHawk.Common;
using BizHawk.Emulation.Common;
using BizHawk.Emulation.Cores.Components.I8048;
namespace BizHawk.Emulation.Cores.Consoles.O2Hawk
{
[Core(
name: CoreNames.O2Hawk,
author: "alyosha")]
public partial class O2Hawk : IEmulator, ISaveRam, IDebuggable, IInputPollable, IRegionable, ISettable<O2Hawk.O2Settings, O2Hawk.O2SyncSettings>, IBoardInfo
{
// memory domains
public byte[] RAM = new byte[0x80];
public byte addr_latch;
public byte kb_byte;
public bool ppu_en, vpp_en, RAM_en, kybrd_en, copy_en, cart_b0, cart_b1;
public ushort rom_bank;
public ushort bank_size;
public byte[] _bios;
public readonly byte[] _rom;
public readonly byte[] header = new byte[0x50];
public byte[] cart_RAM;
public bool has_bat;
public int _frame = 0;
public MapperBase mapper;
private readonly ITraceable _tracer;
public I8048 cpu;
public PPU ppu;
public bool is_pal;
public bool is_G7400;
public bool is_XROM;
[CoreConstructor(VSystemID.Raw.O2)]
public O2Hawk(CoreComm comm, GameInfo game, byte[] rom, O2Settings settings, O2SyncSettings syncSettings)
{
var ser = new BasicServiceProvider(this);
cpu = new I8048
{
ReadMemory = ReadMemory,
WriteMemory = WriteMemory,
PeekMemory = PeekMemory,
DummyReadMemory = ReadMemory,
ReadPort = ReadPort,
WritePort = WritePort,
OnExecFetch = ExecFetch,
};
_settings = settings ?? new O2Settings();
_syncSettings = syncSettings ?? new O2SyncSettings();
is_G7400 = _syncSettings.G7400_Enable;
_controllerDeck = new O2HawkControllerDeck("O2 Controller", "O2 Controller", is_G7400);
_bios = comm.CoreFileProvider.GetFirmwareOrThrow(new("O2", is_G7400 ? "BIOS-G7400" : "BIOS-O2"), "BIOS Not Found, Cannot Load");
Buffer.BlockCopy(rom, 0x100, header, 0, 0x50);
Console.WriteLine(MD5Checksum.ComputePrefixedHex(rom));
Console.WriteLine(SHA1Checksum.ComputePrefixedHex(rom));
_rom = rom;
if (game["XROM"]) { is_XROM = true; }
Setup_Mapper();
_frameHz = 60;
ser.Register<IVideoProvider>(this);
ServiceProvider = ser;
_tracer = new TraceBuffer(cpu.TraceHeader);
ser.Register(_tracer);
ser.Register<IStatable>(new StateSerializer(SyncState));
SetupMemoryDomains();
cpu.SetCallbacks(ReadMemory, PeekMemory, PeekMemory, WriteMemory);
// set up differences between PAL and NTSC systems
if ((game.Region == "US" || game.Region == "EU-US" || game.Region == null) && !is_G7400)
{
is_pal = false;
pic_height = 240;
_frameHz = 60;
ppu = new NTSC_PPU();
}
else
{
is_pal = true;
pic_height = 240;
_frameHz = 50;
ppu = new PAL_PPU();
}
ppu.Core = this;
ppu.set_region(is_pal, is_G7400);
ser.Register<ISoundProvider>(ppu);
_vidbuffer = new int[372 * pic_height];
frame_buffer = new int[320 * pic_height];
HardReset();
}
public DisplayType Region => is_pal ? DisplayType.PAL : DisplayType.NTSC;
private readonly O2HawkControllerDeck _controllerDeck;
public void HardReset()
{
in_vblank = true; // we start off in vblank since the LCD is off
in_vblank_old = true;
ppu.Reset();
cpu.Reset();
RAM = new byte[0x80];
ticker = 0;
// some of these get overwritten, but
addr_latch = 0;
kb_state_row = kb_state_col = 0;
// bank switching carts expect to be in upper bank on boot up, so can't have 0 at ports
WritePort(1, 0xFF);
WritePort(2, 0xFF);
}
public void SoftReset()
{
cpu.Reset();
}
public string BoardName => mapper.GetType().Name;
// TODO: move callbacks to cpu to avoid non-inlinable function call
private void ExecFetch(ushort addr)
{
if (MemoryCallbacks.HasExecutes)
{
uint flags = (uint)MemoryCallbackFlags.AccessExecute;
MemoryCallbacks.CallMemoryCallbacks(addr, 0, flags, "System Bus");
}
}
private void Setup_Mapper()
{
if (is_XROM)
{
mapper = new MapperXROM
{
Core = this
};
}
else
{
mapper = new MapperDefault
{
Core = this
};
}
mapper.Initialize();
// bank size is different for 12 k carts, it uses all 3k per bank. Note that A11 is held low by the CPU during interrupts
// so this means 12k games use the upper 1k outside of vbl
if (_rom.Length == 0x3000)
{
bank_size = 0xC00;
}
else
{
bank_size = 0x800;
}
}
}
}