From f52b02d499c6185a40e3ce4148c1ccab36fc192d Mon Sep 17 00:00:00 2001 From: alyosha-tas Date: Mon, 6 Jul 2020 15:19:57 -0400 Subject: [PATCH] Start wok on G7400 --- .../CPUs/Intel8048/I8048.cs | 6 +- .../Consoles/Magnavox/Odyssey2/O2Hawk.cs | 2 - .../Consoles/Magnavox/Odyssey2/PPU.cs | 7 + .../Videopac G7400/G7400Hawk.ICodeDataLog.cs | 61 + .../Videopac G7400/G7400Hawk.IDebuggable.cs | 24 + .../Videopac G7400/G7400Hawk.IEmulator.cs | 254 +++ .../G7400Hawk.IInputPollable.cs | 24 + .../G7400Hawk.IMemoryDomains.cs | 73 + .../Videopac G7400/G7400Hawk.ISaveRam.cs | 24 + .../Videopac G7400/G7400Hawk.ISettable.cs | 86 + .../Videopac G7400/G7400Hawk.IStatable.cs | 52 + .../Magnavox/Videopac G7400/G7400Hawk.cs | 171 ++ .../Videopac G7400/G7400HawkControllerDeck.cs | 94 + .../Videopac G7400/G7400HawkControllers.cs | 80 + .../Videopac G7400/Mappers/MapperBase.cs | 48 + .../Videopac G7400/Mappers/Mapper_Default.cs | 44 + .../Videopac G7400/Mappers/ReadMe.txt | 3 + .../Magnavox/Videopac G7400/MemoryMap.cs | 179 ++ .../Consoles/Magnavox/Videopac G7400/PPU.cs | 1566 +++++++++++++++++ 19 files changed, 2793 insertions(+), 5 deletions(-) create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ICodeDataLog.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IDebuggable.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IEmulator.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IInputPollable.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IMemoryDomains.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ISaveRam.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ISettable.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IStatable.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400HawkControllerDeck.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400HawkControllers.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/MapperBase.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/Mapper_Default.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/ReadMe.txt create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/MemoryMap.cs create mode 100644 src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/PPU.cs diff --git a/src/BizHawk.Emulation.Cores/CPUs/Intel8048/I8048.cs b/src/BizHawk.Emulation.Cores/CPUs/Intel8048/I8048.cs index 87965d981c..f99b834cb1 100644 --- a/src/BizHawk.Emulation.Cores/CPUs/Intel8048/I8048.cs +++ b/src/BizHawk.Emulation.Cores/CPUs/Intel8048/I8048.cs @@ -8,8 +8,7 @@ namespace BizHawk.Emulation.Cores.Components.I8048 { public sealed partial class I8048 { - public O2Hawk Core { get; set; } - + public int LY; // operations that can take place in an instruction public const ushort IDLE = 0; public const ushort OP = 1; @@ -541,7 +540,7 @@ namespace BizHawk.Emulation.Cores.Components.I8048 Regs[(ushort)(R7 + RB)], Regs[PSW], TotalExecutedCycles, - Core.ppu.LY, + LY, FlagC ? "C" : "c", FlagAC ? "A" : "a", FlagF0 ? "F" : "f", @@ -592,6 +591,7 @@ namespace BizHawk.Emulation.Cores.Components.I8048 ser.Sync(nameof(IRQS), ref IRQS); ser.Sync(nameof(irq_pntr), ref irq_pntr); + ser.Sync(nameof(LY), ref LY); ser.Sync(nameof(EA), ref EA); ser.Sync(nameof(TF), ref TF); ser.Sync(nameof(timer_en), ref timer_en); diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Odyssey2/O2Hawk.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Odyssey2/O2Hawk.cs index 6ed083d8a2..932a1c7f64 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Odyssey2/O2Hawk.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Odyssey2/O2Hawk.cs @@ -74,8 +74,6 @@ namespace BizHawk.Emulation.Cores.Consoles.O2Hawk _frameHz = 60; - cpu.Core = this; - ser.Register(this); ServiceProvider = ser; diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Odyssey2/PPU.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Odyssey2/PPU.cs index 7eb4afb191..48e7c1ed0a 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Odyssey2/PPU.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Odyssey2/PPU.cs @@ -329,6 +329,7 @@ namespace BizHawk.Emulation.Cores.Consoles.O2Hawk cycle = 0; LY++; + Core.cpu.LY = LY; if (LY == LINE_VBL) { @@ -347,6 +348,7 @@ namespace BizHawk.Emulation.Cores.Consoles.O2Hawk if (LY == LINE_MAX) { LY = 0; + Core.cpu.LY = LY; VBL = false; Core.in_vblank = false; Core.cpu.T1 = false; @@ -376,6 +378,7 @@ namespace BizHawk.Emulation.Cores.Consoles.O2Hawk bg_brightness = grid_brightness = grid_fill = LY_ret = cycle = 0; VBL = HBL = lum_en = false; LY = 0; + Core.cpu.LY = LY; AudioReset(); } @@ -1265,6 +1268,7 @@ namespace BizHawk.Emulation.Cores.Consoles.O2Hawk cycle = 0; LY++; + Core.cpu.LY = LY; if (LY == LINE_VBL) { @@ -1283,6 +1287,7 @@ namespace BizHawk.Emulation.Cores.Consoles.O2Hawk if (LY == LINE_MAX) { LY = 0; + Core.cpu.LY = LY; VBL = false; Core.in_vblank = false; Core.cpu.T1 = false; @@ -1346,6 +1351,7 @@ namespace BizHawk.Emulation.Cores.Consoles.O2Hawk cycle = 0; LY++; + Core.cpu.LY = LY; if (LY == LINE_VBL) { @@ -1364,6 +1370,7 @@ namespace BizHawk.Emulation.Cores.Consoles.O2Hawk if (LY == LINE_MAX) { LY = 0; + Core.cpu.LY = LY; VBL = false; Core.in_vblank = false; Core.cpu.T1 = false; diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ICodeDataLog.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ICodeDataLog.cs new file mode 100644 index 0000000000..9173ab3887 --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ICodeDataLog.cs @@ -0,0 +1,61 @@ +using System.IO; +using BizHawk.Emulation.Common; +using BizHawk.Emulation.Cores.Components.I8048; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public partial class G7400Hawk : ICodeDataLogger + { + private ICodeDataLog _cdl; + + public void SetCDL(ICodeDataLog cdl) + { + _cdl = cdl; + if (cdl == null) + this.cpu.CDLCallback = null; + else this.cpu.CDLCallback = CDLCpuCallback; + } + + public void NewCDL(ICodeDataLog cdl) + { + cdl["ROM"] = new byte[MemoryDomains["ROM"].Size]; + cdl["HRAM"] = new byte[MemoryDomains["Zero Page RAM"].Size]; + + cdl["WRAM"] = new byte[MemoryDomains["Main RAM"].Size]; + + if (MemoryDomains.Has("Cart RAM")) + { + cdl["CartRAM"] = new byte[MemoryDomains["Cart RAM"].Size]; + } + + cdl.SubType = "O2"; + cdl.SubVer = 0; + } + + [FeatureNotImplemented] + void ICodeDataLogger.DisassembleCDL(Stream s, ICodeDataLog cdl) + { + } + + public void SetCDL(I8048.eCDLogMemFlags flags, string type, int cdladdr) + { + if (type == null) return; + byte val = (byte)flags; + _cdl[type][cdladdr] |= (byte)flags; + } + + void CDLCpuCallback(ushort addr, I8048.eCDLogMemFlags flags) + { + + if (addr < 0x400) + { + + } + else + { + mapper.MapCDL(addr, flags); + return; + } + } + } +} \ No newline at end of file diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IDebuggable.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IDebuggable.cs new file mode 100644 index 0000000000..ef6bac899b --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IDebuggable.cs @@ -0,0 +1,24 @@ +using System; +using System.Collections.Generic; +using BizHawk.Emulation.Common; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public partial class G7400Hawk : IDebuggable + { + public IDictionary GetCpuFlagsAndRegisters() + => cpu.GetCpuFlagsAndRegisters(); + + public void SetCpuRegister(string register, int value) + => cpu.SetCpuRegister(register, value); + + public IMemoryCallbackSystem MemoryCallbacks { get; } = new MemoryCallbackSystem(new[] { "System Bus" }); + + public bool CanStep(StepType type) => false; + + [FeatureNotImplemented] + public void Step(StepType type) => throw new NotImplementedException(); + + public long TotalExecutedCycles => (long)cpu.TotalExecutedCycles; + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IEmulator.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IEmulator.cs new file mode 100644 index 0000000000..e92977249e --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IEmulator.cs @@ -0,0 +1,254 @@ +using BizHawk.Emulation.Common; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public partial class G7400Hawk : IEmulator, IVideoProvider + { + public IEmulatorServiceProvider ServiceProvider { get; } + + public ControllerDefinition ControllerDefinition => _controllerDeck.Definition; + + public byte controller_state_1, controller_state_2, kb_state_row, kb_state_col; + public bool in_vblank_old; + public bool in_vblank; + public bool vblank_rise; + public uint ticker; + + public bool FrameAdvance(IController controller, bool render, bool rendersound) + { + //Console.WriteLine("-----------------------FRAME-----------------------"); + + if (_tracer.Enabled) + { + cpu.TraceCallback = s => _tracer.Put(s); + } + else + { + cpu.TraceCallback = null; + } + + _frame++; + + if (controller.IsPressed("Power")) + { + HardReset(); + } + + if (controller.IsPressed("Reset")) + { + SoftReset(); + } + + _islag = true; + + do_frame(controller); + + if (_islag) + { + _lagcount++; + } + + return true; + } + + public void do_frame(IController controller) + { + // update the controller state on VBlank + GetControllerState(controller); + + bool frame_chk = true; + //Console.WriteLine("----------FRAME----------"); + + if (is_pal) + { + // PAL timing is: 17.7 / 5 ppu + // and 17.7 / 9 for cpu (divide by 3 externally then by 3 again internally) + while (frame_chk) + { + ticker++; + + if (ticker % 5 == 0) + { + ppu.tick(); + if (ticker % 10 == 0) + { + ppu.Audio_tick(); + } + } + + if ((ticker % 9) == 0) + { + cpu.ExecuteOne(); + } + + if (!in_vblank && in_vblank_old) + { + frame_chk = false; + } + + in_vblank_old = in_vblank; + } + } + else + { + // NTSC is 2 to 1 ppu to cpu ticks + while (frame_chk) + { + ppu.tick(); + ppu.tick(); + ppu.Audio_tick(); + cpu.ExecuteOne(); + + if (!in_vblank && in_vblank_old) + { + frame_chk = false; + } + + in_vblank_old = in_vblank; + } + } + + // send the image on VBlank + SendVideoBuffer(); + } + + public void do_single_step() + { + ppu.tick(); + ppu.tick(); + ppu.Audio_tick(); + cpu.ExecuteOne(); + + } + + public void GetControllerState(IController controller) + { + InputCallbacks.Call(); + controller_state_1 = _controllerDeck.ReadPort1(controller); + controller_state_2 = _controllerDeck.ReadPort2(controller); + + kb_state_row = 8; // nothing pressed + if (controller.IsPressed("0")) { kb_state_row = 0; kb_state_col = 7; } + if (controller.IsPressed("1")) { kb_state_row = 0; kb_state_col = 6; } + if (controller.IsPressed("2")) { kb_state_row = 0; kb_state_col = 5; } + if (controller.IsPressed("3")) { kb_state_row = 0; kb_state_col = 4; } + if (controller.IsPressed("4")) { kb_state_row = 0; kb_state_col = 3; } + if (controller.IsPressed("5")) { kb_state_row = 0; kb_state_col = 2; } + if (controller.IsPressed("6")) { kb_state_row = 0; kb_state_col = 1; } + if (controller.IsPressed("7")) { kb_state_row = 0; kb_state_col = 0; } + if (controller.IsPressed("8")) { kb_state_row = 1; kb_state_col = 7; } + if (controller.IsPressed("9")) { kb_state_row = 1; kb_state_col = 6; } + if (controller.IsPressed("SPC")) { kb_state_row = 1; kb_state_col = 3; } + if (controller.IsPressed("?")) { kb_state_row = 1; kb_state_col = 2; } + if (controller.IsPressed("L")) { kb_state_row = 1; kb_state_col = 1; } + if (controller.IsPressed("P")) { kb_state_row = 1; kb_state_col = 0; } + if (controller.IsPressed("+")) { kb_state_row = 2; kb_state_col = 7; } + if (controller.IsPressed("W")) { kb_state_row = 2; kb_state_col = 6; } + if (controller.IsPressed("E")) { kb_state_row = 2; kb_state_col = 5; } + if (controller.IsPressed("R")) { kb_state_row = 2; kb_state_col = 4; } + if (controller.IsPressed("T")) { kb_state_row = 2; kb_state_col = 3; } + if (controller.IsPressed("U")) { kb_state_row = 2; kb_state_col = 2; } + if (controller.IsPressed("I")) { kb_state_row = 2; kb_state_col = 1; } + if (controller.IsPressed("O")) { kb_state_row = 2; kb_state_col = 0; } + if (controller.IsPressed("Q")) { kb_state_row = 3; kb_state_col = 7; } + if (controller.IsPressed("S")) { kb_state_row = 3; kb_state_col = 6; } + if (controller.IsPressed("D")) { kb_state_row = 3; kb_state_col = 5; } + if (controller.IsPressed("F")) { kb_state_row = 3; kb_state_col = 4; } + if (controller.IsPressed("G")) { kb_state_row = 3; kb_state_col = 3; } + if (controller.IsPressed("H")) { kb_state_row = 3; kb_state_col = 2; } + if (controller.IsPressed("J")) { kb_state_row = 3; kb_state_col = 1; } + if (controller.IsPressed("K")) { kb_state_row = 3; kb_state_col = 0; } + if (controller.IsPressed("A")) { kb_state_row = 4; kb_state_col = 7; } + if (controller.IsPressed("Z")) { kb_state_row = 4; kb_state_col = 6; } + if (controller.IsPressed("X")) { kb_state_row = 4; kb_state_col = 5; } + if (controller.IsPressed("C")) { kb_state_row = 4; kb_state_col = 4; } + if (controller.IsPressed("V")) { kb_state_row = 4; kb_state_col = 3; } + if (controller.IsPressed("B")) { kb_state_row = 4; kb_state_col = 2; } + if (controller.IsPressed("M")) { kb_state_row = 4; kb_state_col = 1; } + if (controller.IsPressed(".")) { kb_state_row = 4; kb_state_col = 0; } + if (controller.IsPressed("-")) { kb_state_row = 5; kb_state_col = 7; } + if (controller.IsPressed("*")) { kb_state_row = 5; kb_state_col = 6; } + if (controller.IsPressed("/")) { kb_state_row = 5; kb_state_col = 5; } + if (controller.IsPressed("=")) { kb_state_row = 5; kb_state_col = 4; } + if (controller.IsPressed("YES")) { kb_state_row = 5; kb_state_col = 3; } + if (controller.IsPressed("NO")) { kb_state_row = 5; kb_state_col = 2; } + if (controller.IsPressed("CLR")) { kb_state_row = 5; kb_state_col = 1; } + if (controller.IsPressed("ENT")) { kb_state_row = 5; kb_state_col = 0; } + + } + + public void KB_Scan() + { + if (kb_byte == kb_state_row) + { + kb_byte &= 0xEF; + kb_byte |= (byte)(kb_state_col << 5); + } + else + { + kb_byte |= 0x10; + } + } + + public int Frame => _frame; + + public string SystemId => "O2"; + + public bool DeterministicEmulation { get; set; } + + public void ResetCounters() + { + _frame = 0; + _lagcount = 0; + _islag = false; + } + + public void Dispose() + { + ppu.DisposeSound(); + } + + public int _frameHz = 60; + + public int[] _vidbuffer; + + public int[] frame_buffer; + + public int[] GetVideoBuffer() + { + return frame_buffer; + } + + public void SendVideoBuffer() + { + for (int j = 0; j < pic_height; j++) + { + for (int i = 0; i < 320; i++) + { + frame_buffer[j * 320 + i] = _vidbuffer[j * 372 + i]; + _vidbuffer[j * 372 + i] = 0; + } + + for (int k = 320; k < 372; k++) + { + _vidbuffer[j * 372 + k] = 0; + } + } + } + + public int pic_height; + + public int VirtualWidth => 320; + public int VirtualHeight => pic_height; + public int BufferWidth => 320; + public int BufferHeight => pic_height; + public int BackgroundColor => unchecked((int)0xFF000000); + public int VsyncNumerator => _frameHz; + public int VsyncDenominator => 1; + + public static readonly uint[] color_palette_BW = { 0xFFFFFFFF , 0xFFAAAAAA, 0xFF555555, 0xFF000000 }; + public static readonly uint[] color_palette_Gr = { 0xFFA4C505, 0xFF88A905, 0xFF1D551D, 0xFF052505 }; + + public uint[] color_palette = new uint[4]; + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IInputPollable.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IInputPollable.cs new file mode 100644 index 0000000000..1dd944e3e3 --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IInputPollable.cs @@ -0,0 +1,24 @@ +using BizHawk.Emulation.Common; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public partial class G7400Hawk : IInputPollable + { + public int LagCount + { + get => _lagcount; + set => _lagcount = value; + } + + public bool IsLagFrame + { + get => _islag; + set => _islag = value; + } + + public IInputCallbackSystem InputCallbacks { get; } = new InputCallbackSystem(); + + public bool _islag = true; + private int _lagcount; + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IMemoryDomains.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IMemoryDomains.cs new file mode 100644 index 0000000000..4d45897836 --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IMemoryDomains.cs @@ -0,0 +1,73 @@ +using System.Collections.Generic; +using BizHawk.Emulation.Common; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public partial class G7400Hawk + { + private IMemoryDomains MemoryDomains; + + public void SetupMemoryDomains() + { + var domains = new List + { + new MemoryDomainDelegate( + "Main RAM", + RAM.Length, + MemoryDomain.Endian.Little, + addr => RAM[addr], + (addr, value) => RAM[addr] = value, + 1), + new MemoryDomainDelegate( + "CPU RAM", + 64, + MemoryDomain.Endian.Little, + addr => (byte)cpu.Regs[addr], + (addr, value) => cpu.Regs[addr] = value, + 1), + new MemoryDomainDelegate( + "System Bus", + 0X1000, + MemoryDomain.Endian.Little, + addr => PeekSystemBus(addr), + (addr, value) => PokeSystemBus(addr, value), + 1), + new MemoryDomainDelegate( + "ROM", + _rom.Length, + MemoryDomain.Endian.Little, + addr => _rom[addr], + (addr, value) => _rom[addr] = value, + 1), + new MemoryDomainDelegate( + "PPU", + 256, + MemoryDomain.Endian.Little, + addr => ppu.PeekReg((int)addr), + (addr, value) => ppu.WriteReg((int)addr, value), + 1) + }; + + if (cart_RAM != null) + { + var CartRam = new MemoryDomainByteArray("Cart RAM", MemoryDomain.Endian.Little, cart_RAM, true, 1); + domains.Add(CartRam); + } + + MemoryDomains = new MemoryDomainList(domains); + (ServiceProvider as BasicServiceProvider).Register(MemoryDomains); + } + + private byte PeekSystemBus(long addr) + { + ushort addr2 = (ushort)(addr & 0xFFF); + return PeekMemory(addr2); + } + + private void PokeSystemBus(long addr, byte value) + { + ushort addr2 = (ushort)(addr & 0xFFF); + WriteMemory(addr2, value); + } + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ISaveRam.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ISaveRam.cs new file mode 100644 index 0000000000..f58cd78a8e --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ISaveRam.cs @@ -0,0 +1,24 @@ +using System; +using BizHawk.Emulation.Common; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public partial class G7400Hawk : ISaveRam + { + public byte[] CloneSaveRam() + { + return (byte[])cart_RAM?.Clone(); + } + + public void StoreSaveRam(byte[] data) + { + if (_syncSettings.Use_SRAM) + { + Buffer.BlockCopy(data, 0, cart_RAM, 0, data.Length); + Console.WriteLine("loading SRAM here"); + } + } + + public bool SaveRamModified => has_bat & _syncSettings.Use_SRAM; + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ISettable.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ISettable.cs new file mode 100644 index 0000000000..a5384ec4cc --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.ISettable.cs @@ -0,0 +1,86 @@ +using System.ComponentModel; +using BizHawk.Common; +using BizHawk.Emulation.Common; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public partial class G7400Hawk : IEmulator, ISettable + { + public G7400Settings GetSettings() + { + return _settings.Clone(); + } + + public G7400SyncSettings GetSyncSettings() + { + return _syncSettings.Clone(); + } + + public PutSettingsDirtyBits PutSettings(G7400Settings o) + { + _settings = o; + return PutSettingsDirtyBits.None; + } + + public PutSettingsDirtyBits PutSyncSettings(G7400SyncSettings o) + { + bool ret = G7400SyncSettings.NeedsReboot(_syncSettings, o); + _syncSettings = o; + return ret ? PutSettingsDirtyBits.RebootCore : PutSettingsDirtyBits.None; + } + + public G7400Settings _settings = new G7400Settings(); + public G7400SyncSettings _syncSettings = new G7400SyncSettings(); + + public class G7400Settings + { + [DisplayName("Display Characters")] + [Description("When true, displays character.")] + [DefaultValue(true)] + public bool Show_Chars { get; set; } + + [DisplayName("Display Quad Characters")] + [Description("When true, displays quad character.")] + [DefaultValue(true)] + public bool Show_Quads { get; set; } + + [DisplayName("Display Sprites")] + [Description("When true, displays sprites.")] + [DefaultValue(true)] + public bool Show_Sprites { get; set; } + + public G7400Settings Clone() + { + return (G7400Settings)MemberwiseClone(); + } + + public G7400Settings() + { + SettingsUtil.SetDefaultValues(this); + } + } + + public class G7400SyncSettings + { + [DisplayName("Use Existing SaveRAM")] + [Description("When true, existing SaveRAM will be loaded at boot up")] + [DefaultValue(true)] + public bool Use_SRAM { get; set; } + + public G7400SyncSettings Clone() + { + return (G7400SyncSettings)MemberwiseClone(); + } + + public G7400SyncSettings() + { + SettingsUtil.SetDefaultValues(this); + } + + public static bool NeedsReboot(G7400SyncSettings x, G7400SyncSettings y) + { + return !DeepEquality.DeepEquals(x, y); + } + } + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IStatable.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IStatable.cs new file mode 100644 index 0000000000..797ed07504 --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.IStatable.cs @@ -0,0 +1,52 @@ +using BizHawk.Common; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public partial class G7400Hawk + { + private void SyncState(Serializer ser) + { + ser.BeginSection("Videopac G7400"); + cpu.SyncState(ser); + mapper.SyncState(ser); + ppu.SyncState(ser); + + ser.Sync("Lag", ref _lagcount); + ser.Sync("Frame", ref _frame); + ser.Sync("IsLag", ref _islag); + _controllerDeck.SyncState(ser); + + ser.Sync(nameof(controller_state_1), ref controller_state_1); + ser.Sync(nameof(controller_state_2), ref controller_state_2); + ser.Sync(nameof(in_vblank), ref in_vblank); + ser.Sync(nameof(in_vblank_old), ref in_vblank_old); + ser.Sync(nameof(vblank_rise), ref vblank_rise); + ser.Sync(nameof(ticker), ref ticker); + + ser.Sync(nameof(RAM_en), ref RAM_en); + ser.Sync(nameof(ppu_en), ref ppu_en); + ser.Sync(nameof(cart_b0), ref cart_b0); + ser.Sync(nameof(cart_b1), ref cart_b1); + ser.Sync(nameof(copy_en), ref copy_en); + ser.Sync(nameof(kybrd_en), ref kybrd_en); + ser.Sync(nameof(rom_bank), ref rom_bank); + ser.Sync(nameof(bank_size), ref bank_size); + + // memory domains + ser.Sync(nameof(RAM), ref RAM, false); + ser.Sync(nameof(_bios), ref _bios, false); + ser.Sync(nameof(addr_latch), ref addr_latch); + ser.Sync(nameof(kb_byte), ref kb_byte); + ser.Sync(nameof(kb_state_row), ref kb_state_row); + ser.Sync(nameof(kb_state_col), ref kb_state_col); + + // probably a better way to do this + if (cart_RAM != null) + { + ser.Sync(nameof(cart_RAM), ref cart_RAM, false); + } + + ser.EndSection(); + } + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.cs new file mode 100644 index 0000000000..456ccaac39 --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400Hawk.cs @@ -0,0 +1,171 @@ +using System; + +using BizHawk.Common.BufferExtensions; +using BizHawk.Emulation.Common; +using BizHawk.Emulation.Cores.Components.I8048; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + [Core( + "G7400Hawk", + "", + isPorted: false, + isReleased: false, + displayName: "Videopac G7400")] + [ServiceNotApplicable(new[] { typeof(IDriveLight) })] + public partial class G7400Hawk : IEmulator, ISaveRam, IDebuggable, IInputPollable, IRegionable, ISettable, IBoardInfo + { + // memory domains + public byte[] RAM = new byte[0x80]; + + public byte addr_latch; + public byte kb_byte; + public bool ppu_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; + + [CoreConstructor("G7400")] + public G7400Hawk(CoreComm comm, GameInfo game, byte[] rom, /*string gameDbFn,*/ object settings, object syncSettings) + { + var ser = new BasicServiceProvider(this); + + cpu = new I8048 + { + ReadMemory = ReadMemory, + WriteMemory = WriteMemory, + PeekMemory = PeekMemory, + DummyReadMemory = ReadMemory, + ReadPort = ReadPort, + WritePort = WritePort, + OnExecFetch = ExecFetch, + }; + + _settings = (G7400Settings)settings ?? new G7400Settings(); + _syncSettings = (G7400SyncSettings)syncSettings ?? new G7400SyncSettings(); + _controllerDeck = new G7400HawkControllerDeck("G7400 Controller", "G7400 Controller"); + + _bios = comm.CoreFileProvider.GetFirmware("G7400", "BIOS", true, "BIOS Not Found, Cannot Load") + ?? throw new MissingFirmwareException("Missing Odyssey2 Bios"); + + Buffer.BlockCopy(rom, 0x100, header, 0, 0x50); + + Console.WriteLine("MD5: " + rom.HashMD5(0, rom.Length)); + Console.WriteLine("SHA1: " + rom.HashSHA1(0, rom.Length)); + _rom = rom; + Setup_Mapper(); + + _frameHz = 60; + + ser.Register(this); + ServiceProvider = ser; + + _settings = (G7400Settings)settings ?? new G7400Settings(); + _syncSettings = (G7400SyncSettings)syncSettings ?? new G7400SyncSettings(); + + _tracer = new TraceBuffer { Header = cpu.TraceHeader }; + ser.Register(_tracer); + ser.Register(new StateSerializer(SyncState)); + SetupMemoryDomains(); + cpu.SetCallbacks(ReadMemory, PeekMemory, PeekMemory, WriteMemory); + + // G7400 is PAL only + is_pal = true; + pic_height = 240; + _frameHz = 50; + ppu = new PAL_PPU(); + + ppu.Core = this; + + ppu.set_region(is_pal); + + ser.Register(ppu); + + _vidbuffer = new int[372 * pic_height]; + frame_buffer = new int[320 * pic_height]; + + HardReset(); + } + + public DisplayType Region => DisplayType.PAL; + + private readonly G7400HawkControllerDeck _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() + { + 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; + } + } + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400HawkControllerDeck.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400HawkControllerDeck.cs new file mode 100644 index 0000000000..77be89ab56 --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400HawkControllerDeck.cs @@ -0,0 +1,94 @@ +using System; +using System.Collections.Generic; +using System.Linq; + +using BizHawk.Common; +using BizHawk.Common.ReflectionExtensions; +using BizHawk.Emulation.Common; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public class G7400HawkControllerDeck + { + public G7400HawkControllerDeck(string controller1Name, string controller2Name) + { + if (!ValidControllerTypes.ContainsKey(controller1Name)) + { + throw new InvalidOperationException("Invalid controller type: " + controller1Name); + } + + if (!ValidControllerTypes.ContainsKey(controller2Name)) + { + throw new InvalidOperationException("Invalid controller type: " + controller2Name); + } + + Port1 = (IPort)Activator.CreateInstance(ValidControllerTypes[controller1Name], 1); + Port2 = (IPort)Activator.CreateInstance(ValidControllerTypes[controller2Name], 2); + + Definition = new ControllerDefinition + { + Name = Port1.Definition.Name, + BoolButtons = Port1.Definition.BoolButtons + .Concat(Port2.Definition.BoolButtons) + .Concat(new[] + { + "0", "1", "2", "3", "4", "5", "6", "7", + "8", "9", "SPC", "?", "L", "P", + "+", "W", "E", "R", "T", "U", "I", "O", + "Q", "S", "D", "F", "G", "H", "J", "K", + "A", "Z", "X", "C", "V", "B", "M", "PERIOD", + "-", "*", "/", "=", "YES", "NO", "CLR", "ENT", + "Reset","Power" + }) + .ToList() + }; + + foreach (var kvp in Port1.Definition.Axes) Definition.Axes.Add(kvp); + } + + public byte ReadPort1(IController c) + { + return Port1.Read(c); + } + + public byte ReadPort2(IController c) + { + return Port2.Read(c); + } + + public ControllerDefinition Definition { get; } + + public void SyncState(Serializer ser) + { + ser.BeginSection(nameof(Port1)); + Port1.SyncState(ser); + ser.EndSection(); + ser.BeginSection(nameof(Port2)); + Port2.SyncState(ser); + ser.EndSection(); + } + + private readonly IPort Port1, Port2; + + private static Dictionary _controllerTypes; + + public static Dictionary ValidControllerTypes + { + get + { + if (_controllerTypes == null) + { + _controllerTypes = typeof(G7400HawkControllerDeck).Assembly + .GetTypes() + .Where(t => typeof(IPort).IsAssignableFrom(t)) + .Where(t => !t.IsAbstract && !t.IsInterface) + .ToDictionary(tkey => tkey.DisplayName()); + } + + return _controllerTypes; + } + } + + public static string DefaultControllerName => typeof(StandardControls).DisplayName(); + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400HawkControllers.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400HawkControllers.cs new file mode 100644 index 0000000000..8554302681 --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/G7400HawkControllers.cs @@ -0,0 +1,80 @@ +using System.ComponentModel; +using System.Linq; + +using BizHawk.Common; +using BizHawk.Emulation.Common; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + /// + /// Represents a G7400 add on + /// + public interface IPort + { + byte Read(IController c); + + ControllerDefinition Definition { get; } + + void SyncState(Serializer ser); + + int PortNum { get; } + } + + [DisplayName("G7400 Controller")] + public class StandardControls : IPort + { + public StandardControls(int portNum) + { + PortNum = portNum; + Definition = new ControllerDefinition + { + Name = "O2 Joystick", + BoolButtons = BaseDefinition + .Select(b => "P" + PortNum + " " + b) + .ToList() + }; + } + + public int PortNum { get; } + + public ControllerDefinition Definition { get; } + + public byte Read(IController c) + { + byte result = 0xFF; + + if (c.IsPressed(Definition.BoolButtons[0])) + { + result -= 1; + } + if (c.IsPressed(Definition.BoolButtons[1])) + { + result -= 4; + } + if (c.IsPressed(Definition.BoolButtons[2])) + { + result -= 8; + } + if (c.IsPressed(Definition.BoolButtons[3])) + { + result -= 2; + } + if (c.IsPressed(Definition.BoolButtons[4])) + { + result -= 16; + } + + return result; + } + + private static readonly string[] BaseDefinition = + { + "Up", "Down", "Left", "Right", "F" + }; + + public void SyncState(Serializer ser) + { + //nothing + } + } +} \ No newline at end of file diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/MapperBase.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/MapperBase.cs new file mode 100644 index 0000000000..380fa97e5d --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/MapperBase.cs @@ -0,0 +1,48 @@ +using BizHawk.Common; +using BizHawk.Emulation.Cores.Components.I8048; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public class MapperBase + { + public G7400Hawk Core { get; set; } + + public virtual byte ReadMemory(ushort addr) => 0; + + public virtual byte PeekMemory(ushort addr) => ReadMemory(addr); + + public virtual void WriteMemory(ushort addr, byte value) + { + } + + public virtual void SyncState(Serializer ser) + { + } + + public virtual void Initialize() + { + } + + public virtual void Mapper_Tick() + { + } + + public virtual void RTC_Get(int value, int index) + { + } + + public virtual void MapCDL(ushort addr, I8048.eCDLogMemFlags flags) + { + } + + protected void SetCDLROM(I8048.eCDLogMemFlags flags, int cdladdr) + { + Core.SetCDL(flags, "ROM", cdladdr); + } + + protected void SetCDLRAM(I8048.eCDLogMemFlags flags, int cdladdr) + { + Core.SetCDL(flags, "CartRAM", cdladdr); + } + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/Mapper_Default.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/Mapper_Default.cs new file mode 100644 index 0000000000..22f8b70f9a --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/Mapper_Default.cs @@ -0,0 +1,44 @@ +using BizHawk.Common; +using BizHawk.Emulation.Cores.Components.I8048; + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + // Default mapper with no bank switching + public class MapperDefault : MapperBase + { + public int ROM_mask; + + public override void Initialize() + { + // roms come in 3 sizes, but the last size is a degenerate case + if (Core._rom.Length == 0x3000) + { + ROM_mask = 0x3FFF; + } + else + { + ROM_mask = Core._rom.Length - 1; + } + } + + public override byte ReadMemory(ushort addr) + { + return Core._rom[addr & ROM_mask]; + } + + public override void MapCDL(ushort addr, I8048.eCDLogMemFlags flags) + { + SetCDLROM(flags, addr); + } + + public override void WriteMemory(ushort addr, byte value) + { + // no mapping hardware available + } + + public override void SyncState(Serializer ser) + { + ser.Sync(nameof(ROM_mask), ref ROM_mask); + } + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/ReadMe.txt b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/ReadMe.txt new file mode 100644 index 0000000000..0c94c8272c --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/Mappers/ReadMe.txt @@ -0,0 +1,3 @@ +TODO: +Official Mappers +Unofficial Mappers diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/MemoryMap.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/MemoryMap.cs new file mode 100644 index 0000000000..09f4b21b5e --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/MemoryMap.cs @@ -0,0 +1,179 @@ +using System; +using BizHawk.Common.NumberExtensions; +using BizHawk.Emulation.Common; + +/* + $0400-$0FFF Cartridge (Only 2K accessible, bit 10 not mapped to cart) + $0000-$03FF BIOS +*/ + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public partial class G7400Hawk + { + public byte ReadMemory(ushort addr) + { + if (MemoryCallbacks.HasReads) + { + uint flags = (uint)MemoryCallbackFlags.AccessRead; + MemoryCallbacks.CallMemoryCallbacks(addr, 0, flags, "System Bus"); + } + + if (addr < 0x400) + { + return _bios[addr]; + } + + return mapper.ReadMemory((ushort)((addr - 0x400) + bank_size * rom_bank)); + } + + public void WriteMemory(ushort addr, byte value) + { + if (MemoryCallbacks.HasWrites) + { + uint flags = (uint)MemoryCallbackFlags.AccessWrite; + MemoryCallbacks.CallMemoryCallbacks(addr, value, flags, "System Bus"); + } + + if (addr < 0x400) + { + + } + else + { + mapper.WriteMemory(addr, value); + } + } + + public byte PeekMemory(ushort addr) + { + if (addr < 0x400) + { + return _bios[addr]; + } + + return mapper.PeekMemory((ushort)((addr - 0x400) + bank_size * rom_bank)); + } + + public byte ReadPort(ushort port) + { + if (port == 0) + { + // BUS, used with external memory and ppu + if (cpu.EA) + { + return addr_latch; + } + + if (RAM_en) + { + if (addr_latch < 0x80) + { + return RAM[addr_latch & 0x7F]; + } + + // voice module would return here + return 0; + } + if (ppu_en) + { + return ppu.ReadReg(addr_latch); + } + + // if neither RAM or PPU is enabled, then a RD pulse from instruction IN A,BUS will latch controller + // onto the bus, but only if they are enabled correctly using port 2 + if (kybrd_en) + { + _islag = false; + if ((kb_byte & 7) == 1) + { + return controller_state_1; + } + if ((kb_byte & 7) == 0) + { + return controller_state_2; + } + } + + Console.WriteLine(cpu.TotalExecutedCycles); + // not sure what happens if this case is reached, probably whatever the last value on the bus is + return 0; + } + + if (port == 1) + { + // various control pins + return (byte)((ppu.lum_en ? 0x80 : 0) | + (copy_en ? 0x40 : 0) | + (0x20) | + (!RAM_en ? 0x10 : 0) | + (!ppu_en ? 0x08 : 0) | + (!kybrd_en ? 0x04 : 0) | + (cart_b1 ? 0x02 : 0) | + (cart_b0 ? 0x01 : 0)); + } + + // keyboard + _islag = false; + return kb_byte; + } + + public void WritePort(ushort port, byte value) + { + if (port == 0) + { + // BUS, used with external memory and ppu + if (cpu.EA) + { + addr_latch = value; + } + else + { + if (RAM_en && !copy_en) + { + if (addr_latch < 0x80) + { + RAM[addr_latch] = value; + } + else + { + // voice module goes here + } + } + + if (ppu_en) + { + ppu.WriteReg(addr_latch, value); + //Console.WriteLine((addr_latch) + " " + value); + } + } + } + else if (port == 1) + { + // various control pins + ppu.lum_en = value.Bit(7); + copy_en = value.Bit(6); + RAM_en = !value.Bit(4); + ppu_en = !value.Bit(3); + kybrd_en = !value.Bit(2); + cart_b1 = value.Bit(1); + cart_b0 = value.Bit(0); + + rom_bank = (ushort)(cart_b0 ? 1 : 0); + rom_bank |= (ushort)(cart_b1 ? 2 : 0); + //rom_bank = (ushort)(rom_bank << 12); + + ppu.bg_brightness = !ppu.lum_en ? 8 : 0; + ppu.grid_brightness = (!ppu.lum_en | ppu.VDC_color.Bit(6)) ? 8 : 0; + + //Console.WriteLine("main ctrl: " + value + " " + ppu.lum_en + " " + ppu_en + " " + RAM_en + " " + cpu.TotalExecutedCycles + " " + ppu.LY + " " + rom_bank); + } + else + { + // keyboard + kb_byte = (byte)(value & 7); + KB_Scan(); + } + } + } +} diff --git a/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/PPU.cs b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/PPU.cs new file mode 100644 index 0000000000..2ebadb901e --- /dev/null +++ b/src/BizHawk.Emulation.Cores/Consoles/Magnavox/Videopac G7400/PPU.cs @@ -0,0 +1,1566 @@ +using System; +using BizHawk.Common; +using BizHawk.Common.NumberExtensions; +using BizHawk.Emulation.Common; + +/* + Notes: Blockout expects STAT to return VBl 1 even outside VBl range. So this is an interrupt request bit, not a status bit (similar to sound) + Also, according to the 8245 data sheet, the hbl status flag starts at roughly halfway through the scanline. This is also + crucial for Blockout to display correctly. However this bit is reset once HBL ends + + Blockout also turns on HBL interrupts and expects them either not to fire (or only one to fire per write to A0) +*/ + +namespace BizHawk.Emulation.Cores.Consoles.G7400Hawk +{ + public class PPU : ISoundProvider + { + public G7400Hawk Core { get; set; } + + // not stated, set on game load + public bool is_pal; + public int LINE_VBL; + public int LINE_MAX; + + public const int HBL_CNT = 45; + public const int GRID_OFST = 24; + public const int OBJ_OFST = 0; + + public byte[] Sprites = new byte[16]; + public byte[] Sprite_Shapes = new byte[32]; + public byte[] Foreground = new byte[48]; + public byte[] Quad_Chars = new byte[64]; + public byte[] Grid_H = new byte[18]; + public byte[] Grid_V = new byte[10]; + public byte[] VDC_col_ret = new byte[8]; + + public byte VDC_ctrl, VDC_status, VDC_collision, VDC_color; + public byte Pixel_Stat; + public int bg_brightness, grid_brightness; + public byte A4_latch, A5_latch; + + public int grid_fill; + public byte grid_fill_col; + public int LY; + public int cycle; + public bool VBL; + public bool HBL; + public bool lum_en; + + public bool latch_x_y; + public bool HBL_req; + public int LY_ret; + + // local variables not stated + int current_pixel_offset; + int double_size; + int right_shift; + int right_shift_even; + int x_base; + + public virtual byte ReadReg(int addr) + { + byte ret = 0; + + if (addr < 0x10) + { + ret = Sprites[addr]; + } + else if (addr < 0x40) + { + ret = Foreground[addr - 0x10]; + } + else if (addr < 0x80) + { + ret = Quad_Chars[addr - 0x40]; + } + else if (addr < 0xA0) + { + ret = Sprite_Shapes[addr - 0x80]; + } + else if (addr == 0xA0) + { + ret = VDC_ctrl; + } + else if (addr == 0xA1) + { + ret = VDC_status; + // reading status clears IRQ request + VDC_status &= 0xF3; + Core.cpu.IRQPending = false; + } + else if (addr == 0xA2) + { + for (int i = 0; i < 8; i++) + { + if (VDC_collision.Bit(i)) + { + ret |= VDC_col_ret[i]; + //VDC_col_ret[i] = 0; + } + } + + // register is reset when read, but not external + for (int i = 0; i < 8; i++) { VDC_col_ret[i] = 0; } + + //Console.WriteLine("col: " + VDC_collision + " " + ret + " " + LY + " " + Core.cpu.TotalExecutedCycles); + } + else if (addr == 0xA3) + { + ret = VDC_color; + } + else if (addr == 0xA4) + { + if (latch_x_y) { ret = A4_latch; } + else { ret = (byte)((LY_ret >= 0) ? LY_ret : 0); } + } + else if (addr == 0xA5) + { + // reading the x reg clears the latch + if (latch_x_y) { ret = A5_latch; latch_x_y = false; VDC_status |= 0x2; } + else { ret = (byte)(cycle); } + } + else if (addr <= 0xAA) + { + ret = AudioReadReg(addr); + } + else if ((addr >= 0xC0) && (addr <= 0xC8)) + { + ret = Grid_H[addr - 0xC0]; + } + else if ((addr >= 0xD0) && (addr <= 0xD8)) + { + ret = Grid_H[addr - 0xD0 + 9]; + } + else if ((addr >= 0xE0) && (addr <= 0xE9)) + { + ret = Grid_V[addr - 0xE0]; + } + + return ret; + } + + //Peek method for memory domains that doesn't effect IRQ + public virtual byte PeekReg(int addr) + { + byte ret = 0; + + if (addr < 0x10) { ret = Sprites[addr]; } + else if (addr < 0x40) { ret = Foreground[addr - 0x10]; } + else if (addr < 0x80) { ret = Quad_Chars[addr - 0x40]; } + else if (addr < 0xA0) { ret = Sprite_Shapes[addr - 0x80]; } + else if (addr == 0xA0) { ret = VDC_ctrl; } + else if (addr == 0xA1) { ret = VDC_status; } + else if (addr == 0xA2) + { + for (int i = 0; i < 8; i++) + { + if (VDC_collision.Bit(i)) + { + ret |= VDC_col_ret[i]; + } + } + } + else if (addr == 0xA3) { ret = VDC_color; } + else if (addr == 0xA4) + { + if (latch_x_y) { ret = A4_latch; } + else { ret = (byte)((LY_ret >= 0) ? LY_ret : 0); } + } + else if (addr == 0xA5) + { + // reading the x reg clears the latch + if (latch_x_y) { ret = A5_latch; } + else { ret = (byte)(cycle); } + } + else if (addr <= 0xAA) { ret = AudioReadReg(addr); } + else if ((addr >= 0xC0) && (addr <= 0xC8)) { ret = Grid_H[addr - 0xC0]; } + else if ((addr >= 0xD0) && (addr <= 0xD8)) { ret = Grid_H[addr - 0xD0 + 9]; } + else if ((addr >= 0xE0) && (addr <= 0xE9)) { ret = Grid_V[addr - 0xE0]; } + + return ret; + } + + public void WriteReg(int addr, byte value) + { + if (addr < 0x10) + { + if (!VDC_ctrl.Bit(5)) { Sprites[addr] = value; } + //Console.WriteLine("spr: " + addr + " " + value + " " + Core.cpu.TotalExecutedCycles); + } + else if (addr < 0x40) + { + // chars position is not effected by last bit + if ((addr % 4) == 0) { value &= 0xFE; } + if (!VDC_ctrl.Bit(5)) { Foreground[addr - 0x10] = value; } + //Console.WriteLine("char: " + addr + " " + value + " " + Core.cpu.TotalExecutedCycles); + } + else if (addr < 0x80) + { + // chars position is not effected by last bit + if ((addr % 4) == 0) { value &= 0xFE; } + if (!VDC_ctrl.Bit(5)) + { + Quad_Chars[addr - 0x40] = value; + + // X and Y are mapped all together + if ((addr % 4) < 2) + { + for (int i = 0; i < 4; i++) + { + Quad_Chars[((addr - 0x40) & 0x30) + (addr % 4) + i * 4] = value; + } + } + } + + //Console.WriteLine("quad: " + (addr - 0x40) + " " + value + " " + Core.cpu.TotalExecutedCycles); + } + else if (addr < 0xA0) + { + Sprite_Shapes[addr - 0x80] = value; + } + else if (addr == 0xA0) + { + //Console.WriteLine("VDC_ctrl: " + value + " " + Core.cpu.TotalExecutedCycles); + if (value.Bit(1) && !VDC_ctrl.Bit(1)) + { + VDC_status &= 0xFD; + latch_x_y = true; + A4_latch = (byte)((LY_ret >= 0) ? LY_ret : 0); + A5_latch = (byte)(cycle); + } + + if (value.Bit(0) && !VDC_ctrl.Bit(0)) + { + HBL_req = true; + } + + VDC_ctrl = value; + + //if (VDC_ctrl.Bit(2)) { Console.WriteLine("sound INT"); } + //if (VDC_ctrl.Bit(0)) { Console.WriteLine("HBL INT"); } + } + else if (addr == 0xA1) + { + // not writable + } + else if (addr == 0xA2) + { + VDC_collision = value; + + //Console.WriteLine("VDC_collide: " + value + " " + Core.cpu.TotalExecutedCycles); + } + else if (addr == 0xA3) + { + VDC_color = value; + //Console.WriteLine("VDC_color: " + value + " " + LY + " " + Core.cpu.TotalExecutedCycles); + grid_brightness = (!lum_en | VDC_color.Bit(6)) ? 8 : 0; + bg_brightness = !lum_en ? 8 : 0; + } + else if (addr == 0xA4) + { + // writing has no effect + } + else if (addr == 0xA5) + { + // writing has no effect + } + else if (addr <= 0xAA) + { + AudioWriteReg(addr, value); + } + else if ((addr >= 0xC0) && (addr <= 0xC8)) + { + if (!VDC_ctrl.Bit(3)) { Grid_H[addr - 0xC0] = value; } else { Console.WriteLine("blocked"); } + } + else if ((addr >= 0xD0) && (addr <= 0xD8)) + { + if (!VDC_ctrl.Bit(3)) { Grid_H[addr - 0xD0 + 9] = value; } else { Console.WriteLine("blocked"); } + } + else if ((addr >= 0xE0) && (addr <= 0xE9)) + { + if (!VDC_ctrl.Bit(3)) { Grid_V[addr - 0xE0] = value; } else { Console.WriteLine("blocked"); } + } + //Console.WriteLine(addr + " " + value + " " + LY + " " + Core.cpu.TotalExecutedCycles); + } + + public virtual void tick() + { + Pixel_Stat = 0; + // drawing cycles + // note: clipping might need to be handled differently between PAL and NTSC + if (cycle < 182) + { + if ((LY < LINE_VBL) && (LY >= 0)) + { + // draw a pixel + process_pixel(); + } + } + else + { + // NOTE: most games expect one less T1 pulse after VBL, maybe some pre-render line + if (cycle == 182 && (LY < LINE_VBL) && (LY > 0)) + { + HBL = true; + // Send T1 pulses + Core.cpu.T1 = true; + } + + if (cycle == 212 && (LY < LINE_VBL)) + { + VDC_status &= 0xFE; + if (VDC_ctrl.Bit(0)) { Core.cpu.IRQPending = false; } + LY_ret = LY_ret + 1; + } + } + + if (cycle == 113 && (LY < LINE_VBL)) + { + VDC_status |= 0x01; + if (VDC_ctrl.Bit(0) && HBL_req) { Core.cpu.IRQPending = true; HBL_req = false; } + } + + cycle++; + + // end of scanline + if (cycle == 228) + { + cycle = 0; + + LY++; + + if (LY == LINE_VBL) + { + VBL = true; + Core.in_vblank = true; + VDC_status |= 0x08; + Core.cpu.IRQPending = true; + Core.cpu.T1 = true; + } + + if (LY >= LINE_VBL) + { + LY_ret = 0; + } + + if (LY == LINE_MAX) + { + LY = 0; + VBL = false; + Core.in_vblank = false; + Core.cpu.T1 = false; + if (Core.is_pal) { Core.cpu.IRQPending = false; } + LY_ret = 0; + } + + if (LY < LINE_VBL) + { + HBL = false; + Core.cpu.T1 = false; + } + } + } + + public virtual void Reset() + { + Sprites = new byte[16]; + Sprite_Shapes = new byte[32]; + Foreground = new byte[48]; + Quad_Chars = new byte[64]; + Grid_H = new byte[18]; + Grid_V = new byte[10]; + + VDC_ctrl = VDC_status = VDC_collision = VDC_color = grid_fill_col = 0; + Pixel_Stat = A4_latch = A5_latch = 0; + bg_brightness = grid_brightness = grid_fill = LY_ret = cycle = 0; + VBL = HBL = lum_en = false; + LY = 0; + + AudioReset(); + } + + public virtual void set_region(bool pal_flag) + { + is_pal = pal_flag; + + LINE_MAX = 312; + LINE_VBL = 240; + } + + public virtual void process_pixel() + { + current_pixel_offset = cycle * 2; + + // background + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_BG[((VDC_color >> 3) & 0x7) + bg_brightness]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_BG[((VDC_color >> 3) & 0x7) + bg_brightness]; + + if (grid_fill > 0) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_BG[(VDC_color & 0x7) + grid_brightness]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_BG[(VDC_color & 0x7) + grid_brightness]; + Pixel_Stat |= grid_fill_col; + grid_fill--; + } + + if (((cycle % 16) == 8) && ((LY - GRID_OFST) >= 0) && VDC_ctrl.Bit(3)) + { + int k = (int)Math.Floor(cycle / 16.0); + int j = (int)Math.Floor((LY - GRID_OFST) / 24.0); + if ((k < 10) && (j < 8)) + { + if (Grid_V[k].Bit(j)) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_BG[(VDC_color & 0x7) + grid_brightness]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_BG[(VDC_color & 0x7) + grid_brightness]; + Pixel_Stat |= 0x10; + if (VDC_ctrl.Bit(7)) { grid_fill = 15; } + else { grid_fill = 1; } + grid_fill_col = 0x10; + } + } + } + + if ((((LY - GRID_OFST) % 24) < 3) && ((cycle - 8) >= 0) && ((LY - GRID_OFST) >= 0) && VDC_ctrl.Bit(3)) + { + int k = (int)Math.Floor((cycle - 8) / 16.0); + int j = (int)Math.Floor((LY - GRID_OFST) / 24.0); + //Console.WriteLine(k + " " + j); + if ((k < 9) && (j < 9)) + { + if (j == 8) + { + if (Grid_H[k + 9].Bit(0)) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_BG[(VDC_color & 0x7) + grid_brightness]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_BG[(VDC_color & 0x7) + grid_brightness]; + Pixel_Stat |= 0x20; + + if (((cycle - 8) % 16) == 15) { grid_fill = 2; grid_fill_col = 0x20; } + } + } + else + { + if (Grid_H[k].Bit(j)) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_BG[(VDC_color & 0x7) + grid_brightness]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_BG[(VDC_color & 0x7) + grid_brightness]; + Pixel_Stat |= 0x20; + if (((cycle - 8) % 16) == 15) { grid_fill = 2; grid_fill_col = 0x20; } + } + } + } + } + + // grid + if (VDC_ctrl.Bit(6) && ((LY - GRID_OFST) >= 0) && (((LY - GRID_OFST) % GRID_OFST) < 3) && VDC_ctrl.Bit(3)) + { + + if (((cycle % 16) == 8) || ((cycle % 16) == 9)) + { + int k = (int)Math.Floor(cycle / 16.0); + int j = (int)Math.Floor((LY - GRID_OFST) / 24.0); + if ((k < 10) && (j < 9)) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_BG[(VDC_color & 0x7) + grid_brightness]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_BG[(VDC_color & 0x7) + grid_brightness]; + Pixel_Stat |= 0x20; + } + } + } + + if (VDC_ctrl.Bit(5)) + + { + // single characters + for (int i = 0; i < 12; i++) + { + if (((LY - OBJ_OFST) >= Foreground[i * 4]) && ((LY - OBJ_OFST) < (Foreground[i * 4] + 8 * 2))) + { + if ((cycle >= Foreground[i * 4 + 1]) && (cycle < (Foreground[i * 4 + 1] + 8))) + { + // sprite is in drawing region, pick a pixel + int offset_y = ((LY - OBJ_OFST) - Foreground[i * 4]) >> 1; + int offset_x = 7 - (cycle - Foreground[i * 4 + 1]); + int char_sel = Foreground[i * 4 + 2]; + + int char_pick = (char_sel - (((~(Foreground[i * 4] >> 1)) + 1) & 0xFF)); + + if (char_pick < 0) + { + char_pick &= 0xFF; + char_pick |= (Foreground[i * 4 + 3] & 1) << 8; + } + else + { + char_pick &= 0xFF; + char_pick |= (~(Foreground[i * 4 + 3] & 1)) << 8; + char_pick &= 0x1FF; + } + + // don't display past the end of a character + int pixel_pick = 0; + + if (((char_pick + 1) & 7) + offset_y < 8) + { + pixel_pick = (Internal_Graphics[(char_pick + offset_y) % 0x200] >> offset_x) & 1; + } + + if (pixel_pick == 1) + { + if (Core._settings.Show_Chars) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Foreground[i * 4 + 3] >> 1) & 0x7]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Foreground[i * 4 + 3] >> 1) & 0x7]; + } + + Pixel_Stat |= 0x80; + } + } + } + } + + // quads + // note: the quads all share X/Y values + for (int i = 3; i >= 0; i--) + { + if (((LY - OBJ_OFST) >= Quad_Chars[i * 16]) && ((LY - OBJ_OFST) < (Quad_Chars[i * 16] + 8 * 2))) + { + if ((cycle >= Quad_Chars[i * 16 + 1]) && (cycle < (Quad_Chars[i * 16 + 1] + 64))) + { + // object is in drawing region, pick a pixel + int offset_y = ((LY - OBJ_OFST) - Quad_Chars[i * 16]) >> 1; + int offset_x = 63 - (cycle - Quad_Chars[i * 16 + 1]); + int quad_num = 3; + while (offset_x > 15) + { + offset_x -= 16; + quad_num--; + } + + if (offset_x > 7) + { + offset_x -= 8; + + int char_sel = Quad_Chars[i * 16 + 4 * quad_num + 2]; + + int char_pick = (char_sel - (((~(Quad_Chars[i * 16] >> 1)) + 1) & 0xFF)); + + if (char_pick < 0) + { + char_pick &= 0xFF; + char_pick |= (Quad_Chars[i * 16 + 4 * quad_num + 3] & 1) << 8; + } + else + { + char_pick &= 0xFF; + char_pick |= (~(Quad_Chars[i * 16 + 4 * quad_num + 3] & 1)) << 8; + char_pick &= 0x1FF; + } + + // don't display past the end of a character + // for quads, this is controlled by the last quad, so need to recalculate the char + int char_sel_3 = Quad_Chars[i * 16 + 4 * 3 + 2]; + + int char_pick_3 = (char_sel_3 - (((~(Quad_Chars[i * 16] >> 1)) + 1) & 0xFF)); + + if (char_pick_3 < 0) + { + char_pick_3 &= 0xFF; + char_pick_3 |= (Quad_Chars[i * 16 + 4 * 3 + 3] & 1) << 8; + } + else + { + char_pick_3 &= 0xFF; + char_pick_3 |= (~(Quad_Chars[i * 16 + 4 * 3 + 3] & 1)) << 8; + char_pick_3 &= 0x1FF; + } + + int pixel_pick = 0; + + if (((char_pick_3 + 1) & 7) + offset_y < 8) + { + pixel_pick = (Internal_Graphics[(char_pick + offset_y) % 0x200] >> offset_x) & 1; + } + + if (pixel_pick == 1) + { + if (Core._settings.Show_Quads) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Quad_Chars[i * 16 + 4 * quad_num + 3] >> 1) & 0x7]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Quad_Chars[i * 16 + 4 * quad_num + 3] >> 1) & 0x7]; + } + + Pixel_Stat |= 0x80; + } + } + } + } + } + + // sprites + for (int i = 3; i >= 0; i--) + { + double_size = Sprites[i * 4 + 2].Bit(2) ? 4 : 2; + right_shift = Sprites[i * 4 + 2].Bit(0) ? 1 : 0; + + if (((LY - OBJ_OFST) >= Sprites[i * 4]) && ((LY - OBJ_OFST) < (Sprites[i * 4] + 8 * double_size))) + { + right_shift_even = (Sprites[i * 4 + 2].Bit(1) && (((Sprites[i * 4] + 8 * double_size - (LY - OBJ_OFST)) % 2) == 0)) ? 1 : 0; + x_base = Sprites[i * 4 + 1]; + + if ((right_shift + right_shift_even) == 0) + { + if ((cycle >= x_base) && (cycle < (x_base + 8 * (double_size / 2)))) + { + // character is in drawing region, pick a pixel + int offset_y = ((LY - OBJ_OFST) - Sprites[i * 4]) >> (double_size / 2); + int offset_x = (cycle - x_base) >> (double_size / 2 - 1); + + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + } + } + else + { + // special shifted cases + // since we are drawing two pixels at a time, we need to be careful that the next background / grid / char pixel + // doesn't overwrite the shifted pixel on the next pass + if ((cycle >= x_base) && (cycle < (x_base + 1 + 8 * (double_size / 2)))) + { + // character is in drawing region, pick a pixel + int offset_y = ((LY - OBJ_OFST) - Sprites[i * 4]) >> (double_size / 2); + int offset_x = (cycle - x_base) >> (double_size / 2 - 1); + + if (double_size == 2) + { + if ((cycle - x_base) == 8) + { + offset_x = 7; + + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + + if ((right_shift + right_shift_even) == 2) + { + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + } + + Pixel_Stat |= (byte)(1 << i); + } + } + else if ((cycle - x_base) == 0) + { + if ((right_shift + right_shift_even) < 2) + { + offset_x = 0; + + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + } + } + else + { + offset_x = cycle - x_base; + + if ((right_shift + right_shift_even) < 2) + { + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> (offset_x - 1)) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + + pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + } + else + { + offset_x -= 1; + + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + } + } + } + else + { + if (((cycle - x_base) >> 1) == 8) + { + if (((cycle - x_base) % 2) == 0) + { + offset_x = 7; + + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + if ((right_shift + right_shift_even) == 2) + { + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + } + + Pixel_Stat |= (byte)(1 << i); + } + } + } + else if (((cycle - x_base) >> 1) == 0) + { + if (((cycle - x_base) % 2) == 1) + { + offset_x = 0; + + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + } + else + { + if ((right_shift + right_shift_even) < 2) + { + offset_x = 0; + + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + } + } + } + else + { + if (((cycle - x_base) % 2) == 1) + { + offset_x = (cycle - x_base) >> 1; + + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + } + else + { + offset_x = (cycle - x_base) >> 1; + + if ((right_shift + right_shift_even) < 2) + { + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> (offset_x - 1)) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + + pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + } + else + { + offset_x -= 1; + + int pixel_pick = (Sprite_Shapes[i * 8 + offset_y] >> offset_x) & 1; + + if (pixel_pick == 1) + { + if (Core._settings.Show_Sprites) + { + Core._vidbuffer[LY * 372 + current_pixel_offset] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + Core._vidbuffer[LY * 372 + current_pixel_offset + 1] = (int)Color_Palette_SPR[(Sprites[i * 4 + 2] >> 3) & 0x7]; + } + + Pixel_Stat |= (byte)(1 << i); + } + } + } + } + } + } + } + } + } + } + + if (Pixel_Stat != 0) + { + // calculate collision + for (int i = 7; i >= 0; i--) + { + for (int j = 0; j < 8; j++) + { + if (Pixel_Stat.Bit(j) & Pixel_Stat.Bit(i) && (j != i)) + { + VDC_col_ret[i] |= (byte)(1 << j); + } + } + } + } + } + + public static readonly byte[] Internal_Graphics = { 0x7C, 0xC6, 0xC6, 0xC6, 0xC6, 0xC6, 0x7C, 00, // 0 0x00 + 0x18, 0x38, 0x18, 0x18, 0x18, 0x18, 0x3C, 00, // 1 0x01 + 0x3C, 0x66, 0x0C, 0x18, 0x30, 0x60, 0x7E, 00, // 2 0x02 + 0x7C, 0xC6, 0x06, 0x3C, 0x06, 0xC6, 0x7C, 00, // 3 0x03 + 0xCC, 0xCC, 0xCC, 0xFE, 0x0C, 0x0C, 0x0C, 00, // 4 0x04 + 0xFE, 0xC0, 0xC0, 0x7C, 0x06, 0xC6, 0x7C, 00, // 5 0x05 + 0x7C, 0xC6, 0xC0, 0xFC, 0xC6, 0xC6, 0x7C, 00, // 6 0x06 + 0xFE, 0x06, 0x0C, 0x18, 0x30, 0x60, 0xC0, 00, // 7 0x07 + 0x7C, 0xC6, 0xC6, 0x7C, 0xC6, 0xC6, 0x7C, 00, // 8 0x08 + 0x7C, 0xC6, 0xC6, 0x7E, 0x06, 0xC6, 0x7C, 00, // 9 0x09 + 0x00, 0x18, 0x18, 0x00, 0x18, 0x18, 0x00, 00, // : 0x0A + 0x18, 0x7E, 0x58, 0x7E, 0x1A, 0x7E, 0x18, 00, // $ 0x0B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 00, // 0x0C + 0x3C, 0x66, 0x0C, 0x18, 0x18, 0x00, 0x18, 00, // ? 0x0D + 0xC0, 0xC0, 0xC0, 0xC0, 0xC0, 0xC0, 0xFE, 00, // L 0x0E + 0xFC, 0xC6, 0xC6, 0xFC, 0xC0, 0xC0, 0xC0, 00, // P 0x0F + 0x00, 0x18, 0x18, 0x7E, 0x18, 0x18, 0x00, 00, // + 0x10 + 0xC6, 0xC6, 0xC6, 0xD6, 0xFE, 0xEE, 0xC6, 00, // W 0x11 + 0xFE, 0xC0, 0xC0, 0xFC, 0xC0, 0xC0, 0xFE, 00, // E 0x12 + 0xFC, 0xC6, 0xC6, 0xFC, 0xD8, 0xCC, 0xC6, 00, // R 0x13 + 0x7E, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 00, // T 0x14 + 0xC6, 0xC6, 0xC6, 0xC6, 0xC6, 0xC6, 0x7C, 00, // U 0x15 + 0x3C, 0x18, 0x18, 0x18, 0x18, 0x18, 0x3C, 00, // I 0x16 + 0x7C, 0xC6, 0xC6, 0xC6, 0xC6, 0xC6, 0x7C, 00, // O 0x17 + 0x7C, 0xC6, 0xC6, 0xC6, 0xDE, 0xCC, 0x76, 00, // Q 0x18 + 0x7C, 0xC6, 0xC0, 0x7C, 0x06, 0xC6, 0x7C, 00, // S 0x19 + 0xFC, 0xC6, 0xC6, 0xC6, 0xC6, 0xC6, 0xFC, 00, // D 0x1A + 0xFE, 0xC0, 0xC0, 0xF8, 0xC0, 0xC0, 0xC0, 00, // F 0x1B + 0x7C, 0xC6, 0xC0, 0xC0, 0xCE, 0xC6, 0x7E, 00, // G 0x1C + 0xC6, 0xC6, 0xC6, 0xFE, 0xC6, 0xC6, 0xC6, 00, // H 0x1D + 0x06, 0x06, 0x06, 0x06, 0x06, 0xC6, 0x7C, 00, // J 0x1E + 0xC6, 0xCC, 0xD8, 0xF0, 0xD8, 0xCC, 0xC6, 00, // K 0x1F + 0x38, 0x6C, 0xC6, 0xC6, 0xFE, 0xC6, 0xC6, 00, // A 0x20 + 0x7E, 0x06, 0x0C, 0x18, 0x30, 0x60, 0x7E, 00, // Z 0x21 + 0xC6, 0xC6, 0x6C, 0x38, 0x6C, 0xC6, 0xC6, 00, // X 0x22 + 0x7C, 0xC6, 0xC0, 0xC0, 0xC0, 0xC6, 0x7C, 00, // C 0x23 + 0xC6, 0xC6, 0xC6, 0xC6, 0xC6, 0x6C, 0x38, 00, // V 0x24 + 0xFC, 0xC6, 0xC6, 0xFC, 0xC6, 0xC6, 0xFC, 00, // B 0x25 + 0xC6, 0xEE, 0xFE, 0xD6, 0xC6, 0xC6, 0xC6, 00, // M 0x26 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x38, 0x38, 00, // . 0x27 + 0x00, 0x00, 0x00, 0x7E, 0x00, 0x00, 0x00, 00, // - 0x28 + 0x00, 0x66, 0x3C, 0x18, 0x3C, 0x66, 0x00, 00, // x 0x29 + 0x00, 0x18, 0x00, 0x7E, 0x00, 0x18, 0x00, 00, // (div) 0x2A + 0x00, 0x00, 0x7E, 0x00, 0x7E, 0x00, 0x00, 00, // = 0x2B + 0x66, 0x66, 0x66, 0x3C, 0x18, 0x18, 0x18, 00, // Y 0x2C + 0xC6, 0xE6, 0xF6, 0xFE, 0xDE, 0xCE, 0xC6, 00, // N 0x2D + 0x03, 0x06, 0x0C, 0x18, 0x30, 0x60, 0xC0, 00, // / 0x2E + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 00, // (box) 0x2F + 0xCE, 0xDB, 0xDB, 0xDB, 0xDB, 0xDB, 0xCE, 00, // 10 0x30 + 0x00, 0x00, 0x3C, 0x7E, 0x7E, 0x7E, 0x3C, 00, // (ball) 0x31 + 0x1C, 0x1C, 0x18, 0x1E, 0x18, 0x18, 0x1C, 00, // (person R) 0x32 + 0x1C, 0x1C, 0x18, 0x1E, 0x18, 0x34, 0x26, 00, // (runner R) 0x33 + 0x38, 0x38, 0x18, 0x78, 0x18, 0x2C, 0x64, 00, // (runner L) 0x34 + 0x38, 0x38, 0x18, 0x78, 0x18, 0x18, 0x38, 00, // (person L) 0x35 + 0x00, 0x18, 0x0C, 0xFE, 0x0C, 0x18, 0x00, 00, // (arrow R) 0x36 + 0x18, 0x3C, 0x7E, 0xFF, 0xFF, 0x18, 0x18, 00, // (tree) 0x37 + 0x03, 0x07, 0x0F, 0x1F, 0x3F, 0x7F, 0xFF, 00, // (ramp R) 0x38 + 0xC0, 0xE0, 0xF0, 0xF8, 0xFC, 0xFE, 0xFF, 00, // (ramp L) 0x39 + 0x38, 0x38, 0x12, 0xFE, 0xB8, 0x28, 0x6C, 00, // (person F) 0x3A + 0xC0, 0x60, 0x30, 0x18, 0x0C, 0x06, 0x03, 00, // \ 0x3B + 0x00, 0x00, 0x0C, 0x08, 0x08, 0xFF, 0x7E, 00, // (boat 1) 0x3C + 0x00, 0x03, 0x63, 0xFF, 0xFF, 0x18, 0x08, 00, // (plane) 0x3D + 0x00, 0x00, 0x00, 0x10, 0x38, 0xFF, 0x7E, 00, // (boat 2) 0x3E + 0x00, 0x00, 0x00, 0x06, 0x6E, 0xFF, 0x7E, 00 // (boat 3) 0x3F + }; + + public static readonly uint[] Color_Palette_SPR = + { + 0xFF676767, // grey + 0xFFFF4141, // light red + 0xFF56FF69, // light green + 0xFFFFCC66, // light yellow + 0xFF3595FF, // light blue + 0xFFDC84D4, // light violet + 0xFF77E6EB, // light blue-green + 0xFFFFFFFF, // white + }; + + public static readonly uint[] Color_Palette_BG = + { + 0xFF000000, // black + 0xFF1A37E0, // blue + 0xFF008000, // green + 0xFF2AAABE, // blue-green + 0xFFC00000, // red + 0xFF94309F, // violet + 0xFF77670B, // yellow + 0xFFBCBCBC, // light grey + 0xFF676767, // grey + 0xFF3595FF, // light blue + 0xFF56FF69, // light green + 0xFF77E6EB, // light blue-green + 0xFFFF4141, // light red + 0xFFDC84D4, // light violet + 0xFFFFCC66, // light yellow + 0xFFFFFFFF, // white + }; + + public static readonly byte[] VPP_Alpha_Numeric = { 0x00, 0x38, 0x44, 0x40, 0x20, 0x10, 0x00, 0x10, 0x00, 0x00, // (reverse ?) 0x00 + 0x00, 0x10, 0x28, 0x00, 0x38, 0x44, 0x7C, 0x44, 0x00, 0x00, // (A hat) 0x01 + 0x00, 0x08, 0x10, 0x3C, 0x20, 0x30, 0x20, 0x3C, 0x00, 0x00, // (E accent /) 0x02 + 0x00, 0x08, 0x14, 0x10, 0x38, 0x10, 0x24, 0x3C, 0x00, 0x00, // (pound) 0x03 + 0x00, 0x10, 0x38, 0x50, 0x38, 0x14, 0x54, 0x38, 0x10, 0x00, // $ 0x04 + 0x00, 0x38, 0x44, 0x40, 0x40, 0x40, 0x44, 0x38, 0x10, 0x20, // (C accent) 0x05 + 0x00, 0x28, 0x28, 0x7C, 0x28, 0x7C, 0x28, 0x28, 0x00, 0x00, // # 0x06 + 0x00, 0x20, 0x18, 0x00, 0x38, 0x44, 0x7C, 0x44, 0x00, 0x00, // (A accent) 0x07 + 0x00, 0x20, 0x18, 0x00, 0x44, 0x44, 0x44, 0x38, 0x00, 0x00, // (U accent) 0x08 + 0x00, 0x10, 0x08, 0x3C, 0x20, 0x30, 0x20, 0x3C, 0x00, 0x00, // (E accent \) 0x09 + 0x00, 0x3C, 0x50, 0x50, 0x58, 0x50, 0x50, 0x3C, 0x00, 0x00, // (E bold) 0x0A + 0x00, 0x08, 0x14, 0x3C, 0x20, 0x30, 0x20, 0x3C, 0x00, 0x00, // (E hat) 0x0B + 0x00, 0x00, 0x10, 0x20, 0x7F, 0x20, 0x10, 0x00, 0x00, 0x00, // (Left Arrow) 0x0C + 0x00, 0x10, 0x38, 0x54, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, // (Up Arrow) 0x0D + 0x00, 0x00, 0x80, 0x40, 0xFE, 0x40, 0x80, 0x00, 0x00, 0x00, // (Right Arrow) 0x0E + 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x54, 0x38, 0x10, 0x00, // (Down Arrow) 0x0F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (degree) 0x10 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (+/-) 0x11 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (e accent /) 0x12 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (e accent ..) 0x13 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (i accent ..) 0x14 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (c accent) 0x15 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (u hat) 0x16 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (a accent) 0x17 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (divide) 0x18 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (e accent \) 0x19 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (e bold) 0x1A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (e hat) 0x1B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (1/4) 0x1C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (1/2) 0x1D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (3/4) 0x1E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (o hat) 0x1F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (space) 0x20 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // ! 0x21 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // " 0x22 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (E accent ..) 0x23 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (a hat) 0x24 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // % 0x25 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // & 0x26 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // ' 0x27 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // ( 0x28 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // ) 0x29 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // * 0x2A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // + 0x2B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // , 0x2C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // - 0x2D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // . 0x2E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // / 0x2F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0 0x30 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 1 0x31 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 2 0x32 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 3 0x33 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 4 0x34 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 5 0x35 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 6 0x36 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 7 0x37 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 8 0x38 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 9 0x39 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // : 0x3A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // ; 0x3B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // < 0x3C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // = 0x3D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // > 0x3E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // ? 0x3F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // @ 0x40 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // A 0x41 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // B 0x42 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // C 0x43 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // D 0x44 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // E 0x45 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // F 0x46 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // G 0x47 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // H 0x48 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // I 0x49 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // J 0x4A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // K 0x4B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // L 0x4C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // M 0x4D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // N 0x4E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // O 0x4F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // P 0x50 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Q 0x51 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // R 0x52 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // S 0x53 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // T 0x54 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // U 0x55 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // V 0x56 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // W 0x57 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // X 0x58 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Y 0x59 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Z 0x5A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // [ 0x5B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // \ 0x5C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // ] 0x5D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (i hat) 0x5E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // _ 0x5F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (long dash) 0x60 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // a 0x61 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // b 0x62 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // c 0x63 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // d 0x64 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // e 0x65 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // f 0x66 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // g 0x67 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // h 0x68 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // i 0x69 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // j 0x6A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // k 0x6B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // l 0x6C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // m 0x6D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // n 0x6E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // o 0x6F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // p 0x70 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // q 0x71 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // r 0x72 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // s 0x73 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // t 0x74 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // u 0x75 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // v 0x76 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // w 0x77 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // x 0x78 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // y 0x79 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // z 0x7A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (left line) 0x7B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // | 0x7C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (right line) 0x7D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // (top dash) 0x7E + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF // (full box) 0x7F + }; + + public static readonly byte[] VPP_Mosaic = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x00 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x01 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x02 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x03 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x04 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x05 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x06 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x07 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x08 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x09 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x0A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x0B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x0C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x0D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x0E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x0F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x10 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x11 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x12 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x13 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x14 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x15 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x16 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x17 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x18 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x19 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x1A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x1B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x1C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x1D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x1E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x1F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x20 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x21 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x22 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x23 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x24 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x25 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x26 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x27 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x28 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x29 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x2A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x2B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x2C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x2D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x2E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x2F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x30 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x31 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x32 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x33 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x34 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x35 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x36 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x37 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x38 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x39 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x3A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x3B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x3C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x3D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x3E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x3F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x40 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x41 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x42 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x43 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x44 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x45 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x46 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x47 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x48 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x49 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x4A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x4B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x4C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x4D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x4E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x4F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x50 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x51 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x52 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x53 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x54 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x55 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x56 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x57 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x58 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x59 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x5A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x5B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x5C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x5D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x5E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x5F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x60 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x61 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x62 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x63 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x64 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x65 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x66 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x67 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x68 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x69 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x6A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x6B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x6C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x6D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x6E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x6F + + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x70 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x71 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x72 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x73 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x74 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x75 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x76 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x77 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x78 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x79 + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x7A + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x7B + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x7C + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x7D + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x7E + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0x7F + }; + + public void SyncState(Serializer ser) + { + ser.Sync(nameof(Sprites), ref Sprites, false); + ser.Sync(nameof(Sprite_Shapes), ref Sprite_Shapes, false); + ser.Sync(nameof(Foreground), ref Foreground, false); + ser.Sync(nameof(Quad_Chars), ref Quad_Chars, false); + ser.Sync(nameof(Grid_H), ref Grid_H, false); + ser.Sync(nameof(Grid_V), ref Grid_V, false); + ser.Sync(nameof(A4_latch), ref A4_latch); + ser.Sync(nameof(A5_latch), ref A5_latch); + + ser.Sync(nameof(VDC_ctrl), ref VDC_ctrl); + ser.Sync(nameof(VDC_status), ref VDC_status); + ser.Sync(nameof(VDC_collision), ref VDC_collision); + ser.Sync(nameof(VDC_col_ret), ref VDC_col_ret, false); + ser.Sync(nameof(VDC_color), ref VDC_color); + ser.Sync(nameof(Pixel_Stat), ref Pixel_Stat); + ser.Sync(nameof(bg_brightness), ref bg_brightness); + ser.Sync(nameof(grid_brightness), ref grid_brightness); + ser.Sync(nameof(lum_en), ref lum_en); + + ser.Sync(nameof(grid_fill), ref grid_fill); + ser.Sync(nameof(grid_fill_col), ref grid_fill_col); + ser.Sync(nameof(LY), ref LY); + ser.Sync(nameof(cycle), ref cycle); + ser.Sync(nameof(VBL), ref VBL); + ser.Sync(nameof(HBL), ref HBL); + + ser.Sync(nameof(latch_x_y), ref latch_x_y); + ser.Sync(nameof(HBL_req), ref HBL_req); + ser.Sync(nameof(LY_ret), ref LY_ret); + + AudioSyncState(ser); + } + + private BlipBuffer _blip_C = new BlipBuffer(15000); + + public byte sample; + + public byte shift_0, shift_1, shift_2, aud_ctrl; + public byte shift_reg_0, shift_reg_1, shift_reg_2; + + public uint master_audio_clock; + + public int tick_cnt, output_bit, shift_cnt; + + public int latched_sample_C; + + public byte AudioReadReg(int addr) + { + byte ret = 0; + + switch (addr) + { + case 0xA7: ret = shift_reg_0; break; + case 0xA8: ret = shift_reg_1; break; + case 0xA9: ret = shift_reg_2; break; + case 0xAA: ret = aud_ctrl; break; + } + //Console.WriteLine("aud read: " + (addr - 0xA7) + " " + ret + " " + Core.cpu.TotalExecutedCycles); + return ret; + } + + public void AudioWriteReg(int addr, byte value) + { + switch (addr) + { + case 0xA7: shift_0 = shift_reg_0 = value; break; + case 0xA8: shift_1 = shift_reg_1 = value; break; + case 0xA9: shift_2 = shift_reg_2 = value; break; + case 0xAA: aud_ctrl = value; break; + } + + //Console.WriteLine("aud write: " + (addr - 0xA7) + " " + value + " " + Core.cpu.TotalExecutedCycles); + } + + public void Audio_tick() + { + int C_final = 0; + + if (aud_ctrl.Bit(7)) + { + tick_cnt++; + if (tick_cnt > (aud_ctrl.Bit(5) ? 455 : 1820)) + { + tick_cnt = 0; + + output_bit = shift_2 & 1; + + shift_2 = (byte)((shift_2 >> 1) | ((shift_1 & 1) << 7)); + shift_1 = (byte)((shift_1 >> 1) | ((shift_0 & 1) << 7)); + shift_0 = (byte)(shift_0 >> 1); + + if (aud_ctrl.Bit(4)) + { + shift_0 |= (byte)(((output_bit.Bit(0) ^ shift_2.Bit(7)) ^ shift_2.Bit(4)) ? 0x80 : 0); + } + + shift_cnt++; + + if (shift_cnt == 24) + { + if (aud_ctrl.Bit(6) && !aud_ctrl.Bit(4)) + { + shift_0 = shift_reg_0; + shift_1 = shift_reg_1; + shift_2 = shift_reg_2; + } + + // audio interrupt for empy shift regs + if (VDC_ctrl.Bit(2)) + { + VDC_status |= 4; + Core.cpu.IRQPending = true; + } + + shift_cnt = 0; + } + } + + C_final = output_bit; + C_final *= ((aud_ctrl & 0xF) + 1) * 400; + } + + if (C_final != latched_sample_C) + { + _blip_C.AddDelta(master_audio_clock, C_final - latched_sample_C); + latched_sample_C = C_final; + } + + master_audio_clock++; + } + + public void AudioReset() + { + master_audio_clock = 0; + + sample = 0; + + shift_cnt = 0; + + _blip_C.SetRates(1792000, 44100); + } + + public void AudioSyncState(Serializer ser) + { + ser.Sync(nameof(master_audio_clock), ref master_audio_clock); + + ser.Sync(nameof(sample), ref sample); + ser.Sync(nameof(latched_sample_C), ref latched_sample_C); + + ser.Sync(nameof(aud_ctrl), ref aud_ctrl); + ser.Sync(nameof(shift_0), ref shift_0); + ser.Sync(nameof(shift_1), ref shift_1); + ser.Sync(nameof(shift_2), ref shift_2); + ser.Sync(nameof(shift_reg_0), ref shift_reg_0); + ser.Sync(nameof(shift_reg_1), ref shift_reg_1); + ser.Sync(nameof(shift_reg_2), ref shift_reg_2); + ser.Sync(nameof(tick_cnt), ref tick_cnt); + ser.Sync(nameof(shift_cnt), ref shift_cnt); + ser.Sync(nameof(output_bit), ref output_bit); + + ser.Sync(nameof(latch_x_y), ref latch_x_y); + } + + public bool CanProvideAsync => false; + + public void SetSyncMode(SyncSoundMode mode) + { + if (mode != SyncSoundMode.Sync) + { + throw new InvalidOperationException("Only Sync mode is supported_"); + } + } + + public SyncSoundMode SyncMode => SyncSoundMode.Sync; + + public void GetSamplesSync(out short[] samples, out int nsamp) + { + _blip_C.EndFrame(master_audio_clock); + + nsamp = _blip_C.SamplesAvailable(); + + samples = new short[nsamp * 2]; + + if (nsamp != 0) + { + _blip_C.ReadSamples(samples, nsamp, true); + } + + for (int i = 0; i < nsamp * 2; i += 2) + { + samples[i + 1] = samples[i]; + } + + master_audio_clock = 0; + } + + public void GetSamplesAsync(short[] samples) + { + throw new NotSupportedException("Async is not available"); + } + + public void DiscardSamples() + { + _blip_C.Clear(); + master_audio_clock = 0; + } + + private void GetSamples(short[] samples) + { + + } + + public void DisposeSound() + { + _blip_C.Clear(); + _blip_C.Dispose(); + _blip_C = null; + } + } + + public class PAL_PPU : PPU + { + public override void tick() + { + Pixel_Stat = 0; + // drawing cycles + // note: clipping might need to be handled differently between PAL and NTSC + if (cycle < 182) + { + if ((LY < LINE_VBL) && (LY >= 0)) + { + // draw a pixel + process_pixel(); + } + } + else + { + // NOTE: most games expect one less T1 pulse after VBL, maybe some pre-render line + if (cycle == 182 && (LY < LINE_VBL) && (LY > 0)) + { + HBL = true; + // Send T1 pulses + Core.cpu.T1 = true; + } + + if (cycle == 212 && (LY < LINE_VBL)) + { + VDC_status &= 0xFE; + if (VDC_ctrl.Bit(0)) { Core.cpu.IRQPending = false; } + LY_ret = LY_ret + 1; + } + } + + if (cycle == 113 && (LY < LINE_VBL)) + { + VDC_status |= 0x01; + if (VDC_ctrl.Bit(0) && HBL_req) { Core.cpu.IRQPending = true; HBL_req = false; } + } + + cycle++; + + // end of scanline + if (cycle == 228) + { + cycle = 0; + + LY++; + + if (LY == LINE_VBL) + { + VBL = true; + Core.in_vblank = true; + VDC_status |= 0x08; + Core.cpu.IRQPending = true; + Core.cpu.T1 = true; + } + + if (LY >= LINE_VBL) + { + LY_ret = 0; + } + + if (LY == LINE_MAX) + { + LY = 0; + VBL = false; + Core.in_vblank = false; + Core.cpu.T1 = false; + if (Core.is_pal) { Core.cpu.IRQPending = false; } + LY_ret = 0; + } + + if (LY < LINE_VBL) + { + HBL = false; + Core.cpu.T1 = false; + } + } + } + } +}