diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/LibGBHawk.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/LibGBHawk.cs
new file mode 100644
index 0000000000..ded3c7c0b8
--- /dev/null
+++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/LibGBHawk.cs
@@ -0,0 +1,188 @@
+using System;
+using System.Runtime.InteropServices;
+using System.Text;
+
+namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
+{
+ ///
+ /// static bindings into GBHawk.dll
+ ///
+ public static class LibGBHawk
+ {
+ # region Core
+ /// opaque state pointer
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern IntPtr GB_create();
+
+ /// opaque state pointer
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern void GB_destroy(IntPtr core);
+
+ ///
+ /// Load BIOS and BASIC image. each must be 16K in size
+ ///
+ /// opaque state pointer
+ /// the rom data, can be disposed of once this function returns
+ /// is it GBC console
+ /// is it in GBA mode
+ /// 0 on success, negative value on failure.
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern int GB_load_bios(IntPtr core, byte[] bios, bool is_GBC, bool GBC_as_GBA);
+
+ ///
+ /// Load ROM image.
+ ///
+ /// opaque state pointer
+ /// the rom data, can be disposed of once this function returns
+ /// length of romdata in bytes
+ /// 0 on success, negative value on failure.
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern int GB_load(IntPtr core, byte[] romdata_1, uint length_1, uint RTC_init, uint RTC_offset);
+
+ ///
+ /// Advance a frame and send controller data.
+ ///
+ /// opaque state pointer
+ /// controller data for player 1
+ /// controller data for player 2
+ /// length of romdata in bytes
+ /// Mapper number to load core with
+ /// 0 on success, negative value on failure.
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern bool GB_frame_advance(IntPtr core, byte ctrl1, byte ctrl2, byte[] kbrows, bool render, bool sound);
+
+ ///
+ /// Get Video data
+ ///
+ /// opaque state pointer
+ /// where to send video to
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern void GB_get_video(IntPtr core, int[] videobuf);
+
+ ///
+ /// Get Video data
+ ///
+ /// opaque state pointer
+ /// where to send left audio to
+ /// number of left samples
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern uint GB_get_audio(IntPtr core, int[] aud_buf, ref uint n_samp);
+
+ #endregion
+
+ #region State Save / Load
+
+ ///
+ /// Save State
+ ///
+ /// opaque state pointer
+ /// save buffer
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern void GB_save_state(IntPtr core, byte[] saver);
+
+ ///
+ /// Load State
+ ///
+ /// opaque state pointer
+ /// load buffer
+ [DllImport("MSXHAWK.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern void GB_load_state(IntPtr core, byte[] loader);
+
+ #endregion
+
+ #region Memory Domain Functions
+
+ ///
+ /// Read the system bus
+ ///
+ /// opaque state pointer
+ /// system bus address
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern byte GB_getsysbus(IntPtr core, int addr);
+
+ ///
+ /// Read the VRAM
+ ///
+ /// opaque state pointer
+ /// vram address
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern byte GB_getvram(IntPtr core, int addr);
+
+ ///
+ /// Read the RAM
+ ///
+ /// opaque state pointer
+ /// ram address
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern byte GB_getram(IntPtr core, int addr);
+
+ #endregion
+
+ #region Tracer
+ ///
+ /// type of the cpu trace callback
+ ///
+ /// type of event
+ [UnmanagedFunctionPointer(CallingConvention.Cdecl)]
+ public delegate void TraceCallback(int t);
+
+ ///
+ /// set a callback for trace logging
+ ///
+ /// opaque state pointer
+ /// null to clear
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern void GB_settracecallback(IntPtr core, TraceCallback callback);
+
+ ///
+ /// get the trace logger header length
+ ///
+ /// opaque state pointer
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern int GB_getheaderlength(IntPtr core);
+
+ ///
+ /// get the trace logger disassembly length, a constant
+ ///
+ /// opaque state pointer
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern int GB_getdisasmlength(IntPtr core);
+
+ ///
+ /// get the trace logger register string length, a constant
+ ///
+ /// opaque state pointer
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern int GB_getregstringlength(IntPtr core);
+
+ ///
+ /// get the trace logger header
+ ///
+ /// opaque state pointer
+ /// pointer to const char *
+ /// null to clear
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern void GB_getheader(IntPtr core, StringBuilder h, int l);
+
+ ///
+ /// get the register state from the cpu
+ ///
+ /// opaque state pointer
+ /// pointer to const char *
+ /// call type
+ /// copy length, must be obtained from appropriate get legnth function
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern void GB_getregisterstate(IntPtr core, StringBuilder h, int t, int l);
+
+ ///
+ /// get the register state from the cpu
+ ///
+ /// opaque state pointer
+ /// pointer to const char *
+ /// call type
+ /// copy length, must be obtained from appropriate get legnth function
+ [DllImport("GBHawk.dll", CallingConvention = CallingConvention.Cdecl)]
+ public static extern void GB_getdisassembly(IntPtr core, StringBuilder h, int t, int l);
+ #endregion
+ }
+}
diff --git a/libHawk/GBHawk/GBHawk/Core.h b/libHawk/GBHawk/GBHawk/Core.h
index ce28e5c5c6..5baab56200 100644
--- a/libHawk/GBHawk/GBHawk/Core.h
+++ b/libHawk/GBHawk/GBHawk/Core.h
@@ -5,11 +5,11 @@
#include "LR35902.h"
#include "GBAudio.h"
-#include "PPU_Base.h"
#include "Memory.h"
#include "Timer.h"
#include "SerialPort.h"
-#include "Mapper_Base.h"
+#include "Mappers.h"
+#include "PPU.h"
namespace GBHawk
{
@@ -18,27 +18,74 @@ namespace GBHawk
public:
GBCore()
{
+
+ };
+
+ PPU* ppu;
+ LR35902 cpu;
+ GBAudio psg;
+ MemoryManager MemMap;
+ Timer timer;
+ SerialPort serialport;
+ Mapper* mapper;
+
+ void Load_BIOS(uint8_t* bios, bool GBC_console, bool GBC_as_GBA)
+ {
+ MemMap.Load_BIOS(bios, GBC_console, GBC_as_GBA);
+ }
+
+ void Load_ROM(uint8_t* ext_rom_1, uint32_t ext_rom_size_1, string MD5, uint32_t RTC_initial, uint32_t RTC_offset)
+ {
+ MemMap.Load_ROM(ext_rom_1, ext_rom_size_1);
+
+ // After we load the ROM we need to initialize the rest of the components (ppu and mapper)
+ // tell the cpu the console type
+ cpu.is_GBC = MemMap.is_GBC;
+
+ //initialize system components
+ // initialize the proper ppu
+ if (MemMap.is_GBC)
+ {
+ if ((MemMap.header[0x43] != 0x80) && (MemMap.header[0x43] != 0xC0))
+ {
+ ppu = new GBC_GB_PPU();
+ }
+ else
+ {
+ ppu = new GBC_PPU();
+ }
+ }
+ else
+ {
+ ppu = new GB_PPU();
+ }
+
+ MemMap.ppu_pntr = &ppu[0];
+
+ // initialize the proper mapper
+ Setup_Mapper(MD5, RTC_initial, RTC_offset);
+
+ // set up pointers
MemMap.cpu_pntr = &cpu;
- MemMap.ppu_pntr = &ppu;
MemMap.psg_pntr = &psg;
MemMap.timer_pntr = &timer;
MemMap.serialport_pntr = &serialport;
cpu.mem_ctrl = &MemMap;
- ppu.FlagI = &cpu.FlagI;
- ppu.in_vblank = &MemMap.in_vblank;
- ppu.cpu_LY = &cpu.LY;
- ppu.REG_FFFF = &MemMap.REG_FFFF;
- ppu.REG_FF0F = &MemMap.REG_FF0F;
- ppu._scanlineCallbackLine = &MemMap._scanlineCallbackLine;
- ppu.OAM = &MemMap.OAM[0];
- ppu.VRAM = &MemMap.VRAM[0];
- ppu.VRAM_Bank = &MemMap.VRAM_Bank;
- ppu.cpu_halted = &cpu.halted;
- ppu._vidbuffer = &MemMap._vidbuffer[0];
- ppu.color_palette = &MemMap.color_palette[0];
- ppu.HDMA_transfer = &MemMap.HDMA_transfer;
- ppu.GBC_compat = &MemMap.GBC_compat;
+ MemMap.ppu_pntr->FlagI = &cpu.FlagI;
+ MemMap.ppu_pntr->in_vblank = &MemMap.in_vblank;
+ MemMap.ppu_pntr->cpu_LY = &cpu.LY;
+ MemMap.ppu_pntr->REG_FFFF = &MemMap.REG_FFFF;
+ MemMap.ppu_pntr->REG_FF0F = &MemMap.REG_FF0F;
+ MemMap.ppu_pntr->_scanlineCallbackLine = &MemMap._scanlineCallbackLine;
+ MemMap.ppu_pntr->OAM = &MemMap.OAM[0];
+ MemMap.ppu_pntr->VRAM = &MemMap.VRAM[0];
+ MemMap.ppu_pntr->VRAM_Bank = &MemMap.VRAM_Bank;
+ MemMap.ppu_pntr->cpu_halted = &cpu.halted;
+ MemMap.ppu_pntr->_vidbuffer = &MemMap._vidbuffer[0];
+ MemMap.ppu_pntr->color_palette = &MemMap.color_palette[0];
+ MemMap.ppu_pntr->HDMA_transfer = &MemMap.HDMA_transfer;
+ MemMap.ppu_pntr->GBC_compat = &MemMap.GBC_compat;
timer.FlagI = &cpu.FlagI;
timer.REG_FFFF = &MemMap.REG_FFFF;
@@ -53,31 +100,13 @@ namespace GBHawk
psg.double_speed = &MemMap.double_speed;
psg.timer_div_reg = &timer.divider_reg;
- mapper.addr_access = &MemMap.addr_access;
- mapper.Acc_X_state = &MemMap.Acc_X_state;
- mapper.Acc_Y_state = &MemMap.Acc_Y_state;
- mapper.ROM_Length = &MemMap.ROM_Length;
- mapper.Cart_RAM_Length = &MemMap.Cart_RAM_Length;
- mapper.ROM = &MemMap.ROM[0];
- mapper.Cart_RAM = &MemMap.Cart_RAM[0];
- };
-
- PPU ppu;
- LR35902 cpu;
- GBAudio psg;
- MemoryManager MemMap;
- Timer timer;
- SerialPort serialport;
- Mapper mapper;
-
- void Load_BIOS(uint8_t* bios, bool GBC_console)
- {
- MemMap.Load_BIOS(bios, GBC_console);
- }
-
- void Load_ROM(uint8_t* ext_rom_1, uint32_t ext_rom_size_1, uint32_t ext_rom_mapper_1, uint8_t* ext_rom_2, uint32_t ext_rom_size_2, uint32_t ext_rom_mapper_2)
- {
- MemMap.Load_ROM(ext_rom_1, ext_rom_size_1, ext_rom_mapper_1, ext_rom_2, ext_rom_size_2, ext_rom_mapper_2);
+ MemMap.mapper_pntr->addr_access = &MemMap.addr_access;
+ MemMap.mapper_pntr->Acc_X_state = &MemMap.Acc_X_state;
+ MemMap.mapper_pntr->Acc_Y_state = &MemMap.Acc_Y_state;
+ MemMap.mapper_pntr->ROM_Length = &MemMap.ROM_Length;
+ MemMap.mapper_pntr->Cart_RAM_Length = &MemMap.Cart_RAM_Length;
+ MemMap.mapper_pntr->ROM = &MemMap.ROM[0];
+ MemMap.mapper_pntr->Cart_RAM = &MemMap.Cart_RAM[0];
}
bool FrameAdvance(uint8_t controller_1, uint8_t controller_2, uint8_t* kb_rows_ptr, bool render, bool rendersound)
@@ -86,7 +115,6 @@ namespace GBHawk
MemMap.controller_byte_1 = controller_1;
MemMap.controller_byte_2 = controller_2;
MemMap.kb_rows = kb_rows_ptr;
- MemMap.start_pressed = (controller_1 & 0x80) > 0;
MemMap.lagged = true;
uint32_t scanlinesPerFrame = 262;
@@ -119,11 +147,226 @@ namespace GBHawk
return psg.master_audio_clock;
}
+ void Setup_Mapper(string MD5, uint32_t RTC_initial, uint32_t RTC_offset)
+ {
+ // setup up mapper based on header entry
+ string mppr;
+
+ switch (MemMap.header[0x47])
+ {
+ case 0x0: mapper = new Mapper_Default(); mppr = "NROM"; break;
+ case 0x1: mapper = new Mapper_MBC1(); mppr = "MBC1"; break;
+ case 0x2: mapper = new Mapper_MBC1(); mppr = "MBC1"; break;
+ case 0x3: mapper = new Mapper_MBC1(); mppr = "MBC1"; MemMap.has_bat = true; break;
+ case 0x5: mapper = new Mapper_MBC2(); mppr = "MBC2"; break;
+ case 0x6: mapper = new Mapper_MBC2(); mppr = "MBC2"; MemMap.has_bat = true; break;
+ case 0x8: mapper = new Mapper_Default(); mppr = "NROM"; break;
+ case 0x9: mapper = new Mapper_Default(); mppr = "NROM"; MemMap.has_bat = true; break;
+ case 0xB: mapper = new Mapper_MMM01(); mppr = "MMM01"; break;
+ case 0xC: mapper = new Mapper_MMM01(); mppr = "MMM01"; break;
+ case 0xD: mapper = new Mapper_MMM01(); mppr = "MMM01"; MemMap.has_bat = true; break;
+ case 0xF: mapper = new Mapper_MBC3(); mppr = "MBC3"; MemMap.has_bat = true; break;
+ case 0x10: mapper = new Mapper_MBC3(); mppr = "MBC3"; MemMap.has_bat = true; break;
+ case 0x11: mapper = new Mapper_MBC3(); mppr = "MBC3"; break;
+ case 0x12: mapper = new Mapper_MBC3(); mppr = "MBC3"; break;
+ case 0x13: mapper = new Mapper_MBC3(); mppr = "MBC3"; MemMap.has_bat = true; break;
+ case 0x19: mapper = new Mapper_MBC5(); mppr = "MBC5"; break;
+ case 0x1A: mapper = new Mapper_MBC5(); mppr = "MBC5"; MemMap.has_bat = true; break;
+ case 0x1B: mapper = new Mapper_MBC5(); mppr = "MBC5"; break;
+ case 0x1C: mapper = new Mapper_MBC5(); mppr = "MBC5"; break;
+ case 0x1D: mapper = new Mapper_MBC5(); mppr = "MBC5"; break;
+ case 0x1E: mapper = new Mapper_MBC5(); mppr = "MBC5"; MemMap.has_bat = true; break;
+ case 0x20: mapper = new Mapper_MBC6(); mppr = "MBC6"; break;
+ case 0x22: mapper = new Mapper_MBC7(); mppr = "MBC7"; MemMap.has_bat = true; break;
+ case 0xFC: mapper = new Mapper_Camera(); mppr = "CAM"; MemMap.has_bat = true; break;
+ case 0xFD: mapper = new Mapper_TAMA5(); mppr = "TAMA5"; MemMap.has_bat = true; break;
+ case 0xFE: mapper = new Mapper_HuC3(); mppr = "HuC3"; break;
+ case 0xFF: mapper = new Mapper_HuC1(); mppr = "HuC1"; break;
+
+ // Bootleg mappers
+ // NOTE: Sachen mapper selection does not account for scrambling, so if another bootleg mapper
+ // identifies itself as 0x31, this will need to be modified
+ case 0x31: mapper = new Mapper_Sachen2(); mppr = "Schn2"; break;
+
+ case 0x4:
+ case 0x7:
+ case 0xA:
+ case 0xE:
+ case 0x14:
+ case 0x15:
+ case 0x16:
+ case 0x17:
+ case 0x18:
+ case 0x1F:
+ case 0x21:
+ default:
+ // mapper not implemented
+ mapper = nullptr;
+ }
+
+ // special case for multi cart mappers
+ if ((MD5 == "97122B9B183AAB4079C8D36A4CE6E9C1") ||
+ (MD5 == "9FB9C42CF52DCFDCFBAD5E61AE1B5777") ||
+ (MD5 == "CF1F58AB72112716D3C615A553B2F481")
+ )
+ {
+ mapper = new Mapper_MBC1_Multi();
+ }
+
+ // Wisdom Tree does not identify their mapper, so use hash instead
+ if ((MD5 == "2C07CAEE51A1F0C91C72C7C6F380B0F6") || // Joshua
+ (MD5 == "37E017C8D1A45BAB609FB5B43FB64337") || // Spiritual Warfare
+ (MD5 == "AB1FA0ED0207B1D0D5F401F0CD17BEBF") || // Exodus
+ (MD5 == "BA2AC3587B3E1B36DE52E740274071B0") || // Bible - KJV
+ (MD5 == "8CDDB8B2DCD3EC1A3FDD770DF8BDA07C") // Bible - NIV
+ )
+ {
+ mapper = new Mapper_WT();
+ mppr = "Wtree";
+ }
+
+ // special case for bootlegs
+ if ((MD5 == "CAE0998A899DF2EE6ABA8E7695C2A096"))
+ {
+ mapper = new Mapper_RM8();
+ }
+ if ((MD5 == "D3C1924D847BC5D125BF54C2076BE27A"))
+ {
+ mapper = new Mapper_Sachen1();
+ mppr = "Schn1";
+ }
+
+ MemMap.Cart_RAM = nullptr;
+
+ switch (MemMap.header[0x49])
+ {
+ case 1:
+ MemMap.Cart_RAM = new uint8_t[0x800];
+ break;
+ case 2:
+ MemMap.Cart_RAM = new uint8_t[0x2000];
+ break;
+ case 3:
+ MemMap.Cart_RAM = new uint8_t[0x8000];
+ break;
+ case 4:
+ MemMap.Cart_RAM = new uint8_t[0x20000];
+ break;
+ case 5:
+ MemMap.Cart_RAM = new uint8_t[0x10000];
+ break;
+ case 0:
+ MemMap.has_bat = false;
+ break;
+ }
+
+ // Sachen maper not known to have RAM
+ if ((mppr == "Schn1") || (mppr == "Schn2"))
+ {
+ MemMap.Cart_RAM = nullptr;
+ MemMap.Use_MT = true;
+ }
+
+ // mbc2 carts have built in RAM
+ if (mppr == "MBC2")
+ {
+ MemMap.Cart_RAM = new uint8_t[0x200];
+ }
+
+ // mbc7 has 256 bytes of RAM, regardless of any header info
+ if (mppr == "MBC7")
+ {
+ MemMap.Cart_RAM = new uint8_t[0x100];
+ MemMap.has_bat = true;
+ }
+
+ // TAMA5 has 0x1000 bytes of RAM, regardless of any header info
+ if (mppr == "TAMA5")
+ {
+ MemMap.Cart_RAM = new uint8_t[0x20];
+ MemMap.has_bat = true;
+ }
+
+ MemMap.Cart_RAM_Length = sizeof(MemMap.Cart_RAM);
+
+ if (MemMap.Cart_RAM != nullptr && (mppr != "MBC7"))
+ {
+ for (uint32_t i = 0; i < MemMap.Cart_RAM_Length; i++)
+ {
+ MemMap.Cart_RAM[i] = 0xFF;
+ }
+ }
+
+ // Extra RTC initialization for mbc3, HuC3, and TAMA5
+ if (mppr == "MBC3")
+ {
+ MemMap.Use_MT = true;
+
+ mapper->RTC_Get(RTC_offset, 5);
+
+ int days = (int)floor(RTC_initial / 86400.0);
+
+ int days_upper = ((days & 0x100) >> 8) | ((days & 0x200) >> 2);
+
+ mapper->RTC_Get(days_upper, 4);
+ mapper->RTC_Get(days & 0xFF, 3);
+
+ int remaining = RTC_initial - (days * 86400);
+
+ int hours = (int)floor(remaining / 3600.0);
+
+ mapper->RTC_Get(hours & 0xFF, 2);
+
+ remaining = remaining - (hours * 3600);
+
+ int minutes = (int)floor(remaining / 60.0);
+
+ mapper->RTC_Get(minutes & 0xFF, 1);
+
+ remaining = remaining - (minutes * 60);
+
+ mapper->RTC_Get(remaining & 0xFF, 0);
+ }
+
+ if (mppr == "HuC3")
+ {
+ MemMap.Use_MT = true;
+
+ int years = (int)floor(RTC_initial / 31536000.0);
+
+ mapper->RTC_Get(years, 24);
+
+ int remaining = RTC_initial - (years * 31536000);
+
+ int days = (int)floor(remaining / 86400.0);
+ int days_upper = (days >> 8) & 0xF;
+
+ mapper->RTC_Get(days_upper, 20);
+ mapper->RTC_Get(days & 0xFF, 12);
+
+ remaining = remaining - (days * 86400);
+
+ int minutes = (int)floor(remaining / 60.0);
+ int minutes_upper = (minutes >> 8) & 0xF;
+
+ mapper->RTC_Get(minutes_upper, 8);
+ mapper->RTC_Get(remaining & 0xFF, 0);
+ }
+
+ if (mppr == "TAMA5")
+ {
+ MemMap.Use_MT = true;
+
+ // currently no date / time input for TAMA5
+
+ }
+ }
+
#pragma region State Save / Load
void SaveState(uint8_t* saver)
{
- saver = ppu.SaveState(saver);
+ saver = ppu->SaveState(saver);
saver = cpu.SaveState(saver);
saver = psg.SaveState(saver);
saver = MemMap.SaveState(saver);
@@ -131,7 +374,7 @@ namespace GBHawk
void LoadState(uint8_t* loader)
{
- loader = ppu.LoadState(loader);
+ loader = ppu->LoadState(loader);
loader = cpu.LoadState(loader);
loader = psg.LoadState(loader);
loader = MemMap.LoadState(loader);
diff --git a/libHawk/GBHawk/GBHawk/GBC_GB_PPU.h b/libHawk/GBHawk/GBHawk/GBC_GB_PPU.h
deleted file mode 100644
index 597c479dcd..0000000000
--- a/libHawk/GBHawk/GBHawk/GBC_GB_PPU.h
+++ /dev/null
@@ -1,1624 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "PPU_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class GBC_GB_PPU : public PPU
- {
- public:
- uint8_t ReadReg(uint32_t addr)
- {
- uint8_t ret = 0;
- //Console.WriteLine(Core.cpu.TotalExecutedCycles);
- switch (addr)
- {
- case 0xFF40: ret = LCDC; break; // LCDC
- case 0xFF41: ret = STAT; break; // STAT
- case 0xFF42: ret = scroll_y; break; // SCY
- case 0xFF43: ret = scroll_x; break; // SCX
- case 0xFF44: ret = LY; break; // LY
- case 0xFF45: ret = LYC; break; // LYC
- case 0xFF46: ret = DMA_addr; break; // DMA
- case 0xFF47: ret = BGP; break; // BGP
- case 0xFF48: ret = obj_pal_0; break; // OBP0
- case 0xFF49: ret = obj_pal_1; break; // OBP1
- case 0xFF4A: ret = window_y; break; // WY
- case 0xFF4B: ret = window_x; break; // WX
-
- // These are GBC specific Regs
- case 0xFF51: ret = HDMA_src_hi; break; // HDMA1
- case 0xFF52: ret = HDMA_src_lo; break; // HDMA2
- case 0xFF53: ret = HDMA_dest_hi; break; // HDMA3
- case 0xFF54: ret = HDMA_dest_lo; break; // HDMA4
- case 0xFF55: ret = HDMA_ctrl(); break; // HDMA5
- case 0xFF68: ret = BG_pal_ret(); break; // BGPI
- case 0xFF69: ret = BG_PAL_read(); break; // BGPD
- case 0xFF6A: ret = OBJ_pal_ret(); break; // OBPI
- case 0xFF6B: ret = OBJ_bytes[OBJ_bytes_index]; break; // OBPD
- }
-
- return ret;
- }
-
- uint8_t BG_PAL_read()
- {
- if (VRAM_access_read)
- {
- return BG_bytes[BG_bytes_index];
- }
- else
- {
- return 0xFF;
- }
- }
-
- void WriteReg(uint32_t addr, uint8_t value)
- {
- switch (addr)
- {
- case 0xFF40: // LCDC
- if (((LCDC & 0x80) > 0) && !((value & 0x80) > 0))
- {
- VRAM_access_read = true;
- VRAM_access_write = true;
- OAM_access_read = true;
- OAM_access_write = true;
- }
-
- if (!((LCDC & 0x80) > 0) && ((value & 0x80) > 0))
- {
- // don't draw for one frame after turning on
- blank_frame = true;
- }
-
- LCDC = value;
- break;
- case 0xFF41: // STAT
- // note that their is no stat interrupt bug in GBC
- STAT = (uint8_t)((value & 0xF8) | (STAT & 7) | 0x80);
-
- if (((STAT & 3) == 0) && ((STAT & 0x8) > 0)) { HBL_INT = true; }
- else { HBL_INT = false; }
- if (((STAT & 3) == 1) && ((STAT & 0x10) > 0)) { VBL_INT = true; }
- else { VBL_INT = false; }
- // OAM not triggered?
- // if (((STAT & 3) == 2) && STAT.Bit(5)) { OAM_INT = true; } else { OAM_INT = false; }
-
- if (((value & 0x40) > 0) && ((LCDC & 0x80) > 0))
- {
- if (LY == LYC) { LYC_INT = true; }
- else { LYC_INT = false; }
- }
- if (!((STAT & 0x40) > 0)) { LYC_INT = false; }
- break;
- case 0xFF42: // SCY
- scroll_y = value;
- break;
- case 0xFF43: // SCX
- scroll_x = value;
- break;
- case 0xFF44: // LY
- LY = 0; /*reset*/
- break;
- case 0xFF45: // LYC
- // tests indicate that latching writes to LYC should take place 4 cycles after the write
- // otherwise tests around LY boundaries will fail
- LYC_t = value;
- LYC_cd = 4;
- break;
- case 0xFF46: // DMA
- DMA_addr = value;
- DMA_start = true;
- DMA_OAM_access = true;
- DMA_clock = 0;
- DMA_inc = 0;
- break;
- case 0xFF47: // BGP
- BGP = value;
- break;
- case 0xFF48: // OBP0
- obj_pal_0 = value;
- break;
- case 0xFF49: // OBP1
- obj_pal_1 = value;
- break;
- case 0xFF4A: // WY
- window_y = value;
- break;
- case 0xFF4B: // WX
- window_x = value;
- break;
-
- // These are GBC specific Regs
- case 0xFF51: // HDMA1
- HDMA_src_hi = value;
- cur_DMA_src = (uint32_t)(((HDMA_src_hi & 0xFF) << 8) | (cur_DMA_src & 0xF0));
- break;
- case 0xFF52: // HDMA2
- HDMA_src_lo = value;
- cur_DMA_src = (uint32_t)((cur_DMA_src & 0xFF00) | (HDMA_src_lo & 0xF0));
- break;
- case 0xFF53: // HDMA3
- HDMA_dest_hi = value;
- cur_DMA_dest = (uint32_t)(((HDMA_dest_hi & 0x1F) << 8) | (cur_DMA_dest & 0xF0));
- break;
- case 0xFF54: // HDMA4
- HDMA_dest_lo = value;
- cur_DMA_dest = (uint32_t)((cur_DMA_dest & 0xFF00) | (HDMA_dest_lo & 0xF0));
- break;
- case 0xFF55: // HDMA5
- if (!HDMA_active)
- {
- HDMA_mode = ((value & 0x80) > 0);
- HDMA_countdown = 4;
- HDMA_tick = 0;
- if (((value & 0x80) > 0))
- {
- // HDMA during HBlank only, but only if screen is on, otherwise DMA immediately one block of data
- // worms armaggedon requires HDMA to fire in hblank mode even if the screen is off.
- HDMA_active = true;
- HBL_HDMA_count = 0x10;
-
- last_HBL = LY - 1;
-
- HBL_test = true;
- HBL_HDMA_go = false;
-
- if (!((LCDC & 0x80) > 0))
- {
- HDMA_run_once = true;
- }
- }
- else
- {
- // HDMA immediately
- HDMA_active = true;
- HDMA_transfer[0] = true;
- }
-
- HDMA_length = ((value & 0x7F) + 1) * 16;
- }
- else
- {
- //terminate the transfer
- if (!((value & 0x80) > 0))
- {
- HDMA_active = false;
- }
- }
-
- break;
- case 0xFF68: // BGPI
- BG_bytes_index = (uint8_t)(value & 0x3F);
- BG_bytes_inc = ((value & 0x80) == 0x80);
- break;
- case 0xFF69: // BGPD
- if (VRAM_access_write)
- {
- BG_transfer_byte = value;
- BG_bytes[BG_bytes_index] = value;
- }
-
- // change the appropriate palette color
- color_compute_BG();
- if (BG_bytes_inc) { BG_bytes_index++; BG_bytes_index &= 0x3F; }
- break;
- case 0xFF6A: // OBPI
- OBJ_bytes_index = (uint8_t)(value & 0x3F);
- OBJ_bytes_inc = ((value & 0x80) == 0x80);
- break;
- case 0xFF6B: // OBPD
- OBJ_transfer_byte = value;
- OBJ_bytes[OBJ_bytes_index] = value;
-
- // change the appropriate palette color
- color_compute_OBJ();
-
- if (OBJ_bytes_inc) { OBJ_bytes_index++; OBJ_bytes_index &= 0x3F; }
- break;
- }
- }
-
- void tick()
- {
- // Do HDMA ticks
- if (HDMA_active)
- {
- if (HDMA_length > 0)
- {
- if (!HDMA_mode)
- {
- if (HDMA_countdown > 0)
- {
- HDMA_countdown--;
- }
- else
- {
- // immediately transfer bytes, 2 bytes per cycles
- if ((HDMA_tick % 2) == 0)
- {
- HDMA_byte = ReadMemory(cur_DMA_src);
- }
- else
- {
- VRAM[(VRAM_Bank[0] * 0x2000) + cur_DMA_dest] = HDMA_byte;
- cur_DMA_dest = (uint8_t)((cur_DMA_dest + 1) & 0x1FFF);
- cur_DMA_src = (uint8_t)((cur_DMA_src + 1) & 0xFFFF);
- HDMA_length--;
- }
-
- HDMA_tick++;
- }
- }
- else
- {
- // only transfer during mode 0, and only 16 bytes at a time
- if (((STAT & 3) == 0) && (LY != last_HBL) && HBL_test && (LY_inc == 1) && (cycle > 4))
- {
- HBL_HDMA_go = true;
- HBL_test = false;
- }
- else if (HDMA_run_once)
- {
- HBL_HDMA_go = true;
- HBL_test = false;
- HDMA_run_once = false;
- }
-
- if (HBL_HDMA_go && (HBL_HDMA_count > 0))
- {
- HDMA_transfer[0] = true;
-
- if (HDMA_countdown > 0)
- {
- HDMA_countdown--;
- }
- else
- {
- if ((HDMA_tick % 2) == 0)
- {
- HDMA_byte = ReadMemory(cur_DMA_src);
- }
- else
- {
- VRAM[(VRAM_Bank[0] * 0x2000) + cur_DMA_dest] = HDMA_byte;
- cur_DMA_dest = (uint32_t)((cur_DMA_dest + 1) & 0x1FFF);
- cur_DMA_src = (uint32_t)((cur_DMA_src + 1) & 0xFFFF);
- HDMA_length--;
- HBL_HDMA_count--;
- }
-
- if ((HBL_HDMA_count == 0) && (HDMA_length != 0))
- {
-
- HBL_test = true;
- last_HBL = LY;
- HBL_HDMA_count = 0x10;
- HBL_HDMA_go = false;
- HDMA_countdown = 4;
- }
-
- HDMA_tick++;
- }
- }
- else
- {
- HDMA_transfer = false;
- }
- }
- }
- else
- {
- HDMA_active = false;
- HDMA_transfer = false;
- }
- }
-
- // the ppu only does anything if it is turned on via bit 7 of LCDC
- if (((LCDC & 0x80) > 0))
- {
- // start the next scanline
- if (cycle == 456)
- {
- // scanline callback
- if ((LY + LY_inc) == _scanlineCallbackLine[0])
- {
- //if (Core._scanlineCallback != null)
- //{
- // Core.GetGPU();
- // Core._scanlineCallback(LCDC);
- //}
- }
-
- cycle = 0;
- LY += LY_inc;
- cpu_LY[0] = LY;
-
- no_scan = false;
-
- if (LY == 0 && LY_inc == 0)
- {
- LY_inc = 1;
- in_vblank[0] = false;
-
- //STAT &= 0xFC;
-
- // special note here, the y coordiate of the window is kept if the window is deactivated
- // meaning it will pick up where it left off if re-enabled later
- // so we don't reset it in the scanline loop
- window_y_tile = 0;
- window_y_latch = window_y;
- window_y_tile_inc = 0;
- window_started = false;
- if (!((LCDC & 0x20) > 0)) { window_is_reset = true; }
- }
-
- // Automatically restore access to VRAM at this time (force end drawing)
- // Who Framed Roger Rabbit seems to run into this.
- VRAM_access_write = true;
- VRAM_access_read = true;
-
- if (LY == 144)
- {
- in_vblank[0] = true;
- }
- }
-
- // exit vblank if LCD went from off to on
- if (LCD_was_off)
- {
- //VBL_INT = false;
- in_vblank[0] = false;
- LCD_was_off = false;
-
- // we exit vblank into mode 0 for 4 cycles
- // but no hblank interrupt, presumably this only happens transitioning from mode 3 to 0
- STAT &= 0xFC;
-
- // also the LCD doesn't turn on right away
- // also, the LCD does not enter mode 2 on scanline 0 when first turned on
- no_scan = true;
- cycle = 8;
- }
-
- // the VBL stat is continuously asserted
- if (LY >= 144)
- {
- if (((STAT & 0x10) > 0))
- {
- if ((cycle >= 4) && (LY == 144))
- {
- VBL_INT = true;
- }
- else if (LY > 144)
- {
- VBL_INT = true;
- }
- }
-
- if ((cycle == 2) && (LY == 144))
- {
- // there is an edge case where a VBL INT is triggered if STAT bit 5 is set
- if (((STAT & 0x20) > 0)) { VBL_INT = true; }
- }
-
- if ((cycle == 4) && (LY == 144))
- {
- HBL_INT = false;
-
- // set STAT mode to 1 (VBlank) and interrupt flag if it is enabled
- STAT &= 0xFC;
- STAT |= 0x01;
-
- if ((REG_FFFF[0] & 1) > 0) { FlagI[0] = true; }
- REG_FF0F[0] |= 0x01;
- }
-
- if ((cycle == 4) && (LY == 144))
- {
- if (((STAT & 0x20) > 0)) { VBL_INT = false; }
- }
-
- if ((cycle == 8) && (LY == 153))
- {
- LY = 0;
- LY_inc = 0;
- cpu_LY[0] = LY;
- }
- }
-
- if (!in_vblank[0])
- {
- if (no_scan)
- {
- // timings are slightly different if we just turned on the LCD
- // there is no mode 2 (presumably it missed the trigger)
- if (cycle < 85)
- {
- if (cycle == 8)
- {
- // clear the sprite table
- for (uint32_t k = 0; k < 10; k++)
- {
- SL_sprites[k * 4] = 0;
- SL_sprites[k * 4 + 1] = 0;
- SL_sprites[k * 4 + 2] = 0;
- SL_sprites[k * 4 + 3] = 0;
- }
-
- if (LY != LYC)
- {
- LYC_INT = false;
- STAT &= 0xFB;
- }
-
- if ((LY == LYC) && !((STAT & 0x4) > 0))
- {
- // set STAT coincidence FLAG and interrupt flag if it is enabled
- STAT |= 0x04;
- if (((STAT & 0x40) > 0)) { LYC_INT = true; }
- }
- }
-
- if (cycle == 84)
- {
- STAT &= 0xFC;
- STAT |= 0x03;
- OAM_INT = false;
-
- OAM_access_read = false;
- OAM_access_write = false;
- VRAM_access_read = false;
- VRAM_access_write = false;
- }
- }
- else
- {
- if (cycle >= 85)
- {
- if (cycle == 85)
- {
- // x-scroll is expected to be latched one cycle later
- // this is fine since nothing has started in the rendering until the second cycle
- // calculate the column number of the tile to start with
- x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
- render_offset = scroll_x % 8;
- }
-
- // render the screen and handle hblank
- render(cycle - 85);
- }
- }
- }
- else
- {
- if (cycle <= 80)
- {
- if (cycle == 2)
- {
- if (LY != 0)
- {
- HBL_INT = false;
-
- if (((STAT & 0x20) > 0)) { OAM_INT = true; }
- }
- }
- else if (cycle == 4)
- {
- // here mode 2 will be set to true and interrupts fired if enabled
- STAT &= 0xFC;
- STAT |= 0x2;
-
- if (LY == 0)
- {
- VBL_INT = false;
- if (((STAT & 0x20) > 0)) { OAM_INT = true; }
- }
- }
-
- if (cycle == 80)
- {
- OAM_access_read = false;
- OAM_access_write = true;
- VRAM_access_read = false;
- }
- else
- {
- // here OAM scanning is performed
- OAM_scan(cycle);
- }
- }
- else if (cycle >= 83)
- {
- if (cycle == 84)
- {
- STAT &= 0xFC;
- STAT |= 0x03;
- OAM_INT = false;
- OAM_access_write = false;
- VRAM_access_write = false;
-
- // x-scroll is expected to be latched one cycle later
- // this is fine since nothing has started in the rendering until the second cycle
- // calculate the column number of the tile to start with
- x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
- render_offset = scroll_x % 8;
- }
-
- // render the screen and handle hblank
- render(cycle - 83);
- }
- }
- }
-
- if (LY_inc == 0)
- {
- if (cycle == 12)
- {
- LYC_INT = false;
- STAT &= 0xFB;
- }
- else if (cycle == 14)
- {
- // Special case of LY = LYC
- if ((LY == LYC) && !((STAT & 0x4) > 0))
- {
- // set STAT coincidence FLAG and interrupt flag if it is enabled
- STAT |= 0x04;
- if (((STAT & 0x40) > 0)) { LYC_INT = true; }
- }
- }
- }
-
- // here LY=LYC will be asserted or cleared (but only if LY isnt 0 as that's a special case)
- if ((cycle == 4) && (LY != 0))
- {
- if (LY_inc == 1)
- {
- LYC_INT = false;
- STAT &= 0xFB;
- }
- }
- else if ((cycle == 6) && (LY != 0))
- {
- if ((LY == LYC) && !((STAT & 0x4) > 0))
- {
- // set STAT coincidence FLAG and interrupt flag if it is enabled
- STAT |= 0x04;
- if (((STAT & 0x40) > 0)) { LYC_INT = true; }
- }
- }
-
- cycle++;
- }
- else
- {
- STAT &= 0xFC;
-
- VBL_INT = LYC_INT = HBL_INT = OAM_INT = false;
-
- in_vblank[0] = true;
-
- LCD_was_off = true;
-
- LY = 0;
- cpu_LY[0] = LY;
-
- cycle = 0;
- }
-
- // assert the STAT IRQ line if the line went from zero to 1
- stat_line = VBL_INT | LYC_INT | HBL_INT | OAM_INT;
-
- if (stat_line && !stat_line_old)
- {
- if ((REG_FFFF[0] & 0x2) > 0) { FlagI[0] = true; }
- REG_FF0F[0] |= 0x02;
- }
-
- stat_line_old = stat_line;
-
- // process latch delays
- //latch_delay();
-
- if (LYC_cd > 0)
- {
- LYC_cd--;
- if (LYC_cd == 0)
- {
- LYC = LYC_t;
-
- if (((LCDC & 0x80) > 0))
- {
- if (LY != LYC) { STAT &= 0xFB; LYC_INT = false; }
- else { STAT |= 0x4; LYC_INT = true; }
- }
- }
- }
- }
-
- // might be needed, not sure yet
- void latch_delay()
- {
- //BGP_l = BGP;
- }
-
- void render(uint32_t render_cycle)
- {
- // we are now in STAT mode 3
- // NOTE: presumably the first necessary sprite is fetched at sprite evaulation
- // i.e. just keeping track of the lowest x-value sprite
- if (render_cycle == 0)
- {
- /*
- OAM_access_read = false;
- OAM_access_write = true;
- VRAM_access_read = false;
- */
- // window X is latched for the scanline, mid-line changes have no effect
- window_x_latch = window_x;
-
- OAM_scan_index = 0;
- read_case = 0;
- internal_cycle = 0;
- pre_render = true;
- pre_render_2 = true;
- tile_inc = 0;
- pixel_counter = -8;
- sl_use_index = 0;
- fetch_sprite = false;
- going_to_fetch = false;
- first_fetch = true;
- consecutive_sprite = -render_offset + 8;
- no_sprites = false;
- evaled_sprites = 0;
- window_pre_render = false;
- window_latch = ((LCDC & 0x20) > 0);
-
- total_counter = 0;
-
- // TODO: If Window is turned on midscanline what happens? When is this check done exactly?
- if ((window_started && window_latch) || (window_is_reset && !window_latch && (LY >= window_y_latch)))
- {
- window_y_tile_inc++;
- if (window_y_tile_inc == 8)
- {
- window_y_tile_inc = 0;
- window_y_tile++;
- window_y_tile %= 32;
- }
- }
- window_started = false;
-
- if (SL_sprites_index == 0) { no_sprites = true; }
- // it is much easier to process sprites if we order them according to the rules of sprite priority first
- if (!no_sprites) { reorder_and_assemble_sprites(); }
- }
-
- // before anything else, we have to check if windowing is in effect
- if (window_latch && !window_started && (LY >= window_y_latch) && (pixel_counter >= (window_x_latch - 7)) && (window_x_latch < 167))
- {
- /*
- Console.Write(LY);
- Console.Write(" ");
- Console.Write(cycle);
- Console.Write(" ");
- Console.Write(window_y_tile);
- Console.Write(" ");
- Console.Write(render_offset);
- Console.Write(" ");
- Console.Write(window_x_latch);
- Console.Write(" ");
- Console.WriteLine(pixel_counter);
- */
-
- if (window_x_latch == 0)
- {
- // if the window starts at zero, we still do the first access to the BG
- // but then restart all over again at the window
- if ((render_offset % 7) <= 6)
- {
- read_case = 9;
- }
- else
- {
- read_case = 10;
- }
- }
- else
- {
- read_case = 4;
- }
-
- window_pre_render = true;
-
- window_counter = 0;
- render_counter = 0;
-
- window_x_tile = (uint32_t)floor((float)(pixel_counter - (window_x_latch - 7)) / 8.0);
-
- window_tile_inc = 0;
- window_started = true;
- window_is_reset = false;
- }
-
- if (!pre_render && !fetch_sprite)
- {
- // start shifting data into the LCD
- if (render_counter >= (render_offset + 8))
- {
- if (((tile_data_latch[2] & 0x20) > 0) && GBC_compat[0])
- {
- pixel = (tile_data_latch[0] & (1 << (render_counter % 8)) > 0) ? 1 : 0;
- pixel |= (tile_data_latch[1] & (1 << (render_counter % 8)) > 0) ? 2 : 0;
- }
- else
- {
- pixel = (tile_data_latch[0] & (1 << (7 - (render_counter % 8))) > 0) ? 1 : 0;
- pixel |= (tile_data_latch[1] & (1 << (7 - (render_counter % 8))) > 0) ? 2 : 0;
- }
-
- uint32_t ref_pixel = pixel;
-
- if (!GBC_compat[0])
- {
- if (((LCDC & 0x1) > 0))
- {
- pixel = (BGP >> (pixel * 2)) & 3;
- }
- else
- {
- pixel = 0;
- }
- }
-
- uint32_t pal_num = tile_data_latch[2] & 0x7;
-
- bool use_sprite = false;
-
- uint32_t s_pixel = 0;
-
- // now we have the BG pixel, we next need the sprite pixel
- if (!no_sprites)
- {
- bool have_sprite = false;
- uint32_t sprite_attr = 0;
-
- if (sprite_present_list[pixel_counter] == 1)
- {
- have_sprite = true;
- s_pixel = sprite_pixel_list[pixel_counter];
- sprite_attr = sprite_attr_list[pixel_counter];
- }
-
- if (have_sprite)
- {
- if (((LCDC & 0x2) > 0))
- {
- if (!((sprite_attr & 0x80) > 0))
- {
- use_sprite = true;
- }
- else if (ref_pixel == 0)
- {
- use_sprite = true;
- }
-
- if (!((LCDC & 0x1) > 0))
- {
- use_sprite = true;
- }
-
- // There is another priority bit in GBC, that can still override sprite priority
- if (((LCDC & 0x1) > 0) && ((tile_data_latch[2] & 0x80) > 0) && (ref_pixel != 0) && GBC_compat[0])
- {
- use_sprite = false;
- }
- }
-
- if (use_sprite)
- {
- pal_num = sprite_attr & 7;
-
- if (!GBC_compat[0])
- {
- pal_num = ((sprite_attr & 0x10) > 0) ? 1 : 0;
-
- if (((sprite_attr & 0x10) > 0))
- {
- pixel = (obj_pal_1 >> (s_pixel * 2)) & 3;
- }
- else
- {
- pixel = (obj_pal_0 >> (s_pixel * 2)) & 3;
- }
- }
- }
- }
- }
-
- // based on sprite priority and pixel values, pick a final pixel color
- if (GBC_compat[0])
- {
- if (use_sprite)
- {
- _vidbuffer[LY * 160 + pixel_counter] = (uint32_t)OBJ_palette[pal_num * 4 + s_pixel];
- }
- else
- {
- _vidbuffer[LY * 160 + pixel_counter] = (uint32_t)BG_palette[pal_num * 4 + pixel];
- }
- }
- else
- {
- if (use_sprite)
- {
- _vidbuffer[LY * 160 + pixel_counter] = (uint32_t)OBJ_palette[pal_num * 4 + pixel];
- }
- else
- {
- _vidbuffer[LY * 160 + pixel_counter] = (uint32_t)BG_palette[pixel];
- }
- }
-
- pixel_counter++;
-
- if (pixel_counter == 160)
- {
- read_case = 8;
- hbl_countdown = 2;
- }
- }
- else if (pixel_counter < 0)
- {
- pixel_counter++;
- }
- render_counter++;
- }
-
- if (!fetch_sprite)
- {
- if (!pre_render_2)
- {
- // before we go on to read case 3, we need to know if we stall there or not
- // Gekkio's tests show that if sprites are at position 0 or 1 (mod 8)
- // then it takes an extra cycle (1 or 2 more t-states) to process them
-
- if (!no_sprites && (pixel_counter < 160))
- {
- for (uint32_t i = 0; i < SL_sprites_index; i++)
- {
- if ((pixel_counter >= (SL_sprites[i * 4 + 1] - 8)) &&
- (pixel_counter < (SL_sprites[i * 4 + 1])) &&
- !((evaled_sprites & (1 << i)) > 0))
- {
- going_to_fetch = true;
- fetch_sprite = true;
- }
- }
- }
- }
-
- switch (read_case)
- {
- case 0: // read a background tile
- if ((internal_cycle % 2) == 1)
- {
- // calculate the row number of the tiles to be fetched
- y_tile = ((uint32_t)floor((float)((uint32_t)scroll_y + LY) / 8.0)) % 32;
-
- temp_fetch = y_tile * 32 + (x_tile + tile_inc) % 32;
- tile_byte = VRAM[0x1800 + (((LCDC & 0x4) > 0) ? 1 : 0) * 0x400 + temp_fetch];
- tile_data[2] = VRAM[0x3800 + (((LCDC & 0x4) > 0) ? 1 : 0) * 0x400 + temp_fetch];
- VRAM_sel = ((tile_data[2] & 0x8) > 0) ? 1 : 0;
-
- BG_V_flip = ((tile_data[2] & 0x40) > 0) & GBC_compat[0];
-
- read_case = 1;
- if (!pre_render)
- {
- tile_inc++;
- }
- }
- break;
-
- case 1: // read from tile graphics (0)
- if ((internal_cycle % 2) == 1)
- {
- y_scroll_offset = (scroll_y + LY) % 8;
-
- if (BG_V_flip)
- {
- y_scroll_offset = 7 - y_scroll_offset;
- }
-
- if (((LCDC & 0x10) > 0))
- {
- tile_data[0] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed byte
- if (((tile_byte & 0x80) > 0))
- {
- tile_byte -= 256;
- }
- tile_data[0] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
- }
-
- read_case = 2;
- }
- break;
-
- case 2: // read from tile graphics (1)
- if ((internal_cycle % 2) == 0)
- {
- pre_render_2 = false;
- }
- else
- {
- y_scroll_offset = (scroll_y + LY) % 8;
-
- if (BG_V_flip)
- {
- y_scroll_offset = 7 - y_scroll_offset;
- }
-
- if (((LCDC & 0x10) > 0))
- {
- // if LCDC somehow changed between the two reads, make sure we have a positive number
- if (tile_byte < 0)
- {
- tile_byte += 256;
- }
- tile_data[1] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed byte
- if (((tile_byte & 0x80) > 0) && tile_byte > 0)
- {
- tile_byte -= 256;
- }
- tile_data[1] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
-
- if (pre_render)
- {
- // here we set up rendering
- pre_render = false;
-
- render_counter = 0;
- latch_counter = 0;
- read_case = 0;
- }
- else
- {
- read_case = 3;
- }
- }
- break;
-
- case 3: // read from sprite data
- if ((internal_cycle % 2) == 1)
- {
- read_case = 0;
- latch_new_data = true;
- }
- break;
-
- case 4: // read from window data
- if ((window_counter % 2) == 1)
- {
- temp_fetch = window_y_tile * 32 + (window_x_tile + window_tile_inc) % 32;
- tile_byte = VRAM[0x1800 + (((LCDC & 0x40) > 0) ? 1 : 0) * 0x400 + temp_fetch];
- tile_data[2] = VRAM[0x3800 + (((LCDC & 0x40) > 0) ? 1 : 0) * 0x400 + temp_fetch];
- VRAM_sel = ((tile_data[2] & 0x8) > 0) ? 1 : 0;
- BG_V_flip = ((tile_data[2] & 0x40) > 0) & GBC_compat[0];
-
- window_tile_inc++;
- read_case = 5;
- }
- window_counter++;
- break;
-
- case 5: // read from tile graphics (for the window)
- if ((window_counter % 2) == 1)
- {
- y_scroll_offset = (window_y_tile_inc) % 8;
-
- if (BG_V_flip)
- {
- y_scroll_offset = 7 - y_scroll_offset;
- }
-
- if (((LCDC & 0x10) > 0))
- {
- tile_data[0] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed byte
- if (((tile_byte & 0x80) > 0))
- {
- tile_byte -= 256;
- }
- tile_data[0] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
- }
-
- read_case = 6;
- }
- window_counter++;
- break;
-
- case 6: // read from tile graphics (for the window)
- if ((window_counter % 2) == 1)
- {
- y_scroll_offset = (window_y_tile_inc) % 8;
-
- if (BG_V_flip)
- {
- y_scroll_offset = 7 - y_scroll_offset;
- }
-
- if (((LCDC & 0x10) > 0))
- {
- // if LCDC somehow changed between the two reads, make sure we have a positive number
- if (tile_byte < 0)
- {
- tile_byte += 256;
- }
- tile_data[1] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed byte
- if (((tile_byte & 0x80) > 0) && tile_byte > 0)
- {
- tile_byte -= 256;
- }
- tile_data[1] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
-
- if (window_pre_render)
- {
- // here we set up rendering
- // unlike for the normal background case, there is no pre-render period for the window
- // so start shifting in data to the screen right away
- if (window_x_latch <= 7)
- {
- if (render_offset == 0)
- {
- read_case = 4;
- }
- else
- {
- read_case = 9 + render_offset - 1;
- }
- render_counter = 8 - render_offset;
-
- render_offset = 7 - window_x_latch;
- }
- else
- {
- render_offset = 0;
- read_case = 4;
- render_counter = 8;
- }
-
- latch_counter = 0;
- latch_new_data = true;
- window_pre_render = false;
- }
- else
- {
- read_case = 7;
- }
- }
- window_counter++;
- break;
-
- case 7: // read from sprite data
- if ((window_counter % 2) == 1)
- {
- read_case = 4;
- latch_new_data = true;
- }
- window_counter++;
- break;
-
- case 8: // done reading, we are now in phase 0
- pre_render = true;
-
- // the other interrupts appear to be delayed by 1 CPU cycle, so do the same here
- if (hbl_countdown > 0)
- {
- hbl_countdown--;
-
- if (hbl_countdown == 0)
- {
- OAM_access_read = true;
- OAM_access_write = true;
- VRAM_access_read = true;
- VRAM_access_write = true;
- }
- else
- {
- STAT &= 0xFC;
- STAT |= 0x00;
-
- if (((STAT & 0x8) > 0)) { HBL_INT = true; }
- }
- }
- break;
-
- case 9:
- // this is a degenerate case for starting the window at 0
- // kevtris' timing doc indicates an additional normal BG access
- // but this information is thrown away, so it's faster to do this then constantly check
- // for it in read case 0
- read_case = 4;
- break;
- case 10:
- case 11:
- case 12:
- case 13:
- case 14:
- case 15:
- case 16:
- case 17:
- read_case--;
- break;
- }
- internal_cycle++;
-
- if (latch_new_data)
- {
- latch_new_data = false;
- tile_data_latch[0] = tile_data[0];
- tile_data_latch[1] = tile_data[1];
- tile_data_latch[2] = tile_data[2];
- }
- }
-
- // every in range sprite takes 6 cycles to process
- // sprites located at x=0 still take 6 cycles to process even though they don't appear on screen
- // sprites above x=168 do not take any cycles to process however
- if (fetch_sprite)
- {
- if (going_to_fetch)
- {
- going_to_fetch = false;
-
- last_eval = 0;
-
- // at this time it is unknown what each cycle does, but we only need to accurately keep track of cycles
- for (uint32_t i = 0; i < SL_sprites_index; i++)
- {
- if ((pixel_counter >= (SL_sprites[i * 4 + 1] - 8)) &&
- (pixel_counter < (SL_sprites[i * 4 + 1])) &&
- !((evaled_sprites & (1 << i)) > 0))
- {
- sprite_fetch_counter += 6;
- evaled_sprites |= (1 << i);
- last_eval = SL_sprites[i * 4 + 1];
- }
- }
-
- // x scroll offsets the penalty table
- // there is no penalty if the next sprites to be fetched are within the currentfetch block (8 pixels)
- if (first_fetch || (last_eval >= consecutive_sprite))
- {
- if (((last_eval + render_offset) % 8) == 0) { sprite_fetch_counter += 5; }
- else if (((last_eval + render_offset) % 8) == 1) { sprite_fetch_counter += 4; }
- else if (((last_eval + render_offset) % 8) == 2) { sprite_fetch_counter += 3; }
- else if (((last_eval + render_offset) % 8) == 3) { sprite_fetch_counter += 2; }
- else if (((last_eval + render_offset) % 8) == 4) { sprite_fetch_counter += 1; }
- else if (((last_eval + render_offset) % 8) == 5) { sprite_fetch_counter += 0; }
- else if (((last_eval + render_offset) % 8) == 6) { sprite_fetch_counter += 0; }
- else if (((last_eval + render_offset) % 8) == 7) { sprite_fetch_counter += 0; }
-
- consecutive_sprite = (uint32_t)floor((double)(last_eval + render_offset) / 8.0) * 8 + 8 - render_offset;
-
- // special case exists here for sprites at zero with non-zero x-scroll. Not sure exactly the reason for it.
- if (last_eval == 0 && render_offset != 0)
- {
- sprite_fetch_counter += render_offset;
- }
- }
-
- total_counter += sprite_fetch_counter;
-
- first_fetch = false;
- }
- else
- {
- sprite_fetch_counter--;
- if (sprite_fetch_counter == 0)
- {
- fetch_sprite = false;
- }
- }
- }
-
- }
-
- void process_sprite()
- {
- uint32_t y;
- uint32_t VRAM_temp = (((SL_sprites[sl_use_index * 4 + 3] & 0x8) > 0) && GBC_compat[0]) ? 1 : 0;
-
- if ((SL_sprites[sl_use_index * 4 + 3] & 0x40) > 0)
- {
- if (((LCDC & 0x4) > 0))
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- y = 15 - y;
- sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
- sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
- }
- else
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- y = 7 - y;
- sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
- sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
- }
- }
- else
- {
- if (((LCDC & 0x4) > 0))
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
- sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
- }
- else
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
- sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
- }
- }
-
- if ((SL_sprites[sl_use_index * 4 + 3] & 0x20) > 0)
- {
- uint32_t b0, b1, b2, b3, b4, b5, b6, b7 = 0;
- for (uint32_t i = 0; i < 2; i++)
- {
- b0 = (sprite_sel[i] & 0x01) << 7;
- b1 = (sprite_sel[i] & 0x02) << 5;
- b2 = (sprite_sel[i] & 0x04) << 3;
- b3 = (sprite_sel[i] & 0x08) << 1;
- b4 = (sprite_sel[i] & 0x10) >> 1;
- b5 = (sprite_sel[i] & 0x20) >> 3;
- b6 = (sprite_sel[i] & 0x40) >> 5;
- b7 = (sprite_sel[i] & 0x80) >> 7;
-
- sprite_sel[i] = (uint8_t)(b0 | b1 | b2 | b3 | b4 | b5 | b6 | b7);
- }
- }
- }
-
- // normal DMA moves twice as fast in double speed mode on GBC
- // So give it it's own function so we can seperate it from PPU tick
- void DMA_tick()
- {
- // Note that DMA is halted when the CPU is halted
- if (DMA_start && !cpu_halted)
- {
- if (DMA_clock >= 4)
- {
- DMA_OAM_access = false;
- if ((DMA_clock % 4) == 1)
- {
- // the cpu can't access memory during this time, but we still need the ppu to be able to.
- DMA_start = false;
- // Gekkio reports that A14 being high on DMA transfers always represent WRAM accesses
- // So transfers nominally from higher memory areas are actually still from there (i.e. FF -> DF)
- uint8_t DMA_actual = DMA_addr;
- if (DMA_addr > 0xDF) { DMA_actual &= 0xDF; }
- DMA_byte = ReadMemory((uint32_t)((DMA_actual << 8) + DMA_inc));
- DMA_start = true;
- }
- else if ((DMA_clock % 4) == 3)
- {
- OAM[DMA_inc] = DMA_byte;
-
- if (DMA_inc < (0xA0 - 1)) { DMA_inc++; }
- }
- }
-
- DMA_clock++;
-
- if (DMA_clock == 648)
- {
- DMA_start = false;
- DMA_OAM_access = true;
- }
- }
- }
-
- // order sprites according to x coordinate
- // note that for sprites of equal x coordinate, priority goes to first on the list
- void reorder_and_assemble_sprites()
- {
- sprite_ordered_index = 0;
-
- // In CGB mode, sprites are ordered solely based on their position in OAM, so they are already ordered
-
- if (GBC_compat[0])
- {
- for (uint32_t j = 0; j < SL_sprites_index; j++)
- {
- sl_use_index = j;
- process_sprite();
- SL_sprites_ordered[sprite_ordered_index * 4] = SL_sprites[j * 4 + 1];
- SL_sprites_ordered[sprite_ordered_index * 4 + 1] = sprite_sel[0];
- SL_sprites_ordered[sprite_ordered_index * 4 + 2] = sprite_sel[1];
- SL_sprites_ordered[sprite_ordered_index * 4 + 3] = SL_sprites[j * 4 + 3];
- sprite_ordered_index++;
- }
- }
- else
- {
- for (int i = 0; i < 256; i++)
- {
- for (int j = 0; j < SL_sprites_index; j++)
- {
- if (SL_sprites[j * 4 + 1] == i)
- {
- sl_use_index = j;
- process_sprite();
- SL_sprites_ordered[sprite_ordered_index * 4] = SL_sprites[j * 4 + 1];
- SL_sprites_ordered[sprite_ordered_index * 4 + 1] = sprite_sel[0];
- SL_sprites_ordered[sprite_ordered_index * 4 + 2] = sprite_sel[1];
- SL_sprites_ordered[sprite_ordered_index * 4 + 3] = SL_sprites[j * 4 + 3];
- sprite_ordered_index++;
- }
- }
- }
- }
-
- bool have_pixel = false;
- uint8_t s_pixel = 0;
- uint8_t sprite_attr = 0;
-
- for (uint32_t i = 0; i < 160; i++)
- {
- have_pixel = false;
- for (uint32_t j = 0; j < SL_sprites_index; j++)
- {
- if ((i >= (SL_sprites_ordered[j * 4] - 8)) &&
- (i < SL_sprites_ordered[j * 4]) &&
- !have_pixel)
- {
- // we can use the current sprite, so pick out a pixel for it
- uint32_t t_index = i - (SL_sprites_ordered[j * 4] - 8);
-
- t_index = 7 - t_index;
-
- sprite_data[0] = (uint8_t)((SL_sprites_ordered[j * 4 + 1] >> t_index) & 1);
- sprite_data[1] = (uint8_t)(((SL_sprites_ordered[j * 4 + 2] >> t_index) & 1) << 1);
-
- s_pixel = (uint8_t)(sprite_data[0] + sprite_data[1]);
- sprite_attr = (uint8_t)SL_sprites_ordered[j * 4 + 3];
-
- // pixel color of 0 is transparent, so if this is the case we don't have a pixel
- if (s_pixel != 0)
- {
- have_pixel = true;
- }
- }
- }
-
- if (have_pixel)
- {
- sprite_present_list[i] = 1;
- sprite_pixel_list[i] = s_pixel;
- sprite_attr_list[i] = sprite_attr;
- }
- else
- {
- sprite_present_list[i] = 0;
- }
- }
- }
-
- void OAM_scan(uint32_t OAM_cycle)
- {
- // we are now in STAT mode 2
- // TODO: maybe stat mode 2 flags are set at cycle 0 on visible scanlines?
- if (OAM_cycle == 0)
- {
- OAM_access_read = false;
- OAM_access_write = false;
-
- OAM_scan_index = 0;
- SL_sprites_index = 0;
- write_sprite = 0;
- }
-
- // the gameboy has 80 cycles to scan through 40 sprites, picking out the first 10 it finds to draw
- // the following is a guessed at implmenentation based on how NES does it, it's probably pretty close
- if (OAM_cycle < 10)
- {
- // start by clearing the sprite table (probably just clears X on hardware, but let's be safe here.)
- SL_sprites[OAM_cycle * 4] = 0;
- SL_sprites[OAM_cycle * 4 + 1] = 0;
- SL_sprites[OAM_cycle * 4 + 2] = 0;
- SL_sprites[OAM_cycle * 4 + 3] = 0;
- }
- else
- {
- if (write_sprite == 0)
- {
- if (OAM_scan_index < 40)
- {
- uint32_t temp = DMA_OAM_access ? OAM[OAM_scan_index * 4] : (uint32_t)0xFF;
- // (sprite Y - 16) equals LY, we have a sprite
- if ((temp - 16) <= LY &&
- ((temp - 16) + 8 + (((LCDC & 0x4) > 0) ? 8 : 0)) > LY)
- {
- // always pick the first 10 in range sprites
- if (SL_sprites_index < 10)
- {
- SL_sprites[SL_sprites_index * 4] = temp;
-
- write_sprite = 1;
- }
- else
- {
- // if we already have 10 sprites, there's nothing to do, increment the index
- OAM_scan_index++;
- }
- }
- else
- {
- OAM_scan_index++;
- }
- }
- }
- else
- {
- uint32_t temp2 = DMA_OAM_access ? OAM[OAM_scan_index * 4 + write_sprite] : (uint32_t)0xFF;
- SL_sprites[SL_sprites_index * 4 + write_sprite] = temp2;
- write_sprite++;
-
- if (write_sprite == 4)
- {
- write_sprite = 0;
- SL_sprites_index++;
- OAM_scan_index++;
- }
- }
- }
- }
-
- void color_compute_BG()
- {
- uint32_t R;
- uint32_t G;
- uint32_t B;
-
- if ((BG_bytes_index % 2) == 0)
- {
- R = (uint32_t)(BG_bytes[BG_bytes_index] & 0x1F);
- G = (uint32_t)(((BG_bytes[BG_bytes_index] & 0xE0) | ((BG_bytes[BG_bytes_index + 1] & 0x03) << 8)) >> 5);
- B = (uint32_t)((BG_bytes[BG_bytes_index + 1] & 0x7C) >> 2);
- }
- else
- {
- R = (uint32_t)(BG_bytes[BG_bytes_index - 1] & 0x1F);
- G = (uint32_t)(((BG_bytes[BG_bytes_index - 1] & 0xE0) | ((BG_bytes[BG_bytes_index] & 0x03) << 8)) >> 5);
- B = (uint32_t)((BG_bytes[BG_bytes_index] & 0x7C) >> 2);
- }
-
- uint32_t retR = ((R * 13 + G * 2 + B) >> 1) & 0xFF;
- uint32_t retG = ((G * 3 + B) << 1) & 0xFF;
- uint32_t retB = ((R * 3 + G * 2 + B * 11) >> 1) & 0xFF;
-
- BG_palette[BG_bytes_index >> 1] = (uint32_t)(0xFF000000 | (retR << 16) | (retG << 8) | retB);
- }
-
- void color_compute_OBJ()
- {
- uint32_t R;
- uint32_t G;
- uint32_t B;
-
- if ((OBJ_bytes_index % 2) == 0)
- {
- R = (uint32_t)(OBJ_bytes[OBJ_bytes_index] & 0x1F);
- G = (uint32_t)(((OBJ_bytes[OBJ_bytes_index] & 0xE0) | ((OBJ_bytes[OBJ_bytes_index + 1] & 0x03) << 8)) >> 5);
- B = (uint32_t)((OBJ_bytes[OBJ_bytes_index + 1] & 0x7C) >> 2);
- }
- else
- {
- R = (uint32_t)(OBJ_bytes[OBJ_bytes_index - 1] & 0x1F);
- G = (uint32_t)(((OBJ_bytes[OBJ_bytes_index - 1] & 0xE0) | ((OBJ_bytes[OBJ_bytes_index] & 0x03) << 8)) >> 5);
- B = (uint32_t)((OBJ_bytes[OBJ_bytes_index] & 0x7C) >> 2);
- }
-
- uint32_t retR = ((R * 13 + G * 2 + B) >> 1) & 0xFF;
- uint32_t retG = ((G * 3 + B) << 1) & 0xFF;
- uint32_t retB = ((R * 3 + G * 2 + B * 11) >> 1) & 0xFF;
-
- OBJ_palette[OBJ_bytes_index >> 1] = (uint32_t)(0xFF000000 | (retR << 16) | (retG << 8) | retB);
- }
-
- void Reset()
- {
- LCDC = 0;
- STAT = 0x80;
- scroll_y = 0;
- scroll_x = 0;
- LY = 0;
- LYC = 0;
- DMA_addr = 0;
- BGP = 0xFF;
- obj_pal_0 = 0;
- obj_pal_1 = 0;
- window_y = 0x0;
- window_x = 0x0;
- window_x_latch = 0xFF;
- window_y_latch = 0xFF;
- LY_inc = 1;
- no_scan = false;
- OAM_access_read = true;
- VRAM_access_read = true;
- OAM_access_write = true;
- VRAM_access_write = true;
- DMA_OAM_access = true;
-
- cycle = 0;
- LYC_INT = false;
- HBL_INT = false;
- VBL_INT = false;
- OAM_INT = false;
-
- stat_line = false;
- stat_line_old = false;
-
- window_counter = 0;
- window_pre_render = false;
- window_started = false;
- window_tile_inc = 0;
- window_y_tile = 0;
- window_x_tile = 0;
- window_y_tile_inc = 0;
-
- BG_bytes_inc = false;
- OBJ_bytes_inc = false;
- BG_bytes_index = 0;
- OBJ_bytes_index = 0;
- BG_transfer_byte = 0;
- OBJ_transfer_byte = 0;
-
- HDMA_src_hi = 0;
- HDMA_src_lo = 0;
- HDMA_dest_hi = 0;
- HDMA_dest_lo = 0;
-
- VRAM_sel = 0;
- BG_V_flip = false;
- HDMA_active = false;
- HDMA_mode = false;
- cur_DMA_src = 0;
- cur_DMA_dest = 0;
- HDMA_length = 0;
- HDMA_countdown = 0;
- HBL_HDMA_count = 0;
- last_HBL = 0;
- HBL_HDMA_go = false;
- HBL_test = false;
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/GBC_PPU.h b/libHawk/GBHawk/GBHawk/GBC_PPU.h
deleted file mode 100644
index 572f2f6681..0000000000
--- a/libHawk/GBHawk/GBHawk/GBC_PPU.h
+++ /dev/null
@@ -1,1570 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "PPU_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class GBC_PPU : public PPU
- {
- public:
-
- uint8_t ReadReg(uint32_t addr)
- {
- uint8_t ret = 0;
- //Console.WriteLine(Core.cpu.TotalExecutedCycles);
- switch (addr)
- {
- case 0xFF40: ret = LCDC; break; // LCDC
- case 0xFF41: ret = STAT; break; // STAT
- case 0xFF42: ret = scroll_y; break; // SCY
- case 0xFF43: ret = scroll_x; break; // SCX
- case 0xFF44: ret = LY; break; // LY
- case 0xFF45: ret = LYC; break; // LYC
- case 0xFF46: ret = DMA_addr; break; // DMA
- case 0xFF47: ret = BGP; break; // BGP
- case 0xFF48: ret = obj_pal_0; break; // OBP0
- case 0xFF49: ret = obj_pal_1; break; // OBP1
- case 0xFF4A: ret = window_y; break; // WY
- case 0xFF4B: ret = window_x; break; // WX
-
- // These are GBC specific Regs
- case 0xFF51: ret = HDMA_src_hi; break; // HDMA1
- case 0xFF52: ret = HDMA_src_lo; break; // HDMA2
- case 0xFF53: ret = HDMA_dest_hi; break; // HDMA3
- case 0xFF54: ret = HDMA_dest_lo; break; // HDMA4
- case 0xFF55: ret = HDMA_ctrl(); break; // HDMA5
- case 0xFF68: ret = BG_pal_ret(); break; // BGPI
- case 0xFF69: ret = BG_PAL_read(); break; // BGPD
- case 0xFF6A: ret = OBJ_pal_ret(); break; // OBPI
- case 0xFF6B: ret = OBJ_bytes[OBJ_bytes_index]; break; // OBPD
- }
-
- return ret;
- }
-
- uint8_t BG_PAL_read()
- {
- if (VRAM_access_read)
- {
- return BG_bytes[BG_bytes_index];
- }
- else
- {
- return 0xFF;
- }
- }
-
- void WriteReg(uint32_t addr, uint8_t value)
- {
- switch (addr)
- {
- case 0xFF40: // LCDC
- if (((LCDC & 0x80) > 0) && !((value & 0x80) > 0))
- {
- VRAM_access_read = true;
- VRAM_access_write = true;
- OAM_access_read = true;
- OAM_access_write = true;
- }
-
- if (!((LCDC & 0x80) > 0) && ((value & 0x80) > 0))
- {
- // don't draw for one frame after turning on
- blank_frame = true;
- }
-
- LCDC = value;
- break;
- case 0xFF41: // STAT
- // note that their is no stat interrupt bug in GBC
- STAT = (uint8_t)((value & 0xF8) | (STAT & 7) | 0x80);
-
- if (((STAT & 3) == 0) && ((STAT & 0x8) > 0)) { HBL_INT = true; }
- else { HBL_INT = false; }
- if (((STAT & 3) == 1) && ((STAT & 0x10) > 0)) { VBL_INT = true; }
- else { VBL_INT = false; }
- // OAM not triggered?
- // if (((STAT & 3) == 2) && STAT.Bit(5)) { OAM_INT = true; } else { OAM_INT = false; }
-
- if (((value & 0x40) > 0) && ((LCDC & 0x80) > 0))
- {
- if (LY == LYC) { LYC_INT = true; }
- else { LYC_INT = false; }
- }
- if (!((STAT & 0x40) > 0)) { LYC_INT = false; }
- break;
- case 0xFF42: // SCY
- scroll_y = value;
- break;
- case 0xFF43: // SCX
- scroll_x = value;
- break;
- case 0xFF44: // LY
- LY = 0; /*reset*/
- break;
- case 0xFF45: // LYC
- // tests indicate that latching writes to LYC should take place 4 cycles after the write
- // otherwise tests around LY boundaries will fail
- LYC_t = value;
- LYC_cd = 4;
- break;
- case 0xFF46: // DMA
- DMA_addr = value;
- DMA_start = true;
- DMA_OAM_access = true;
- DMA_clock = 0;
- DMA_inc = 0;
- break;
- case 0xFF47: // BGP
- BGP = value;
- break;
- case 0xFF48: // OBP0
- obj_pal_0 = value;
- break;
- case 0xFF49: // OBP1
- obj_pal_1 = value;
- break;
- case 0xFF4A: // WY
- window_y = value;
- break;
- case 0xFF4B: // WX
- window_x = value;
- break;
-
- // These are GBC specific Regs
- case 0xFF51: // HDMA1
- HDMA_src_hi = value;
- cur_DMA_src = (uint32_t)(((HDMA_src_hi & 0xFF) << 8) | (cur_DMA_src & 0xF0));
- break;
- case 0xFF52: // HDMA2
- HDMA_src_lo = value;
- cur_DMA_src = (uint32_t)((cur_DMA_src & 0xFF00) | (HDMA_src_lo & 0xF0));
- break;
- case 0xFF53: // HDMA3
- HDMA_dest_hi = value;
- cur_DMA_dest = (uint32_t)(((HDMA_dest_hi & 0x1F) << 8) | (cur_DMA_dest & 0xF0));
- break;
- case 0xFF54: // HDMA4
- HDMA_dest_lo = value;
- cur_DMA_dest = (uint32_t)((cur_DMA_dest & 0xFF00) | (HDMA_dest_lo & 0xF0));
- break;
- case 0xFF55: // HDMA5
- if (!HDMA_active)
- {
- HDMA_mode = ((value & 0x80) > 0);
- HDMA_countdown = 4;
- HDMA_tick = 0;
- if (((value & 0x80) > 0))
- {
- // HDMA during HBlank only, but only if screen is on, otherwise DMA immediately one block of data
- // worms armaggedon requires HDMA to fire in hblank mode even if the screen is off.
- HDMA_active = true;
- HBL_HDMA_count = 0x10;
-
- last_HBL = LY - 1;
-
- HBL_test = true;
- HBL_HDMA_go = false;
-
- if (!((LCDC & 0x80) > 0))
- {
- HDMA_run_once = true;
- }
- }
- else
- {
- // HDMA immediately
- HDMA_active = true;
- HDMA_transfer[0] = true;
- }
-
- HDMA_length = ((value & 0x7F) + 1) * 16;
- }
- else
- {
- //terminate the transfer
- if (!((value & 0x80) > 0))
- {
- HDMA_active = false;
- }
- }
-
- break;
- case 0xFF68: // BGPI
- BG_bytes_index = (uint8_t)(value & 0x3F);
- BG_bytes_inc = ((value & 0x80) == 0x80);
- break;
- case 0xFF69: // BGPD
- if (VRAM_access_write)
- {
- BG_transfer_byte = value;
- BG_bytes[BG_bytes_index] = value;
- }
-
- // change the appropriate palette color
- color_compute_BG();
- if (BG_bytes_inc) { BG_bytes_index++; BG_bytes_index &= 0x3F; }
- break;
- case 0xFF6A: // OBPI
- OBJ_bytes_index = (uint8_t)(value & 0x3F);
- OBJ_bytes_inc = ((value & 0x80) == 0x80);
- break;
- case 0xFF6B: // OBPD
- OBJ_transfer_byte = value;
- OBJ_bytes[OBJ_bytes_index] = value;
-
- // change the appropriate palette color
- color_compute_OBJ();
-
- if (OBJ_bytes_inc) { OBJ_bytes_index++; OBJ_bytes_index &= 0x3F; }
- break;
- }
- }
-
- void tick()
- {
- // Do HDMA ticks
- if (HDMA_active)
- {
- if (HDMA_length > 0)
- {
- if (!HDMA_mode)
- {
- if (HDMA_countdown > 0)
- {
- HDMA_countdown--;
- }
- else
- {
- // immediately transfer bytes, 2 bytes per cycles
- if ((HDMA_tick % 2) == 0)
- {
- HDMA_byte = ReadMemory(cur_DMA_src);
- }
- else
- {
- VRAM[(VRAM_Bank[0] * 0x2000) + cur_DMA_dest] = HDMA_byte;
- cur_DMA_dest = (uint8_t)((cur_DMA_dest + 1) & 0x1FFF);
- cur_DMA_src = (uint8_t)((cur_DMA_src + 1) & 0xFFFF);
- HDMA_length--;
- }
-
- HDMA_tick++;
- }
- }
- else
- {
- // only transfer during mode 0, and only 16 bytes at a time
- if (((STAT & 3) == 0) && (LY != last_HBL) && HBL_test && (LY_inc == 1) && (cycle > 4))
- {
- HBL_HDMA_go = true;
- HBL_test = false;
- }
- else if (HDMA_run_once)
- {
- HBL_HDMA_go = true;
- HBL_test = false;
- HDMA_run_once = false;
- }
-
- if (HBL_HDMA_go && (HBL_HDMA_count > 0))
- {
- HDMA_transfer[0] = true;
-
- if (HDMA_countdown > 0)
- {
- HDMA_countdown--;
- }
- else
- {
- if ((HDMA_tick % 2) == 0)
- {
- HDMA_byte = ReadMemory(cur_DMA_src);
- }
- else
- {
- VRAM[(VRAM_Bank[0] * 0x2000) + cur_DMA_dest] = HDMA_byte;
- cur_DMA_dest = (uint32_t)((cur_DMA_dest + 1) & 0x1FFF);
- cur_DMA_src = (uint32_t)((cur_DMA_src + 1) & 0xFFFF);
- HDMA_length--;
- HBL_HDMA_count--;
- }
-
- if ((HBL_HDMA_count == 0) && (HDMA_length != 0))
- {
-
- HBL_test = true;
- last_HBL = LY;
- HBL_HDMA_count = 0x10;
- HBL_HDMA_go = false;
- HDMA_countdown = 4;
- }
-
- HDMA_tick++;
- }
- }
- else
- {
- HDMA_transfer = false;
- }
- }
- }
- else
- {
- HDMA_active = false;
- HDMA_transfer = false;
- }
- }
-
- // the ppu only does anything if it is turned on via bit 7 of LCDC
- if (((LCDC & 0x80) > 0))
- {
- // start the next scanline
- if (cycle == 456)
- {
- // scanline callback
- if ((LY + LY_inc) == _scanlineCallbackLine[0])
- {
- //if (Core._scanlineCallback != null)
- //{
- // Core.GetGPU();
- // Core._scanlineCallback(LCDC);
- //}
- }
-
- cycle = 0;
- LY += LY_inc;
- cpu_LY[0] = LY;
-
- no_scan = false;
-
- if (LY == 0 && LY_inc == 0)
- {
- LY_inc = 1;
- in_vblank[0] = false;
-
- //STAT &= 0xFC;
-
- // special note here, the y coordiate of the window is kept if the window is deactivated
- // meaning it will pick up where it left off if re-enabled later
- // so we don't reset it in the scanline loop
- window_y_tile = 0;
- window_y_latch = window_y;
- window_y_tile_inc = 0;
- window_started = false;
- if (!((LCDC & 0x20) > 0)) { window_is_reset = true; }
- }
-
- // Automatically restore access to VRAM at this time (force end drawing)
- // Who Framed Roger Rabbit seems to run into this.
- VRAM_access_write = true;
- VRAM_access_read = true;
-
- if (LY == 144)
- {
- in_vblank[0] = true;
- }
- }
-
- // exit vblank if LCD went from off to on
- if (LCD_was_off)
- {
- //VBL_INT = false;
- in_vblank[0] = false;
- LCD_was_off = false;
-
- // we exit vblank into mode 0 for 4 cycles
- // but no hblank interrupt, presumably this only happens transitioning from mode 3 to 0
- STAT &= 0xFC;
-
- // also the LCD doesn't turn on right away
- // also, the LCD does not enter mode 2 on scanline 0 when first turned on
- no_scan = true;
- cycle = 8;
- }
-
- // the VBL stat is continuously asserted
- if (LY >= 144)
- {
- if (((STAT & 0x10) > 0))
- {
- if ((cycle >= 4) && (LY == 144))
- {
- VBL_INT = true;
- }
- else if (LY > 144)
- {
- VBL_INT = true;
- }
- }
-
- if ((cycle == 2) && (LY == 144))
- {
- // there is an edge case where a VBL INT is triggered if STAT bit 5 is set
- if (((STAT & 0x20) > 0)) { VBL_INT = true; }
- }
-
- if ((cycle == 4) && (LY == 144))
- {
- HBL_INT = false;
-
- // set STAT mode to 1 (VBlank) and interrupt flag if it is enabled
- STAT &= 0xFC;
- STAT |= 0x01;
-
- if ((REG_FFFF[0] & 1) > 0) { FlagI[0] = true; }
- REG_FF0F[0] |= 0x01;
- }
-
- if ((cycle == 4) && (LY == 144))
- {
- if (((STAT & 0x20) > 0)) { VBL_INT = false; }
- }
-
- if ((cycle == 8) && (LY == 153))
- {
- LY = 0;
- LY_inc = 0;
- cpu_LY[0] = LY;
- }
- }
-
- if (!in_vblank[0])
- {
- if (no_scan)
- {
- // timings are slightly different if we just turned on the LCD
- // there is no mode 2 (presumably it missed the trigger)
- if (cycle < 85)
- {
- if (cycle == 8)
- {
- // clear the sprite table
- for (uint32_t k = 0; k < 10; k++)
- {
- SL_sprites[k * 4] = 0;
- SL_sprites[k * 4 + 1] = 0;
- SL_sprites[k * 4 + 2] = 0;
- SL_sprites[k * 4 + 3] = 0;
- }
-
- if (LY != LYC)
- {
- LYC_INT = false;
- STAT &= 0xFB;
- }
-
- if ((LY == LYC) && !((STAT & 0x4) > 0))
- {
- // set STAT coincidence FLAG and interrupt flag if it is enabled
- STAT |= 0x04;
- if (((STAT & 0x40) > 0)) { LYC_INT = true; }
- }
- }
-
- if (cycle == 84)
- {
- STAT &= 0xFC;
- STAT |= 0x03;
- OAM_INT = false;
-
- OAM_access_read = false;
- OAM_access_write = false;
- VRAM_access_read = false;
- VRAM_access_write = false;
- }
- }
- else
- {
- if (cycle >= 85)
- {
- if (cycle == 85)
- {
- // x-scroll is expected to be latched one cycle later
- // this is fine since nothing has started in the rendering until the second cycle
- // calculate the column number of the tile to start with
- x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
- render_offset = scroll_x % 8;
- }
-
- // render the screen and handle hblank
- render(cycle - 85);
- }
- }
- }
- else
- {
- if (cycle <= 80)
- {
- if (cycle == 2)
- {
- if (LY != 0)
- {
- HBL_INT = false;
-
- if (((STAT & 0x20) > 0)) { OAM_INT = true; }
- }
- }
- else if (cycle == 4)
- {
- // here mode 2 will be set to true and interrupts fired if enabled
- STAT &= 0xFC;
- STAT |= 0x2;
-
- if (LY == 0)
- {
- VBL_INT = false;
- if (((STAT & 0x20) > 0)) { OAM_INT = true; }
- }
- }
-
- if (cycle == 80)
- {
- OAM_access_read = false;
- OAM_access_write = true;
- VRAM_access_read = false;
- }
- else
- {
- // here OAM scanning is performed
- OAM_scan(cycle);
- }
- }
- else if (cycle >= 83)
- {
- if (cycle == 84)
- {
- STAT &= 0xFC;
- STAT |= 0x03;
- OAM_INT = false;
- OAM_access_write = false;
- VRAM_access_write = false;
-
- // x-scroll is expected to be latched one cycle later
- // this is fine since nothing has started in the rendering until the second cycle
- // calculate the column number of the tile to start with
- x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
- render_offset = scroll_x % 8;
- }
-
- // render the screen and handle hblank
- render(cycle - 83);
- }
- }
- }
-
- if (LY_inc == 0)
- {
- if (cycle == 12)
- {
- LYC_INT = false;
- STAT &= 0xFB;
- }
- else if (cycle == 14)
- {
- // Special case of LY = LYC
- if ((LY == LYC) && !((STAT & 0x4) > 0))
- {
- // set STAT coincidence FLAG and interrupt flag if it is enabled
- STAT |= 0x04;
- if (((STAT & 0x40) > 0)) { LYC_INT = true; }
- }
- }
- }
-
- // here LY=LYC will be asserted or cleared (but only if LY isnt 0 as that's a special case)
- if ((cycle == 4) && (LY != 0))
- {
- if (LY_inc == 1)
- {
- LYC_INT = false;
- STAT &= 0xFB;
- }
- }
- else if ((cycle == 6) && (LY != 0))
- {
- if ((LY == LYC) && !((STAT & 0x4) > 0))
- {
- // set STAT coincidence FLAG and interrupt flag if it is enabled
- STAT |= 0x04;
- if (((STAT & 0x40) > 0)) { LYC_INT = true; }
- }
- }
-
- cycle++;
- }
- else
- {
- STAT &= 0xFC;
-
- VBL_INT = LYC_INT = HBL_INT = OAM_INT = false;
-
- in_vblank[0] = true;
-
- LCD_was_off = true;
-
- LY = 0;
- cpu_LY[0] = LY;
-
- cycle = 0;
- }
-
- // assert the STAT IRQ line if the line went from zero to 1
- stat_line = VBL_INT | LYC_INT | HBL_INT | OAM_INT;
-
- if (stat_line && !stat_line_old)
- {
- if ((REG_FFFF[0] & 0x2) > 0) { FlagI[0] = true; }
- REG_FF0F[0] |= 0x02;
-
- //if (LY == 46)
- //{
- //Console.Write(VBL_INT + " " + LYC_INT + " " + HBL_INT + " " + OAM_INT + " " + LY + " ");
- //Console.Write(render_offset + " " + scroll_x + " " + total_counter + " ");
- //Console.WriteLine(STAT + " " + cycle + " " + Core.cpu.TotalExecutedCycles);
- //}
- }
-
- stat_line_old = stat_line;
-
- // process latch delays
- //latch_delay();
-
- if (LYC_cd > 0)
- {
- LYC_cd--;
- if (LYC_cd == 0)
- {
- LYC = LYC_t;
-
- if (((LCDC & 0x80) > 0))
- {
- if (LY != LYC) { STAT &= 0xFB; LYC_INT = false; }
- else { STAT |= 0x4; LYC_INT = true; }
- }
- }
- }
- }
-
- // might be needed, not sure yet
- void latch_delay()
- {
- //BGP_l = BGP;
- }
-
- void render(uint32_t render_cycle)
- {
- // we are now in STAT mode 3
- // NOTE: presumably the first necessary sprite is fetched at sprite evaulation
- // i.e. just keeping track of the lowest x-value sprite
- if (render_cycle == 0)
- {
- /*
- OAM_access_read = false;
- OAM_access_write = true;
- VRAM_access_read = false;
- */
- // window X is latched for the scanline, mid-line changes have no effect
- window_x_latch = window_x;
-
- OAM_scan_index = 0;
- read_case = 0;
- internal_cycle = 0;
- pre_render = true;
- pre_render_2 = true;
- tile_inc = 0;
- pixel_counter = -8;
- sl_use_index = 0;
- fetch_sprite = false;
- going_to_fetch = false;
- first_fetch = true;
- consecutive_sprite = -render_offset + 8;
- no_sprites = false;
- evaled_sprites = 0;
- window_pre_render = false;
- window_latch = ((LCDC & 0x20) > 0);
-
- total_counter = 0;
-
- // TODO: If Window is turned on midscanline what happens? When is this check done exactly?
- if ((window_started && window_latch) || (window_is_reset && !window_latch && (LY >= window_y_latch)))
- {
- window_y_tile_inc++;
- if (window_y_tile_inc == 8)
- {
- window_y_tile_inc = 0;
- window_y_tile++;
- window_y_tile %= 32;
- }
- }
- window_started = false;
-
- if (SL_sprites_index == 0) { no_sprites = true; }
- // it is much easier to process sprites if we order them according to the rules of sprite priority first
- if (!no_sprites) { reorder_and_assemble_sprites(); }
- }
-
- // before anything else, we have to check if windowing is in effect
- if (window_latch && !window_started && (LY >= window_y_latch) && (pixel_counter >= (window_x_latch - 7)) && (window_x_latch < 167))
- {
- /*
- Console.Write(LY);
- Console.Write(" ");
- Console.Write(cycle);
- Console.Write(" ");
- Console.Write(window_y_tile);
- Console.Write(" ");
- Console.Write(render_offset);
- Console.Write(" ");
- Console.Write(window_x_latch);
- Console.Write(" ");
- Console.WriteLine(pixel_counter);
- */
-
- if (window_x_latch == 0)
- {
- // if the window starts at zero, we still do the first access to the BG
- // but then restart all over again at the window
- if ((render_offset % 7) <= 6)
- {
- read_case = 9;
- }
- else
- {
- read_case = 10;
- }
- }
- else
- {
- read_case = 4;
- }
-
- window_pre_render = true;
-
- window_counter = 0;
- render_counter = 0;
-
- window_x_tile = (uint32_t)floor((float)(pixel_counter - (window_x_latch - 7)) / 8.0);
-
- window_tile_inc = 0;
- window_started = true;
- window_is_reset = false;
- }
-
- if (!pre_render && !fetch_sprite)
- {
- // start shifting data into the LCD
- if (render_counter >= (render_offset + 8))
- {
- if ((tile_data_latch[2] & 0x20) > 0)
- {
- pixel = (tile_data_latch[0] & (1 << (render_counter % 8)) > 0) ? 1 : 0;
- pixel |= (tile_data_latch[1] & (1 << (render_counter % 8)) > 0) ? 2 : 0;
- }
- else
- {
- pixel = (tile_data_latch[0] & (1 << (7 - (render_counter % 8))) > 0) ? 1 : 0;
- pixel |= (tile_data_latch[1] & (1 << (7 - (render_counter % 8))) > 0) ? 2 : 0;
- }
-
- uint32_t ref_pixel = pixel;
-
- uint32_t pal_num = tile_data_latch[2] & 0x7;
-
- bool use_sprite = false;
-
- uint32_t s_pixel = 0;
-
- // now we have the BG pixel, we next need the sprite pixel
- if (!no_sprites)
- {
- bool have_sprite = false;
- uint32_t sprite_attr = 0;
-
- if (sprite_present_list[pixel_counter] == 1)
- {
- have_sprite = true;
- s_pixel = sprite_pixel_list[pixel_counter];
- sprite_attr = sprite_attr_list[pixel_counter];
- }
-
- if (have_sprite)
- {
- if (((LCDC & 0x2) > 0))
- {
- if (!((sprite_attr & 0x80) > 0))
- {
- use_sprite = true;
- }
- else if (ref_pixel == 0)
- {
- use_sprite = true;
- }
-
- if (!((LCDC & 0x1) > 0))
- {
- use_sprite = true;
- }
-
- // There is another priority bit in GBC, that can still override sprite priority
- if (((LCDC & 0x1) > 0) && ((tile_data_latch[2] & 0x80) > 0) && (ref_pixel != 0))
- {
- use_sprite = false;
- }
- }
-
- if (use_sprite)
- {
- pal_num = sprite_attr & 7;
- }
- }
- }
-
- // based on sprite priority and pixel values, pick a final pixel color
- if (use_sprite)
- {
- _vidbuffer[LY * 160 + pixel_counter] = (int)OBJ_palette[pal_num * 4 + s_pixel];
- }
- else
- {
- _vidbuffer[LY * 160 + pixel_counter] = (int)BG_palette[pal_num * 4 + pixel];
- }
-
- pixel_counter++;
-
- if (pixel_counter == 160)
- {
- read_case = 8;
- hbl_countdown = 2;
- }
- }
- else if (pixel_counter < 0)
- {
- pixel_counter++;
- }
- render_counter++;
- }
-
- if (!fetch_sprite)
- {
- if (!pre_render_2)
- {
- // before we go on to read case 3, we need to know if we stall there or not
- // Gekkio's tests show that if sprites are at position 0 or 1 (mod 8)
- // then it takes an extra cycle (1 or 2 more t-states) to process them
-
- if (!no_sprites && (pixel_counter < 160))
- {
- for (uint32_t i = 0; i < SL_sprites_index; i++)
- {
- if ((pixel_counter >= (SL_sprites[i * 4 + 1] - 8)) &&
- (pixel_counter < (SL_sprites[i * 4 + 1])) &&
- !((evaled_sprites & (1 << i)) > 0))
- {
- going_to_fetch = true;
- fetch_sprite = true;
- }
- }
- }
- }
-
- switch (read_case)
- {
- case 0: // read a background tile
- if ((internal_cycle % 2) == 1)
- {
- // calculate the row number of the tiles to be fetched
- y_tile = ((uint32_t)floor((float)((uint32_t)scroll_y + LY) / 8.0)) % 32;
-
- temp_fetch = y_tile * 32 + (x_tile + tile_inc) % 32;
- tile_byte = VRAM[0x1800 + (((LCDC & 0x4) > 0) ? 1 : 0) * 0x400 + temp_fetch];
- tile_data[2] = VRAM[0x3800 + (((LCDC & 0x4) > 0) ? 1 : 0) * 0x400 + temp_fetch];
- VRAM_sel = ((tile_data[2] & 0x8) > 0) ? 1 : 0;
-
- BG_V_flip = ((tile_data[2] & 0x40) > 0);
-
- read_case = 1;
- if (!pre_render)
- {
- tile_inc++;
- }
- }
- break;
-
- case 1: // read from tile graphics (0)
- if ((internal_cycle % 2) == 1)
- {
- y_scroll_offset = (scroll_y + LY) % 8;
-
- if (BG_V_flip)
- {
- y_scroll_offset = 7 - y_scroll_offset;
- }
-
- if (((LCDC & 0x10) > 0))
- {
- tile_data[0] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed byte
- if (((tile_byte & 0x80) > 0))
- {
- tile_byte -= 256;
- }
- tile_data[0] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
- }
-
- read_case = 2;
- }
- break;
-
- case 2: // read from tile graphics (1)
- if ((internal_cycle % 2) == 0)
- {
- pre_render_2 = false;
- }
- else
- {
- y_scroll_offset = (scroll_y + LY) % 8;
-
- if (BG_V_flip)
- {
- y_scroll_offset = 7 - y_scroll_offset;
- }
-
- if (((LCDC & 0x10) > 0))
- {
- // if LCDC somehow changed between the two reads, make sure we have a positive number
- if (tile_byte < 0)
- {
- tile_byte += 256;
- }
- tile_data[1] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed byte
- if (((tile_byte & 0x80) > 0) && tile_byte > 0)
- {
- tile_byte -= 256;
- }
- tile_data[1] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
-
- if (pre_render)
- {
- // here we set up rendering
- pre_render = false;
-
- render_counter = 0;
- latch_counter = 0;
- read_case = 0;
- }
- else
- {
- read_case = 3;
- }
- }
- break;
-
- case 3: // read from sprite data
- if ((internal_cycle % 2) == 1)
- {
- read_case = 0;
- latch_new_data = true;
- }
- break;
-
- case 4: // read from window data
- if ((window_counter % 2) == 1)
- {
- temp_fetch = window_y_tile * 32 + (window_x_tile + window_tile_inc) % 32;
- tile_byte = VRAM[0x1800 + (((LCDC & 0x40) > 0) ? 1 : 0) * 0x400 + temp_fetch];
- tile_data[2] = VRAM[0x3800 + (((LCDC & 0x40) > 0) ? 1 : 0) * 0x400 + temp_fetch];
- VRAM_sel = ((tile_data[2] & 0x8) > 0) ? 1 : 0;
- BG_V_flip = ((tile_data[2] & 0x40) > 0);
-
- window_tile_inc++;
- read_case = 5;
- }
- window_counter++;
- break;
-
- case 5: // read from tile graphics (for the window)
- if ((window_counter % 2) == 1)
- {
- y_scroll_offset = (window_y_tile_inc) % 8;
-
- if (BG_V_flip)
- {
- y_scroll_offset = 7 - y_scroll_offset;
- }
-
- if (((LCDC & 0x10) > 0))
- {
- tile_data[0] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed byte
- if (((tile_byte & 0x80) > 0))
- {
- tile_byte -= 256;
- }
- tile_data[0] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
- }
-
- read_case = 6;
- }
- window_counter++;
- break;
-
- case 6: // read from tile graphics (for the window)
- if ((window_counter % 2) == 1)
- {
- y_scroll_offset = (window_y_tile_inc) % 8;
-
- if (BG_V_flip)
- {
- y_scroll_offset = 7 - y_scroll_offset;
- }
-
- if (((LCDC & 0x10) > 0))
- {
- // if LCDC somehow changed between the two reads, make sure we have a positive number
- if (tile_byte < 0)
- {
- tile_byte += 256;
- }
- tile_data[1] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed byte
- if (((tile_byte & 0x80) > 0) && tile_byte > 0)
- {
- tile_byte -= 256;
- }
- tile_data[1] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
-
- if (window_pre_render)
- {
- // here we set up rendering
- // unlike for the normal background case, there is no pre-render period for the window
- // so start shifting in data to the screen right away
- if (window_x_latch <= 7)
- {
- if (render_offset == 0)
- {
- read_case = 4;
- }
- else
- {
- read_case = 9 + render_offset - 1;
- }
- render_counter = 8 - render_offset;
-
- render_offset = 7 - window_x_latch;
- }
- else
- {
- render_offset = 0;
- read_case = 4;
- render_counter = 8;
- }
-
- latch_counter = 0;
- latch_new_data = true;
- window_pre_render = false;
- }
- else
- {
- read_case = 7;
- }
- }
- window_counter++;
- break;
-
- case 7: // read from sprite data
- if ((window_counter % 2) == 1)
- {
- read_case = 4;
- latch_new_data = true;
- }
- window_counter++;
- break;
-
- case 8: // done reading, we are now in phase 0
- pre_render = true;
-
- // the other interrupts appear to be delayed by 1 CPU cycle, so do the same here
- if (hbl_countdown > 0)
- {
- hbl_countdown--;
-
- if (hbl_countdown == 0)
- {
- OAM_access_read = true;
- OAM_access_write = true;
- VRAM_access_read = true;
- VRAM_access_write = true;
- }
- else
- {
- STAT &= 0xFC;
- STAT |= 0x00;
-
- if (((STAT & 0x8) > 0)) { HBL_INT = true; }
- }
- }
- break;
-
- case 9:
- // this is a degenerate case for starting the window at 0
- // kevtris' timing doc indicates an additional normal BG access
- // but this information is thrown away, so it's faster to do this then constantly check
- // for it in read case 0
- read_case = 4;
- break;
- case 10:
- case 11:
- case 12:
- case 13:
- case 14:
- case 15:
- case 16:
- case 17:
- read_case--;
- break;
- }
- internal_cycle++;
-
- if (latch_new_data)
- {
- latch_new_data = false;
- tile_data_latch[0] = tile_data[0];
- tile_data_latch[1] = tile_data[1];
- tile_data_latch[2] = tile_data[2];
- }
- }
-
- // every in range sprite takes 6 cycles to process
- // sprites located at x=0 still take 6 cycles to process even though they don't appear on screen
- // sprites above x=168 do not take any cycles to process however
- if (fetch_sprite)
- {
- if (going_to_fetch)
- {
- going_to_fetch = false;
-
- last_eval = 0;
-
- // at this time it is unknown what each cycle does, but we only need to accurately keep track of cycles
- for (uint32_t i = 0; i < SL_sprites_index; i++)
- {
- if ((pixel_counter >= (SL_sprites[i * 4 + 1] - 8)) &&
- (pixel_counter < (SL_sprites[i * 4 + 1])) &&
- !((evaled_sprites & (1 << i)) > 0))
- {
- sprite_fetch_counter += 6;
- evaled_sprites |= (1 << i);
- last_eval = SL_sprites[i * 4 + 1];
- }
- }
-
- // x scroll offsets the penalty table
- // there is no penalty if the next sprites to be fetched are within the currentfetch block (8 pixels)
- if (first_fetch || (last_eval >= consecutive_sprite))
- {
- if (((last_eval + render_offset) % 8) == 0) { sprite_fetch_counter += 5; }
- else if (((last_eval + render_offset) % 8) == 1) { sprite_fetch_counter += 4; }
- else if (((last_eval + render_offset) % 8) == 2) { sprite_fetch_counter += 3; }
- else if (((last_eval + render_offset) % 8) == 3) { sprite_fetch_counter += 2; }
- else if (((last_eval + render_offset) % 8) == 4) { sprite_fetch_counter += 1; }
- else if (((last_eval + render_offset) % 8) == 5) { sprite_fetch_counter += 0; }
- else if (((last_eval + render_offset) % 8) == 6) { sprite_fetch_counter += 0; }
- else if (((last_eval + render_offset) % 8) == 7) { sprite_fetch_counter += 0; }
-
- consecutive_sprite = (uint32_t)floor((double)(last_eval + render_offset) / 8.0) * 8 + 8 - render_offset;
-
- // special case exists here for sprites at zero with non-zero x-scroll. Not sure exactly the reason for it.
- if (last_eval == 0 && render_offset != 0)
- {
- sprite_fetch_counter += render_offset;
- }
- }
-
- total_counter += sprite_fetch_counter;
-
- first_fetch = false;
- }
- else
- {
- sprite_fetch_counter--;
- if (sprite_fetch_counter == 0)
- {
- fetch_sprite = false;
- }
- }
- }
-
- }
-
- void process_sprite()
- {
- uint32_t y;
- uint32_t VRAM_temp = ((SL_sprites[sl_use_index * 4 + 3] & 0x8) > 0) ? 1 : 0;
-
- if ((SL_sprites[sl_use_index * 4 + 3] & 0x40) > 0)
- {
- if (((LCDC & 0x4) > 0))
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- y = 15 - y;
- sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
- sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
- }
- else
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- y = 7 - y;
- sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
- sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
- }
- }
- else
- {
- if (((LCDC & 0x4) > 0))
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
- sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
- }
- else
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
- sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
- }
- }
-
- if ((SL_sprites[sl_use_index * 4 + 3] & 0x20) > 0)
- {
- uint32_t b0, b1, b2, b3, b4, b5, b6, b7 = 0;
- for (uint32_t i = 0; i < 2; i++)
- {
- b0 = (sprite_sel[i] & 0x01) << 7;
- b1 = (sprite_sel[i] & 0x02) << 5;
- b2 = (sprite_sel[i] & 0x04) << 3;
- b3 = (sprite_sel[i] & 0x08) << 1;
- b4 = (sprite_sel[i] & 0x10) >> 1;
- b5 = (sprite_sel[i] & 0x20) >> 3;
- b6 = (sprite_sel[i] & 0x40) >> 5;
- b7 = (sprite_sel[i] & 0x80) >> 7;
-
- sprite_sel[i] = (uint8_t)(b0 | b1 | b2 | b3 | b4 | b5 | b6 | b7);
- }
- }
- }
-
- // normal DMA moves twice as fast in double speed mode on GBC
- // So give it it's own function so we can seperate it from PPU tick
- void DMA_tick()
- {
- // Note that DMA is halted when the CPU is halted
- if (DMA_start && !cpu_halted)
- {
- if (DMA_clock >= 4)
- {
- DMA_OAM_access = false;
- if ((DMA_clock % 4) == 1)
- {
- // the cpu can't access memory during this time, but we still need the ppu to be able to.
- DMA_start = false;
- // Gekkio reports that A14 being high on DMA transfers always represent WRAM accesses
- // So transfers nominally from higher memory areas are actually still from there (i.e. FF -> DF)
- uint8_t DMA_actual = DMA_addr;
- if (DMA_addr > 0xDF) { DMA_actual &= 0xDF; }
- DMA_byte = ReadMemory((uint32_t)((DMA_actual << 8) + DMA_inc));
- DMA_start = true;
- }
- else if ((DMA_clock % 4) == 3)
- {
- OAM[DMA_inc] = DMA_byte;
-
- if (DMA_inc < (0xA0 - 1)) { DMA_inc++; }
- }
- }
-
- DMA_clock++;
-
- if (DMA_clock == 648)
- {
- DMA_start = false;
- DMA_OAM_access = true;
- }
- }
- }
-
- // order sprites according to x coordinate
- // note that for sprites of equal x coordinate, priority goes to first on the list
- void reorder_and_assemble_sprites()
- {
- sprite_ordered_index = 0;
-
- // In CGB mode, sprites are ordered solely based on their position in OAM, so they are already ordered
- for (uint32_t j = 0; j < SL_sprites_index; j++)
- {
- sl_use_index = j;
- process_sprite();
- SL_sprites_ordered[sprite_ordered_index * 4] = SL_sprites[j * 4 + 1];
- SL_sprites_ordered[sprite_ordered_index * 4 + 1] = sprite_sel[0];
- SL_sprites_ordered[sprite_ordered_index * 4 + 2] = sprite_sel[1];
- SL_sprites_ordered[sprite_ordered_index * 4 + 3] = SL_sprites[j * 4 + 3];
- sprite_ordered_index++;
- }
-
- bool have_pixel = false;
- uint8_t s_pixel = 0;
- uint8_t sprite_attr = 0;
-
- for (uint32_t i = 0; i < 160; i++)
- {
- have_pixel = false;
- for (uint32_t j = 0; j < SL_sprites_index; j++)
- {
- if ((i >= (SL_sprites_ordered[j * 4] - 8)) &&
- (i < SL_sprites_ordered[j * 4]) &&
- !have_pixel)
- {
- // we can use the current sprite, so pick out a pixel for it
- uint32_t t_index = i - (SL_sprites_ordered[j * 4] - 8);
-
- t_index = 7 - t_index;
-
- sprite_data[0] = (uint8_t)((SL_sprites_ordered[j * 4 + 1] >> t_index) & 1);
- sprite_data[1] = (uint8_t)(((SL_sprites_ordered[j * 4 + 2] >> t_index) & 1) << 1);
-
- s_pixel = (uint8_t)(sprite_data[0] + sprite_data[1]);
- sprite_attr = (uint8_t)SL_sprites_ordered[j * 4 + 3];
-
- // pixel color of 0 is transparent, so if this is the case we don't have a pixel
- if (s_pixel != 0)
- {
- have_pixel = true;
- }
- }
- }
-
- if (have_pixel)
- {
- sprite_present_list[i] = 1;
- sprite_pixel_list[i] = s_pixel;
- sprite_attr_list[i] = sprite_attr;
- }
- else
- {
- sprite_present_list[i] = 0;
- }
- }
- }
-
- void OAM_scan(uint32_t OAM_cycle)
- {
- // we are now in STAT mode 2
- // TODO: maybe stat mode 2 flags are set at cycle 0 on visible scanlines?
- if (OAM_cycle == 0)
- {
- OAM_access_read = false;
- OAM_access_write = false;
-
- OAM_scan_index = 0;
- SL_sprites_index = 0;
- write_sprite = 0;
- }
-
- // the gameboy has 80 cycles to scan through 40 sprites, picking out the first 10 it finds to draw
- // the following is a guessed at implmenentation based on how NES does it, it's probably pretty close
- if (OAM_cycle < 10)
- {
- // start by clearing the sprite table (probably just clears X on hardware, but let's be safe here.)
- SL_sprites[OAM_cycle * 4] = 0;
- SL_sprites[OAM_cycle * 4 + 1] = 0;
- SL_sprites[OAM_cycle * 4 + 2] = 0;
- SL_sprites[OAM_cycle * 4 + 3] = 0;
- }
- else
- {
- if (write_sprite == 0)
- {
- if (OAM_scan_index < 40)
- {
- uint32_t temp = DMA_OAM_access ? OAM[OAM_scan_index * 4] : (uint32_t)0xFF;
- // (sprite Y - 16) equals LY, we have a sprite
- if ((temp - 16) <= LY &&
- ((temp - 16) + 8 + (((LCDC & 0x4) > 0) ? 8 : 0)) > LY)
- {
- // always pick the first 10 in range sprites
- if (SL_sprites_index < 10)
- {
- SL_sprites[SL_sprites_index * 4] = temp;
-
- write_sprite = 1;
- }
- else
- {
- // if we already have 10 sprites, there's nothing to do, increment the index
- OAM_scan_index++;
- }
- }
- else
- {
- OAM_scan_index++;
- }
- }
- }
- else
- {
- uint32_t temp2 = DMA_OAM_access ? OAM[OAM_scan_index * 4 + write_sprite] : (uint32_t)0xFF;
- SL_sprites[SL_sprites_index * 4 + write_sprite] = temp2;
- write_sprite++;
-
- if (write_sprite == 4)
- {
- write_sprite = 0;
- SL_sprites_index++;
- OAM_scan_index++;
- }
- }
- }
- }
-
- void color_compute_BG()
- {
- uint32_t R;
- uint32_t G;
- uint32_t B;
-
- if ((BG_bytes_index % 2) == 0)
- {
- R = (uint32_t)(BG_bytes[BG_bytes_index] & 0x1F);
- G = (uint32_t)(((BG_bytes[BG_bytes_index] & 0xE0) | ((BG_bytes[BG_bytes_index + 1] & 0x03) << 8)) >> 5);
- B = (uint32_t)((BG_bytes[BG_bytes_index + 1] & 0x7C) >> 2);
- }
- else
- {
- R = (uint32_t)(BG_bytes[BG_bytes_index - 1] & 0x1F);
- G = (uint32_t)(((BG_bytes[BG_bytes_index - 1] & 0xE0) | ((BG_bytes[BG_bytes_index] & 0x03) << 8)) >> 5);
- B = (uint32_t)((BG_bytes[BG_bytes_index] & 0x7C) >> 2);
- }
-
- uint32_t retR = ((R * 13 + G * 2 + B) >> 1) & 0xFF;
- uint32_t retG = ((G * 3 + B) << 1) & 0xFF;
- uint32_t retB = ((R * 3 + G * 2 + B * 11) >> 1) & 0xFF;
-
- BG_palette[BG_bytes_index >> 1] = (uint32_t)(0xFF000000 | (retR << 16) | (retG << 8) | retB);
- }
-
- void color_compute_OBJ()
- {
- uint32_t R;
- uint32_t G;
- uint32_t B;
-
- if ((OBJ_bytes_index % 2) == 0)
- {
- R = (uint32_t)(OBJ_bytes[OBJ_bytes_index] & 0x1F);
- G = (uint32_t)(((OBJ_bytes[OBJ_bytes_index] & 0xE0) | ((OBJ_bytes[OBJ_bytes_index + 1] & 0x03) << 8)) >> 5);
- B = (uint32_t)((OBJ_bytes[OBJ_bytes_index + 1] & 0x7C) >> 2);
- }
- else
- {
- R = (uint32_t)(OBJ_bytes[OBJ_bytes_index - 1] & 0x1F);
- G = (uint32_t)(((OBJ_bytes[OBJ_bytes_index - 1] & 0xE0) | ((OBJ_bytes[OBJ_bytes_index] & 0x03) << 8)) >> 5);
- B = (uint32_t)((OBJ_bytes[OBJ_bytes_index] & 0x7C) >> 2);
- }
-
- uint32_t retR = ((R * 13 + G * 2 + B) >> 1) & 0xFF;
- uint32_t retG = ((G * 3 + B) << 1) & 0xFF;
- uint32_t retB = ((R * 3 + G * 2 + B * 11) >> 1) & 0xFF;
-
- OBJ_palette[OBJ_bytes_index >> 1] = (uint32_t)(0xFF000000 | (retR << 16) | (retG << 8) | retB);
- }
-
- void Reset()
- {
- LCDC = 0;
- STAT = 0x80;
- scroll_y = 0;
- scroll_x = 0;
- LY = 0;
- LYC = 0;
- DMA_addr = 0;
- BGP = 0xFF;
- obj_pal_0 = 0;
- obj_pal_1 = 0;
- window_y = 0x0;
- window_x = 0x0;
- window_x_latch = 0xFF;
- window_y_latch = 0xFF;
- LY_inc = 1;
- no_scan = false;
- OAM_access_read = true;
- VRAM_access_read = true;
- OAM_access_write = true;
- VRAM_access_write = true;
- DMA_OAM_access = true;
-
- cycle = 0;
- LYC_INT = false;
- HBL_INT = false;
- VBL_INT = false;
- OAM_INT = false;
-
- stat_line = false;
- stat_line_old = false;
-
- window_counter = 0;
- window_pre_render = false;
- window_started = false;
- window_tile_inc = 0;
- window_y_tile = 0;
- window_x_tile = 0;
- window_y_tile_inc = 0;
-
- BG_bytes_inc = false;
- OBJ_bytes_inc = false;
- BG_bytes_index = 0;
- OBJ_bytes_index = 0;
- BG_transfer_byte = 0;
- OBJ_transfer_byte = 0;
-
- HDMA_src_hi = 0;
- HDMA_src_lo = 0;
- HDMA_dest_hi = 0;
- HDMA_dest_lo = 0;
-
- VRAM_sel = 0;
- BG_V_flip = false;
- HDMA_active = false;
- HDMA_mode = false;
- cur_DMA_src = 0;
- cur_DMA_dest = 0;
- HDMA_length = 0;
- HDMA_countdown = 0;
- HBL_HDMA_count = 0;
- last_HBL = 0;
- HBL_HDMA_go = false;
- HBL_test = false;
- }
-
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/GBHawk.cpp b/libHawk/GBHawk/GBHawk/GBHawk.cpp
index 33660260d8..cbe5671bf9 100644
--- a/libHawk/GBHawk/GBHawk/GBHawk.cpp
+++ b/libHawk/GBHawk/GBHawk/GBHawk.cpp
@@ -26,16 +26,18 @@ GBHawk_EXPORT void GB_destroy(GBCore* p)
std::free(p);
}
-// load bios and basic into the core
-GBHawk_EXPORT void GB_load_bios(GBCore* p, uint8_t* bios, bool GBC_console)
+// load bios into the core
+GBHawk_EXPORT void GB_load_bios(GBCore* p, uint8_t* bios, bool GBC_console, bool GBC_as_GBA)
{
- p->Load_BIOS(bios, GBC_console);
+ p->Load_BIOS(bios, GBC_console, GBC_as_GBA);
}
// load a rom into the core
-GBHawk_EXPORT void GB_load(GBCore* p, uint8_t* rom_1, uint32_t size_1, uint32_t mapper_1, uint8_t* rom_2, uint32_t size_2, uint8_t mapper_2)
+GBHawk_EXPORT void GB_load(GBCore* p, uint8_t* rom_1, uint32_t size_1, uint32_t RTC_initial, uint32_t RTC_offset)
{
- p->Load_ROM(rom_1, size_1, mapper_1, rom_2, size_2, mapper_2);
+ string MD5;
+
+ p->Load_ROM(rom_1, size_1, MD5, RTC_initial, RTC_offset);
}
// advance a frame
diff --git a/libHawk/GBHawk/GBHawk/GBHawk.vcxproj b/libHawk/GBHawk/GBHawk/GBHawk.vcxproj
index bcadc24fba..011645ac17 100644
--- a/libHawk/GBHawk/GBHawk/GBHawk.vcxproj
+++ b/libHawk/GBHawk/GBHawk/GBHawk.vcxproj
@@ -153,39 +153,19 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
+
-
+
-
+
diff --git a/libHawk/GBHawk/GBHawk/GB_PPU.h b/libHawk/GBHawk/GBHawk/GB_PPU.h
deleted file mode 100644
index b041d4d3d4..0000000000
--- a/libHawk/GBHawk/GBHawk/GB_PPU.h
+++ /dev/null
@@ -1,1245 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "PPU_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class GB_PPU : public PPU
- {
- public:
- uint8_t ReadReg(uint32_t addr)
- {
- uint8_t ret = 0;
-
- switch (addr)
- {
- case 0xFF40: ret = LCDC; break; // LCDC
- case 0xFF41: ret = STAT; break; // STAT
- case 0xFF42: ret = scroll_y; break; // SCY
- case 0xFF43: ret = scroll_x; break; // SCX
- case 0xFF44: ret = LY; break; // LY
- case 0xFF45: ret = LYC; break; // LYC
- case 0xFF46: ret = DMA_addr; break; // DMA
- case 0xFF47: ret = BGP; break; // BGP
- case 0xFF48: ret = obj_pal_0; break; // OBP0
- case 0xFF49: ret = obj_pal_1; break; // OBP1
- case 0xFF4A: ret = window_y; break; // WY
- case 0xFF4B: ret = window_x; break; // WX
- }
-
- return ret;
- }
-
- void WriteReg(uint32_t addr, uint8_t value)
- {
- //Console.WriteLine((addr - 0xFF40) + " " + value + " " + LY + " " + cycle + " " + LCDC.Bit(7));
- switch (addr)
- {
- case 0xFF40: // LCDC
- if (((LCDC & 0x80) > 0) && !((value & 0x80) > 0))
- {
- VRAM_access_read = true;
- VRAM_access_write = true;
- OAM_access_read = true;
- OAM_access_write = true;
-
- clear_screen = true;
- }
-
- if (!((LCDC & 0x80) > 0) && ((value & 0x80) > 0))
- {
- // don't draw for one frame after turning on
- blank_frame = true;
- }
-
- LCDC = value;
- break;
- case 0xFF41: // STAT
- // writing to STAT during mode 0 or 1 causes a STAT IRQ
- // this appears to be a glitchy LYC compare
- if (((LCDC & 0x80) > 0))
- {
- if (((STAT & 3) == 0) || ((STAT & 3) == 1))
- {
- LYC_INT = true;
- //if (Core.REG_FFFF.Bit(1)) { Core.cpu.FlagI = true; }
- //Core.REG_FF0F |= 0x02;
- }
- else
- {
- if (((value & 0x40) > 0))
- {
- if (LY == LYC) { LYC_INT = true; }
- else { LYC_INT = false; }
- }
- }
- }
- STAT = (uint8_t)((value & 0xF8) | (STAT & 7) | 0x80);
-
- //if (!STAT.Bit(6)) { LYC_INT = false; }
- if (!((STAT & 0x10) > 0)) { VBL_INT = false; }
- break;
- case 0xFF42: // SCY
- scroll_y = value;
- break;
- case 0xFF43: // SCX
- scroll_x = value;
- break;
- case 0xFF44: // LY
- LY = 0; /*reset*/
- break;
- case 0xFF45: // LYC
- LYC = value;
- if (((LCDC & 0x80) > 0))
- {
- if (LY != LYC) { STAT &= 0xFB; LYC_INT = false; }
- else { STAT |= 0x4; LYC_INT = true; }
-
- // special case: the transition from 153 -> 0 acts strange
- // the comparison to 153 expects to be true for longer then the value of LY expects to be 153
- // this appears to be fixed in CGB
- if ((LY_inc == 0) && cycle == 8)
- {
- if (153 != LYC) { STAT &= 0xFB; LYC_INT = false; }
- else { STAT |= 0x4; LYC_INT = true; }
- }
- }
- break;
- case 0xFF46: // DMA
- DMA_addr = value;
- DMA_start = true;
- DMA_OAM_access = true;
- DMA_clock = 0;
- DMA_inc = 0;
- break;
- case 0xFF47: // BGP
- BGP = value;
- break;
- case 0xFF48: // OBP0
- obj_pal_0 = value;
- break;
- case 0xFF49: // OBP1
- obj_pal_1 = value;
- break;
- case 0xFF4A: // WY
- window_y = value;
- break;
- case 0xFF4B: // WX
- window_x = value;
- break;
- }
- }
-
- void tick()
- {
- // the ppu only does anything if it is turned on via bit 7 of LCDC
- if (((LCDC & 0x80) > 0))
- {
- // start the next scanline
- if (cycle == 456)
- {
- // scanline callback
- if ((LY + LY_inc) == _scanlineCallbackLine[0])
- {
- //if (Core._scanlineCallback != null)
- //{
- //Core.GetGPU();
- //Core._scanlineCallback(LCDC);
- //}
- }
-
- cycle = 0;
- LY += LY_inc;
- cpu_LY[0] = LY;
-
- no_scan = false;
-
- if (LY == 0 && LY_inc == 0)
- {
- LY_inc = 1;
- in_vblank[0] = false;
-
- STAT &= 0xFC;
-
- // special note here, the y coordiate of the window is kept if the window is deactivated
- // meaning it will pick up where it left off if re-enabled later
- // so we don't reset it in the scanline loop
- window_y_tile = 0;
- window_y_latch = window_y;
- window_y_tile_inc = 0;
- window_started = false;
- if (!((LCDC & 0x20) > 0)) { window_is_reset = true; }
- }
-
- // Automatically restore access to VRAM at this time (force end drawing)
- // Who Framed Roger Rabbit seems to run into this.
- VRAM_access_write = true;
- VRAM_access_read = true;
-
- if (LY == 144)
- {
- in_vblank[0] = true;
- }
- }
-
- // exit vblank if LCD went from off to on
- if (LCD_was_off)
- {
- //VBL_INT = false;
- in_vblank[0] = false;
- LCD_was_off = false;
-
- // we exit vblank into mode 0 for 4 cycles
- // but no hblank interrupt, presumably this only happens transitioning from mode 3 to 0
- STAT &= 0xFC;
-
- // also the LCD doesn't turn on right away
-
- // also, the LCD does not enter mode 2 on scanline 0 when first turned on
- no_scan = true;
- cycle = 8;
- }
-
- // the VBL stat is continuously asserted
- if (LY >= 144)
- {
- if (((STAT & 0x10) > 0))
- {
- if ((cycle >= 4) && (LY == 144))
- {
- VBL_INT = true;
- }
- else if (LY > 144)
- {
- VBL_INT = true;
- }
- }
-
- if ((cycle == 2) && (LY == 144))
- {
- // there is an edge case where a VBL INT is triggered if STAT bit 5 is set
- if (((STAT & 0x20) > 0)) { VBL_INT = true; }
- }
-
- if ((cycle == 4) && (LY == 144))
- {
- HBL_INT = false;
-
- // set STAT mode to 1 (VBlank) and interrupt flag if it is enabled
- STAT &= 0xFC;
- STAT |= 0x01;
-
- if (((REG_FFFF[0] & 0x1) > 0)) { FlagI[0] = true; }
- REG_FF0F[0] |= 0x01;
- }
-
- if ((cycle == 4) && (LY == 144))
- {
- if (((STAT & 0x20) > 0)) { VBL_INT = false; }
- }
-
- if ((cycle == 6) && (LY == 153))
- {
- LY = 0;
- LY_inc = 0;
- cpu_LY[0] = LY;
- }
- }
-
- if (!in_vblank[0])
- {
- if (no_scan)
- {
- // timings are slightly different if we just turned on the LCD
- // there is no mode 2 (presumably it missed the trigger)
- if (cycle < 85)
- {
- if (cycle == 8)
- {
- // clear the sprite table
- for (uint32_t k = 0; k < 10; k++)
- {
- SL_sprites[k * 4] = 0;
- SL_sprites[k * 4 + 1] = 0;
- SL_sprites[k * 4 + 2] = 0;
- SL_sprites[k * 4 + 3] = 0;
- }
-
- if (LY != LYC)
- {
- LYC_INT = false;
- STAT &= 0xFB;
- }
-
- if ((LY == LYC) && !((STAT & 0x4) > 0))
- {
- // set STAT coincidence FLAG and interrupt flag if it is enabled
- STAT |= 0x04;
- if (((STAT & 0x40) > 0)) { LYC_INT = true; }
- }
- }
-
- if (cycle == 84)
- {
- STAT &= 0xFC;
- STAT |= 0x03;
- OAM_INT = false;
-
- OAM_access_read = false;
- OAM_access_write = false;
- VRAM_access_read = false;
- VRAM_access_write = false;
- }
- }
- else
- {
- if (cycle >= 85)
- {
- if (cycle == 85)
- {
- // x-scroll is expected to be latched one cycle later
- // this is fine since nothing has started in the rendering until the second cycle
- // calculate the column number of the tile to start with
- x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
- render_offset = scroll_x % 8;
- }
-
- // render the screen and handle hblank
- render(cycle - 85);
- }
- }
- }
- else
- {
- if (cycle <= 80)
- {
- if (cycle == 2)
- {
- if (LY != 0)
- {
- HBL_INT = false;
- if (((STAT & 0x20) > 0)) { OAM_INT = true; }
- }
- }
- else if (cycle == 4)
- {
- // apparently, writes can make it to OAM one cycle longer then reads
- OAM_access_write = false;
-
- // here mode 2 will be set to true and interrupts fired if enabled
- STAT &= 0xFC;
- STAT |= 0x2;
-
- if (LY == 0)
- {
- VBL_INT = false;
- if (((STAT & 0x20) > 0)) { OAM_INT = true; }
- }
- }
-
- if (cycle == 80)
- {
- OAM_access_read = false;
- OAM_access_write = true;
- VRAM_access_read = false;
- }
- else
- {
- // here OAM scanning is performed
- OAM_scan(cycle);
- }
- }
- else if (cycle >= 83)
- {
- if (cycle == 84)
- {
- STAT &= 0xFC;
- STAT |= 0x03;
- OAM_INT = false;
- OAM_access_write = false;
- VRAM_access_write = false;
-
- // x-scroll is expected to be latched one cycle later
- // this is fine since nothing has started in the rendering until the second cycle
- // calculate the column number of the tile to start with
- x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
- render_offset = scroll_x % 8;
- }
-
- // render the screen and handle hblank
- render(cycle - 83);
- }
- }
- }
-
- if (LY_inc == 0)
- {
- if (cycle == 10)
- {
- LYC_INT = false;
- STAT &= 0xFB;
- }
- else if (cycle == 12)
- {
- // Special case of LY = LYC
- if ((LY == LYC) && !((STAT & 0x4) > 0))
- {
- // set STAT coincidence FLAG and interrupt flag if it is enabled
- STAT |= 0x04;
- if (((STAT & 0x40) > 0)) { LYC_INT = true; }
- }
- }
- }
-
- // here LY=LYC will be asserted or cleared (but only if LY isnt 0 as that's a special case)
- if ((cycle == 2) && (LY != 0))
- {
- if (LY_inc == 1)
- {
- LYC_INT = false;
- STAT &= 0xFB;
- }
- }
- else if ((cycle == 4) && (LY != 0))
- {
- if ((LY == LYC) && !((STAT & 0x4) > 0))
- {
- // set STAT coincidence FLAG and interrupt flag if it is enabled
- STAT |= 0x04;
- if (((STAT & 0x40) > 0)) { LYC_INT = true; }
- }
- }
-
- cycle++;
- }
- else
- {
- STAT &= 0xFC;
-
- VBL_INT = LYC_INT = HBL_INT = OAM_INT = false;
-
- in_vblank[0] = true;
-
- LCD_was_off = true;
-
- LY = 0;
- cpu_LY[0] = LY;
-
- cycle = 0;
- }
-
- // assert the STAT IRQ line if the line went from zero to 1
- stat_line = VBL_INT | LYC_INT | HBL_INT | OAM_INT;
-
- if (stat_line && !stat_line_old)
- {
- if (((REG_FFFF[0] & 0x2) > 0)) { FlagI[0] = true; }
- REG_FF0F[0] |= 0x02;
- }
-
- stat_line_old = stat_line;
-
- // process latch delays
- //latch_delay();
- }
-
- // might be needed, not sure yet
- void latch_delay()
- {
-
- }
-
- void render(uint32_t render_cycle)
- {
- // we are now in STAT mode 3
- // NOTE: presumably the first necessary sprite is fetched at sprite evaulation
- // i.e. just keeping track of the lowest x-value sprite
- if (render_cycle == 0)
- {
- /*
- OAM_access_read = false;
- OAM_access_write = true;
- VRAM_access_read = false;
- */
- // window X is latched for the scanline, mid-line changes have no effect
- window_x_latch = window_x;
-
- OAM_scan_index = 0;
- read_case = 0;
- internal_cycle = 0;
- pre_render = true;
- pre_render_2 = true;
- tile_inc = 0;
- pixel_counter = -8;
- sl_use_index = 0;
- fetch_sprite = false;
- going_to_fetch = false;
- first_fetch = true;
- consecutive_sprite = -render_offset + 8;
- no_sprites = false;
- evaled_sprites = 0;
- window_pre_render = false;
- window_latch = ((LCDC & 0x20) > 0);
-
- total_counter = 0;
-
- // TODO: If Window is turned on midscanline what happens? When is this check done exactly?
- if ((window_started && window_latch) || (window_is_reset && !window_latch && (LY >= window_y_latch)))
- {
- window_y_tile_inc++;
- if (window_y_tile_inc == 8)
- {
- window_y_tile_inc = 0;
- window_y_tile++;
- window_y_tile %= 32;
- }
- }
- window_started = false;
-
- if (SL_sprites_index == 0) { no_sprites = true; }
- // it is much easier to process sprites if we order them according to the rules of sprite priority first
- if (!no_sprites) { reorder_and_assemble_sprites(); }
- }
-
- // before anything else, we have to check if windowing is in effect
- if (window_latch && !window_started && (LY >= window_y_latch) && (pixel_counter >= (window_x_latch - 7)) && (window_x_latch < 167))
- {
- /*
- Console.Write(LY);
- Console.Write(" ");
- Console.Write(cycle);
- Console.Write(" ");
- Console.Write(window_y_tile_inc);
- Console.Write(" ");
- Console.Write(window_x_latch);
- Console.Write(" ");
- Console.WriteLine(pixel_counter);
- */
- if (window_x_latch == 0)
- {
- // if the window starts at zero, we still do the first access to the BG
- // but then restart all over again at the window
- if ((render_offset % 7) <= 6)
- {
- read_case = 9;
- }
- else
- {
- read_case = 10;
- }
- }
- else
- {
- read_case = 4;
- }
-
- window_pre_render = true;
-
- window_counter = 0;
- render_counter = 0;
-
- window_x_tile = (uint32_t)floor((float)(pixel_counter - (window_x_latch - 7)) / 8.0);
-
- window_tile_inc = 0;
- window_started = true;
- window_is_reset = false;
- }
-
- if (!pre_render && !fetch_sprite)
- {
- // start shifting data into the LCD
- if (render_counter >= (render_offset + 8))
- {
-
- pixel = (tile_data_latch[0] & (1 << (7 - (render_counter % 8))) > 0) ? 1 : 0;
- pixel |= (tile_data_latch[1] & (1 << (7 - (render_counter % 8))) > 0) ? 2 : 0;
-
- uint32_t ref_pixel = pixel;
- if (((LCDC & 0x1) > 0))
- {
- pixel = (BGP >> (pixel * 2)) & 3;
- }
- else
- {
- pixel = 0;
- }
-
- // now we have the BG pixel, we next need the sprite pixel
- if (!no_sprites)
- {
- bool have_sprite = false;
- uint32_t s_pixel = 0;
- uint32_t sprite_attr = 0;
-
- if (sprite_present_list[pixel_counter] == 1)
- {
- have_sprite = true;
- s_pixel = sprite_pixel_list[pixel_counter];
- sprite_attr = sprite_attr_list[pixel_counter];
- }
-
- if (have_sprite)
- {
- bool use_sprite = false;
- if (((LCDC & 0x2) > 0))
- {
- if (!((sprite_attr & 0x80) > 0))
- {
- use_sprite = true;
- }
- else if (ref_pixel == 0)
- {
- use_sprite = true;
- }
-
- if (!((LCDC & 0x1) > 0))
- {
- use_sprite = true;
- }
- }
-
- if (use_sprite)
- {
- if (((sprite_attr & 0x10) > 0))
- {
- pixel = (obj_pal_1 >> (s_pixel * 2)) & 3;
- }
- else
- {
- pixel = (obj_pal_0 >> (s_pixel * 2)) & 3;
- }
- }
- }
- }
-
- // based on sprite priority and pixel values, pick a final pixel color
- _vidbuffer[LY * 160 + pixel_counter] = (uint32_t)color_palette[pixel];
- pixel_counter++;
-
- if (pixel_counter == 160)
- {
- read_case = 8;
- }
- }
- else if (pixel_counter < 0)
- {
- pixel_counter++;
- }
- render_counter++;
- }
-
- if (!fetch_sprite)
- {
- if (!pre_render_2)
- {
- // before we go on to read case 3, we need to know if we stall there or not
- // Gekkio's tests show that if sprites are at position 0 or 1 (mod 8)
- // then it takes an extra cycle (1 or 2 more t-states) to process them
- // Also, on DMG only, this process only runs if sprites are on in the LCDC (on GBC it always runs)
- if (!no_sprites && (pixel_counter < 160) && ((LCDC & 0x2) > 0))
- {
- for (uint32_t i = 0; i < SL_sprites_index; i++)
- {
- if ((pixel_counter >= (SL_sprites[i * 4 + 1] - 8)) &&
- (pixel_counter < (SL_sprites[i * 4 + 1])) &&
- !((evaled_sprites & (1 << i)) > 0))
- {
- going_to_fetch = true;
- fetch_sprite = true;
- }
- }
- }
- }
-
- switch (read_case)
- {
- case 0: // read a background tile
- if ((internal_cycle % 2) == 1)
- {
- // calculate the row number of the tiles to be fetched
- y_tile = ((uint32_t)floor((float)((uint32_t)scroll_y + LY) / 8.0)) % 32;
-
- temp_fetch = y_tile * 32 + (x_tile + tile_inc) % 32;
- tile_byte = VRAM[0x1800 + (((LCDC & 0x8) > 0) ? 1 : 0) * 0x400 + temp_fetch];
-
- read_case = 1;
- if (!pre_render)
- {
- tile_inc++;
- }
- }
- break;
-
- case 1: // read from tile graphics (0)
- if ((internal_cycle % 2) == 1)
- {
- y_scroll_offset = (scroll_y + LY) % 8;
-
- if (((LCDC & 0x10) > 0))
- {
- tile_data[0] = VRAM[tile_byte * 16 + y_scroll_offset * 2];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed uint8_t
- if (((tile_byte & 0x80) > 0))
- {
- tile_byte -= 256;
- }
- tile_data[0] = VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2];
- }
-
- read_case = 2;
- }
- break;
-
- case 2: // read from tile graphics (1)
- if ((internal_cycle % 2) == 0)
- {
- pre_render_2 = false;
- }
- else
- {
- y_scroll_offset = (scroll_y + LY) % 8;
-
- if (((LCDC & 0x10) > 0))
- {
- // if LCDC somehow changed between the two reads, make sure we have a positive number
- if (tile_byte < 0)
- {
- tile_byte += 256;
- }
-
- tile_data[1] = VRAM[tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed uint8_t
- if (((tile_byte & 0x80) > 0) && tile_byte > 0)
- {
- tile_byte -= 256;
- }
-
- tile_data[1] = VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
-
- if (pre_render)
- {
- // here we set up rendering
- pre_render = false;
-
- render_counter = 0;
- latch_counter = 0;
- read_case = 0;
- }
- else
- {
- read_case = 3;
- }
- }
- break;
-
- case 3: // read from sprite data
- if ((internal_cycle % 2) == 1)
- {
- read_case = 0;
- latch_new_data = true;
- }
- break;
-
- case 4: // read from window data
- if ((window_counter % 2) == 1)
- {
- temp_fetch = window_y_tile * 32 + (window_x_tile + window_tile_inc) % 32;
- tile_byte = VRAM[0x1800 + (((LCDC & 0x40) > 0) ? 1 : 0) * 0x400 + temp_fetch];
-
- window_tile_inc++;
- read_case = 5;
- }
- window_counter++;
- break;
-
- case 5: // read from tile graphics (for the window)
- if ((window_counter % 2) == 1)
- {
- y_scroll_offset = (window_y_tile_inc) % 8;
-
- if (((LCDC & 0x10) > 0))
- {
-
- tile_data[0] = VRAM[tile_byte * 16 + y_scroll_offset * 2];
-
- }
- else
- {
- // same as before except now tile uint8_t represents a signed uint8_t
- if (((tile_byte & 0x80) > 0))
- {
- tile_byte -= 256;
- }
-
- tile_data[0] = VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2];
- }
-
- read_case = 6;
- }
- window_counter++;
- break;
-
- case 6: // read from tile graphics (for the window)
- if ((window_counter % 2) == 1)
- {
- y_scroll_offset = (window_y_tile_inc) % 8;
- if (((LCDC & 0x10) > 0))
- {
- // if LCDC somehow changed between the two reads, make sure we have a positive number
- if (tile_byte < 0)
- {
- tile_byte += 256;
- }
-
- tile_data[1] = VRAM[tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
- else
- {
- // same as before except now tile uint8_t represents a signed uint8_t
- if (((tile_byte & 0x80) > 0) && tile_byte > 0)
- {
- tile_byte -= 256;
- }
-
- tile_data[1] = VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
- }
-
- if (window_pre_render)
- {
- // here we set up rendering
- // unlike for the normal background case, there is no pre-render period for the window
- // so start shifting in data to the screen right away
- if (window_x_latch <= 7)
- {
- if (render_offset == 0)
- {
- read_case = 4;
- }
- else
- {
- read_case = 9 + render_offset - 1;
- }
- render_counter = 8 - render_offset;
-
- render_offset = 7 - window_x_latch;
- }
- else
- {
- render_offset = 0;
- read_case = 4;
- render_counter = 8;
- }
-
- latch_counter = 0;
- latch_new_data = true;
- window_pre_render = false;
- }
- else
- {
- read_case = 7;
- }
- }
- window_counter++;
- break;
-
- case 7: // read from sprite data
- if ((window_counter % 2) == 1)
- {
- read_case = 4;
- latch_new_data = true;
- }
- window_counter++;
- break;
-
- case 8: // done reading, we are now in phase 0
- pre_render = true;
-
- STAT &= 0xFC;
- STAT |= 0x00;
-
- if (((STAT & 0x8) > 0)) { HBL_INT = true; }
-
- OAM_access_read = true;
- OAM_access_write = true;
- VRAM_access_read = true;
- VRAM_access_write = true;
- break;
-
- case 9:
- // this is a degenerate case for starting the window at 0
- // kevtris' timing doc indicates an additional normal BG access
- // but this information is thrown away, so it's faster to do this then constantly check
- // for it in read case 0
- read_case = 4;
- break;
- case 10:
- case 11:
- case 12:
- case 13:
- case 14:
- case 15:
- case 16:
- case 17:
- read_case--;
- break;
- }
- internal_cycle++;
-
- if (latch_new_data)
- {
- latch_new_data = false;
- tile_data_latch[0] = tile_data[0];
- tile_data_latch[1] = tile_data[1];
- }
- }
-
- // every in range sprite takes 6 cycles to process
- // sprites located at x=0 still take 6 cycles to process even though they don't appear on screen
- // sprites above x=168 do not take any cycles to process however
- if (fetch_sprite)
- {
- if (going_to_fetch)
- {
- going_to_fetch = false;
-
- last_eval = 0;
-
- // at this time it is unknown what each cycle does, but we only need to accurately keep track of cycles
- for (uint32_t i = 0; i < SL_sprites_index; i++)
- {
- if ((pixel_counter >= (SL_sprites[i * 4 + 1] - 8)) &&
- (pixel_counter < (SL_sprites[i * 4 + 1])) &&
- !((evaled_sprites & (1 << i)) > 0))
- {
- sprite_fetch_counter += 6;
- evaled_sprites |= (1 << i);
- last_eval = SL_sprites[i * 4 + 1];
- }
- }
-
- // x scroll offsets the penalty table
- // there is no penalty if the next sprites to be fetched are within the currentfetch block (8 pixels)
- if (first_fetch || (last_eval >= consecutive_sprite))
- {
- if (((last_eval + render_offset) % 8) == 0) { sprite_fetch_counter += 5; }
- else if (((last_eval + render_offset) % 8) == 1) { sprite_fetch_counter += 4; }
- else if (((last_eval + render_offset) % 8) == 2) { sprite_fetch_counter += 3; }
- else if (((last_eval + render_offset) % 8) == 3) { sprite_fetch_counter += 2; }
- else if (((last_eval + render_offset) % 8) == 4) { sprite_fetch_counter += 1; }
- else if (((last_eval + render_offset) % 8) == 5) { sprite_fetch_counter += 0; }
- else if (((last_eval + render_offset) % 8) == 6) { sprite_fetch_counter += 0; }
- else if (((last_eval + render_offset) % 8) == 7) { sprite_fetch_counter += 0; }
-
- consecutive_sprite = (uint32_t)floor((double)((uint32_t)last_eval + render_offset) / 8.0) * 8 + 8 - render_offset;
-
- // special case exists here for sprites at zero with non-zero x-scroll. Not sure exactly the reason for it.
- if (last_eval == 0 && render_offset != 0)
- {
- sprite_fetch_counter += render_offset;
- }
- }
-
- total_counter += sprite_fetch_counter;
-
- first_fetch = false;
- }
- else
- {
- sprite_fetch_counter--;
- if (sprite_fetch_counter == 0)
- {
- fetch_sprite = false;
- }
- }
- }
- }
-
- void process_sprite()
- {
- uint32_t y;
-
- if ((SL_sprites[sl_use_index * 4 + 3] & 0x40) > 0)
- {
- if (((LCDC & 0x4) > 0))
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- y = 15 - y;
- sprite_sel[0] = VRAM[(SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
- sprite_sel[1] = VRAM[(SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
- }
- else
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- y = 7 - y;
- sprite_sel[0] = VRAM[SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
- sprite_sel[1] = VRAM[SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
- }
- }
- else
- {
- if (((LCDC & 0x4) > 0))
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- sprite_sel[0] = VRAM[(SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
- sprite_sel[1] = VRAM[(SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
- }
- else
- {
- y = LY - (SL_sprites[sl_use_index * 4] - 16);
- sprite_sel[0] = VRAM[SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
- sprite_sel[1] = VRAM[SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
- }
- }
-
- if ((SL_sprites[sl_use_index * 4 + 3] & 0x20) > 0)
- {
- uint32_t b0, b1, b2, b3, b4, b5, b6, b7 = 0;
- for (uint32_t i = 0; i < 2; i++)
- {
- b0 = (sprite_sel[i] & 0x01) << 7;
- b1 = (sprite_sel[i] & 0x02) << 5;
- b2 = (sprite_sel[i] & 0x04) << 3;
- b3 = (sprite_sel[i] & 0x08) << 1;
- b4 = (sprite_sel[i] & 0x10) >> 1;
- b5 = (sprite_sel[i] & 0x20) >> 3;
- b6 = (sprite_sel[i] & 0x40) >> 5;
- b7 = (sprite_sel[i] & 0x80) >> 7;
-
- sprite_sel[i] = (uint8_t)(b0 | b1 | b2 | b3 | b4 | b5 | b6 | b7);
- }
- }
- }
-
- // normal DMA moves twice as fast in double speed mode on GBC
- // So give it it's own function so we can seperate it from PPU tick
- void DMA_tick()
- {
- // Note that DMA is halted when the CPU is halted
- if (DMA_start && !cpu_halted[0])
- {
- if (DMA_clock >= 4)
- {
- DMA_OAM_access = false;
- if ((DMA_clock % 4) == 1)
- {
- // the cpu can't access memory during this time, but we still need the ppu to be able to.
- DMA_start = false;
- // Gekkio reports that A14 being high on DMA transfers always represent WRAM accesses
- // So transfers nominally from higher memory areas are actually still from there (i.e. FF -> DF)
- uint8_t DMA_actual = DMA_addr;
- if (DMA_addr > 0xDF) { DMA_actual &= 0xDF; }
- DMA_byte = ReadMemory((uint32_t)((DMA_actual << 8) + DMA_inc));
- DMA_start = true;
- }
- else if ((DMA_clock % 4) == 3)
- {
- OAM[DMA_inc] = DMA_byte;
-
- if (DMA_inc < (0xA0 - 1)) { DMA_inc++; }
- }
- }
-
- DMA_clock++;
-
- if (DMA_clock == 648)
- {
- DMA_start = false;
- DMA_OAM_access = true;
- }
- }
- }
-
- // order sprites according to x coordinate
- // note that for sprites of equal x coordinate, priority goes to first on the list
- void reorder_and_assemble_sprites()
- {
- sprite_ordered_index = 0;
-
- for (uint32_t i = 0; i < 256; i++)
- {
- for (uint32_t j = 0; j < SL_sprites_index; j++)
- {
- if (SL_sprites[j * 4 + 1] == i)
- {
- sl_use_index = j;
- process_sprite();
- SL_sprites_ordered[sprite_ordered_index * 4] = SL_sprites[j * 4 + 1];
- SL_sprites_ordered[sprite_ordered_index * 4 + 1] = sprite_sel[0];
- SL_sprites_ordered[sprite_ordered_index * 4 + 2] = sprite_sel[1];
- SL_sprites_ordered[sprite_ordered_index * 4 + 3] = SL_sprites[j * 4 + 3];
- sprite_ordered_index++;
- }
- }
- }
-
- bool have_pixel = false;
- uint8_t s_pixel = 0;
- uint8_t sprite_attr = 0;
-
- for (uint32_t i = 0; i < 160; i++)
- {
- have_pixel = false;
- for (uint32_t j = 0; j < SL_sprites_index; j++)
- {
- if ((i >= (SL_sprites_ordered[j * 4] - 8)) &&
- (i < SL_sprites_ordered[j * 4]) &&
- !have_pixel)
- {
- // we can use the current sprite, so pick out a pixel for it
- uint32_t t_index = i - (SL_sprites_ordered[j * 4] - 8);
-
- t_index = 7 - t_index;
-
- sprite_data[0] = (uint8_t)((SL_sprites_ordered[j * 4 + 1] >> t_index) & 1);
- sprite_data[1] = (uint8_t)(((SL_sprites_ordered[j * 4 + 2] >> t_index) & 1) << 1);
-
- s_pixel = (uint8_t)(sprite_data[0] + sprite_data[1]);
- sprite_attr = (uint8_t)SL_sprites_ordered[j * 4 + 3];
-
- // pixel color of 0 is transparent, so if this is the case we don't have a pixel
- if (s_pixel != 0)
- {
- have_pixel = true;
- }
- }
- }
-
- if (have_pixel)
- {
- sprite_present_list[i] = 1;
- sprite_pixel_list[i] = s_pixel;
- sprite_attr_list[i] = sprite_attr;
- }
- else
- {
- sprite_present_list[i] = 0;
- }
- }
- }
-
- void OAM_scan(uint32_t OAM_cycle)
- {
- // we are now in STAT mode 2
- // TODO: maybe stat mode 2 flags are set at cycle 0 on visible scanlines?
- if (OAM_cycle == 0)
- {
- OAM_access_read = false;
-
- OAM_scan_index = 0;
- SL_sprites_index = 0;
- write_sprite = 0;
- }
-
- // the gameboy has 80 cycles to scan through 40 sprites, picking out the first 10 it finds to draw
- // the following is a guessed at implmenentation based on how NES does it, it's probably pretty close
- if (OAM_cycle < 10)
- {
- // start by clearing the sprite table (probably just clears X on hardware, but let's be safe here.)
- SL_sprites[OAM_cycle * 4] = 0;
- SL_sprites[OAM_cycle * 4 + 1] = 0;
- SL_sprites[OAM_cycle * 4 + 2] = 0;
- SL_sprites[OAM_cycle * 4 + 3] = 0;
- }
- else
- {
- if (write_sprite == 0)
- {
- if (OAM_scan_index < 40)
- {
- uint32_t temp = DMA_OAM_access ? OAM[OAM_scan_index * 4] : (uint32_t)0xFF;
- // (sprite Y - 16) equals LY, we have a sprite
- if ((temp - 16) <= LY &&
- ((temp - 16) + 8 + (((LCDC & 0x4) > 0) ? 8 : 0)) > LY)
- {
- // always pick the first 10 in range sprites
- if (SL_sprites_index < 10)
- {
- SL_sprites[SL_sprites_index * 4] = temp;
-
- write_sprite = 1;
- }
- else
- {
- // if we already have 10 sprites, there's nothing to do, increment the index
- OAM_scan_index++;
- }
- }
- else
- {
- OAM_scan_index++;
- }
- }
- }
- else
- {
- uint32_t temp2 = DMA_OAM_access ? OAM[OAM_scan_index * 4 + write_sprite] : (uint32_t)0xFF;
- SL_sprites[SL_sprites_index * 4 + write_sprite] = temp2;
- write_sprite++;
-
- if (write_sprite == 4)
- {
- write_sprite = 0;
- SL_sprites_index++;
- OAM_scan_index++;
- }
- }
- }
- }
-
- void Reset()
- {
- LCDC = 0;
- STAT = 0x80;
- scroll_y = 0;
- scroll_x = 0;
- LY = 0;
- LYC = 0;
- DMA_addr = 0xFF;
- BGP = 0xFF;
- obj_pal_0 = 0xFF;
- obj_pal_1 = 0xFF;
- window_y = 0x0;
- window_x = 0x0;
- window_x_latch = 0xFF;
- window_y_latch = 0xFF;
- LY_inc = 1;
- no_scan = false;
- OAM_access_read = true;
- VRAM_access_read = true;
- OAM_access_write = true;
- VRAM_access_write = true;
- DMA_OAM_access = true;
-
- cycle = 0;
- LYC_INT = false;
- HBL_INT = false;
- VBL_INT = false;
- OAM_INT = false;
-
- stat_line = false;
- stat_line_old = false;
-
- window_counter = 0;
- window_pre_render = false;
- window_started = false;
- window_is_reset = true;
- window_tile_inc = 0;
- window_y_tile = 0;
- window_x_tile = 0;
- window_y_tile_inc = 0;
- }
- };
-}
\ No newline at end of file
diff --git a/libHawk/GBHawk/GBHawk/Mapper_Base.h b/libHawk/GBHawk/GBHawk/Mapper_Base.h
deleted file mode 100644
index 0d97efd88f..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_Base.h
+++ /dev/null
@@ -1,350 +0,0 @@
-#include
-#include
-#include
-#include
-
-using namespace std;
-
-namespace GBHawk
-{
- class MemoryManager;
-
- class Mapper
- {
- public:
-
- Mapper()
- {
-
- }
-
- uint8_t* ROM = nullptr;
- uint8_t* Cart_RAM = nullptr;
- uint32_t* ROM_Length = nullptr;
- uint32_t* Cart_RAM_Length = nullptr;
- uint32_t* addr_access = nullptr;
- uint32_t* Acc_X_state = nullptr;
- uint32_t* Acc_Y_state = nullptr;
-
- // Generic Mapper Variables
- bool RAM_enable;
- bool sel_mode;
- bool IR_signal;
- uint32_t ROM_bank;
- uint32_t RAM_bank;
- uint32_t ROM_mask;
- uint32_t RAM_mask;
-
- // Common
- bool halt;
- uint32_t RTC_timer;
- uint32_t RTC_low_clock;
-
- // HuC3
- bool timer_read;
- uint8_t control;
- uint8_t chip_read;
- uint32_t time_val_shift;
- uint32_t time;
- uint32_t RTC_seconds;
-
- // MBC3
- uint8_t RTC_regs[5] = {};
- uint8_t RTC_regs_latch[5] = {};
- bool RTC_regs_latch_wr;
- uint32_t RTC_offset;
-
- // camera
- bool regs_enable;
- uint8_t regs_cam[0x80] = {};
-
- // sachen
- bool locked, locked_GBC, finished, reg_access;
- uint32_t ROM_bank_mask;
- uint32_t BASE_ROM_Bank;
- uint32_t addr_last;
- uint32_t counter;
-
- // TAMA5
- uint8_t RTC_regs_TAMA[10] = {};
- uint32_t ctrl;
- uint32_t RAM_addr_low;
- uint32_t RAM_addr_high;
- uint32_t RAM_val_low;
- uint32_t RAM_val_high;
- uint8_t Chip_return_low;
- uint8_t Chip_return_high;
-
- // MBC7
- bool RAM_enable_1, RAM_enable_2, is_erased;
- uint8_t acc_x_low;
- uint8_t acc_x_high;
- uint8_t acc_y_low;
- uint8_t acc_y_high;
- // EEPROM related
- bool CS_prev;
- bool CLK_prev;
- bool DI_prev;
- bool DO;
- bool instr_read;
- bool perf_instr;
- bool WR_EN;
- bool countdown_start;
- uint32_t instr_bit_counter;
- uint32_t instr;
- uint32_t EE_addr;
- uint32_t instr_case;
- uint32_t instr_clocks;
- uint32_t EE_value;
- uint32_t countdown;
-
-
-
- virtual uint8_t ReadMemory(uint32_t addr)
- {
- return 0;
- }
-
- virtual uint8_t PeekMemory(uint32_t addr)
- {
- return 0;
- }
-
- virtual void WriteMemory(uint32_t addr, uint8_t value)
- {
- }
-
- virtual void PokeMemory(uint32_t addr, uint8_t value)
- {
- }
-
-
- virtual void Dispose()
- {
- }
-
- virtual void Reset()
- {
- }
-
- virtual void Mapper_Tick()
- {
- }
-
- virtual void RTC_Get(int value, int index)
- {
- }
- /*
- virtual void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- }
-
- protected void SetCDLROM(LR35902.eCDLogMemFlags flags, int cdladdr)
- {
- Core.SetCDL(flags, "ROM", cdladdr);
- }
-
- protected void SetCDLRAM(LR35902.eCDLogMemFlags flags, int cdladdr)
- {
- Core.SetCDL(flags, "CartRAM", cdladdr);
- }
- */
-
- #pragma region State Save / Load
-
- uint8_t* SaveState(uint8_t* saver)
- {
- saver = bool_saver(RAM_enable, saver);
- saver = bool_saver(sel_mode, saver);
- saver = bool_saver(IR_signal, saver);
- saver = int_saver(ROM_bank, saver);
- saver = int_saver(RAM_bank, saver);
- saver = int_saver(ROM_mask, saver);
- saver = int_saver(RAM_mask, saver);
-
- saver = bool_saver(halt, saver);
- saver = int_saver(RTC_timer, saver);
- saver = int_saver(RTC_low_clock, saver);
-
- saver = bool_saver(timer_read, saver);
- saver = byte_saver(control, saver);
- saver = byte_saver(chip_read, saver);
- saver = int_saver(time_val_shift, saver);
- saver = int_saver(time, saver);
- saver = int_saver(RTC_seconds, saver);
-
- for (int i = 0; i < 5; i++) { saver = byte_saver(RTC_regs[i], saver); }
- for (int i = 0; i < 5; i++) { saver = byte_saver(RTC_regs_latch[i], saver); }
- saver = bool_saver(RTC_regs_latch_wr, saver);
- saver = int_saver(RTC_offset, saver);
-
- saver = bool_saver(regs_enable, saver);
- for (int i = 0; i < 5; i++) { saver = byte_saver(regs_cam[i], saver); }
-
- saver = bool_saver(locked, saver);
- saver = bool_saver(locked_GBC, saver);
- saver = bool_saver(finished, saver);
- saver = bool_saver(reg_access, saver);
- saver = int_saver(ROM_bank_mask, saver);
- saver = int_saver(BASE_ROM_Bank, saver);
- saver = int_saver(addr_last, saver);
- saver = int_saver(counter, saver);
-
- for (int i = 0; i < 10; i++) { saver = byte_saver(RTC_regs_TAMA[i], saver); }
- saver = byte_saver(Chip_return_low, saver);
- saver = byte_saver(Chip_return_high, saver);
- saver = int_saver(ctrl, saver);
- saver = int_saver(RAM_addr_low, saver);
- saver = int_saver(RAM_addr_high, saver);
- saver = int_saver(RAM_val_low, saver);
- saver = int_saver(RAM_val_high, saver);
-
- saver = bool_saver(RAM_enable_1, saver);
- saver = bool_saver(RAM_enable_2, saver);
- saver = bool_saver(is_erased, saver);
- saver = byte_saver(acc_x_low, saver);
- saver = byte_saver(acc_x_high, saver);
- saver = byte_saver(acc_y_low, saver);
- saver = byte_saver(acc_y_high, saver);
- // EEPROM related
- saver = bool_saver(CS_prev, saver);
- saver = bool_saver(CLK_prev, saver);
- saver = bool_saver(DI_prev, saver);
- saver = bool_saver(DO, saver);
- saver = bool_saver(instr_read, saver);
- saver = bool_saver(perf_instr, saver);
- saver = bool_saver(WR_EN, saver);
- saver = bool_saver(countdown_start, saver);
- saver = int_saver(instr_bit_counter, saver);
- saver = int_saver(instr, saver);
- saver = int_saver(EE_addr, saver);
- saver = int_saver(instr_case, saver);
- saver = int_saver(instr_clocks, saver);
- saver = int_saver(EE_value, saver);
- saver = int_saver(countdown, saver);
-
- return saver;
- }
-
- uint8_t* LoadState(uint8_t* loader)
- {
- loader = bool_loader(&RAM_enable, loader);
- loader = bool_loader(&sel_mode, loader);
- loader = bool_loader(&IR_signal, loader);
- loader = int_loader(&ROM_bank, loader);
- loader = int_loader(&RAM_bank, loader);
- loader = int_loader(&ROM_mask, loader);
- loader = int_loader(&RAM_mask, loader);
-
- loader = bool_loader(&halt, loader);
- loader = int_loader(&RTC_timer, loader);
- loader = int_loader(&RTC_low_clock, loader);
-
- loader = bool_loader(&timer_read, loader);
- loader = byte_loader(&control, loader);
- loader = byte_loader(&chip_read, loader);
- loader = int_loader(&time_val_shift, loader);
- loader = int_loader(&time, loader);
- loader = int_loader(&RTC_seconds, loader);
-
- for (int i = 0; i < 5; i++) { loader = byte_loader(&RTC_regs[i], loader); }
- for (int i = 0; i < 5; i++) { loader = byte_loader(&RTC_regs_latch[i], loader); }
- loader = bool_loader(&RTC_regs_latch_wr, loader);
- loader = int_loader(&RTC_offset, loader);
-
- loader = bool_loader(®s_enable, loader);
- for (int i = 0; i < 5; i++) { loader = byte_loader(®s_cam[i], loader); }
-
- loader = bool_loader(&locked, loader);
- loader = bool_loader(&locked_GBC, loader);
- loader = bool_loader(&finished, loader);
- loader = bool_loader(®_access, loader);
- loader = int_loader(&ROM_bank_mask, loader);
- loader = int_loader(&BASE_ROM_Bank, loader);
- loader = int_loader(&addr_last, loader);
- loader = int_loader(&counter, loader);
-
- for (int i = 0; i < 10; i++) { loader = byte_loader(&RTC_regs_TAMA[i], loader); }
- loader = byte_loader(&Chip_return_low, loader);
- loader = byte_loader(&Chip_return_high, loader);
- loader = int_loader(&ctrl, loader);
- loader = int_loader(&RAM_addr_low, loader);
- loader = int_loader(&RAM_addr_high, loader);
- loader = int_loader(&RAM_val_low, loader);
- loader = int_loader(&RAM_val_high, loader);
-
- loader = bool_loader(&RAM_enable_1, loader);
- loader = bool_loader(&RAM_enable_2, loader);
- loader = bool_loader(&is_erased, loader);
- loader = byte_loader(&acc_x_low, loader);
- loader = byte_loader(&acc_x_high, loader);
- loader = byte_loader(&acc_y_low, loader);
- loader = byte_loader(&acc_y_high, loader);
- // EEPROM related
- loader = bool_loader(&CS_prev, loader);
- loader = bool_loader(&CLK_prev, loader);
- loader = bool_loader(&DI_prev, loader);
- loader = bool_loader(&DO, loader);
- loader = bool_loader(&instr_read, loader);
- loader = bool_loader(&perf_instr, loader);
- loader = bool_loader(&WR_EN, loader);
- loader = bool_loader(&countdown_start, loader);
- loader = int_loader(&instr_bit_counter, loader);
- loader = int_loader(&instr, loader);
- loader = int_loader(&EE_addr, loader);
- loader = int_loader(&instr_case, loader);
- loader = int_loader(&instr_clocks, loader);
- loader = int_loader(&EE_value, loader);
- loader = int_loader(&countdown, loader);
-
- return loader;
- }
-
- uint8_t* bool_saver(bool to_save, uint8_t* saver)
- {
- *saver = (uint8_t)(to_save ? 1 : 0); saver++;
-
- return saver;
- }
-
- uint8_t* byte_saver(uint8_t to_save, uint8_t* saver)
- {
- *saver = to_save; saver++;
-
- return saver;
- }
-
- uint8_t* int_saver(uint32_t to_save, uint8_t* saver)
- {
- *saver = (uint8_t)(to_save & 0xFF); saver++; *saver = (uint8_t)((to_save >> 8) & 0xFF); saver++;
- *saver = (uint8_t)((to_save >> 16) & 0xFF); saver++; *saver = (uint8_t)((to_save >> 24) & 0xFF); saver++;
-
- return saver;
- }
-
- uint8_t* bool_loader(bool* to_load, uint8_t* loader)
- {
- to_load[0] = *to_load == 1; loader++;
-
- return loader;
- }
-
- uint8_t* byte_loader(uint8_t* to_load, uint8_t* loader)
- {
- to_load[0] = *loader; loader++;
-
- return loader;
- }
-
- uint8_t* int_loader(uint32_t* to_load, uint8_t* loader)
- {
- to_load[0] = *loader; loader++; to_load[0] |= (*loader << 8); loader++;
- to_load[0] |= (*loader << 16); loader++; to_load[0] |= (*loader << 24); loader++;
-
- return loader;
- }
-
- #pragma endregion
-
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_Camera.h b/libHawk/GBHawk/GBHawk/Mapper_Camera.h
deleted file mode 100644
index 683af45d2b..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_Camera.h
+++ /dev/null
@@ -1,148 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_Camera : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 1;
- RAM_bank = 0;
- RAM_enable = false;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
-
- RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
-
- regs_enable = false;
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- return ROM[addr];
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else
- {
- if (regs_enable)
- {
- if ((addr & 0x7F) == 0)
- {
- return 0;// regs[0];
- }
- else
- {
- return 0;
- }
- }
- else
- {
- if (/*RAM_enable && */(((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
- }
- else
- {
- return 0xFF;
- }
- }
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- // lowest bank is fixed, but is still effected by mode
- SetCDLROM(flags, addr);
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else
- {
- if (!regs_enable)
- {
- if ((((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
- }
- else
- {
- return;
- }
- }
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x8000)
- {
- if (addr < 0x2000)
- {
- RAM_enable = (value & 0xF) == 0xA;
- }
- else if (addr < 0x4000)
- {
- ROM_bank = value;
- ROM_bank &= ROM_mask;
- //Console.WriteLine(addr + " " + value + " " + ROM_mask + " " + ROM_bank);
- }
- else if (addr < 0x6000)
- {
- if ((value & 0x10) == 0x10)
- {
- regs_enable = true;
- }
- else
- {
- regs_enable = false;
- RAM_bank = value & RAM_mask;
- }
- }
- }
- else
- {
- if (regs_enable)
- {
- regs_cam[(addr & 0x7F)] = (uint8_t)(value & 0x7);
- }
- else
- {
- if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
- }
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_Default.h b/libHawk/GBHawk/GBHawk/Mapper_Default.h
deleted file mode 100644
index a81c2c1d39..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_Default.h
+++ /dev/null
@@ -1,87 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_Default : Mapper
- {
- public:
-
- void Reset()
- {
- // nothing to initialize
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x8000)
- {
- return ROM[addr];
- }
- else
- {
- if (Cart_RAM_Length > 0)
- {
- return Cart_RAM[addr - 0xA000];
- }
- else
- {
- return 0;
- }
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x8000)
- {
- SetCDLROM(flags, addr);
- }
- else
- {
- if (Cart_RAM != null)
- {
- SetCDLRAM(flags, addr - 0xA000);
- }
- else
- {
- return;
- }
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x8000)
- {
- // no mapping hardware available
- }
- else
- {
- if (Cart_RAM_Length > 0)
- {
- Cart_RAM[addr - 0xA000] = value;
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_HuC1.h b/libHawk/GBHawk/GBHawk/Mapper_HuC1.h
deleted file mode 100644
index ce459c465c..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_HuC1.h
+++ /dev/null
@@ -1,180 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_HuC1 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 0;
- RAM_bank = 0;
- RAM_enable = false;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
-
- // some games have sizes that result in a degenerate ROM, account for it here
- if (ROM_mask > 4) { ROM_mask |= 3; }
-
- RAM_mask = 0;
- if (Cart_RAM_Length[0] > 0)
- {
- RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
- if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
- }
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- return ROM[addr];
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else if ((addr >= 0xA000) && (addr < 0xC000))
- {
- if (RAM_enable)
- {
- if (Cart_RAM_Length[0] > 0)
- {
- if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
- {
- return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
- }
- else
- {
- return 0xFF;
- }
- }
- else
- {
- return 0xFF;
- }
- }
- else
- {
- // when RAM isn't enabled, reading from this area will return IR sensor reading
- // for now we'll assume it never sees light (0xC0)
- return 0xC0;
- }
- }
- else
- {
- return 0xFF;
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- SetCDLROM(flags, addr);
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else if ((addr >= 0xA000) && (addr < 0xC000))
- {
- if (RAM_enable)
- {
- if (Cart_RAM != null)
- {
- if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
- {
- SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
- }
- else
- {
- return;
- }
- }
- else
- {
- return;
- }
- }
- else
- {
- return;
- }
- }
- else
- {
- return;
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x8000)
- {
- if (addr < 0x2000)
- {
- RAM_enable = (value & 0xF) != 0xE;
- }
- else if (addr < 0x4000)
- {
- value &= 0x3F;
-
- ROM_bank &= 0xC0;
- ROM_bank |= value;
- ROM_bank &= ROM_mask;
- }
- else if (addr < 0x6000)
- {
- RAM_bank = value & 3;
- RAM_bank &= RAM_mask;
- }
- }
- else
- {
- if (RAM_enable)
- {
- if (Cart_RAM_Length[0] > 0)
- {
- if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
- {
- Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
- }
- }
- }
- else
- {
- // I don't know if other bits here have an effect
- if (value == 1)
- {
- IR_signal = true;
- }
- else if (value == 0)
- {
- IR_signal = false;
- }
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_HuC3.h b/libHawk/GBHawk/GBHawk/Mapper_HuC3.h
deleted file mode 100644
index 701516f527..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_HuC3.h
+++ /dev/null
@@ -1,278 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_HuC3 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 0;
- RAM_bank = 0;
- RAM_enable = false;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
- control = 0;
- chip_read = 1;
- timer_read = false;
- time_val_shift = 0;
-
- // some games have sizes that result in a degenerate ROM, account for it here
- if (ROM_mask > 4) { ROM_mask |= 3; }
-
- RAM_mask = 0;
- if (Cart_RAM_Length[0] > 0)
- {
- RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
- if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
- }
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- return ROM[addr];
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else if ((addr >= 0xA000) && (addr < 0xC000))
- {
- if ((control >= 0xB) && (control < 0xE))
- {
- if (control == 0xD)
- {
- return 1;
- }
- return chip_read;
- }
-
- if (RAM_enable)
- {
- if (Cart_RAM_Length[0] > 0)
- {
- if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
- {
- return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
- }
- else
- {
- return 0xFF;
- }
- }
- else
- {
- return 0xFF;
- }
- }
- else
- {
- // what to return if RAM not enabled and controller not selected?
- return 0xFF;
- }
- }
- else
- {
- return 0xFF;
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- SetCDLROM(flags, addr);
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else if ((addr >= 0xA000) && (addr < 0xC000))
- {
- if (RAM_enable)
- {
- if (Cart_RAM != null)
- {
- if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
- {
- SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
- }
- else
- {
- return;
- }
- }
- else
- {
- return;
- }
- }
- else
- {
- return;
- }
- }
- else
- {
- return;
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x8000)
- {
- if (addr < 0x2000)
- {
- RAM_enable = (value & 0xA) == 0xA;
- control = value;
- }
- else if (addr < 0x4000)
- {
- if (value == 0) { value = 1; }
-
- ROM_bank = value;
- ROM_bank &= ROM_mask;
- }
- else if (addr < 0x6000)
- {
- RAM_bank = value;
- RAM_bank &= 0xF;
- RAM_bank &= RAM_mask;
- }
- }
- else
- {
- if (RAM_enable && ((control < 0xB) || (control > 0xE)))
- {
- if (Cart_RAM_Length[0] > 0)
- {
- if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
- {
- Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
- }
- }
- }
-
- if (control == 0xB)
- {
- switch (value & 0xF0)
- {
- case 0x10:
- if (timer_read)
- {
- // return timer value
- chip_read = (uint8_t)((time >> time_val_shift) & 0xF);
- time_val_shift += 4;
- if (time_val_shift == 28) { time_val_shift = 0; }
- }
- break;
- case 0x20:
- break;
- case 0x30:
- if (!timer_read)
- {
- // write to timer
- if (time_val_shift == 0) { time = 0; }
- if (time_val_shift < 28)
- {
- time |= (uint32_t)((value & 0x0F) << time_val_shift);
- time_val_shift += 4;
- if (time_val_shift == 28) { timer_read = true; }
- }
- }
- break;
- case 0x40:
- // other commands
- switch (value & 0xF)
- {
- case 0x0:
- time_val_shift = 0;
- break;
- case 0x3:
- timer_read = false;
- time_val_shift = 0;
- break;
- case 0x7:
- timer_read = true;
- time_val_shift = 0;
- break;
- case 0xF:
- break;
- }
- break;
- case 0x50:
- break;
- case 0x60:
- timer_read = true;
- break;
- }
- }
- else if (control == 0xC)
- {
- // maybe IR
- }
- else if (control == 0xD)
- {
- // maybe IR
- }
- }
- }
-
- void RTC_Get(uint32_t value, uint32_t index)
- {
- time |= (uint32_t)((value & 0xFF) << index);
- }
-
- void Mapper_Tick()
- {
- RTC_timer++;
-
- if (RTC_timer == 128)
- {
- RTC_timer = 0;
-
- RTC_low_clock++;
-
- if (RTC_low_clock == 32768)
- {
- RTC_low_clock = 0;
-
- RTC_seconds++;
- if (RTC_seconds > 59)
- {
- RTC_seconds = 0;
- time++;
- if ((time & 0xFFF) > 1439)
- {
- time -= 1440;
- time += (1 << 12);
- if ((time >> 12) > 365)
- {
- time -= (365 << 12);
- time += (1 << 24);
- }
- }
- }
- }
- }
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_MBC1.h b/libHawk/GBHawk/GBHawk/Mapper_MBC1.h
deleted file mode 100644
index 70f2ffa303..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_MBC1.h
+++ /dev/null
@@ -1,186 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_MBC1 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 1;
- RAM_bank = 0;
- RAM_enable = false;
- sel_mode = false;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
-
- // some games have sizes that result in a degenerate ROM, account for it here
- if (ROM_mask > 4) { ROM_mask |= 3; }
-
- RAM_mask = 0;
- if (Cart_RAM_Length[0] > 0)
- {
- RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
- if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
- }
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- // lowest bank is fixed, but is still effected by mode
- if (sel_mode)
- {
- return ROM[(ROM_bank & 0x60) * 0x4000 + addr];
- }
- else
- {
- return ROM[addr];
- }
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else
- {
- if (Cart_RAM_Length[0] > 0)
- {
- if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
- }
- else
- {
- return 0xFF;
- }
-
- }
- else
- {
- return 0xFF;
- }
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- // lowest bank is fixed, but is still effected by mode
- if (sel_mode)
- {
- SetCDLROM(flags, (ROM_bank & 0x60) * 0x4000 + addr);
- }
- else
- {
- SetCDLROM(flags, addr);
- }
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else
- {
- if (Cart_RAM != null)
- {
- if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
- }
- else
- {
- return;
- }
-
- }
- else
- {
- return;
- }
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x8000)
- {
- if (addr < 0x2000)
- {
- RAM_enable = (value & 0xF) == 0xA;
- }
- else if (addr < 0x4000)
- {
- value &= 0x1F;
-
- // writing zero gets translated to 1
- if (value == 0) { value = 1; }
-
- ROM_bank &= 0xE0;
- ROM_bank |= value;
- ROM_bank &= ROM_mask;
- }
- else if (addr < 0x6000)
- {
- if (sel_mode && (Cart_RAM_Length[0] > 0))
- {
- RAM_bank = value & 3;
- RAM_bank &= RAM_mask;
- }
- else
- {
- ROM_bank &= 0x1F;
- ROM_bank |= ((value & 3) << 5);
- ROM_bank &= ROM_mask;
- }
- }
- else
- {
- sel_mode = (value & 1) > 0;
-
- if (sel_mode && (Cart_RAM_Length[0] > 0))
- {
- ROM_bank &= 0x1F;
- ROM_bank &= ROM_mask;
- }
- else
- {
- RAM_bank = 0;
- }
- }
- }
- else
- {
- if (Cart_RAM_Length[0] > 0)
- {
- if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
- }
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_MBC1_Multi.h b/libHawk/GBHawk/GBHawk/Mapper_MBC1_Multi.h
deleted file mode 100644
index d829666a91..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_MBC1_Multi.h
+++ /dev/null
@@ -1,182 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_MBC1_Multi : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 1;
- RAM_bank = 0;
- RAM_enable = false;
- sel_mode = false;
- ROM_mask = (ROM_Length[0] / 0x4000 * 2) - 1; // due to how mapping works, we want a 1 bit higher mask
- RAM_mask = 0;
- if (Cart_RAM_Length[0] > 0)
- {
- RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
- if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
- }
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- // lowest bank is fixed, but is still effected by mode
- if (sel_mode)
- {
- return ROM[((ROM_bank & 0x60) >> 1) * 0x4000 + addr];
- }
- else
- {
- return ROM[addr];
- }
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + (((ROM_bank & 0x60) >> 1) | (ROM_bank & 0xF)) * 0x4000];
- }
- else
- {
- if (Cart_RAM_Length[0] > 0)
- {
- if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
- }
- else
- {
- return 0xFF;
- }
-
- }
- else
- {
- return 0;
- }
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- // lowest bank is fixed, but is still effected by mode
- if (sel_mode)
- {
- SetCDLROM(flags, ((ROM_bank & 0x60) >> 1) * 0x4000 + addr);
- }
- else
- {
- SetCDLROM(flags, addr);
- }
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + (((ROM_bank & 0x60) >> 1) | (ROM_bank & 0xF)) * 0x4000);
- }
- else
- {
- if (Cart_RAM != null)
- {
- if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
- }
- else
- {
- return;
- }
-
- }
- else
- {
- return;
- }
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x8000)
- {
- if (addr < 0x2000)
- {
- RAM_enable = ((value & 0xA) == 0xA);
- }
- else if (addr < 0x4000)
- {
- value &= 0x1F;
-
- // writing zero gets translated to 1
- if (value == 0) { value = 1; }
-
- ROM_bank &= 0xE0;
- ROM_bank |= value;
- ROM_bank &= ROM_mask;
- }
- else if (addr < 0x6000)
- {
- if (sel_mode && (Cart_RAM_Length[0] > 0))
- {
- RAM_bank = value & 3;
- RAM_bank &= RAM_mask;
- }
- else
- {
- ROM_bank &= 0x1F;
- ROM_bank |= ((value & 3) << 5);
- ROM_bank &= ROM_mask;
- }
- }
- else
- {
- sel_mode = (value & 1) > 0;
-
- if (sel_mode && (Cart_RAM_Length[0] > 0))
- {
- ROM_bank &= 0x1F;
- ROM_bank &= ROM_mask;
- }
- else
- {
- RAM_bank = 0;
- }
- }
- }
- else
- {
- if (Cart_RAM_Length[0] > 0)
- {
- if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
- }
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_MBC2.h b/libHawk/GBHawk/GBHawk/Mapper_MBC2.h
deleted file mode 100644
index 9df42496b4..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_MBC2.h
+++ /dev/null
@@ -1,111 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_MBC2 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 1;
- RAM_bank = 0;
- RAM_enable = false;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- return ROM[addr];
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else if ((addr >= 0xA000) && (addr < 0xA200))
- {
- if (RAM_enable)
- {
- return Cart_RAM[addr - 0xA000];
- }
- return 0xFF;
- }
- else
- {
- return 0xFF;
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- SetCDLROM(flags, addr);
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else if ((addr >= 0xA000) && (addr < 0xA200))
- {
- if (RAM_enable)
- {
- SetCDLRAM(flags, addr - 0xA000);
- }
- return;
- }
- else
- {
- return;
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x2000)
- {
- if ((addr & 0x100) == 0)
- {
- RAM_enable = ((value & 0xA) == 0xA);
- }
- }
- else if (addr < 0x4000)
- {
- if ((addr & 0x100) > 0)
- {
- ROM_bank = value & 0xF & ROM_mask;
- if (ROM_bank==0) { ROM_bank = 1; }
- }
- }
- else if ((addr >= 0xA000) && (addr < 0xA200))
- {
- if (RAM_enable)
- {
- Cart_RAM[addr - 0xA000] = (uint8_t)(value & 0xF);
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
\ No newline at end of file
diff --git a/libHawk/GBHawk/GBHawk/Mapper_MBC3.h b/libHawk/GBHawk/GBHawk/Mapper_MBC3.h
deleted file mode 100644
index 09c2532e37..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_MBC3.h
+++ /dev/null
@@ -1,266 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_MBC3 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 1;
- RAM_bank = 0;
- RAM_enable = false;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
-
- // some games have sizes that result in a degenerate ROM, account for it here
- if (ROM_mask > 4) { ROM_mask |= 3; }
-
- RAM_mask = 0;
- if (Cart_RAM_Length[0] > 0)
- {
- RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
- if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
- }
-
- RTC_regs_latch[0] = 0;
- RTC_regs_latch[1] = 0;
- RTC_regs_latch[2] = 0;
- RTC_regs_latch[3] = 0;
- RTC_regs_latch[4] = 0;
-
- RTC_regs_latch_wr = true;
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- return ROM[addr];
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else
- {
- if (RAM_enable)
- {
- if ((Cart_RAM_Length[0] > 0) && (RAM_bank <= RAM_mask))
- {
- if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
- {
- return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
- }
- else
- {
- return 0xFF;
- }
- }
-
- if ((RAM_bank >= 8) && (RAM_bank <= 0xC))
- {
- //Console.WriteLine("reg: " + (RAM_bank - 8) + " value: " + RTC_regs_latch[RAM_bank - 8] + " cpu: " + Core.cpu.TotalExecutedCycles);
- return RTC_regs_latch[RAM_bank - 8];
- }
- else
- {
- return 0x0;
- }
- }
- else
- {
- return 0x0;
- }
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- SetCDLROM(flags, addr);
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else
- {
- if (RAM_enable)
- {
- if ((Cart_RAM != null) && (RAM_bank <= RAM_mask))
- {
- if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
- {
- SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
- }
- else
- {
- return;
- }
- }
-
- if ((RAM_bank >= 8) && (RAM_bank <= 0xC))
- {
- return;
- }
- else
- {
- return;
- }
- }
- else
- {
- return;
- }
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x8000)
- {
- if (addr < 0x2000)
- {
- RAM_enable = ((value & 0xA) == 0xA);
- }
- else if (addr < 0x4000)
- {
- value &= 0x7F;
-
- // writing zero gets translated to 1
- if (value == 0) { value = 1; }
-
- ROM_bank = value;
- ROM_bank &= ROM_mask;
- }
- else if (addr < 0x6000)
- {
- RAM_bank = value;
- }
- else
- {
- if (!RTC_regs_latch_wr && ((value & 1) == 1))
- {
- for (uint32_t i = 0; i < 5; i++)
- {
- RTC_regs_latch[i] = RTC_regs[i];
- }
- }
-
- RTC_regs_latch_wr = (value & 1) > 0;
- }
- }
- else
- {
- if (RAM_enable)
- {
- if ((Cart_RAM_Length[0] > 0) && (RAM_bank <= RAM_mask))
- {
- if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
- {
- Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
- }
- }
- else if ((RAM_bank >= 8) && (RAM_bank <= 0xC))
- {
- RTC_regs[RAM_bank - 8] = value;
-
- if ((RAM_bank - 8) == 0) { RTC_low_clock = RTC_timer = 0; }
-
- halt = (RTC_regs[4] & 0x40) > 0;
- }
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
-
- void RTC_Get(uint32_t value, uint32_t index)
- {
- if (index < 5)
- {
- RTC_regs[index] = (uint8_t)value;
- }
- else
- {
- RTC_offset = value;
- }
- }
-
- void Mapper_Tick()
- {
- if (!halt)
- {
- RTC_timer++;
-
- if (RTC_timer == 128)
- {
- RTC_timer = 0;
-
- RTC_low_clock++;
-
- if (RTC_low_clock == 32768)
- {
- RTC_low_clock = 0;
- RTC_timer = RTC_offset;
-
- RTC_regs[0]++;
-
- if (RTC_regs[0] > 59)
- {
- RTC_regs[0] = 0;
- RTC_regs[1]++;
- if (RTC_regs[1] > 59)
- {
- RTC_regs[1] = 0;
- RTC_regs[2]++;
- if (RTC_regs[2] > 23)
- {
- RTC_regs[2] = 0;
- if (RTC_regs[3] < 0xFF)
- {
- RTC_regs[3]++;
- }
- else
- {
- RTC_regs[3] = 0;
-
- if ((RTC_regs[4] & 1) == 0)
- {
- RTC_regs[4] |= 1;
- }
- else
- {
- RTC_regs[4] &= 0xFE;
- RTC_regs[4] |= 0x80;
- }
- }
- }
- }
- }
- }
- }
- }
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_MBC5.h b/libHawk/GBHawk/GBHawk/Mapper_MBC5.h
deleted file mode 100644
index 204c539853..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_MBC5.h
+++ /dev/null
@@ -1,152 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_MBC5 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 1;
- RAM_bank = 0;
- RAM_enable = false;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
-
- // some games have sizes that result in a degenerate ROM, account for it here
- if (ROM_mask > 4) { ROM_mask |= 3; }
- if (ROM_mask > 0x100) { ROM_mask |= 0xFF; }
-
- RAM_mask = 0;
- if (Cart_RAM_Length[0] > 0)
- {
- RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
- if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
- }
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- return ROM[addr];
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else
- {
- if (Cart_RAM_Length[0] > 0)
- {
- if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
- }
- else
- {
- return 0xFF;
- }
-
- }
- else
- {
- return 0xFF;
- }
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- SetCDLROM(flags, addr);
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else
- {
- if (Cart_RAM != null)
- {
- if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
- }
- else
- {
- return;
- }
-
- }
- else
- {
- return;
- }
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x8000)
- {
- if (addr < 0x2000)
- {
- RAM_enable = (value & 0xF) == 0xA;
- }
- else if (addr < 0x3000)
- {
- value &= 0xFF;
-
- ROM_bank &= 0x100;
- ROM_bank |= value;
- ROM_bank &= ROM_mask;
- }
- else if (addr < 0x4000)
- {
- value &= 1;
-
- ROM_bank &= 0xFF;
- ROM_bank |= (value << 8);
- ROM_bank &= ROM_mask;
- }
- else if (addr < 0x6000)
- {
- RAM_bank = value & 0xF;
- RAM_bank &= RAM_mask;
- }
- }
- else
- {
- if (Cart_RAM_Length[0] > 0)
- {
- if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
- {
- Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
- }
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_MBC6.h b/libHawk/GBHawk/GBHawk/Mapper_MBC6.h
deleted file mode 100644
index 8accdb045f..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_MBC6.h
+++ /dev/null
@@ -1,87 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_MBC6 : Mapper
- {
- public:
-
- void Reset()
- {
- // nothing to initialize
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x8000)
- {
- return ROM[addr];
- }
- else
- {
- if (Cart_RAM_Length[0] > 0)
- {
- return Cart_RAM[addr - 0xA000];
- }
- else
- {
- return 0;
- }
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x8000)
- {
- SetCDLROM(flags, addr);
- }
- else
- {
- if (Cart_RAM != null)
- {
- SetCDLRAM(flags, addr - 0xA000);
- }
- else
- {
- return;
- }
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x8000)
- {
- // no mapping hardware available
- }
- else
- {
- if (Cart_RAM_Length[0] > 0)
- {
- Cart_RAM[addr - 0xA000] = value;
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_MBC7.h b/libHawk/GBHawk/GBHawk/Mapper_MBC7.h
deleted file mode 100644
index 972e41fdd6..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_MBC7.h
+++ /dev/null
@@ -1,432 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_MBC7 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 1;
- RAM_enable_1 = RAM_enable_2 = false;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
-
- // some games have sizes that result in a degenerate ROM, account for it here
- if (ROM_mask > 4) { ROM_mask |= 3; }
-
- acc_x_low = 0;
- acc_x_high = 0x80;
- acc_y_low = 0;
- acc_y_high = 0x80;
-
- // reset acceerometer
- is_erased = false;
-
- // EEPROM related
- CS_prev = CLK_prev = DI_prev = DO = instr_read = perf_instr = WR_EN = countdown_start = false;
- instr_bit_counter = instr = EE_addr = instr_case = instr_clocks = EE_value = countdown = 0;
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- return ROM[addr];
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else if (addr < 0xA000)
- {
- return 0xFF;
- }
- else if (addr < 0xB000)
- {
- if (RAM_enable_1 && RAM_enable_2)
- {
- return Register_Access_Read(addr);
- }
- else
- {
- return 0xFF;
- }
- }
- else
- {
- return 0xFF;
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- SetCDLROM(flags, addr);
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else if (addr < 0xA000)
- {
- return;
- }
- else if (addr < 0xB000)
- {
- if (RAM_enable_1 && RAM_enable_2)
- {
- return;
- }
- else
- {
- return;
- }
- }
- else
- {
- return;
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0xA000)
- {
- if (addr < 0x2000)
- {
- RAM_enable_1 = (value & 0xF) == 0xA;
- }
- else if (addr < 0x4000)
- {
- value &= 0xFF;
-
- //Console.WriteLine(Core.cpu.TotalExecutedCycles);
- //Console.WriteLine(value);
-
- ROM_bank &= 0x100;
- ROM_bank |= value;
- ROM_bank &= ROM_mask;
- }
- else if (addr < 0x6000)
- {
- RAM_enable_2 = (value & 0xF0) == 0x40;
- }
- }
- else
- {
- if (RAM_enable_1 && RAM_enable_2)
- {
- Register_Access_Write(addr, value);
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
-
- uint8_t Register_Access_Read(uint32_t addr)
- {
- if ((addr & 0xA0F0) == 0xA000)
- {
- return 0xFF;
- }
- else if ((addr & 0xA0F0) == 0xA010)
- {
- return 0xFF;
- }
- else if ((addr & 0xA0F0) == 0xA020)
- {
- return acc_x_low;
- }
- else if ((addr & 0xA0F0) == 0xA030)
- {
- return acc_x_high;
- }
- else if ((addr & 0xA0F0) == 0xA040)
- {
- return acc_y_low;
- }
- else if ((addr & 0xA0F0) == 0xA050)
- {
- return acc_y_high;
- }
- else if ((addr & 0xA0F0) == 0xA060)
- {
- return 0xFF;
- }
- else if ((addr & 0xA0F0) == 0xA070)
- {
- return 0xFF;
- }
- else if ((addr & 0xA0F0) == 0xA080)
- {
- return (uint8_t)((CS_prev ? 0x80 : 0) |
- (CLK_prev ? 0x40 : 0) |
- (DI_prev ? 2 : 0) |
- (DO ? 1 : 0));
- }
- else
- {
- return 0xFF;
- }
- }
-
- void Register_Access_Write(uint32_t addr, uint8_t value)
- {
- if ((addr & 0xA0F0) == 0xA000)
- {
- if (value == 0x55)
- {
- //Console.WriteLine("Erasing ACC");
-
- is_erased = true;
- acc_x_low = 0x00;
- acc_x_high = 0x80;
- acc_y_low = 0x00;
- acc_y_high = 0x80;
- }
- }
- else if ((addr & 0xA0F0) == 0xA010)
- {
- if ((value == 0xAA) && is_erased)
- {
- // latch new accelerometer values
- //Console.WriteLine("Latching ACC");
- acc_x_low = (uint8_t)(Acc_X_state[0] & 0xFF);
- acc_x_high = (uint8_t)((Acc_X_state[0] & 0xFF00) >> 8);
- acc_y_low = (uint8_t)(Acc_Y_state[0] & 0xFF);
- acc_y_high = (uint8_t)((Acc_Y_state[0] & 0xFF00) >> 8);
- }
- }
- else if ((addr & 0xA0F0) == 0xA080)
- {
- // EEPROM writes
- EEPROM_write(value);
- }
- }
-
- void EEPROM_write(uint8_t value)
- {
- bool CS = (value & 0x80) > 0;
- bool CLK = (value & 0x40) > 0;
- bool DI = (value & 0x2) > 0;
-
- // if we deselect the chip, complete instructions or countdown and stop
- if (!CS)
- {
- CS_prev = CS;
- CLK_prev = CLK;
- DI_prev = DI;
-
- DO = true;
- countdown_start = false;
- perf_instr = false;
- instr_read = false;
-
- //Console.Write("Chip De-selected: ");
- //Console.WriteLine(Core.cpu.TotalExecutedCycles);
- }
-
- if (!instr_read && !perf_instr)
- {
- // if we aren't performing an operation or reading an incoming instruction, we are waiting for one
- // this is signalled by CS and DI both being 1 while CLK goes from 0 to 1
- if (CLK && !CLK_prev && DI && CS)
- {
- instr_read = true;
- instr_bit_counter = 0;
- instr = 0;
- DO = false;
- //Console.Write("Initiating command: ");
- //Console.WriteLine(Core.cpu.TotalExecutedCycles);
- }
- }
- else if (instr_read && CLK && !CLK_prev)
- {
- // all instructions are 10 bits long
- instr = (instr << 1) | ((value & 2) >> 1);
-
- instr_bit_counter++;
- if (instr_bit_counter == 10)
- {
- instr_read = false;
- instr_clocks = 0;
- EE_addr = instr & 0x7F;
- EE_value = 0;
-
- switch (instr & 0x300)
- {
- case 0x0:
- switch (instr & 0xC0)
- {
- case 0x0: // disable writes
- instr_case = 0;
- WR_EN = false;
- DO = true;
- break;
- case 0x40: // fill mem with value
- instr_case = 1;
- perf_instr = true;
- break;
- case 0x80: // fill mem with FF
- instr_case = 2;
- if (WR_EN)
- {
- for (uint32_t i = 0; i < 256; i++)
- {
- Cart_RAM[i] = 0xFF;
- }
- }
- DO = true;
- break;
- case 0xC0: // enable writes
- instr_case = 3;
- WR_EN = true;
- DO = true;
- break;
- }
- break;
- case 0x100: // write to address
- instr_case = 4;
- perf_instr = true;
- break;
- case 0x200: // read from address
- instr_case = 5;
- perf_instr = true;
- break;
- case 0x300: // set address to FF
- instr_case = 6;
- if (WR_EN)
- {
- Cart_RAM[EE_addr * 2] = 0xFF;
- Cart_RAM[EE_addr * 2 + 1] = 0xFF;
- }
- DO = true;
- break;
- }
-
- //Console.Write("Selected Command: ");
- //Console.Write(instr_case);
- //Console.Write(" ");
- //Console.WriteLine(Core.cpu.TotalExecutedCycles);
- }
- }
- else if (perf_instr && CLK && !CLK_prev)
- {
- //Console.Write("Command In progress, Cycle: ");
- //Console.Write(instr_clocks);
- //Console.Write(" ");
- //Console.WriteLine(Core.cpu.TotalExecutedCycles);
-
- // for commands that require additional clocking
- switch (instr_case)
- {
- case 1:
- EE_value = (EE_value << 1) | ((value & 2) >> 1);
-
- if (instr_clocks == 15)
- {
- if (WR_EN)
- {
- for (uint32_t i = 0; i < 128; i++)
- {
- Cart_RAM[i * 2] = (uint8_t)(EE_value & 0xFF);
- Cart_RAM[i * 2 + 1] = (uint8_t)((EE_value & 0xFF00) >> 8);
- }
- }
- instr_case = 7;
- countdown = 8;
- }
- break;
-
- case 4:
- EE_value = (EE_value << 1) | ((value & 2) >> 1);
-
- if (instr_clocks == 15)
- {
- if (WR_EN)
- {
- Cart_RAM[EE_addr * 2] = (uint8_t)(EE_value & 0xFF);
- Cart_RAM[EE_addr * 2 + 1] = (uint8_t)((EE_value & 0xFF00) >> 8);
- }
- instr_case = 7;
- countdown = 8;
- }
- break;
-
- case 5:
- if ((instr_clocks >= 0) && (instr_clocks <= 7))
- {
- DO = ((Cart_RAM[EE_addr * 2 + 1] >> (7 - instr_clocks)) & 1) == 1;
- }
- else if ((instr_clocks >= 8) && (instr_clocks <= 15))
- {
- DO = ((Cart_RAM[EE_addr * 2] >> (15 - instr_clocks)) & 1) == 1;
- }
-
- if (instr_clocks == 15)
- {
- instr_case = 7;
- countdown = 8;
- }
- break;
-
- case 6:
-
- instr_case = 7;
- countdown = 8;
- break;
-
- case 7:
- // completed operations take time, so countdown a bit here.
- // not cycle accurate for operations like writing to all of the EEPROM, but good enough
-
- break;
- }
-
- if (instr_case == 7)
- {
- perf_instr = false;
- countdown_start = true;
- }
-
- instr_clocks++;
- }
- else if (countdown_start)
- {
- countdown--;
- if (countdown == 0)
- {
- countdown_start = false;
- DO = true;
-
- //Console.Write("Command Complete: ");
- //Console.WriteLine(Core.cpu.TotalExecutedCycles);
- }
- }
-
- CS_prev = CS;
- CLK_prev = CLK;
- DI_prev = DI;
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_MMM01.h b/libHawk/GBHawk/GBHawk/Mapper_MMM01.h
deleted file mode 100644
index ed1fe9bcab..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_MMM01.h
+++ /dev/null
@@ -1,87 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_MMM01 : Mapper
- {
- public:
-
- void Reset()
- {
- // nothing to initialize
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x8000)
- {
- return ROM[addr];
- }
- else
- {
- if (Cart_RAM_Length[0] > 0)
- {
- return Cart_RAM[addr - 0xA000];
- }
- else
- {
- return 0;
- }
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x8000)
- {
- SetCDLROM(flags, addr);
- }
- else
- {
- if (Cart_RAM != null)
- {
- SetCDLRAM(flags, addr - 0xA000);
- }
- else
- {
- return;
- }
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x8000)
- {
- // no mapping hardware available
- }
- else
- {
- if (Cart_RAM_Length[0] > 0)
- {
- Cart_RAM[addr - 0xA000] = value;
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_RockMan8.h b/libHawk/GBHawk/GBHawk/Mapper_RockMan8.h
deleted file mode 100644
index 5d719633b2..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_RockMan8.h
+++ /dev/null
@@ -1,87 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_RM8 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 1;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
-
- // some games have sizes that result in a degenerate ROM, account for it here
- if (ROM_mask > 4) { ROM_mask |= 3; }
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- // lowest bank is fixed
- return ROM[addr];
-
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else
- {
- return 0xFF;
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- // lowest bank is fixed
- SetCDLROM(flags, addr);
-
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else
- {
- return;
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if ((addr >= 0x2000) && (addr < 0x4000))
- {
- value &= 0x1F;
-
- if (value == 0) { value = 1; }
-
- // in hhugboy they just subtract 8, but to me looks like bits 4 and 5 are just swapped (and bit 4 is unused?)
- ROM_bank = ((value & 0xF) | ((value & 0x10) >> 1))& ROM_mask;
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_Sachen_MMC1.h b/libHawk/GBHawk/GBHawk/Mapper_Sachen_MMC1.h
deleted file mode 100644
index 1e08fe778d..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_Sachen_MMC1.h
+++ /dev/null
@@ -1,167 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_Sachen1 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 1;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
- BASE_ROM_Bank = 0;
- ROM_bank_mask = 0xFF;
- locked = true;
- reg_access = false;
- addr_last = 0;
- counter = 0;
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- if (locked)
- {
- // header is scrambled
- if ((addr >= 0x100) && (addr < 0x200))
- {
- uint32_t temp0 = (addr & 1);
- uint32_t temp1 = (addr & 2);
- uint32_t temp4 = (addr & 0x10);
- uint32_t temp6 = (addr & 0x40);
-
- temp0 = temp0 << 6;
- temp1 = temp1 << 3;
- temp4 = temp4 >> 3;
- temp6 = temp6 >> 6;
-
- addr &= 0x1AC;
- addr |= (uint32_t)(temp0 | temp1 | temp4 | temp6);
- }
- addr |= 0x80;
- }
-
- return ROM[addr + BASE_ROM_Bank * 0x4000];
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else
- {
- return 0xFF;
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- if (locked)
- {
- // header is scrambled
- if ((addr >= 0x100) && (addr < 0x200))
- {
- uint32_t temp0 = (addr & 1);
- uint32_t temp1 = (addr & 2);
- uint32_t temp4 = (addr & 0x10);
- uint32_t temp6 = (addr & 0x40);
-
- temp0 = temp0 << 6;
- temp1 = temp1 << 3;
- temp4 = temp4 >> 3;
- temp6 = temp6 >> 6;
-
- addr &= 0x1AC;
- addr |= (uint32_t)(temp0 | temp1 | temp4 | temp6);
- }
- addr |= 0x80;
- }
-
- SetCDLROM(flags, addr + BASE_ROM_Bank * 0x4000);
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else
- {
- return;
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x2000)
- {
- if (reg_access)
- {
- BASE_ROM_Bank = value;
- }
- }
- else if (addr < 0x4000)
- {
- ROM_bank = (value > 0) ? value : 1;
-
- if ((value & 0x30) == 0x30)
- {
- reg_access = true;
- }
- else
- {
- reg_access = false;
- }
- }
- else if (addr < 0x6000)
- {
- if (reg_access)
- {
- ROM_bank_mask = value;
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
-
- void Mapper_Tick()
- {
- if (locked)
- {
- if (((addr_access[0] & 0x8000) == 0) && ((addr_last & 0x8000) > 0) && (addr_access[0] >= 0x100))
- {
- counter++;
- }
-
- if (addr_access[0] >= 0x100)
- {
- addr_last = addr_access[0];
- }
-
- if (counter == 0x30)
- {
- locked = false;
- }
- }
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_Sachen_MMC2.h b/libHawk/GBHawk/GBHawk/Mapper_Sachen_MMC2.h
deleted file mode 100644
index 9d688a86d3..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_Sachen_MMC2.h
+++ /dev/null
@@ -1,200 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_Sachen2 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 1;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
- BASE_ROM_Bank = 0;
- ROM_bank_mask = 0;
- locked = true;
- locked_GBC = false;
- finished = false;
- reg_access = false;
- addr_last = 0;
- counter = 0;
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- // header is scrambled
- if ((addr >= 0x100) && (addr < 0x200))
- {
- uint32_t temp0 = (addr & 1);
- uint32_t temp1 = (addr & 2);
- uint32_t temp4 = (addr & 0x10);
- uint32_t temp6 = (addr & 0x40);
-
- temp0 = temp0 << 6;
- temp1 = temp1 << 3;
- temp4 = temp4 >> 3;
- temp6 = temp6 >> 6;
-
- addr &= 0x1AC;
- addr |= (uint32_t)(temp0 | temp1 | temp4 | temp6);
- }
-
- if (locked_GBC) { addr |= 0x80; }
-
- return ROM[addr + BASE_ROM_Bank * 0x4000];
- }
- else if (addr < 0x8000)
- {
- uint32_t temp_bank = (ROM_bank & ~ROM_bank_mask) | (ROM_bank_mask & BASE_ROM_Bank);
- temp_bank &= ROM_mask;
-
- return ROM[(addr - 0x4000) + temp_bank * 0x4000];
- }
- else
- {
- return 0xFF;
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- // header is scrambled
- if ((addr >= 0x100) && (addr < 0x200))
- {
- uint32_t temp0 = (addr & 1);
- uint32_t temp1 = (addr & 2);
- uint32_t temp4 = (addr & 0x10);
- uint32_t temp6 = (addr & 0x40);
-
- temp0 = temp0 << 6;
- temp1 = temp1 << 3;
- temp4 = temp4 >> 3;
- temp6 = temp6 >> 6;
-
- addr &= 0x1AC;
- addr |= (uint32_t)(temp0 | temp1 | temp4 | temp6);
- }
-
- if (locked_GBC) { addr |= 0x80; }
-
- SetCDLROM(flags, addr + BASE_ROM_Bank * 0x4000);
- }
- else if (addr < 0x8000)
- {
- uint32_t temp_bank = (ROM_bank & ~ROM_bank_mask) | (ROM_bank_mask & BASE_ROM_Bank);
- temp_bank &= ROM_mask;
-
- SetCDLROM(flags, (addr - 0x4000) + temp_bank * 0x4000);
- }
- else
- {
- return;
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x2000)
- {
- if (reg_access)
- {
- BASE_ROM_Bank = value;
- }
- }
- else if (addr < 0x4000)
- {
- ROM_bank = (value > 0) ? (value) : 1;
-
- if ((value & 0x30) == 0x30)
- {
- reg_access = true;
- }
- else
- {
- reg_access = false;
- }
- }
- else if (addr < 0x6000)
- {
- if (reg_access)
- {
- ROM_bank_mask = value;
- }
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
-
- void Mapper_Tick()
- {
- if (locked)
- {
- if (((addr_access[0] & 0x8000) == 0) && ((addr_last & 0x8000) > 0) && (addr_access[0] >= 0x100))
- {
- counter++;
- }
-
- if (addr_access[0] >= 0x100)
- {
- addr_last = addr_access[0];
- }
-
- if (counter == 0x30)
- {
- locked = false;
- locked_GBC = true;
- counter = 0;
- }
- }
- else if (locked_GBC)
- {
- if (((addr_access[0] & 0x8000) == 0) && ((addr_last & 0x8000) > 0) && (addr_access[0] >= 0x100))
- {
- counter++;
- }
-
- if (addr_access[0] >= 0x100)
- {
- addr_last = addr_access[0];
- }
-
- if (counter == 0x30)
- {
- locked_GBC = false;
- finished = true;
- }
-
- // The above condition seems to never be reached as described in the mapper notes
- // so for now add this one
-
- if ((addr_access[0] == 0x133) && (counter == 1))
- {
- locked_GBC = false;
- finished = true;
- }
- }
- }
- };
-}
\ No newline at end of file
diff --git a/libHawk/GBHawk/GBHawk/Mapper_TAMA5.h b/libHawk/GBHawk/GBHawk/Mapper_TAMA5.h
deleted file mode 100644
index 135a63f31a..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_TAMA5.h
+++ /dev/null
@@ -1,255 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_TAMA5 : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 0;
- RAM_bank = 0;
- ROM_mask = ROM_Length[0] / 0x4000 - 1;
-
- // some games have sizes that result in a degenerate ROM, account for it here
- if (ROM_mask > 4) { ROM_mask |= 3; }
-
- RAM_mask = 0;
- if (Cart_RAM_Length[0] > 0)
- {
- RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
- if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
- }
-
- RAM_addr_low = RAM_addr_high = RAM_val_low = RAM_val_high = 0;
- Chip_return_low = Chip_return_high = 0;
- halt = false;
-
- ctrl = 0;
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x4000)
- {
- return ROM[addr];
- }
- else if (addr < 0x8000)
- {
- return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
- }
- else
- {
-
- switch (ctrl)
- {
- case 0xA:
- // The game won't proceed unless this value (anded with 3) is 1
- // see bank 0: 0x1A7D to 0x1A89
- return 1;
- case 0xC:
- //Console.WriteLine("read low: " + Chip_return_low);
- return Chip_return_low;
- case 0xD:
- //Console.WriteLine("read high: " + Chip_return_high);
- return Chip_return_high;
- }
-
- return 0x0;
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x4000)
- {
- SetCDLROM(flags, addr);
- }
- else if (addr < 0x8000)
- {
- SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
- }
- else
- {
-
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr == 0xA000)
- {
- switch (ctrl)
- {
- case 0:
- ROM_bank &= 0xF0;
- ROM_bank |= (value & 0xF);
- break;
- case 1:
- ROM_bank &= 0x0F;
- ROM_bank |= ((value & 0x1) << 4);
- break;
- case 4:
- RAM_val_low = (value & 0xF);
- break;
- case 5:
- RAM_val_high = (value & 0xF);
- //Cart_RAM[(RAM_addr_high << 4) | RAM_addr_low] = (uint8_t)((RAM_val_high << 4) | RAM_val_low);
- break;
- case 6:
- RAM_addr_high = (value & 1);
-
- switch ((value & 0xE) >> 1)
- {
- case 0:
- // write to RAM
- Cart_RAM[(RAM_addr_high << 4) | RAM_addr_low] = (uint8_t)((RAM_val_high << 4) | RAM_val_low);
- break;
- case 1:
- // read from RAM
- Chip_return_high = (uint8_t)(Cart_RAM[(RAM_addr_high << 4) | RAM_addr_low] >> 4);
- Chip_return_low = (uint8_t)(Cart_RAM[(RAM_addr_high << 4) | RAM_addr_low] & 0xF);
- break;
- case 2:
- // read from RTC registers
- if (RAM_addr_low == 3)
- {
- Chip_return_high = RTC_regs_TAMA[2];
- Chip_return_low = RTC_regs_TAMA[1];
- }
- else if (RAM_addr_low == 6)
- {
- Chip_return_high = RTC_regs_TAMA[4];
- Chip_return_low = RTC_regs_TAMA[3];
- }
- else
- {
- Chip_return_high = 1;
- Chip_return_low = 1;
- }
- break;
- case 3:
- // write to RTC registers (probably wrong, not well tested)
- if (RAM_addr_low == 3)
- {
- RTC_regs_TAMA[2] = (uint8_t)(RAM_val_high & 0xF);
- RTC_regs_TAMA[1] = (uint8_t)(RAM_val_low & 0xF);
- }
- else if (RAM_addr_low == 6)
- {
- RTC_regs_TAMA[4] = (uint8_t)(RAM_val_high & 0xF);
- RTC_regs_TAMA[3] = (uint8_t)(RAM_val_low & 0xF);
- }
- else
- {
-
- }
- break;
- case 4:
- // read from seconds register (time changes are checked when it rolls over)
- Chip_return_low = (uint8_t)(RTC_regs_TAMA[0] & 0xF);
- break;
- }
-
- //Console.WriteLine("CTRL: " + (value >> 1) + " RAM_high:" + RAM_addr_high + " RAM_low: " + RAM_addr_low + " val: " + (uint8_t)((RAM_val_high << 4) | RAM_val_low) + " Cpu: " + Core.cpu.TotalExecutedCycles);
- break;
- case 7:
- RAM_addr_low = (value & 0xF);
-
- //Console.WriteLine(" RAM_low:" + RAM_addr_low + " Cpu: " + Core.cpu.TotalExecutedCycles);
- break;
- }
- }
- else if (addr == 0xA001)
- {
- ctrl = value;
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
-
- void RTC_Get(uint32_t value, uint32_t index)
- {
- if (index < 10)
- {
- RTC_regs_TAMA[index] = (uint8_t)value;
- }
- else
- {
- RTC_offset = value;
- }
- }
-
- void Mapper_Tick()
- {
- if (!halt)
- {
- RTC_timer++;
-
- if (RTC_timer == 128)
- {
- RTC_timer = 0;
-
- RTC_low_clock++;
-
- if (RTC_low_clock == 32768)
- {
- RTC_low_clock = 0;
- RTC_timer = RTC_offset;
-
- RTC_regs_TAMA[0]++;
-
- if (RTC_regs_TAMA[0] > 59)
- {
- RTC_regs_TAMA[0] = 0;
- RTC_regs_TAMA[1]++;
- // 1's digit of minutes
- if (RTC_regs_TAMA[1] > 9)
- {
- RTC_regs_TAMA[1] = 0;
- RTC_regs_TAMA[2]++;
- // 10's digit of minutes
- if (RTC_regs_TAMA[2] > 5)
- {
- RTC_regs_TAMA[2] = 0;
- RTC_regs_TAMA[3]++;
- // 1's digit of hours
- if (RTC_regs_TAMA[3] > 9)
- {
- RTC_regs_TAMA[3] = 0;
- RTC_regs_TAMA[4]++;
- // 10's digit of hours
- if (RTC_regs_TAMA[4] > 2)
- {
- RTC_regs_TAMA[4] = 0;
- RTC_regs_TAMA[5]++;
- }
- }
- }
- }
- }
- }
- }
- }
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mapper_WisdomTree.h b/libHawk/GBHawk/GBHawk/Mapper_WisdomTree.h
deleted file mode 100644
index 61f2185fa4..0000000000
--- a/libHawk/GBHawk/GBHawk/Mapper_WisdomTree.h
+++ /dev/null
@@ -1,72 +0,0 @@
-#include
-#include
-#include
-#include
-#include
-
-#include "Mapper_Base.h"
-
-using namespace std;
-
-namespace GBHawk
-{
- class Mapper_WT : Mapper
- {
- public:
-
- void Reset()
- {
- ROM_bank = 0;
- ROM_mask = ROM_Length[0] / 0x8000 - 1;
-
- // some games have sizes that result in a degenerate ROM, account for it here
- if (ROM_mask > 4) { ROM_mask |= 3; }
- if (ROM_mask > 0x100) { ROM_mask |= 0xFF; }
- }
-
- uint8_t ReadMemory(uint32_t addr)
- {
- if (addr < 0x8000)
- {
- return ROM[ROM_bank * 0x8000 + addr];
- }
- else
- {
- return 0xFF;
- }
- }
-
- /*
- void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
- {
- if (addr < 0x8000)
- {
- SetCDLROM(flags, ROM_bank * 0x8000 + addr);
- }
- else
- {
- return;
- }
- }
- */
-
- uint8_t PeekMemory(uint32_t addr)
- {
- return ReadMemory(addr);
- }
-
- void WriteMemory(uint32_t addr, uint8_t value)
- {
- if (addr < 0x4000)
- {
- ROM_bank = ((addr << 1) & 0x1ff) >> 1;
- ROM_bank &= ROM_mask;
- }
- }
-
- void PokeMemory(uint32_t addr, uint8_t value)
- {
- WriteMemory(addr, value);
- }
- };
-}
diff --git a/libHawk/GBHawk/GBHawk/Mappers.h b/libHawk/GBHawk/GBHawk/Mappers.h
new file mode 100644
index 0000000000..6aeadcb6ea
--- /dev/null
+++ b/libHawk/GBHawk/GBHawk/Mappers.h
@@ -0,0 +1,3190 @@
+#include
+#include
+#include
+#include
+
+using namespace std;
+
+namespace GBHawk
+{
+ class Mapper
+ {
+ public:
+ #pragma region mapper base
+
+ Mapper()
+ {
+
+ }
+
+ uint8_t* ROM = nullptr;
+ uint8_t* Cart_RAM = nullptr;
+ uint32_t* ROM_Length = nullptr;
+ uint32_t* Cart_RAM_Length = nullptr;
+ uint32_t* addr_access = nullptr;
+ uint32_t* Acc_X_state = nullptr;
+ uint32_t* Acc_Y_state = nullptr;
+
+ // Generic Mapper Variables
+ bool RAM_enable;
+ bool sel_mode;
+ bool IR_signal;
+ uint32_t ROM_bank;
+ uint32_t RAM_bank;
+ uint32_t ROM_mask;
+ uint32_t RAM_mask;
+
+ // Common
+ bool halt;
+ uint32_t RTC_timer;
+ uint32_t RTC_low_clock;
+
+ // HuC3
+ bool timer_read;
+ uint8_t control;
+ uint8_t chip_read;
+ uint32_t time_val_shift;
+ uint32_t time;
+ uint32_t RTC_seconds;
+
+ // MBC3
+ uint8_t RTC_regs[5] = {};
+ uint8_t RTC_regs_latch[5] = {};
+ bool RTC_regs_latch_wr;
+ uint32_t RTC_offset;
+
+ // camera
+ bool regs_enable;
+ uint8_t regs_cam[0x80] = {};
+
+ // sachen
+ bool locked, locked_GBC, finished, reg_access;
+ uint32_t ROM_bank_mask;
+ uint32_t BASE_ROM_Bank;
+ uint32_t addr_last;
+ uint32_t counter;
+
+ // TAMA5
+ uint8_t RTC_regs_TAMA[10] = {};
+ uint32_t ctrl;
+ uint32_t RAM_addr_low;
+ uint32_t RAM_addr_high;
+ uint32_t RAM_val_low;
+ uint32_t RAM_val_high;
+ uint8_t Chip_return_low;
+ uint8_t Chip_return_high;
+
+ // MBC7
+ bool RAM_enable_1, RAM_enable_2, is_erased;
+ uint8_t acc_x_low;
+ uint8_t acc_x_high;
+ uint8_t acc_y_low;
+ uint8_t acc_y_high;
+ // EEPROM related
+ bool CS_prev;
+ bool CLK_prev;
+ bool DI_prev;
+ bool DO;
+ bool instr_read;
+ bool perf_instr;
+ bool WR_EN;
+ bool countdown_start;
+ uint32_t instr_bit_counter;
+ uint32_t instr;
+ uint32_t EE_addr;
+ uint32_t instr_case;
+ uint32_t instr_clocks;
+ uint32_t EE_value;
+ uint32_t countdown;
+
+
+
+ virtual uint8_t ReadMemory(uint32_t addr)
+ {
+ return 0;
+ }
+
+ virtual uint8_t PeekMemory(uint32_t addr)
+ {
+ return 0;
+ }
+
+ virtual void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ }
+
+ virtual void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ }
+
+
+ virtual void Dispose()
+ {
+ }
+
+ virtual void Reset()
+ {
+ }
+
+ virtual void Mapper_Tick()
+ {
+ }
+
+ virtual void RTC_Get(int value, int index)
+ {
+ }
+ /*
+ virtual void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ }
+
+ protected void SetCDLROM(LR35902.eCDLogMemFlags flags, int cdladdr)
+ {
+ Core.SetCDL(flags, "ROM", cdladdr);
+ }
+
+ protected void SetCDLRAM(LR35902.eCDLogMemFlags flags, int cdladdr)
+ {
+ Core.SetCDL(flags, "CartRAM", cdladdr);
+ }
+ */
+ #pragma endregion
+
+ #pragma region State Save / Load
+
+ uint8_t* SaveState(uint8_t* saver)
+ {
+ saver = bool_saver(RAM_enable, saver);
+ saver = bool_saver(sel_mode, saver);
+ saver = bool_saver(IR_signal, saver);
+ saver = int_saver(ROM_bank, saver);
+ saver = int_saver(RAM_bank, saver);
+ saver = int_saver(ROM_mask, saver);
+ saver = int_saver(RAM_mask, saver);
+
+ saver = bool_saver(halt, saver);
+ saver = int_saver(RTC_timer, saver);
+ saver = int_saver(RTC_low_clock, saver);
+
+ saver = bool_saver(timer_read, saver);
+ saver = byte_saver(control, saver);
+ saver = byte_saver(chip_read, saver);
+ saver = int_saver(time_val_shift, saver);
+ saver = int_saver(time, saver);
+ saver = int_saver(RTC_seconds, saver);
+
+ for (int i = 0; i < 5; i++) { saver = byte_saver(RTC_regs[i], saver); }
+ for (int i = 0; i < 5; i++) { saver = byte_saver(RTC_regs_latch[i], saver); }
+ saver = bool_saver(RTC_regs_latch_wr, saver);
+ saver = int_saver(RTC_offset, saver);
+
+ saver = bool_saver(regs_enable, saver);
+ for (int i = 0; i < 5; i++) { saver = byte_saver(regs_cam[i], saver); }
+
+ saver = bool_saver(locked, saver);
+ saver = bool_saver(locked_GBC, saver);
+ saver = bool_saver(finished, saver);
+ saver = bool_saver(reg_access, saver);
+ saver = int_saver(ROM_bank_mask, saver);
+ saver = int_saver(BASE_ROM_Bank, saver);
+ saver = int_saver(addr_last, saver);
+ saver = int_saver(counter, saver);
+
+ for (int i = 0; i < 10; i++) { saver = byte_saver(RTC_regs_TAMA[i], saver); }
+ saver = byte_saver(Chip_return_low, saver);
+ saver = byte_saver(Chip_return_high, saver);
+ saver = int_saver(ctrl, saver);
+ saver = int_saver(RAM_addr_low, saver);
+ saver = int_saver(RAM_addr_high, saver);
+ saver = int_saver(RAM_val_low, saver);
+ saver = int_saver(RAM_val_high, saver);
+
+ saver = bool_saver(RAM_enable_1, saver);
+ saver = bool_saver(RAM_enable_2, saver);
+ saver = bool_saver(is_erased, saver);
+ saver = byte_saver(acc_x_low, saver);
+ saver = byte_saver(acc_x_high, saver);
+ saver = byte_saver(acc_y_low, saver);
+ saver = byte_saver(acc_y_high, saver);
+ // EEPROM related
+ saver = bool_saver(CS_prev, saver);
+ saver = bool_saver(CLK_prev, saver);
+ saver = bool_saver(DI_prev, saver);
+ saver = bool_saver(DO, saver);
+ saver = bool_saver(instr_read, saver);
+ saver = bool_saver(perf_instr, saver);
+ saver = bool_saver(WR_EN, saver);
+ saver = bool_saver(countdown_start, saver);
+ saver = int_saver(instr_bit_counter, saver);
+ saver = int_saver(instr, saver);
+ saver = int_saver(EE_addr, saver);
+ saver = int_saver(instr_case, saver);
+ saver = int_saver(instr_clocks, saver);
+ saver = int_saver(EE_value, saver);
+ saver = int_saver(countdown, saver);
+
+ return saver;
+ }
+
+ uint8_t* LoadState(uint8_t* loader)
+ {
+ loader = bool_loader(&RAM_enable, loader);
+ loader = bool_loader(&sel_mode, loader);
+ loader = bool_loader(&IR_signal, loader);
+ loader = int_loader(&ROM_bank, loader);
+ loader = int_loader(&RAM_bank, loader);
+ loader = int_loader(&ROM_mask, loader);
+ loader = int_loader(&RAM_mask, loader);
+
+ loader = bool_loader(&halt, loader);
+ loader = int_loader(&RTC_timer, loader);
+ loader = int_loader(&RTC_low_clock, loader);
+
+ loader = bool_loader(&timer_read, loader);
+ loader = byte_loader(&control, loader);
+ loader = byte_loader(&chip_read, loader);
+ loader = int_loader(&time_val_shift, loader);
+ loader = int_loader(&time, loader);
+ loader = int_loader(&RTC_seconds, loader);
+
+ for (int i = 0; i < 5; i++) { loader = byte_loader(&RTC_regs[i], loader); }
+ for (int i = 0; i < 5; i++) { loader = byte_loader(&RTC_regs_latch[i], loader); }
+ loader = bool_loader(&RTC_regs_latch_wr, loader);
+ loader = int_loader(&RTC_offset, loader);
+
+ loader = bool_loader(®s_enable, loader);
+ for (int i = 0; i < 5; i++) { loader = byte_loader(®s_cam[i], loader); }
+
+ loader = bool_loader(&locked, loader);
+ loader = bool_loader(&locked_GBC, loader);
+ loader = bool_loader(&finished, loader);
+ loader = bool_loader(®_access, loader);
+ loader = int_loader(&ROM_bank_mask, loader);
+ loader = int_loader(&BASE_ROM_Bank, loader);
+ loader = int_loader(&addr_last, loader);
+ loader = int_loader(&counter, loader);
+
+ for (int i = 0; i < 10; i++) { loader = byte_loader(&RTC_regs_TAMA[i], loader); }
+ loader = byte_loader(&Chip_return_low, loader);
+ loader = byte_loader(&Chip_return_high, loader);
+ loader = int_loader(&ctrl, loader);
+ loader = int_loader(&RAM_addr_low, loader);
+ loader = int_loader(&RAM_addr_high, loader);
+ loader = int_loader(&RAM_val_low, loader);
+ loader = int_loader(&RAM_val_high, loader);
+
+ loader = bool_loader(&RAM_enable_1, loader);
+ loader = bool_loader(&RAM_enable_2, loader);
+ loader = bool_loader(&is_erased, loader);
+ loader = byte_loader(&acc_x_low, loader);
+ loader = byte_loader(&acc_x_high, loader);
+ loader = byte_loader(&acc_y_low, loader);
+ loader = byte_loader(&acc_y_high, loader);
+ // EEPROM related
+ loader = bool_loader(&CS_prev, loader);
+ loader = bool_loader(&CLK_prev, loader);
+ loader = bool_loader(&DI_prev, loader);
+ loader = bool_loader(&DO, loader);
+ loader = bool_loader(&instr_read, loader);
+ loader = bool_loader(&perf_instr, loader);
+ loader = bool_loader(&WR_EN, loader);
+ loader = bool_loader(&countdown_start, loader);
+ loader = int_loader(&instr_bit_counter, loader);
+ loader = int_loader(&instr, loader);
+ loader = int_loader(&EE_addr, loader);
+ loader = int_loader(&instr_case, loader);
+ loader = int_loader(&instr_clocks, loader);
+ loader = int_loader(&EE_value, loader);
+ loader = int_loader(&countdown, loader);
+
+ return loader;
+ }
+
+ uint8_t* bool_saver(bool to_save, uint8_t* saver)
+ {
+ *saver = (uint8_t)(to_save ? 1 : 0); saver++;
+
+ return saver;
+ }
+
+ uint8_t* byte_saver(uint8_t to_save, uint8_t* saver)
+ {
+ *saver = to_save; saver++;
+
+ return saver;
+ }
+
+ uint8_t* int_saver(uint32_t to_save, uint8_t* saver)
+ {
+ *saver = (uint8_t)(to_save & 0xFF); saver++; *saver = (uint8_t)((to_save >> 8) & 0xFF); saver++;
+ *saver = (uint8_t)((to_save >> 16) & 0xFF); saver++; *saver = (uint8_t)((to_save >> 24) & 0xFF); saver++;
+
+ return saver;
+ }
+
+ uint8_t* bool_loader(bool* to_load, uint8_t* loader)
+ {
+ to_load[0] = *to_load == 1; loader++;
+
+ return loader;
+ }
+
+ uint8_t* byte_loader(uint8_t* to_load, uint8_t* loader)
+ {
+ to_load[0] = *loader; loader++;
+
+ return loader;
+ }
+
+ uint8_t* int_loader(uint32_t* to_load, uint8_t* loader)
+ {
+ to_load[0] = *loader; loader++; to_load[0] |= (*loader << 8); loader++;
+ to_load[0] |= (*loader << 16); loader++; to_load[0] |= (*loader << 24); loader++;
+
+ return loader;
+ }
+
+ #pragma endregion
+
+ };
+
+ #pragma region Camera
+
+ class Mapper_Camera : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 1;
+ RAM_bank = 0;
+ RAM_enable = false;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+
+ RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
+
+ regs_enable = false;
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ return ROM[addr];
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else
+ {
+ if (regs_enable)
+ {
+ if ((addr & 0x7F) == 0)
+ {
+ return 0;// regs[0];
+ }
+ else
+ {
+ return 0;
+ }
+ }
+ else
+ {
+ if (/*RAM_enable && */(((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ // lowest bank is fixed, but is still effected by mode
+ SetCDLROM(flags, addr);
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else
+ {
+ if (!regs_enable)
+ {
+ if ((((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
+ }
+ else
+ {
+ return;
+ }
+ }
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x8000)
+ {
+ if (addr < 0x2000)
+ {
+ RAM_enable = (value & 0xF) == 0xA;
+ }
+ else if (addr < 0x4000)
+ {
+ ROM_bank = value;
+ ROM_bank &= ROM_mask;
+ //Console.WriteLine(addr + " " + value + " " + ROM_mask + " " + ROM_bank);
+ }
+ else if (addr < 0x6000)
+ {
+ if ((value & 0x10) == 0x10)
+ {
+ regs_enable = true;
+ }
+ else
+ {
+ regs_enable = false;
+ RAM_bank = value & RAM_mask;
+ }
+ }
+ }
+ else
+ {
+ if (regs_enable)
+ {
+ regs_cam[(addr & 0x7F)] = (uint8_t)(value & 0x7);
+ }
+ else
+ {
+ if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
+ }
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region Default
+
+ class Mapper_Default : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ // nothing to initialize
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x8000)
+ {
+ return ROM[addr];
+ }
+ else
+ {
+ if (Cart_RAM_Length > 0)
+ {
+ return Cart_RAM[addr - 0xA000];
+ }
+ else
+ {
+ return 0;
+ }
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x8000)
+ {
+ SetCDLROM(flags, addr);
+ }
+ else
+ {
+ if (Cart_RAM != null)
+ {
+ SetCDLRAM(flags, addr - 0xA000);
+ }
+ else
+ {
+ return;
+ }
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x8000)
+ {
+ // no mapping hardware available
+ }
+ else
+ {
+ if (Cart_RAM_Length > 0)
+ {
+ Cart_RAM[addr - 0xA000] = value;
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region HuC1
+
+ class Mapper_HuC1 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 0;
+ RAM_bank = 0;
+ RAM_enable = false;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+
+ // some games have sizes that result in a degenerate ROM, account for it here
+ if (ROM_mask > 4) { ROM_mask |= 3; }
+
+ RAM_mask = 0;
+ if (Cart_RAM_Length[0] > 0)
+ {
+ RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
+ if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
+ }
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ return ROM[addr];
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else if ((addr >= 0xA000) && (addr < 0xC000))
+ {
+ if (RAM_enable)
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
+ {
+ return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+ else
+ {
+ // when RAM isn't enabled, reading from this area will return IR sensor reading
+ // for now we'll assume it never sees light (0xC0)
+ return 0xC0;
+ }
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ SetCDLROM(flags, addr);
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else if ((addr >= 0xA000) && (addr < 0xC000))
+ {
+ if (RAM_enable)
+ {
+ if (Cart_RAM != null)
+ {
+ if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
+ {
+ SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
+ }
+ else
+ {
+ return;
+ }
+ }
+ else
+ {
+ return;
+ }
+ }
+ else
+ {
+ return;
+ }
+ }
+ else
+ {
+ return;
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x8000)
+ {
+ if (addr < 0x2000)
+ {
+ RAM_enable = (value & 0xF) != 0xE;
+ }
+ else if (addr < 0x4000)
+ {
+ value &= 0x3F;
+
+ ROM_bank &= 0xC0;
+ ROM_bank |= value;
+ ROM_bank &= ROM_mask;
+ }
+ else if (addr < 0x6000)
+ {
+ RAM_bank = value & 3;
+ RAM_bank &= RAM_mask;
+ }
+ }
+ else
+ {
+ if (RAM_enable)
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
+ {
+ Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
+ }
+ }
+ }
+ else
+ {
+ // I don't know if other bits here have an effect
+ if (value == 1)
+ {
+ IR_signal = true;
+ }
+ else if (value == 0)
+ {
+ IR_signal = false;
+ }
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+ #pragma endregion
+
+ #pragma region huC3
+
+ class Mapper_HuC3 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 0;
+ RAM_bank = 0;
+ RAM_enable = false;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+ control = 0;
+ chip_read = 1;
+ timer_read = false;
+ time_val_shift = 0;
+
+ // some games have sizes that result in a degenerate ROM, account for it here
+ if (ROM_mask > 4) { ROM_mask |= 3; }
+
+ RAM_mask = 0;
+ if (Cart_RAM_Length[0] > 0)
+ {
+ RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
+ if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
+ }
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ return ROM[addr];
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else if ((addr >= 0xA000) && (addr < 0xC000))
+ {
+ if ((control >= 0xB) && (control < 0xE))
+ {
+ if (control == 0xD)
+ {
+ return 1;
+ }
+ return chip_read;
+ }
+
+ if (RAM_enable)
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
+ {
+ return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+ else
+ {
+ // what to return if RAM not enabled and controller not selected?
+ return 0xFF;
+ }
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ SetCDLROM(flags, addr);
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else if ((addr >= 0xA000) && (addr < 0xC000))
+ {
+ if (RAM_enable)
+ {
+ if (Cart_RAM != null)
+ {
+ if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
+ {
+ SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
+ }
+ else
+ {
+ return;
+ }
+ }
+ else
+ {
+ return;
+ }
+ }
+ else
+ {
+ return;
+ }
+ }
+ else
+ {
+ return;
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x8000)
+ {
+ if (addr < 0x2000)
+ {
+ RAM_enable = (value & 0xA) == 0xA;
+ control = value;
+ }
+ else if (addr < 0x4000)
+ {
+ if (value == 0) { value = 1; }
+
+ ROM_bank = value;
+ ROM_bank &= ROM_mask;
+ }
+ else if (addr < 0x6000)
+ {
+ RAM_bank = value;
+ RAM_bank &= 0xF;
+ RAM_bank &= RAM_mask;
+ }
+ }
+ else
+ {
+ if (RAM_enable && ((control < 0xB) || (control > 0xE)))
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
+ {
+ Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
+ }
+ }
+ }
+
+ if (control == 0xB)
+ {
+ switch (value & 0xF0)
+ {
+ case 0x10:
+ if (timer_read)
+ {
+ // return timer value
+ chip_read = (uint8_t)((time >> time_val_shift) & 0xF);
+ time_val_shift += 4;
+ if (time_val_shift == 28) { time_val_shift = 0; }
+ }
+ break;
+ case 0x20:
+ break;
+ case 0x30:
+ if (!timer_read)
+ {
+ // write to timer
+ if (time_val_shift == 0) { time = 0; }
+ if (time_val_shift < 28)
+ {
+ time |= (uint32_t)((value & 0x0F) << time_val_shift);
+ time_val_shift += 4;
+ if (time_val_shift == 28) { timer_read = true; }
+ }
+ }
+ break;
+ case 0x40:
+ // other commands
+ switch (value & 0xF)
+ {
+ case 0x0:
+ time_val_shift = 0;
+ break;
+ case 0x3:
+ timer_read = false;
+ time_val_shift = 0;
+ break;
+ case 0x7:
+ timer_read = true;
+ time_val_shift = 0;
+ break;
+ case 0xF:
+ break;
+ }
+ break;
+ case 0x50:
+ break;
+ case 0x60:
+ timer_read = true;
+ break;
+ }
+ }
+ else if (control == 0xC)
+ {
+ // maybe IR
+ }
+ else if (control == 0xD)
+ {
+ // maybe IR
+ }
+ }
+ }
+
+ void RTC_Get(uint32_t value, uint32_t index)
+ {
+ time |= (uint32_t)((value & 0xFF) << index);
+ }
+
+ void Mapper_Tick()
+ {
+ RTC_timer++;
+
+ if (RTC_timer == 128)
+ {
+ RTC_timer = 0;
+
+ RTC_low_clock++;
+
+ if (RTC_low_clock == 32768)
+ {
+ RTC_low_clock = 0;
+
+ RTC_seconds++;
+ if (RTC_seconds > 59)
+ {
+ RTC_seconds = 0;
+ time++;
+ if ((time & 0xFFF) > 1439)
+ {
+ time -= 1440;
+ time += (1 << 12);
+ if ((time >> 12) > 365)
+ {
+ time -= (365 << 12);
+ time += (1 << 24);
+ }
+ }
+ }
+ }
+ }
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region MBC1
+
+ class Mapper_MBC1 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 1;
+ RAM_bank = 0;
+ RAM_enable = false;
+ sel_mode = false;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+
+ // some games have sizes that result in a degenerate ROM, account for it here
+ if (ROM_mask > 4) { ROM_mask |= 3; }
+
+ RAM_mask = 0;
+ if (Cart_RAM_Length[0] > 0)
+ {
+ RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
+ if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
+ }
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ // lowest bank is fixed, but is still effected by mode
+ if (sel_mode)
+ {
+ return ROM[(ROM_bank & 0x60) * 0x4000 + addr];
+ }
+ else
+ {
+ return ROM[addr];
+ }
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
+ }
+ else
+ {
+ return 0xFF;
+ }
+
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ // lowest bank is fixed, but is still effected by mode
+ if (sel_mode)
+ {
+ SetCDLROM(flags, (ROM_bank & 0x60) * 0x4000 + addr);
+ }
+ else
+ {
+ SetCDLROM(flags, addr);
+ }
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else
+ {
+ if (Cart_RAM != null)
+ {
+ if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
+ }
+ else
+ {
+ return;
+ }
+
+ }
+ else
+ {
+ return;
+ }
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x8000)
+ {
+ if (addr < 0x2000)
+ {
+ RAM_enable = (value & 0xF) == 0xA;
+ }
+ else if (addr < 0x4000)
+ {
+ value &= 0x1F;
+
+ // writing zero gets translated to 1
+ if (value == 0) { value = 1; }
+
+ ROM_bank &= 0xE0;
+ ROM_bank |= value;
+ ROM_bank &= ROM_mask;
+ }
+ else if (addr < 0x6000)
+ {
+ if (sel_mode && (Cart_RAM_Length[0] > 0))
+ {
+ RAM_bank = value & 3;
+ RAM_bank &= RAM_mask;
+ }
+ else
+ {
+ ROM_bank &= 0x1F;
+ ROM_bank |= ((value & 3) << 5);
+ ROM_bank &= ROM_mask;
+ }
+ }
+ else
+ {
+ sel_mode = (value & 1) > 0;
+
+ if (sel_mode && (Cart_RAM_Length[0] > 0))
+ {
+ ROM_bank &= 0x1F;
+ ROM_bank &= ROM_mask;
+ }
+ else
+ {
+ RAM_bank = 0;
+ }
+ }
+ }
+ else
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
+ }
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region MBC1_Multi
+
+ class Mapper_MBC1_Multi : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 1;
+ RAM_bank = 0;
+ RAM_enable = false;
+ sel_mode = false;
+ ROM_mask = (ROM_Length[0] / 0x4000 * 2) - 1; // due to how mapping works, we want a 1 bit higher mask
+ RAM_mask = 0;
+ if (Cart_RAM_Length[0] > 0)
+ {
+ RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
+ if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
+ }
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ // lowest bank is fixed, but is still effected by mode
+ if (sel_mode)
+ {
+ return ROM[((ROM_bank & 0x60) >> 1) * 0x4000 + addr];
+ }
+ else
+ {
+ return ROM[addr];
+ }
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + (((ROM_bank & 0x60) >> 1) | (ROM_bank & 0xF)) * 0x4000];
+ }
+ else
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
+ }
+ else
+ {
+ return 0xFF;
+ }
+
+ }
+ else
+ {
+ return 0;
+ }
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ // lowest bank is fixed, but is still effected by mode
+ if (sel_mode)
+ {
+ SetCDLROM(flags, ((ROM_bank & 0x60) >> 1) * 0x4000 + addr);
+ }
+ else
+ {
+ SetCDLROM(flags, addr);
+ }
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + (((ROM_bank & 0x60) >> 1) | (ROM_bank & 0xF)) * 0x4000);
+ }
+ else
+ {
+ if (Cart_RAM != null)
+ {
+ if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
+ }
+ else
+ {
+ return;
+ }
+
+ }
+ else
+ {
+ return;
+ }
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x8000)
+ {
+ if (addr < 0x2000)
+ {
+ RAM_enable = ((value & 0xA) == 0xA);
+ }
+ else if (addr < 0x4000)
+ {
+ value &= 0x1F;
+
+ // writing zero gets translated to 1
+ if (value == 0) { value = 1; }
+
+ ROM_bank &= 0xE0;
+ ROM_bank |= value;
+ ROM_bank &= ROM_mask;
+ }
+ else if (addr < 0x6000)
+ {
+ if (sel_mode && (Cart_RAM_Length[0] > 0))
+ {
+ RAM_bank = value & 3;
+ RAM_bank &= RAM_mask;
+ }
+ else
+ {
+ ROM_bank &= 0x1F;
+ ROM_bank |= ((value & 3) << 5);
+ ROM_bank &= ROM_mask;
+ }
+ }
+ else
+ {
+ sel_mode = (value & 1) > 0;
+
+ if (sel_mode && (Cart_RAM_Length[0] > 0))
+ {
+ ROM_bank &= 0x1F;
+ ROM_bank &= ROM_mask;
+ }
+ else
+ {
+ RAM_bank = 0;
+ }
+ }
+ }
+ else
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
+ }
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region MBC2
+
+ class Mapper_MBC2 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 1;
+ RAM_bank = 0;
+ RAM_enable = false;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ return ROM[addr];
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else if ((addr >= 0xA000) && (addr < 0xA200))
+ {
+ if (RAM_enable)
+ {
+ return Cart_RAM[addr - 0xA000];
+ }
+ return 0xFF;
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ SetCDLROM(flags, addr);
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else if ((addr >= 0xA000) && (addr < 0xA200))
+ {
+ if (RAM_enable)
+ {
+ SetCDLRAM(flags, addr - 0xA000);
+ }
+ return;
+ }
+ else
+ {
+ return;
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x2000)
+ {
+ if ((addr & 0x100) == 0)
+ {
+ RAM_enable = ((value & 0xA) == 0xA);
+ }
+ }
+ else if (addr < 0x4000)
+ {
+ if ((addr & 0x100) > 0)
+ {
+ ROM_bank = value & 0xF & ROM_mask;
+ if (ROM_bank == 0) { ROM_bank = 1; }
+ }
+ }
+ else if ((addr >= 0xA000) && (addr < 0xA200))
+ {
+ if (RAM_enable)
+ {
+ Cart_RAM[addr - 0xA000] = (uint8_t)(value & 0xF);
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region MBC3
+
+ class Mapper_MBC3 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 1;
+ RAM_bank = 0;
+ RAM_enable = false;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+
+ // some games have sizes that result in a degenerate ROM, account for it here
+ if (ROM_mask > 4) { ROM_mask |= 3; }
+
+ RAM_mask = 0;
+ if (Cart_RAM_Length[0] > 0)
+ {
+ RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
+ if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
+ }
+
+ RTC_regs_latch[0] = 0;
+ RTC_regs_latch[1] = 0;
+ RTC_regs_latch[2] = 0;
+ RTC_regs_latch[3] = 0;
+ RTC_regs_latch[4] = 0;
+
+ RTC_regs_latch_wr = true;
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ return ROM[addr];
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else
+ {
+ if (RAM_enable)
+ {
+ if ((Cart_RAM_Length[0] > 0) && (RAM_bank <= RAM_mask))
+ {
+ if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
+ {
+ return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ if ((RAM_bank >= 8) && (RAM_bank <= 0xC))
+ {
+ //Console.WriteLine("reg: " + (RAM_bank - 8) + " value: " + RTC_regs_latch[RAM_bank - 8] + " cpu: " + Core.cpu.TotalExecutedCycles);
+ return RTC_regs_latch[RAM_bank - 8];
+ }
+ else
+ {
+ return 0x0;
+ }
+ }
+ else
+ {
+ return 0x0;
+ }
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ SetCDLROM(flags, addr);
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else
+ {
+ if (RAM_enable)
+ {
+ if ((Cart_RAM != null) && (RAM_bank <= RAM_mask))
+ {
+ if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
+ {
+ SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
+ }
+ else
+ {
+ return;
+ }
+ }
+
+ if ((RAM_bank >= 8) && (RAM_bank <= 0xC))
+ {
+ return;
+ }
+ else
+ {
+ return;
+ }
+ }
+ else
+ {
+ return;
+ }
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x8000)
+ {
+ if (addr < 0x2000)
+ {
+ RAM_enable = ((value & 0xA) == 0xA);
+ }
+ else if (addr < 0x4000)
+ {
+ value &= 0x7F;
+
+ // writing zero gets translated to 1
+ if (value == 0) { value = 1; }
+
+ ROM_bank = value;
+ ROM_bank &= ROM_mask;
+ }
+ else if (addr < 0x6000)
+ {
+ RAM_bank = value;
+ }
+ else
+ {
+ if (!RTC_regs_latch_wr && ((value & 1) == 1))
+ {
+ for (uint32_t i = 0; i < 5; i++)
+ {
+ RTC_regs_latch[i] = RTC_regs[i];
+ }
+ }
+
+ RTC_regs_latch_wr = (value & 1) > 0;
+ }
+ }
+ else
+ {
+ if (RAM_enable)
+ {
+ if ((Cart_RAM_Length[0] > 0) && (RAM_bank <= RAM_mask))
+ {
+ if (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0])
+ {
+ Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
+ }
+ }
+ else if ((RAM_bank >= 8) && (RAM_bank <= 0xC))
+ {
+ RTC_regs[RAM_bank - 8] = value;
+
+ if ((RAM_bank - 8) == 0) { RTC_low_clock = RTC_timer = 0; }
+
+ halt = (RTC_regs[4] & 0x40) > 0;
+ }
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+
+ void RTC_Get(uint32_t value, uint32_t index)
+ {
+ if (index < 5)
+ {
+ RTC_regs[index] = (uint8_t)value;
+ }
+ else
+ {
+ RTC_offset = value;
+ }
+ }
+
+ void Mapper_Tick()
+ {
+ if (!halt)
+ {
+ RTC_timer++;
+
+ if (RTC_timer == 128)
+ {
+ RTC_timer = 0;
+
+ RTC_low_clock++;
+
+ if (RTC_low_clock == 32768)
+ {
+ RTC_low_clock = 0;
+ RTC_timer = RTC_offset;
+
+ RTC_regs[0]++;
+
+ if (RTC_regs[0] > 59)
+ {
+ RTC_regs[0] = 0;
+ RTC_regs[1]++;
+ if (RTC_regs[1] > 59)
+ {
+ RTC_regs[1] = 0;
+ RTC_regs[2]++;
+ if (RTC_regs[2] > 23)
+ {
+ RTC_regs[2] = 0;
+ if (RTC_regs[3] < 0xFF)
+ {
+ RTC_regs[3]++;
+ }
+ else
+ {
+ RTC_regs[3] = 0;
+
+ if ((RTC_regs[4] & 1) == 0)
+ {
+ RTC_regs[4] |= 1;
+ }
+ else
+ {
+ RTC_regs[4] &= 0xFE;
+ RTC_regs[4] |= 0x80;
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region MBC5
+
+ class Mapper_MBC5 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 1;
+ RAM_bank = 0;
+ RAM_enable = false;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+
+ // some games have sizes that result in a degenerate ROM, account for it here
+ if (ROM_mask > 4) { ROM_mask |= 3; }
+ if (ROM_mask > 0x100) { ROM_mask |= 0xFF; }
+
+ RAM_mask = 0;
+ if (Cart_RAM_Length[0] > 0)
+ {
+ RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
+ if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
+ }
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ return ROM[addr];
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ return Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000];
+ }
+ else
+ {
+ return 0xFF;
+ }
+
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ SetCDLROM(flags, addr);
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else
+ {
+ if (Cart_RAM != null)
+ {
+ if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ SetCDLRAM(flags, (addr - 0xA000) + RAM_bank * 0x2000);
+ }
+ else
+ {
+ return;
+ }
+
+ }
+ else
+ {
+ return;
+ }
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x8000)
+ {
+ if (addr < 0x2000)
+ {
+ RAM_enable = (value & 0xF) == 0xA;
+ }
+ else if (addr < 0x3000)
+ {
+ value &= 0xFF;
+
+ ROM_bank &= 0x100;
+ ROM_bank |= value;
+ ROM_bank &= ROM_mask;
+ }
+ else if (addr < 0x4000)
+ {
+ value &= 1;
+
+ ROM_bank &= 0xFF;
+ ROM_bank |= (value << 8);
+ ROM_bank &= ROM_mask;
+ }
+ else if (addr < 0x6000)
+ {
+ RAM_bank = value & 0xF;
+ RAM_bank &= RAM_mask;
+ }
+ }
+ else
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ if (RAM_enable && (((addr - 0xA000) + RAM_bank * 0x2000) < Cart_RAM_Length[0]))
+ {
+ Cart_RAM[(addr - 0xA000) + RAM_bank * 0x2000] = value;
+ }
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region MBC6
+
+ class Mapper_MBC6 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ // nothing to initialize
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x8000)
+ {
+ return ROM[addr];
+ }
+ else
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ return Cart_RAM[addr - 0xA000];
+ }
+ else
+ {
+ return 0;
+ }
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x8000)
+ {
+ SetCDLROM(flags, addr);
+ }
+ else
+ {
+ if (Cart_RAM != null)
+ {
+ SetCDLRAM(flags, addr - 0xA000);
+ }
+ else
+ {
+ return;
+ }
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x8000)
+ {
+ // no mapping hardware available
+ }
+ else
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ Cart_RAM[addr - 0xA000] = value;
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region MBC7
+
+ class Mapper_MBC7 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 1;
+ RAM_enable_1 = RAM_enable_2 = false;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+
+ // some games have sizes that result in a degenerate ROM, account for it here
+ if (ROM_mask > 4) { ROM_mask |= 3; }
+
+ acc_x_low = 0;
+ acc_x_high = 0x80;
+ acc_y_low = 0;
+ acc_y_high = 0x80;
+
+ // reset acceerometer
+ is_erased = false;
+
+ // EEPROM related
+ CS_prev = CLK_prev = DI_prev = DO = instr_read = perf_instr = WR_EN = countdown_start = false;
+ instr_bit_counter = instr = EE_addr = instr_case = instr_clocks = EE_value = countdown = 0;
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ return ROM[addr];
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else if (addr < 0xA000)
+ {
+ return 0xFF;
+ }
+ else if (addr < 0xB000)
+ {
+ if (RAM_enable_1 && RAM_enable_2)
+ {
+ return Register_Access_Read(addr);
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ SetCDLROM(flags, addr);
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else if (addr < 0xA000)
+ {
+ return;
+ }
+ else if (addr < 0xB000)
+ {
+ if (RAM_enable_1 && RAM_enable_2)
+ {
+ return;
+ }
+ else
+ {
+ return;
+ }
+ }
+ else
+ {
+ return;
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0xA000)
+ {
+ if (addr < 0x2000)
+ {
+ RAM_enable_1 = (value & 0xF) == 0xA;
+ }
+ else if (addr < 0x4000)
+ {
+ value &= 0xFF;
+
+ //Console.WriteLine(Core.cpu.TotalExecutedCycles);
+ //Console.WriteLine(value);
+
+ ROM_bank &= 0x100;
+ ROM_bank |= value;
+ ROM_bank &= ROM_mask;
+ }
+ else if (addr < 0x6000)
+ {
+ RAM_enable_2 = (value & 0xF0) == 0x40;
+ }
+ }
+ else
+ {
+ if (RAM_enable_1 && RAM_enable_2)
+ {
+ Register_Access_Write(addr, value);
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+
+ uint8_t Register_Access_Read(uint32_t addr)
+ {
+ if ((addr & 0xA0F0) == 0xA000)
+ {
+ return 0xFF;
+ }
+ else if ((addr & 0xA0F0) == 0xA010)
+ {
+ return 0xFF;
+ }
+ else if ((addr & 0xA0F0) == 0xA020)
+ {
+ return acc_x_low;
+ }
+ else if ((addr & 0xA0F0) == 0xA030)
+ {
+ return acc_x_high;
+ }
+ else if ((addr & 0xA0F0) == 0xA040)
+ {
+ return acc_y_low;
+ }
+ else if ((addr & 0xA0F0) == 0xA050)
+ {
+ return acc_y_high;
+ }
+ else if ((addr & 0xA0F0) == 0xA060)
+ {
+ return 0xFF;
+ }
+ else if ((addr & 0xA0F0) == 0xA070)
+ {
+ return 0xFF;
+ }
+ else if ((addr & 0xA0F0) == 0xA080)
+ {
+ return (uint8_t)((CS_prev ? 0x80 : 0) |
+ (CLK_prev ? 0x40 : 0) |
+ (DI_prev ? 2 : 0) |
+ (DO ? 1 : 0));
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ void Register_Access_Write(uint32_t addr, uint8_t value)
+ {
+ if ((addr & 0xA0F0) == 0xA000)
+ {
+ if (value == 0x55)
+ {
+ //Console.WriteLine("Erasing ACC");
+
+ is_erased = true;
+ acc_x_low = 0x00;
+ acc_x_high = 0x80;
+ acc_y_low = 0x00;
+ acc_y_high = 0x80;
+ }
+ }
+ else if ((addr & 0xA0F0) == 0xA010)
+ {
+ if ((value == 0xAA) && is_erased)
+ {
+ // latch new accelerometer values
+ //Console.WriteLine("Latching ACC");
+ acc_x_low = (uint8_t)(Acc_X_state[0] & 0xFF);
+ acc_x_high = (uint8_t)((Acc_X_state[0] & 0xFF00) >> 8);
+ acc_y_low = (uint8_t)(Acc_Y_state[0] & 0xFF);
+ acc_y_high = (uint8_t)((Acc_Y_state[0] & 0xFF00) >> 8);
+ }
+ }
+ else if ((addr & 0xA0F0) == 0xA080)
+ {
+ // EEPROM writes
+ EEPROM_write(value);
+ }
+ }
+
+ void EEPROM_write(uint8_t value)
+ {
+ bool CS = (value & 0x80) > 0;
+ bool CLK = (value & 0x40) > 0;
+ bool DI = (value & 0x2) > 0;
+
+ // if we deselect the chip, complete instructions or countdown and stop
+ if (!CS)
+ {
+ CS_prev = CS;
+ CLK_prev = CLK;
+ DI_prev = DI;
+
+ DO = true;
+ countdown_start = false;
+ perf_instr = false;
+ instr_read = false;
+
+ //Console.Write("Chip De-selected: ");
+ //Console.WriteLine(Core.cpu.TotalExecutedCycles);
+ }
+
+ if (!instr_read && !perf_instr)
+ {
+ // if we aren't performing an operation or reading an incoming instruction, we are waiting for one
+ // this is signalled by CS and DI both being 1 while CLK goes from 0 to 1
+ if (CLK && !CLK_prev && DI && CS)
+ {
+ instr_read = true;
+ instr_bit_counter = 0;
+ instr = 0;
+ DO = false;
+ //Console.Write("Initiating command: ");
+ //Console.WriteLine(Core.cpu.TotalExecutedCycles);
+ }
+ }
+ else if (instr_read && CLK && !CLK_prev)
+ {
+ // all instructions are 10 bits long
+ instr = (instr << 1) | ((value & 2) >> 1);
+
+ instr_bit_counter++;
+ if (instr_bit_counter == 10)
+ {
+ instr_read = false;
+ instr_clocks = 0;
+ EE_addr = instr & 0x7F;
+ EE_value = 0;
+
+ switch (instr & 0x300)
+ {
+ case 0x0:
+ switch (instr & 0xC0)
+ {
+ case 0x0: // disable writes
+ instr_case = 0;
+ WR_EN = false;
+ DO = true;
+ break;
+ case 0x40: // fill mem with value
+ instr_case = 1;
+ perf_instr = true;
+ break;
+ case 0x80: // fill mem with FF
+ instr_case = 2;
+ if (WR_EN)
+ {
+ for (uint32_t i = 0; i < 256; i++)
+ {
+ Cart_RAM[i] = 0xFF;
+ }
+ }
+ DO = true;
+ break;
+ case 0xC0: // enable writes
+ instr_case = 3;
+ WR_EN = true;
+ DO = true;
+ break;
+ }
+ break;
+ case 0x100: // write to address
+ instr_case = 4;
+ perf_instr = true;
+ break;
+ case 0x200: // read from address
+ instr_case = 5;
+ perf_instr = true;
+ break;
+ case 0x300: // set address to FF
+ instr_case = 6;
+ if (WR_EN)
+ {
+ Cart_RAM[EE_addr * 2] = 0xFF;
+ Cart_RAM[EE_addr * 2 + 1] = 0xFF;
+ }
+ DO = true;
+ break;
+ }
+
+ //Console.Write("Selected Command: ");
+ //Console.Write(instr_case);
+ //Console.Write(" ");
+ //Console.WriteLine(Core.cpu.TotalExecutedCycles);
+ }
+ }
+ else if (perf_instr && CLK && !CLK_prev)
+ {
+ //Console.Write("Command In progress, Cycle: ");
+ //Console.Write(instr_clocks);
+ //Console.Write(" ");
+ //Console.WriteLine(Core.cpu.TotalExecutedCycles);
+
+ // for commands that require additional clocking
+ switch (instr_case)
+ {
+ case 1:
+ EE_value = (EE_value << 1) | ((value & 2) >> 1);
+
+ if (instr_clocks == 15)
+ {
+ if (WR_EN)
+ {
+ for (uint32_t i = 0; i < 128; i++)
+ {
+ Cart_RAM[i * 2] = (uint8_t)(EE_value & 0xFF);
+ Cart_RAM[i * 2 + 1] = (uint8_t)((EE_value & 0xFF00) >> 8);
+ }
+ }
+ instr_case = 7;
+ countdown = 8;
+ }
+ break;
+
+ case 4:
+ EE_value = (EE_value << 1) | ((value & 2) >> 1);
+
+ if (instr_clocks == 15)
+ {
+ if (WR_EN)
+ {
+ Cart_RAM[EE_addr * 2] = (uint8_t)(EE_value & 0xFF);
+ Cart_RAM[EE_addr * 2 + 1] = (uint8_t)((EE_value & 0xFF00) >> 8);
+ }
+ instr_case = 7;
+ countdown = 8;
+ }
+ break;
+
+ case 5:
+ if ((instr_clocks >= 0) && (instr_clocks <= 7))
+ {
+ DO = ((Cart_RAM[EE_addr * 2 + 1] >> (7 - instr_clocks)) & 1) == 1;
+ }
+ else if ((instr_clocks >= 8) && (instr_clocks <= 15))
+ {
+ DO = ((Cart_RAM[EE_addr * 2] >> (15 - instr_clocks)) & 1) == 1;
+ }
+
+ if (instr_clocks == 15)
+ {
+ instr_case = 7;
+ countdown = 8;
+ }
+ break;
+
+ case 6:
+
+ instr_case = 7;
+ countdown = 8;
+ break;
+
+ case 7:
+ // completed operations take time, so countdown a bit here.
+ // not cycle accurate for operations like writing to all of the EEPROM, but good enough
+
+ break;
+ }
+
+ if (instr_case == 7)
+ {
+ perf_instr = false;
+ countdown_start = true;
+ }
+
+ instr_clocks++;
+ }
+ else if (countdown_start)
+ {
+ countdown--;
+ if (countdown == 0)
+ {
+ countdown_start = false;
+ DO = true;
+
+ //Console.Write("Command Complete: ");
+ //Console.WriteLine(Core.cpu.TotalExecutedCycles);
+ }
+ }
+
+ CS_prev = CS;
+ CLK_prev = CLK;
+ DI_prev = DI;
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region MMM01
+
+ class Mapper_MMM01 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ // nothing to initialize
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x8000)
+ {
+ return ROM[addr];
+ }
+ else
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ return Cart_RAM[addr - 0xA000];
+ }
+ else
+ {
+ return 0;
+ }
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x8000)
+ {
+ SetCDLROM(flags, addr);
+ }
+ else
+ {
+ if (Cart_RAM != null)
+ {
+ SetCDLRAM(flags, addr - 0xA000);
+ }
+ else
+ {
+ return;
+ }
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x8000)
+ {
+ // no mapping hardware available
+ }
+ else
+ {
+ if (Cart_RAM_Length[0] > 0)
+ {
+ Cart_RAM[addr - 0xA000] = value;
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region RockMan8
+
+ class Mapper_RM8 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 1;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+
+ // some games have sizes that result in a degenerate ROM, account for it here
+ if (ROM_mask > 4) { ROM_mask |= 3; }
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ // lowest bank is fixed
+ return ROM[addr];
+
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ // lowest bank is fixed
+ SetCDLROM(flags, addr);
+
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else
+ {
+ return;
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if ((addr >= 0x2000) && (addr < 0x4000))
+ {
+ value &= 0x1F;
+
+ if (value == 0) { value = 1; }
+
+ // in hhugboy they just subtract 8, but to me looks like bits 4 and 5 are just swapped (and bit 4 is unused?)
+ ROM_bank = ((value & 0xF) | ((value & 0x10) >> 1))& ROM_mask;
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region Sachen_MMC1
+
+ class Mapper_Sachen1 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 1;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+ BASE_ROM_Bank = 0;
+ ROM_bank_mask = 0xFF;
+ locked = true;
+ reg_access = false;
+ addr_last = 0;
+ counter = 0;
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ if (locked)
+ {
+ // header is scrambled
+ if ((addr >= 0x100) && (addr < 0x200))
+ {
+ uint32_t temp0 = (addr & 1);
+ uint32_t temp1 = (addr & 2);
+ uint32_t temp4 = (addr & 0x10);
+ uint32_t temp6 = (addr & 0x40);
+
+ temp0 = temp0 << 6;
+ temp1 = temp1 << 3;
+ temp4 = temp4 >> 3;
+ temp6 = temp6 >> 6;
+
+ addr &= 0x1AC;
+ addr |= (uint32_t)(temp0 | temp1 | temp4 | temp6);
+ }
+ addr |= 0x80;
+ }
+
+ return ROM[addr + BASE_ROM_Bank * 0x4000];
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ if (locked)
+ {
+ // header is scrambled
+ if ((addr >= 0x100) && (addr < 0x200))
+ {
+ uint32_t temp0 = (addr & 1);
+ uint32_t temp1 = (addr & 2);
+ uint32_t temp4 = (addr & 0x10);
+ uint32_t temp6 = (addr & 0x40);
+
+ temp0 = temp0 << 6;
+ temp1 = temp1 << 3;
+ temp4 = temp4 >> 3;
+ temp6 = temp6 >> 6;
+
+ addr &= 0x1AC;
+ addr |= (uint32_t)(temp0 | temp1 | temp4 | temp6);
+ }
+ addr |= 0x80;
+ }
+
+ SetCDLROM(flags, addr + BASE_ROM_Bank * 0x4000);
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else
+ {
+ return;
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x2000)
+ {
+ if (reg_access)
+ {
+ BASE_ROM_Bank = value;
+ }
+ }
+ else if (addr < 0x4000)
+ {
+ ROM_bank = (value > 0) ? value : 1;
+
+ if ((value & 0x30) == 0x30)
+ {
+ reg_access = true;
+ }
+ else
+ {
+ reg_access = false;
+ }
+ }
+ else if (addr < 0x6000)
+ {
+ if (reg_access)
+ {
+ ROM_bank_mask = value;
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+
+ void Mapper_Tick()
+ {
+ if (locked)
+ {
+ if (((addr_access[0] & 0x8000) == 0) && ((addr_last & 0x8000) > 0) && (addr_access[0] >= 0x100))
+ {
+ counter++;
+ }
+
+ if (addr_access[0] >= 0x100)
+ {
+ addr_last = addr_access[0];
+ }
+
+ if (counter == 0x30)
+ {
+ locked = false;
+ }
+ }
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region Sachen_MMC2
+
+ class Mapper_Sachen2 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 1;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+ BASE_ROM_Bank = 0;
+ ROM_bank_mask = 0;
+ locked = true;
+ locked_GBC = false;
+ finished = false;
+ reg_access = false;
+ addr_last = 0;
+ counter = 0;
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ // header is scrambled
+ if ((addr >= 0x100) && (addr < 0x200))
+ {
+ uint32_t temp0 = (addr & 1);
+ uint32_t temp1 = (addr & 2);
+ uint32_t temp4 = (addr & 0x10);
+ uint32_t temp6 = (addr & 0x40);
+
+ temp0 = temp0 << 6;
+ temp1 = temp1 << 3;
+ temp4 = temp4 >> 3;
+ temp6 = temp6 >> 6;
+
+ addr &= 0x1AC;
+ addr |= (uint32_t)(temp0 | temp1 | temp4 | temp6);
+ }
+
+ if (locked_GBC) { addr |= 0x80; }
+
+ return ROM[addr + BASE_ROM_Bank * 0x4000];
+ }
+ else if (addr < 0x8000)
+ {
+ uint32_t temp_bank = (ROM_bank & ~ROM_bank_mask) | (ROM_bank_mask & BASE_ROM_Bank);
+ temp_bank &= ROM_mask;
+
+ return ROM[(addr - 0x4000) + temp_bank * 0x4000];
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ // header is scrambled
+ if ((addr >= 0x100) && (addr < 0x200))
+ {
+ uint32_t temp0 = (addr & 1);
+ uint32_t temp1 = (addr & 2);
+ uint32_t temp4 = (addr & 0x10);
+ uint32_t temp6 = (addr & 0x40);
+
+ temp0 = temp0 << 6;
+ temp1 = temp1 << 3;
+ temp4 = temp4 >> 3;
+ temp6 = temp6 >> 6;
+
+ addr &= 0x1AC;
+ addr |= (uint32_t)(temp0 | temp1 | temp4 | temp6);
+ }
+
+ if (locked_GBC) { addr |= 0x80; }
+
+ SetCDLROM(flags, addr + BASE_ROM_Bank * 0x4000);
+ }
+ else if (addr < 0x8000)
+ {
+ uint32_t temp_bank = (ROM_bank & ~ROM_bank_mask) | (ROM_bank_mask & BASE_ROM_Bank);
+ temp_bank &= ROM_mask;
+
+ SetCDLROM(flags, (addr - 0x4000) + temp_bank * 0x4000);
+ }
+ else
+ {
+ return;
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x2000)
+ {
+ if (reg_access)
+ {
+ BASE_ROM_Bank = value;
+ }
+ }
+ else if (addr < 0x4000)
+ {
+ ROM_bank = (value > 0) ? (value) : 1;
+
+ if ((value & 0x30) == 0x30)
+ {
+ reg_access = true;
+ }
+ else
+ {
+ reg_access = false;
+ }
+ }
+ else if (addr < 0x6000)
+ {
+ if (reg_access)
+ {
+ ROM_bank_mask = value;
+ }
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+
+ void Mapper_Tick()
+ {
+ if (locked)
+ {
+ if (((addr_access[0] & 0x8000) == 0) && ((addr_last & 0x8000) > 0) && (addr_access[0] >= 0x100))
+ {
+ counter++;
+ }
+
+ if (addr_access[0] >= 0x100)
+ {
+ addr_last = addr_access[0];
+ }
+
+ if (counter == 0x30)
+ {
+ locked = false;
+ locked_GBC = true;
+ counter = 0;
+ }
+ }
+ else if (locked_GBC)
+ {
+ if (((addr_access[0] & 0x8000) == 0) && ((addr_last & 0x8000) > 0) && (addr_access[0] >= 0x100))
+ {
+ counter++;
+ }
+
+ if (addr_access[0] >= 0x100)
+ {
+ addr_last = addr_access[0];
+ }
+
+ if (counter == 0x30)
+ {
+ locked_GBC = false;
+ finished = true;
+ }
+
+ // The above condition seems to never be reached as described in the mapper notes
+ // so for now add this one
+
+ if ((addr_access[0] == 0x133) && (counter == 1))
+ {
+ locked_GBC = false;
+ finished = true;
+ }
+ }
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region TAMA5
+
+ class Mapper_TAMA5 : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 0;
+ RAM_bank = 0;
+ ROM_mask = ROM_Length[0] / 0x4000 - 1;
+
+ // some games have sizes that result in a degenerate ROM, account for it here
+ if (ROM_mask > 4) { ROM_mask |= 3; }
+
+ RAM_mask = 0;
+ if (Cart_RAM_Length[0] > 0)
+ {
+ RAM_mask = Cart_RAM_Length[0] / 0x2000 - 1;
+ if (Cart_RAM_Length[0] == 0x800) { RAM_mask = 0; }
+ }
+
+ RAM_addr_low = RAM_addr_high = RAM_val_low = RAM_val_high = 0;
+ Chip_return_low = Chip_return_high = 0;
+ halt = false;
+
+ ctrl = 0;
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x4000)
+ {
+ return ROM[addr];
+ }
+ else if (addr < 0x8000)
+ {
+ return ROM[(addr - 0x4000) + ROM_bank * 0x4000];
+ }
+ else
+ {
+
+ switch (ctrl)
+ {
+ case 0xA:
+ // The game won't proceed unless this value (anded with 3) is 1
+ // see bank 0: 0x1A7D to 0x1A89
+ return 1;
+ case 0xC:
+ //Console.WriteLine("read low: " + Chip_return_low);
+ return Chip_return_low;
+ case 0xD:
+ //Console.WriteLine("read high: " + Chip_return_high);
+ return Chip_return_high;
+ }
+
+ return 0x0;
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x4000)
+ {
+ SetCDLROM(flags, addr);
+ }
+ else if (addr < 0x8000)
+ {
+ SetCDLROM(flags, (addr - 0x4000) + ROM_bank * 0x4000);
+ }
+ else
+ {
+
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr == 0xA000)
+ {
+ switch (ctrl)
+ {
+ case 0:
+ ROM_bank &= 0xF0;
+ ROM_bank |= (value & 0xF);
+ break;
+ case 1:
+ ROM_bank &= 0x0F;
+ ROM_bank |= ((value & 0x1) << 4);
+ break;
+ case 4:
+ RAM_val_low = (value & 0xF);
+ break;
+ case 5:
+ RAM_val_high = (value & 0xF);
+ //Cart_RAM[(RAM_addr_high << 4) | RAM_addr_low] = (uint8_t)((RAM_val_high << 4) | RAM_val_low);
+ break;
+ case 6:
+ RAM_addr_high = (value & 1);
+
+ switch ((value & 0xE) >> 1)
+ {
+ case 0:
+ // write to RAM
+ Cart_RAM[(RAM_addr_high << 4) | RAM_addr_low] = (uint8_t)((RAM_val_high << 4) | RAM_val_low);
+ break;
+ case 1:
+ // read from RAM
+ Chip_return_high = (uint8_t)(Cart_RAM[(RAM_addr_high << 4) | RAM_addr_low] >> 4);
+ Chip_return_low = (uint8_t)(Cart_RAM[(RAM_addr_high << 4) | RAM_addr_low] & 0xF);
+ break;
+ case 2:
+ // read from RTC registers
+ if (RAM_addr_low == 3)
+ {
+ Chip_return_high = RTC_regs_TAMA[2];
+ Chip_return_low = RTC_regs_TAMA[1];
+ }
+ else if (RAM_addr_low == 6)
+ {
+ Chip_return_high = RTC_regs_TAMA[4];
+ Chip_return_low = RTC_regs_TAMA[3];
+ }
+ else
+ {
+ Chip_return_high = 1;
+ Chip_return_low = 1;
+ }
+ break;
+ case 3:
+ // write to RTC registers (probably wrong, not well tested)
+ if (RAM_addr_low == 3)
+ {
+ RTC_regs_TAMA[2] = (uint8_t)(RAM_val_high & 0xF);
+ RTC_regs_TAMA[1] = (uint8_t)(RAM_val_low & 0xF);
+ }
+ else if (RAM_addr_low == 6)
+ {
+ RTC_regs_TAMA[4] = (uint8_t)(RAM_val_high & 0xF);
+ RTC_regs_TAMA[3] = (uint8_t)(RAM_val_low & 0xF);
+ }
+ else
+ {
+
+ }
+ break;
+ case 4:
+ // read from seconds register (time changes are checked when it rolls over)
+ Chip_return_low = (uint8_t)(RTC_regs_TAMA[0] & 0xF);
+ break;
+ }
+
+ //Console.WriteLine("CTRL: " + (value >> 1) + " RAM_high:" + RAM_addr_high + " RAM_low: " + RAM_addr_low + " val: " + (uint8_t)((RAM_val_high << 4) | RAM_val_low) + " Cpu: " + Core.cpu.TotalExecutedCycles);
+ break;
+ case 7:
+ RAM_addr_low = (value & 0xF);
+
+ //Console.WriteLine(" RAM_low:" + RAM_addr_low + " Cpu: " + Core.cpu.TotalExecutedCycles);
+ break;
+ }
+ }
+ else if (addr == 0xA001)
+ {
+ ctrl = value;
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+
+ void RTC_Get(uint32_t value, uint32_t index)
+ {
+ if (index < 10)
+ {
+ RTC_regs_TAMA[index] = (uint8_t)value;
+ }
+ else
+ {
+ RTC_offset = value;
+ }
+ }
+
+ void Mapper_Tick()
+ {
+ if (!halt)
+ {
+ RTC_timer++;
+
+ if (RTC_timer == 128)
+ {
+ RTC_timer = 0;
+
+ RTC_low_clock++;
+
+ if (RTC_low_clock == 32768)
+ {
+ RTC_low_clock = 0;
+ RTC_timer = RTC_offset;
+
+ RTC_regs_TAMA[0]++;
+
+ if (RTC_regs_TAMA[0] > 59)
+ {
+ RTC_regs_TAMA[0] = 0;
+ RTC_regs_TAMA[1]++;
+ // 1's digit of minutes
+ if (RTC_regs_TAMA[1] > 9)
+ {
+ RTC_regs_TAMA[1] = 0;
+ RTC_regs_TAMA[2]++;
+ // 10's digit of minutes
+ if (RTC_regs_TAMA[2] > 5)
+ {
+ RTC_regs_TAMA[2] = 0;
+ RTC_regs_TAMA[3]++;
+ // 1's digit of hours
+ if (RTC_regs_TAMA[3] > 9)
+ {
+ RTC_regs_TAMA[3] = 0;
+ RTC_regs_TAMA[4]++;
+ // 10's digit of hours
+ if (RTC_regs_TAMA[4] > 2)
+ {
+ RTC_regs_TAMA[4] = 0;
+ RTC_regs_TAMA[5]++;
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region Wisdom Tree
+
+ class Mapper_WT : public Mapper
+ {
+ public:
+
+ void Reset()
+ {
+ ROM_bank = 0;
+ ROM_mask = ROM_Length[0] / 0x8000 - 1;
+
+ // some games have sizes that result in a degenerate ROM, account for it here
+ if (ROM_mask > 4) { ROM_mask |= 3; }
+ if (ROM_mask > 0x100) { ROM_mask |= 0xFF; }
+ }
+
+ uint8_t ReadMemory(uint32_t addr)
+ {
+ if (addr < 0x8000)
+ {
+ return ROM[ROM_bank * 0x8000 + addr];
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ /*
+ void MapCDL(uint32_t addr, LR35902.eCDLogMemFlags flags)
+ {
+ if (addr < 0x8000)
+ {
+ SetCDLROM(flags, ROM_bank * 0x8000 + addr);
+ }
+ else
+ {
+ return;
+ }
+ }
+ */
+
+ uint8_t PeekMemory(uint32_t addr)
+ {
+ return ReadMemory(addr);
+ }
+
+ void WriteMemory(uint32_t addr, uint8_t value)
+ {
+ if (addr < 0x4000)
+ {
+ ROM_bank = ((addr << 1) & 0x1ff) >> 1;
+ ROM_bank &= ROM_mask;
+ }
+ }
+
+ void PokeMemory(uint32_t addr, uint8_t value)
+ {
+ WriteMemory(addr, value);
+ }
+ };
+
+ #pragma endregion
+}
diff --git a/libHawk/GBHawk/GBHawk/Memory.cpp b/libHawk/GBHawk/GBHawk/Memory.cpp
index a5fa665fb3..c912d03c1e 100644
--- a/libHawk/GBHawk/GBHawk/Memory.cpp
+++ b/libHawk/GBHawk/GBHawk/Memory.cpp
@@ -5,9 +5,9 @@
#include "Memory.h"
#include "LR35902.h"
-#include "PPU_Base.h"
+#include "PPU.h"
#include "GBAudio.h"
-#include "Mapper_Base.h"
+#include "Mappers.h"
#include "SerialPort.h"
#include "Timer.h"
diff --git a/libHawk/GBHawk/GBHawk/Memory.h b/libHawk/GBHawk/GBHawk/Memory.h
index 69d3b9840d..d58e97d60c 100644
--- a/libHawk/GBHawk/GBHawk/Memory.h
+++ b/libHawk/GBHawk/GBHawk/Memory.h
@@ -28,7 +28,7 @@ namespace GBHawk
void WriteMemory(uint32_t addr, uint8_t value);
uint8_t Read_Registers(uint32_t addr);
void Write_Registers(uint32_t addr, uint8_t value);
-
+
#pragma region Declarations
PPU* ppu_pntr = nullptr;
@@ -51,9 +51,7 @@ namespace GBHawk
uint8_t* kb_rows;
// State
- bool PortDEEnabled = false;
bool lagged;
- bool start_pressed;
bool is_GBC;
bool GBC_compat;
bool speed_switch, double_speed;
@@ -61,6 +59,8 @@ namespace GBHawk
bool GB_bios_register;
bool HDMA_transfer;
bool _islag;
+ bool Use_MT;
+ bool has_bat;
uint8_t REG_FFFF, REG_FF0F, REG_FF0F_OLD;
uint8_t _scanlineCallbackLine;
@@ -80,13 +80,14 @@ namespace GBHawk
uint8_t ZP_RAM[0x80] = {};
uint8_t RAM[0x8000] = {};
- uint8_t VRAM[0x10000] = {};
- uint8_t OAM[0x10000] = {};
- uint8_t cart_ram[0x8000] = {};
- uint8_t unmapped[0x400] = {};
+ uint8_t VRAM[0x4000] = {};
+ uint8_t OAM[0xA0] = {};
+ uint8_t header[0x50] = {};
uint32_t _vidbuffer[160 * 144] = {};
uint32_t color_palette[4] = { 0xFFFFFFFF , 0xFFAAAAAA, 0xFF555555, 0xFF000000 };
+ const uint8_t GBA_override[13] = { 0xFF, 0x00, 0xCD, 0x03, 0x35, 0xAA, 0x31, 0x90, 0x94, 0x00, 0x00, 0x00, 0x00 };
+
uint32_t FrameBuffer[160 * 144] = {};
#pragma endregion
@@ -94,12 +95,25 @@ namespace GBHawk
#pragma region Functions
// NOTE: only called when checks pass that the files are correct
- void Load_BIOS(uint8_t* bios, bool GBC_console)
+ void Load_BIOS(uint8_t* bios, bool GBC_console, bool GBC_as_GBA)
{
if (GBC_console)
{
bios_rom = new uint8_t[2304];
memcpy(bios_rom, bios, 2304);
+ is_GBC = true;
+
+ // set up IR variables if it's GBC
+ IR_mask = 0; IR_reg = 0x3E; IR_receive = 2; IR_self = 2; IR_signal = 2;
+
+ if (GBC_as_GBA)
+ {
+ for (int i = 0; i < 13; i++)
+ {
+ bios_rom[i + 0xF3] = (uint8_t)((GBA_override[i] + bios_rom[i + 0xF3]) & 0xFF);
+ }
+ IR_mask = 2;
+ }
}
else
{
@@ -108,14 +122,15 @@ namespace GBHawk
}
}
- void Load_ROM(uint8_t* ext_rom_1, uint32_t ext_rom_size_1, uint32_t ext_rom_mapper_1, uint8_t* ext_rom_2, uint32_t ext_rom_size_2, uint32_t ext_rom_mapper_2)
+ void Load_ROM(uint8_t* ext_rom_1, uint32_t ext_rom_size_1)
{
ROM = new uint8_t[ext_rom_size_1];
memcpy(ROM, ext_rom_1, ext_rom_size_1);
ROM_Length = ext_rom_size_1;
- ROM_Mapper = ext_rom_mapper_1;
+
+ std::memcpy(header, ext_rom_1 + 0x100, 0x50);
}
// Switch Speed (GBC only)
@@ -163,24 +178,20 @@ namespace GBHawk
uint8_t* SaveState(uint8_t* saver)
{
- *saver = (uint8_t)(PortDEEnabled ? 1 : 0); saver++;
*saver = (uint8_t)(lagged ? 1 : 0); saver++;
- *saver = (uint8_t)(start_pressed ? 1 : 0); saver++;
std::memcpy(saver, &RAM, 0x10000); saver += 0x10000;
- std::memcpy(saver, &cart_ram, 0x8000); saver += 0x8000;
+ //std::memcpy(saver, &cart_ram, 0x8000); saver += 0x8000;
return saver;
}
uint8_t* LoadState(uint8_t* loader)
{
- PortDEEnabled = *loader == 1; loader++;
lagged = *loader == 1; loader++;
- start_pressed = *loader == 1; loader++;
std::memcpy(&RAM, loader, 0x10000); loader += 0x10000;
- std::memcpy(&cart_ram, loader, 0x8000); loader += 0x8000;
+ //std::memcpy(&cart_ram, loader, 0x8000); loader += 0x8000;
return loader;
}
diff --git a/libHawk/GBHawk/GBHawk/PPU_Base.cpp b/libHawk/GBHawk/GBHawk/PPU.cpp
similarity index 89%
rename from libHawk/GBHawk/GBHawk/PPU_Base.cpp
rename to libHawk/GBHawk/GBHawk/PPU.cpp
index 5057933fb1..0d18d7dc1d 100644
--- a/libHawk/GBHawk/GBHawk/PPU_Base.cpp
+++ b/libHawk/GBHawk/GBHawk/PPU.cpp
@@ -3,7 +3,7 @@
#include
#include "Memory.h"
-#include "PPU_Base.h"
+#include "PPU.h"
using namespace std;
diff --git a/libHawk/GBHawk/GBHawk/PPU.h b/libHawk/GBHawk/GBHawk/PPU.h
new file mode 100644
index 0000000000..fdac6838fa
--- /dev/null
+++ b/libHawk/GBHawk/GBHawk/PPU.h
@@ -0,0 +1,5003 @@
+#include
+#include
+#include
+#include
+
+using namespace std;
+
+namespace GBHawk
+{
+ class MemoryManager;
+
+ class PPU
+ {
+ public:
+ #pragma region PPU Base
+
+ PPU()
+ {
+
+ }
+
+ uint8_t ReadMemory(uint32_t);
+
+ MemoryManager* mem_ctrl;
+
+ // pointers not stated
+ bool* FlagI = nullptr;
+ bool* in_vblank = nullptr;
+ bool* cpu_halted = nullptr;
+ bool* HDMA_transfer = nullptr;
+ bool* GBC_compat = nullptr;
+
+ uint8_t* cpu_LY = nullptr;
+ uint8_t* REG_FFFF = nullptr;
+ uint8_t* REG_FF0F = nullptr;
+ uint8_t* _scanlineCallbackLine = nullptr;
+ uint8_t* OAM = nullptr;
+ uint8_t* VRAM = nullptr;
+ uint32_t* VRAM_Bank = nullptr;
+ uint32_t* _vidbuffer = nullptr;
+ uint32_t* color_palette = nullptr;
+
+ uint32_t BG_palette[32] = {};
+ uint32_t OBJ_palette[32] = {};
+
+ bool HDMA_active;
+ bool clear_screen;
+
+ // register variables
+ uint8_t LCDC;
+ uint8_t STAT;
+ uint8_t scroll_y;
+ uint8_t scroll_x;
+ uint8_t LY;
+ uint8_t LY_actual;
+ uint8_t LY_inc;
+ uint8_t LYC;
+ uint8_t DMA_addr;
+ uint8_t BGP;
+ uint8_t obj_pal_0;
+ uint8_t obj_pal_1;
+ uint8_t window_y;
+ uint8_t window_x;
+ bool DMA_start;
+ uint32_t DMA_clock;
+ uint32_t DMA_inc;
+ uint8_t DMA_byte;
+
+ // state variables
+ uint32_t cycle;
+ bool LYC_INT;
+ bool HBL_INT;
+ bool VBL_INT;
+ bool OAM_INT;
+ bool LCD_was_off;
+ bool stat_line;
+ bool stat_line_old;
+ // OAM scan
+ bool DMA_OAM_access;
+ bool OAM_access_read;
+ bool OAM_access_write;
+ uint32_t OAM_scan_index;
+ uint32_t SL_sprites_index;
+ uint32_t SL_sprites[40] = {};
+ uint32_t write_sprite;
+ bool no_scan;
+ // render
+ bool VRAM_access_read;
+ bool VRAM_access_write;
+ uint32_t read_case;
+ uint32_t internal_cycle;
+ uint32_t y_tile;
+ uint32_t y_scroll_offset;
+ uint32_t x_tile;
+ uint32_t x_scroll_offset;
+ uint32_t tile_byte;
+ uint32_t sprite_fetch_cycles;
+ bool fetch_sprite;
+ bool going_to_fetch;
+ bool first_fetch;
+ uint32_t sprite_fetch_counter;
+ uint8_t sprite_attr_list[160] = {};
+ uint8_t sprite_pixel_list[160] = {};
+ uint8_t sprite_present_list[160] = {};
+ uint32_t temp_fetch;
+ uint32_t tile_inc;
+ bool pre_render;
+ bool pre_render_2;
+ uint8_t tile_data[3] = {};
+ uint8_t tile_data_latch[3] = {};
+ uint32_t latch_counter;
+ bool latch_new_data;
+ uint32_t render_counter;
+ uint32_t render_offset;
+ int32_t pixel_counter;
+ uint32_t pixel;
+ uint8_t sprite_data[2] = {};
+ uint8_t sprite_sel[2] = {};
+ uint32_t sl_use_index;
+ bool no_sprites;
+ uint32_t SL_sprites_ordered[40] = {}; // (x_end, data_low, data_high, attr)
+ uint32_t evaled_sprites;
+ uint32_t sprite_ordered_index;
+ bool blank_frame;
+ bool window_latch;
+ int32_t consecutive_sprite;
+ uint32_t last_eval;
+
+ uint32_t total_counter;
+ // windowing state
+ uint32_t window_counter;
+ bool window_pre_render;
+ bool window_started;
+ bool window_is_reset;
+ uint32_t window_tile_inc;
+ uint32_t window_y_tile;
+ uint32_t window_x_tile;
+ uint32_t window_y_tile_inc;
+ uint32_t window_x_latch;
+ uint32_t window_y_latch;
+
+ uint32_t hbl_countdown;
+
+ // The following are GBC specific variables
+ // individual uint8_t used in palette colors
+ uint8_t BG_bytes[64] = {};
+ uint8_t OBJ_bytes[64] = {};
+ bool BG_bytes_inc;
+ bool OBJ_bytes_inc;
+ uint8_t BG_bytes_index;
+ uint8_t OBJ_bytes_index;
+ uint8_t BG_transfer_byte;
+ uint8_t OBJ_transfer_byte;
+
+ // HDMA is unique to GBC, do it as part of the PPU tick
+ uint8_t HDMA_src_hi;
+ uint8_t HDMA_src_lo;
+ uint8_t HDMA_dest_hi;
+ uint8_t HDMA_dest_lo;
+ uint32_t HDMA_tick;
+ uint8_t HDMA_byte;
+
+ // controls for tile attributes
+ uint32_t VRAM_sel;
+ bool BG_V_flip;
+ bool HDMA_mode;
+ bool HDMA_run_once;
+ uint32_t cur_DMA_src;
+ uint32_t cur_DMA_dest;
+ uint32_t HDMA_length;
+ uint32_t HDMA_countdown;
+ uint32_t HBL_HDMA_count;
+ uint32_t last_HBL;
+ bool HBL_HDMA_go;
+ bool HBL_test;
+ uint8_t LYC_t;
+ uint32_t LYC_cd;
+
+ // accessors for derived values (GBC only)
+ uint8_t BG_pal_ret() { return (uint8_t)(((BG_bytes_inc ? 1 : 0) << 7) | (BG_bytes_index & 0x3F)); }
+
+ uint8_t OBJ_pal_ret() { return (uint8_t)(((OBJ_bytes_inc ? 1 : 0) << 7) | (OBJ_bytes_index & 0x3F)); }
+
+ uint8_t HDMA_ctrl() { return (uint8_t)(((HDMA_active ? 0 : 1) << 7) | ((HDMA_length >> 4) - 1)); }
+
+ virtual uint8_t ReadReg(uint32_t addr)
+ {
+ return 0;
+ }
+
+ virtual void WriteReg(uint32_t addr, uint8_t value)
+ {
+
+ }
+
+ virtual void tick()
+ {
+
+ }
+
+ // might be needed, not sure yet
+ virtual void latch_delay()
+ {
+
+ }
+
+ virtual void render(uint32_t render_cycle)
+ {
+
+ }
+
+ virtual void process_sprite()
+ {
+
+ }
+
+ // normal DMA moves twice as fast in double speed mode on GBC
+ // So give it it's own function so we can seperate it from PPU tick
+ virtual void DMA_tick()
+ {
+
+ }
+
+ virtual void OAM_scan(uint32_t OAM_cycle)
+ {
+
+ }
+
+ virtual void Reset()
+ {
+
+ }
+
+ // order sprites according to x coordinate
+ // note that for sprites of equal x coordinate, priority goes to first on the list
+ virtual void reorder_and_assemble_sprites()
+ {
+
+ }
+
+ virtual void color_compute_BG()
+ {
+
+ }
+
+ void color_compute_OBJ()
+ {
+
+ }
+
+ #pragma endregion
+
+ #pragma region State Save / Load
+
+ uint8_t* SaveState(uint8_t* saver)
+ {
+ for (int i = 0; i < 32; i++) { saver = int_saver(BG_palette[i], saver); }
+ for (int i = 0; i < 32; i++) { saver = int_saver(OBJ_palette[i], saver); }
+ for (int i = 0; i < 40; i++) { saver = int_saver(SL_sprites[i], saver); }
+
+ for (int i = 0; i < 160; i++) { saver = byte_saver(sprite_attr_list[i], saver); }
+ for (int i = 0; i < 160; i++) { saver = byte_saver(sprite_pixel_list[i], saver); }
+ for (int i = 0; i < 160; i++) { saver = byte_saver(sprite_present_list[i], saver); }
+ for (int i = 0; i < 3; i++) { saver = byte_saver(tile_data[i], saver); }
+ for (int i = 0; i < 3; i++) { saver = byte_saver(tile_data_latch[i], saver); }
+ for (int i = 0; i < 2; i++) { saver = byte_saver(sprite_data[i], saver); }
+ for (int i = 0; i < 2; i++) { saver = byte_saver(sprite_sel[i], saver); }
+ for (int i = 0; i < 40; i++) { saver = int_saver(SL_sprites_ordered[i], saver); }
+
+ saver = bool_saver(HDMA_active, saver);
+ saver = bool_saver(clear_screen, saver);
+
+ saver = byte_saver(LCDC, saver);
+ saver = byte_saver(STAT, saver);
+ saver = byte_saver(scroll_y, saver);
+ saver = byte_saver(scroll_x, saver);
+ saver = byte_saver(LY, saver);
+ saver = byte_saver(LY_actual, saver);
+ saver = byte_saver(LY_inc, saver);
+ saver = byte_saver(LYC, saver);
+ saver = byte_saver(DMA_addr, saver);
+ saver = byte_saver(BGP, saver);
+ saver = byte_saver(obj_pal_0, saver);
+ saver = byte_saver(obj_pal_1, saver);
+ saver = byte_saver(window_y, saver);
+ saver = byte_saver(window_x, saver);
+ saver = bool_saver(DMA_start, saver);
+ saver = int_saver(DMA_clock, saver);
+ saver = int_saver(DMA_inc, saver);
+ saver = byte_saver(DMA_byte, saver);
+
+ saver = int_saver(cycle, saver);
+ saver = bool_saver(LYC_INT, saver);
+ saver = bool_saver(HBL_INT, saver);
+ saver = bool_saver(VBL_INT, saver);
+ saver = bool_saver(OAM_INT, saver);
+ saver = bool_saver(stat_line, saver);
+ saver = bool_saver(stat_line_old, saver);
+ saver = bool_saver(LCD_was_off, saver);
+ saver = int_saver(OAM_scan_index, saver);
+ saver = int_saver(SL_sprites_index, saver);
+ saver = int_saver(write_sprite, saver);
+ saver = bool_saver(no_scan, saver);
+
+ saver = bool_saver(DMA_OAM_access, saver);
+ saver = bool_saver(OAM_access_read, saver);
+ saver = bool_saver(OAM_access_write, saver);
+ saver = bool_saver(VRAM_access_read, saver);
+ saver = bool_saver(VRAM_access_write, saver);
+
+ saver = int_saver(read_case, saver);
+ saver = int_saver(internal_cycle, saver);
+ saver = int_saver(y_tile, saver);
+ saver = int_saver(y_scroll_offset, saver);
+ saver = int_saver(x_tile, saver);
+ saver = int_saver(x_scroll_offset, saver);
+ saver = int_saver(tile_byte, saver);
+ saver = int_saver(sprite_fetch_cycles, saver);
+ saver = bool_saver(fetch_sprite, saver);
+ saver = bool_saver(going_to_fetch, saver);
+ saver = bool_saver(first_fetch, saver);
+ saver = int_saver(sprite_fetch_counter, saver);
+
+ saver = int_saver(temp_fetch, saver);
+ saver = int_saver(tile_inc, saver);
+ saver = bool_saver(pre_render, saver);
+ saver = bool_saver(pre_render_2, saver);
+ saver = int_saver(latch_counter, saver);
+ saver = bool_saver(latch_new_data, saver);
+ saver = int_saver(render_counter, saver);
+ saver = int_saver(render_offset, saver);
+ saver = int_saver(pixel_counter, saver);
+ saver = int_saver(pixel, saver);
+ saver = int_saver(sl_use_index, saver);
+ saver = bool_saver(no_sprites, saver);
+ saver = int_saver(evaled_sprites, saver);
+ saver = int_saver(sprite_ordered_index, saver);
+ saver = bool_saver(blank_frame, saver);
+ saver = bool_saver(window_latch, saver);
+ saver = int_saver(consecutive_sprite, saver);
+ saver = int_saver(last_eval, saver);
+
+ saver = int_saver(window_counter, saver);
+ saver = bool_saver(window_pre_render, saver);
+ saver = bool_saver(window_started, saver);
+ saver = bool_saver(window_is_reset, saver);
+ saver = int_saver(window_tile_inc, saver);
+ saver = int_saver(window_y_tile, saver);
+ saver = int_saver(window_x_tile, saver);
+ saver = int_saver(window_y_tile_inc, saver);
+ saver = int_saver(window_x_latch, saver);
+ saver = int_saver(window_y_latch, saver);
+
+ saver = int_saver(hbl_countdown, saver);
+
+ // The following are GBC specific variables
+ for (int i = 0; i < 64; i++) { saver = byte_saver(BG_bytes[i], saver); }
+ for (int i = 0; i < 64; i++) { saver = byte_saver(OBJ_bytes[i], saver); }
+
+ saver = byte_saver(BG_transfer_byte, saver);
+ saver = byte_saver(OBJ_transfer_byte, saver);
+ saver = byte_saver(HDMA_src_hi, saver);
+ saver = byte_saver(HDMA_src_lo, saver);
+ saver = byte_saver(HDMA_dest_hi, saver);
+ saver = byte_saver(HDMA_dest_lo, saver);
+ saver = int_saver(HDMA_tick, saver);
+ saver = byte_saver(HDMA_byte, saver);
+
+ saver = int_saver(VRAM_sel, saver);
+ saver = bool_saver(BG_V_flip, saver);
+ saver = bool_saver(HDMA_mode, saver);
+ saver = bool_saver(HDMA_run_once, saver);
+ saver = int_saver(cur_DMA_src, saver);
+ saver = int_saver(cur_DMA_dest, saver);
+ saver = int_saver(HDMA_length, saver);
+ saver = int_saver(HDMA_countdown, saver);
+ saver = int_saver(HBL_HDMA_count, saver);
+ saver = int_saver(last_HBL, saver);
+ saver = bool_saver(HBL_HDMA_go, saver);
+ saver = bool_saver(HBL_test, saver);
+
+ saver = bool_saver(BG_bytes_inc, saver);
+ saver = bool_saver(OBJ_bytes_inc, saver);
+ saver = byte_saver(BG_bytes_index, saver);
+ saver = byte_saver(OBJ_bytes_index, saver);
+
+ saver = byte_saver(LYC_t, saver);
+ saver = int_saver(LYC_cd, saver);
+
+ return saver;
+ }
+
+ uint8_t* LoadState(uint8_t* loader)
+ {
+ for (int i = 0; i < 32; i++) { loader = int_loader(&BG_palette[i], loader); }
+ for (int i = 0; i < 32; i++) { loader = int_loader(&OBJ_palette[i], loader); }
+ for (int i = 0; i < 40; i++) { loader = int_loader(&SL_sprites[i], loader); }
+
+ for (int i = 0; i < 160; i++) { loader = byte_loader(&sprite_attr_list[i], loader); }
+ for (int i = 0; i < 160; i++) { loader = byte_loader(&sprite_pixel_list[i], loader); }
+ for (int i = 0; i < 160; i++) { loader = byte_loader(&sprite_present_list[i], loader); }
+ for (int i = 0; i < 3; i++) { loader = byte_loader(&tile_data[i], loader); }
+ for (int i = 0; i < 3; i++) { loader = byte_loader(&tile_data_latch[i], loader); }
+ for (int i = 0; i < 2; i++) { loader = byte_loader(&sprite_data[i], loader); }
+ for (int i = 0; i < 2; i++) { loader = byte_loader(&sprite_sel[i], loader); }
+ for (int i = 0; i < 40; i++) { loader = int_loader(&SL_sprites_ordered[i], loader); }
+
+ loader = bool_loader(&HDMA_active, loader);
+ loader = bool_loader(&clear_screen, loader);
+
+ loader = byte_loader(&LCDC, loader);
+ loader = byte_loader(&STAT, loader);
+ loader = byte_loader(&scroll_y, loader);
+ loader = byte_loader(&scroll_x, loader);
+ loader = byte_loader(&LY, loader);
+ loader = byte_loader(&LY_actual, loader);
+ loader = byte_loader(&LY_inc, loader);
+ loader = byte_loader(&LYC, loader);
+ loader = byte_loader(&DMA_addr, loader);
+ loader = byte_loader(&BGP, loader);
+ loader = byte_loader(&obj_pal_0, loader);
+ loader = byte_loader(&obj_pal_1, loader);
+ loader = byte_loader(&window_y, loader);
+ loader = byte_loader(&window_x, loader);
+ loader = bool_loader(&DMA_start, loader);
+ loader = int_loader(&DMA_clock, loader);
+ loader = int_loader(&DMA_inc, loader);
+ loader = byte_loader(&DMA_byte, loader);
+
+ loader = int_loader(&cycle, loader);
+ loader = bool_loader(&LYC_INT, loader);
+ loader = bool_loader(&HBL_INT, loader);
+ loader = bool_loader(&VBL_INT, loader);
+ loader = bool_loader(&OAM_INT, loader);
+ loader = bool_loader(&stat_line, loader);
+ loader = bool_loader(&stat_line_old, loader);
+ loader = bool_loader(&LCD_was_off, loader);
+ loader = int_loader(&OAM_scan_index, loader);
+ loader = int_loader(&SL_sprites_index, loader);
+ loader = int_loader(&write_sprite, loader);
+ loader = bool_loader(&no_scan, loader);
+
+ loader = bool_loader(&DMA_OAM_access, loader);
+ loader = bool_loader(&OAM_access_read, loader);
+ loader = bool_loader(&OAM_access_write, loader);
+ loader = bool_loader(&VRAM_access_read, loader);
+ loader = bool_loader(&VRAM_access_write, loader);
+
+ loader = int_loader(&read_case, loader);
+ loader = int_loader(&internal_cycle, loader);
+ loader = int_loader(&y_tile, loader);
+ loader = int_loader(&y_scroll_offset, loader);
+ loader = int_loader(&x_tile, loader);
+ loader = int_loader(&x_scroll_offset, loader);
+ loader = int_loader(&tile_byte, loader);
+ loader = int_loader(&sprite_fetch_cycles, loader);
+ loader = bool_loader(&fetch_sprite, loader);
+ loader = bool_loader(&going_to_fetch, loader);
+ loader = bool_loader(&first_fetch, loader);
+ loader = int_loader(&sprite_fetch_counter, loader);
+
+ loader = int_loader(&temp_fetch, loader);
+ loader = int_loader(&tile_inc, loader);
+ loader = bool_loader(&pre_render, loader);
+ loader = bool_loader(&pre_render_2, loader);
+ loader = int_loader(&latch_counter, loader);
+ loader = bool_loader(&latch_new_data, loader);
+ loader = int_loader(&render_counter, loader);
+ loader = int_loader(&render_offset, loader);
+ loader = sint_loader(&pixel_counter, loader);
+ loader = int_loader(&pixel, loader);
+ loader = int_loader(&sl_use_index, loader);
+ loader = bool_loader(&no_sprites, loader);
+ loader = int_loader(&evaled_sprites, loader);
+ loader = int_loader(&sprite_ordered_index, loader);
+ loader = bool_loader(&blank_frame, loader);
+ loader = bool_loader(&window_latch, loader);
+ loader = sint_loader(&consecutive_sprite, loader);
+ loader = int_loader(&last_eval, loader);
+
+ loader = int_loader(&window_counter, loader);
+ loader = bool_loader(&window_pre_render, loader);
+ loader = bool_loader(&window_started, loader);
+ loader = bool_loader(&window_is_reset, loader);
+ loader = int_loader(&window_tile_inc, loader);
+ loader = int_loader(&window_y_tile, loader);
+ loader = int_loader(&window_x_tile, loader);
+ loader = int_loader(&window_y_tile_inc, loader);
+ loader = int_loader(&window_x_latch, loader);
+ loader = int_loader(&window_y_latch, loader);
+
+ loader = int_loader(&hbl_countdown, loader);
+
+ // The following are GBC specific variables
+ for (int i = 0; i < 64; i++) { loader = byte_loader(&BG_bytes[i], loader); }
+ for (int i = 0; i < 64; i++) { loader = byte_loader(&OBJ_bytes[i], loader); }
+
+ loader = byte_loader(&BG_transfer_byte, loader);
+ loader = byte_loader(&OBJ_transfer_byte, loader);
+ loader = byte_loader(&HDMA_src_hi, loader);
+ loader = byte_loader(&HDMA_src_lo, loader);
+ loader = byte_loader(&HDMA_dest_hi, loader);
+ loader = byte_loader(&HDMA_dest_lo, loader);
+ loader = int_loader(&HDMA_tick, loader);
+ loader = byte_loader(&HDMA_byte, loader);
+
+ loader = int_loader(&VRAM_sel, loader);
+ loader = bool_loader(&BG_V_flip, loader);
+ loader = bool_loader(&HDMA_mode, loader);
+ loader = bool_loader(&HDMA_run_once, loader);
+ loader = int_loader(&cur_DMA_src, loader);
+ loader = int_loader(&cur_DMA_dest, loader);
+ loader = int_loader(&HDMA_length, loader);
+ loader = int_loader(&HDMA_countdown, loader);
+ loader = int_loader(&HBL_HDMA_count, loader);
+ loader = int_loader(&last_HBL, loader);
+ loader = bool_loader(&HBL_HDMA_go, loader);
+ loader = bool_loader(&HBL_test, loader);
+
+ loader = bool_loader(&BG_bytes_inc, loader);
+ loader = bool_loader(&OBJ_bytes_inc, loader);
+ loader = byte_loader(&BG_bytes_index, loader);
+ loader = byte_loader(&OBJ_bytes_index, loader);
+
+ loader = byte_loader(&LYC_t, loader);
+ loader = int_loader(&LYC_cd, loader);
+
+ return loader;
+ }
+
+ uint8_t* bool_saver(bool to_save, uint8_t* saver)
+ {
+ *saver = (uint8_t)(to_save ? 1 : 0); saver++;
+
+ return saver;
+ }
+
+ uint8_t* byte_saver(uint8_t to_save, uint8_t* saver)
+ {
+ *saver = to_save; saver++;
+
+ return saver;
+ }
+
+ uint8_t* int_saver(uint32_t to_save, uint8_t* saver)
+ {
+ *saver = (uint8_t)(to_save & 0xFF); saver++; *saver = (uint8_t)((to_save >> 8) & 0xFF); saver++;
+ *saver = (uint8_t)((to_save >> 16) & 0xFF); saver++; *saver = (uint8_t)((to_save >> 24) & 0xFF); saver++;
+
+ return saver;
+ }
+
+ uint8_t* bool_loader(bool* to_load, uint8_t* loader)
+ {
+ to_load[0] = *to_load == 1; loader++;
+
+ return loader;
+ }
+
+ uint8_t* byte_loader(uint8_t* to_load, uint8_t* loader)
+ {
+ to_load[0] = *loader; loader++;
+
+ return loader;
+ }
+
+ uint8_t* int_loader(uint32_t* to_load, uint8_t* loader)
+ {
+ to_load[0] = *loader; loader++; to_load[0] |= (*loader << 8); loader++;
+ to_load[0] |= (*loader << 16); loader++; to_load[0] |= (*loader << 24); loader++;
+
+ return loader;
+ }
+
+ uint8_t* sint_loader(int32_t* to_load, uint8_t* loader)
+ {
+ to_load[0] = *loader; loader++; to_load[0] |= (*loader << 8); loader++;
+ to_load[0] |= (*loader << 16); loader++; to_load[0] |= (*loader << 24); loader++;
+
+ return loader;
+ }
+
+ #pragma endregion
+ };
+
+ #pragma region GB_PPU
+
+ class GB_PPU : public PPU
+ {
+ public:
+
+ GB_PPU() {}
+
+ uint8_t ReadReg(uint32_t addr)
+ {
+ uint8_t ret = 0;
+
+ switch (addr)
+ {
+ case 0xFF40: ret = LCDC; break; // LCDC
+ case 0xFF41: ret = STAT; break; // STAT
+ case 0xFF42: ret = scroll_y; break; // SCY
+ case 0xFF43: ret = scroll_x; break; // SCX
+ case 0xFF44: ret = LY; break; // LY
+ case 0xFF45: ret = LYC; break; // LYC
+ case 0xFF46: ret = DMA_addr; break; // DMA
+ case 0xFF47: ret = BGP; break; // BGP
+ case 0xFF48: ret = obj_pal_0; break; // OBP0
+ case 0xFF49: ret = obj_pal_1; break; // OBP1
+ case 0xFF4A: ret = window_y; break; // WY
+ case 0xFF4B: ret = window_x; break; // WX
+ }
+
+ return ret;
+ }
+
+ void WriteReg(uint32_t addr, uint8_t value)
+ {
+ //Console.WriteLine((addr - 0xFF40) + " " + value + " " + LY + " " + cycle + " " + LCDC.Bit(7));
+ switch (addr)
+ {
+ case 0xFF40: // LCDC
+ if (((LCDC & 0x80) > 0) && !((value & 0x80) > 0))
+ {
+ VRAM_access_read = true;
+ VRAM_access_write = true;
+ OAM_access_read = true;
+ OAM_access_write = true;
+
+ clear_screen = true;
+ }
+
+ if (!((LCDC & 0x80) > 0) && ((value & 0x80) > 0))
+ {
+ // don't draw for one frame after turning on
+ blank_frame = true;
+ }
+
+ LCDC = value;
+ break;
+ case 0xFF41: // STAT
+ // writing to STAT during mode 0 or 1 causes a STAT IRQ
+ // this appears to be a glitchy LYC compare
+ if (((LCDC & 0x80) > 0))
+ {
+ if (((STAT & 3) == 0) || ((STAT & 3) == 1))
+ {
+ LYC_INT = true;
+ //if (Core.REG_FFFF.Bit(1)) { Core.cpu.FlagI = true; }
+ //Core.REG_FF0F |= 0x02;
+ }
+ else
+ {
+ if (((value & 0x40) > 0))
+ {
+ if (LY == LYC) { LYC_INT = true; }
+ else { LYC_INT = false; }
+ }
+ }
+ }
+ STAT = (uint8_t)((value & 0xF8) | (STAT & 7) | 0x80);
+
+ //if (!STAT.Bit(6)) { LYC_INT = false; }
+ if (!((STAT & 0x10) > 0)) { VBL_INT = false; }
+ break;
+ case 0xFF42: // SCY
+ scroll_y = value;
+ break;
+ case 0xFF43: // SCX
+ scroll_x = value;
+ break;
+ case 0xFF44: // LY
+ LY = 0; /*reset*/
+ break;
+ case 0xFF45: // LYC
+ LYC = value;
+ if (((LCDC & 0x80) > 0))
+ {
+ if (LY != LYC) { STAT &= 0xFB; LYC_INT = false; }
+ else { STAT |= 0x4; LYC_INT = true; }
+
+ // special case: the transition from 153 -> 0 acts strange
+ // the comparison to 153 expects to be true for longer then the value of LY expects to be 153
+ // this appears to be fixed in CGB
+ if ((LY_inc == 0) && cycle == 8)
+ {
+ if (153 != LYC) { STAT &= 0xFB; LYC_INT = false; }
+ else { STAT |= 0x4; LYC_INT = true; }
+ }
+ }
+ break;
+ case 0xFF46: // DMA
+ DMA_addr = value;
+ DMA_start = true;
+ DMA_OAM_access = true;
+ DMA_clock = 0;
+ DMA_inc = 0;
+ break;
+ case 0xFF47: // BGP
+ BGP = value;
+ break;
+ case 0xFF48: // OBP0
+ obj_pal_0 = value;
+ break;
+ case 0xFF49: // OBP1
+ obj_pal_1 = value;
+ break;
+ case 0xFF4A: // WY
+ window_y = value;
+ break;
+ case 0xFF4B: // WX
+ window_x = value;
+ break;
+ }
+ }
+
+ void tick()
+ {
+ // the ppu only does anything if it is turned on via bit 7 of LCDC
+ if (((LCDC & 0x80) > 0))
+ {
+ // start the next scanline
+ if (cycle == 456)
+ {
+ // scanline callback
+ if ((LY + LY_inc) == _scanlineCallbackLine[0])
+ {
+ //if (Core._scanlineCallback != null)
+ //{
+ //Core.GetGPU();
+ //Core._scanlineCallback(LCDC);
+ //}
+ }
+
+ cycle = 0;
+ LY += LY_inc;
+ cpu_LY[0] = LY;
+
+ no_scan = false;
+
+ if (LY == 0 && LY_inc == 0)
+ {
+ LY_inc = 1;
+ in_vblank[0] = false;
+
+ STAT &= 0xFC;
+
+ // special note here, the y coordiate of the window is kept if the window is deactivated
+ // meaning it will pick up where it left off if re-enabled later
+ // so we don't reset it in the scanline loop
+ window_y_tile = 0;
+ window_y_latch = window_y;
+ window_y_tile_inc = 0;
+ window_started = false;
+ if (!((LCDC & 0x20) > 0)) { window_is_reset = true; }
+ }
+
+ // Automatically restore access to VRAM at this time (force end drawing)
+ // Who Framed Roger Rabbit seems to run into this.
+ VRAM_access_write = true;
+ VRAM_access_read = true;
+
+ if (LY == 144)
+ {
+ in_vblank[0] = true;
+ }
+ }
+
+ // exit vblank if LCD went from off to on
+ if (LCD_was_off)
+ {
+ //VBL_INT = false;
+ in_vblank[0] = false;
+ LCD_was_off = false;
+
+ // we exit vblank into mode 0 for 4 cycles
+ // but no hblank interrupt, presumably this only happens transitioning from mode 3 to 0
+ STAT &= 0xFC;
+
+ // also the LCD doesn't turn on right away
+
+ // also, the LCD does not enter mode 2 on scanline 0 when first turned on
+ no_scan = true;
+ cycle = 8;
+ }
+
+ // the VBL stat is continuously asserted
+ if (LY >= 144)
+ {
+ if (((STAT & 0x10) > 0))
+ {
+ if ((cycle >= 4) && (LY == 144))
+ {
+ VBL_INT = true;
+ }
+ else if (LY > 144)
+ {
+ VBL_INT = true;
+ }
+ }
+
+ if ((cycle == 2) && (LY == 144))
+ {
+ // there is an edge case where a VBL INT is triggered if STAT bit 5 is set
+ if (((STAT & 0x20) > 0)) { VBL_INT = true; }
+ }
+
+ if ((cycle == 4) && (LY == 144))
+ {
+ HBL_INT = false;
+
+ // set STAT mode to 1 (VBlank) and interrupt flag if it is enabled
+ STAT &= 0xFC;
+ STAT |= 0x01;
+
+ if (((REG_FFFF[0] & 0x1) > 0)) { FlagI[0] = true; }
+ REG_FF0F[0] |= 0x01;
+ }
+
+ if ((cycle == 4) && (LY == 144))
+ {
+ if (((STAT & 0x20) > 0)) { VBL_INT = false; }
+ }
+
+ if ((cycle == 6) && (LY == 153))
+ {
+ LY = 0;
+ LY_inc = 0;
+ cpu_LY[0] = LY;
+ }
+ }
+
+ if (!in_vblank[0])
+ {
+ if (no_scan)
+ {
+ // timings are slightly different if we just turned on the LCD
+ // there is no mode 2 (presumably it missed the trigger)
+ if (cycle < 85)
+ {
+ if (cycle == 8)
+ {
+ // clear the sprite table
+ for (uint32_t k = 0; k < 10; k++)
+ {
+ SL_sprites[k * 4] = 0;
+ SL_sprites[k * 4 + 1] = 0;
+ SL_sprites[k * 4 + 2] = 0;
+ SL_sprites[k * 4 + 3] = 0;
+ }
+
+ if (LY != LYC)
+ {
+ LYC_INT = false;
+ STAT &= 0xFB;
+ }
+
+ if ((LY == LYC) && !((STAT & 0x4) > 0))
+ {
+ // set STAT coincidence FLAG and interrupt flag if it is enabled
+ STAT |= 0x04;
+ if (((STAT & 0x40) > 0)) { LYC_INT = true; }
+ }
+ }
+
+ if (cycle == 84)
+ {
+ STAT &= 0xFC;
+ STAT |= 0x03;
+ OAM_INT = false;
+
+ OAM_access_read = false;
+ OAM_access_write = false;
+ VRAM_access_read = false;
+ VRAM_access_write = false;
+ }
+ }
+ else
+ {
+ if (cycle >= 85)
+ {
+ if (cycle == 85)
+ {
+ // x-scroll is expected to be latched one cycle later
+ // this is fine since nothing has started in the rendering until the second cycle
+ // calculate the column number of the tile to start with
+ x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
+ render_offset = scroll_x % 8;
+ }
+
+ // render the screen and handle hblank
+ render(cycle - 85);
+ }
+ }
+ }
+ else
+ {
+ if (cycle <= 80)
+ {
+ if (cycle == 2)
+ {
+ if (LY != 0)
+ {
+ HBL_INT = false;
+ if (((STAT & 0x20) > 0)) { OAM_INT = true; }
+ }
+ }
+ else if (cycle == 4)
+ {
+ // apparently, writes can make it to OAM one cycle longer then reads
+ OAM_access_write = false;
+
+ // here mode 2 will be set to true and interrupts fired if enabled
+ STAT &= 0xFC;
+ STAT |= 0x2;
+
+ if (LY == 0)
+ {
+ VBL_INT = false;
+ if (((STAT & 0x20) > 0)) { OAM_INT = true; }
+ }
+ }
+
+ if (cycle == 80)
+ {
+ OAM_access_read = false;
+ OAM_access_write = true;
+ VRAM_access_read = false;
+ }
+ else
+ {
+ // here OAM scanning is performed
+ OAM_scan(cycle);
+ }
+ }
+ else if (cycle >= 83)
+ {
+ if (cycle == 84)
+ {
+ STAT &= 0xFC;
+ STAT |= 0x03;
+ OAM_INT = false;
+ OAM_access_write = false;
+ VRAM_access_write = false;
+
+ // x-scroll is expected to be latched one cycle later
+ // this is fine since nothing has started in the rendering until the second cycle
+ // calculate the column number of the tile to start with
+ x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
+ render_offset = scroll_x % 8;
+ }
+
+ // render the screen and handle hblank
+ render(cycle - 83);
+ }
+ }
+ }
+
+ if (LY_inc == 0)
+ {
+ if (cycle == 10)
+ {
+ LYC_INT = false;
+ STAT &= 0xFB;
+ }
+ else if (cycle == 12)
+ {
+ // Special case of LY = LYC
+ if ((LY == LYC) && !((STAT & 0x4) > 0))
+ {
+ // set STAT coincidence FLAG and interrupt flag if it is enabled
+ STAT |= 0x04;
+ if (((STAT & 0x40) > 0)) { LYC_INT = true; }
+ }
+ }
+ }
+
+ // here LY=LYC will be asserted or cleared (but only if LY isnt 0 as that's a special case)
+ if ((cycle == 2) && (LY != 0))
+ {
+ if (LY_inc == 1)
+ {
+ LYC_INT = false;
+ STAT &= 0xFB;
+ }
+ }
+ else if ((cycle == 4) && (LY != 0))
+ {
+ if ((LY == LYC) && !((STAT & 0x4) > 0))
+ {
+ // set STAT coincidence FLAG and interrupt flag if it is enabled
+ STAT |= 0x04;
+ if (((STAT & 0x40) > 0)) { LYC_INT = true; }
+ }
+ }
+
+ cycle++;
+ }
+ else
+ {
+ STAT &= 0xFC;
+
+ VBL_INT = LYC_INT = HBL_INT = OAM_INT = false;
+
+ in_vblank[0] = true;
+
+ LCD_was_off = true;
+
+ LY = 0;
+ cpu_LY[0] = LY;
+
+ cycle = 0;
+ }
+
+ // assert the STAT IRQ line if the line went from zero to 1
+ stat_line = VBL_INT | LYC_INT | HBL_INT | OAM_INT;
+
+ if (stat_line && !stat_line_old)
+ {
+ if (((REG_FFFF[0] & 0x2) > 0)) { FlagI[0] = true; }
+ REG_FF0F[0] |= 0x02;
+ }
+
+ stat_line_old = stat_line;
+
+ // process latch delays
+ //latch_delay();
+ }
+
+ // might be needed, not sure yet
+ void latch_delay()
+ {
+
+ }
+
+ void render(uint32_t render_cycle)
+ {
+ // we are now in STAT mode 3
+ // NOTE: presumably the first necessary sprite is fetched at sprite evaulation
+ // i.e. just keeping track of the lowest x-value sprite
+ if (render_cycle == 0)
+ {
+ /*
+ OAM_access_read = false;
+ OAM_access_write = true;
+ VRAM_access_read = false;
+ */
+ // window X is latched for the scanline, mid-line changes have no effect
+ window_x_latch = window_x;
+
+ OAM_scan_index = 0;
+ read_case = 0;
+ internal_cycle = 0;
+ pre_render = true;
+ pre_render_2 = true;
+ tile_inc = 0;
+ pixel_counter = -8;
+ sl_use_index = 0;
+ fetch_sprite = false;
+ going_to_fetch = false;
+ first_fetch = true;
+ consecutive_sprite = (int32_t)(render_offset * (-1)) + 8;
+ no_sprites = false;
+ evaled_sprites = 0;
+ window_pre_render = false;
+ window_latch = ((LCDC & 0x20) > 0);
+
+ total_counter = 0;
+
+ // TODO: If Window is turned on midscanline what happens? When is this check done exactly?
+ if ((window_started && window_latch) || (window_is_reset && !window_latch && (LY >= window_y_latch)))
+ {
+ window_y_tile_inc++;
+ if (window_y_tile_inc == 8)
+ {
+ window_y_tile_inc = 0;
+ window_y_tile++;
+ window_y_tile %= 32;
+ }
+ }
+ window_started = false;
+
+ if (SL_sprites_index == 0) { no_sprites = true; }
+ // it is much easier to process sprites if we order them according to the rules of sprite priority first
+ if (!no_sprites) { reorder_and_assemble_sprites(); }
+ }
+
+ // before anything else, we have to check if windowing is in effect
+ if (window_latch && !window_started && (LY >= window_y_latch) && (pixel_counter >= (int32_t)(window_x_latch - 7)) && (window_x_latch < 167))
+ {
+ /*
+ Console.Write(LY);
+ Console.Write(" ");
+ Console.Write(cycle);
+ Console.Write(" ");
+ Console.Write(window_y_tile_inc);
+ Console.Write(" ");
+ Console.Write(window_x_latch);
+ Console.Write(" ");
+ Console.WriteLine(pixel_counter);
+ */
+ if (window_x_latch == 0)
+ {
+ // if the window starts at zero, we still do the first access to the BG
+ // but then restart all over again at the window
+ if ((render_offset % 7) <= 6)
+ {
+ read_case = 9;
+ }
+ else
+ {
+ read_case = 10;
+ }
+ }
+ else
+ {
+ read_case = 4;
+ }
+
+ window_pre_render = true;
+
+ window_counter = 0;
+ render_counter = 0;
+
+ window_x_tile = (uint32_t)floor((float)(pixel_counter - (window_x_latch - 7)) / 8.0);
+
+ window_tile_inc = 0;
+ window_started = true;
+ window_is_reset = false;
+ }
+
+ if (!pre_render && !fetch_sprite)
+ {
+ // start shifting data into the LCD
+ if (render_counter >= (render_offset + 8))
+ {
+
+ pixel = (tile_data_latch[0] & (1 << (7 - (render_counter % 8)))) > 0 ? 1 : 0;
+ pixel |= (tile_data_latch[1] & (1 << (7 - (render_counter % 8)))) > 0 ? 2 : 0;
+
+ uint32_t ref_pixel = pixel;
+ if (((LCDC & 0x1) > 0))
+ {
+ pixel = (BGP >> (pixel * 2)) & 3;
+ }
+ else
+ {
+ pixel = 0;
+ }
+
+ // now we have the BG pixel, we next need the sprite pixel
+ if (!no_sprites)
+ {
+ bool have_sprite = false;
+ uint32_t s_pixel = 0;
+ uint32_t sprite_attr = 0;
+
+ if (sprite_present_list[pixel_counter] == 1)
+ {
+ have_sprite = true;
+ s_pixel = sprite_pixel_list[pixel_counter];
+ sprite_attr = sprite_attr_list[pixel_counter];
+ }
+
+ if (have_sprite)
+ {
+ bool use_sprite = false;
+ if (((LCDC & 0x2) > 0))
+ {
+ if (!((sprite_attr & 0x80) > 0))
+ {
+ use_sprite = true;
+ }
+ else if (ref_pixel == 0)
+ {
+ use_sprite = true;
+ }
+
+ if (!((LCDC & 0x1) > 0))
+ {
+ use_sprite = true;
+ }
+ }
+
+ if (use_sprite)
+ {
+ if (((sprite_attr & 0x10) > 0))
+ {
+ pixel = (obj_pal_1 >> (s_pixel * 2)) & 3;
+ }
+ else
+ {
+ pixel = (obj_pal_0 >> (s_pixel * 2)) & 3;
+ }
+ }
+ }
+ }
+
+ // based on sprite priority and pixel values, pick a final pixel color
+ _vidbuffer[LY * 160 + pixel_counter] = (uint32_t)color_palette[pixel];
+ pixel_counter++;
+
+ if (pixel_counter == 160)
+ {
+ read_case = 8;
+ }
+ }
+ else if (pixel_counter < 0)
+ {
+ pixel_counter++;
+ }
+ render_counter++;
+ }
+
+ if (!fetch_sprite)
+ {
+ if (!pre_render_2)
+ {
+ // before we go on to read case 3, we need to know if we stall there or not
+ // Gekkio's tests show that if sprites are at position 0 or 1 (mod 8)
+ // then it takes an extra cycle (1 or 2 more t-states) to process them
+ // Also, on DMG only, this process only runs if sprites are on in the LCDC (on GBC it always runs)
+ if (!no_sprites && (pixel_counter < 160) && ((LCDC & 0x2) > 0))
+ {
+ for (uint32_t i = 0; i < SL_sprites_index; i++)
+ {
+ if ((pixel_counter >= (int32_t)(SL_sprites[i * 4 + 1] - 8)) &&
+ (pixel_counter < (int32_t)(SL_sprites[i * 4 + 1])) &&
+ !((evaled_sprites & (1 << i)) > 0))
+ {
+ going_to_fetch = true;
+ fetch_sprite = true;
+ }
+ }
+ }
+ }
+
+ switch (read_case)
+ {
+ case 0: // read a background tile
+ if ((internal_cycle % 2) == 1)
+ {
+ // calculate the row number of the tiles to be fetched
+ y_tile = ((uint32_t)floor((float)((uint32_t)scroll_y + LY) / 8.0)) % 32;
+
+ temp_fetch = y_tile * 32 + (x_tile + tile_inc) % 32;
+ tile_byte = VRAM[0x1800 + (((LCDC & 0x8) > 0) ? 1 : 0) * 0x400 + temp_fetch];
+
+ read_case = 1;
+ if (!pre_render)
+ {
+ tile_inc++;
+ }
+ }
+ break;
+
+ case 1: // read from tile graphics (0)
+ if ((internal_cycle % 2) == 1)
+ {
+ y_scroll_offset = (scroll_y + LY) % 8;
+
+ if (((LCDC & 0x10) > 0))
+ {
+ tile_data[0] = VRAM[tile_byte * 16 + y_scroll_offset * 2];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed uint8_t
+ if (((tile_byte & 0x80) > 0))
+ {
+ tile_byte -= 256;
+ }
+ tile_data[0] = VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2];
+ }
+
+ read_case = 2;
+ }
+ break;
+
+ case 2: // read from tile graphics (1)
+ if ((internal_cycle % 2) == 0)
+ {
+ pre_render_2 = false;
+ }
+ else
+ {
+ y_scroll_offset = (scroll_y + LY) % 8;
+
+ if (((LCDC & 0x10) > 0))
+ {
+ // if LCDC somehow changed between the two reads, make sure we have a positive number
+ if (tile_byte < 0)
+ {
+ tile_byte += 256;
+ }
+
+ tile_data[1] = VRAM[tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed uint8_t
+ if (((tile_byte & 0x80) > 0) && tile_byte > 0)
+ {
+ tile_byte -= 256;
+ }
+
+ tile_data[1] = VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+
+ if (pre_render)
+ {
+ // here we set up rendering
+ pre_render = false;
+
+ render_counter = 0;
+ latch_counter = 0;
+ read_case = 0;
+ }
+ else
+ {
+ read_case = 3;
+ }
+ }
+ break;
+
+ case 3: // read from sprite data
+ if ((internal_cycle % 2) == 1)
+ {
+ read_case = 0;
+ latch_new_data = true;
+ }
+ break;
+
+ case 4: // read from window data
+ if ((window_counter % 2) == 1)
+ {
+ temp_fetch = window_y_tile * 32 + (window_x_tile + window_tile_inc) % 32;
+ tile_byte = VRAM[0x1800 + (((LCDC & 0x40) > 0) ? 1 : 0) * 0x400 + temp_fetch];
+
+ window_tile_inc++;
+ read_case = 5;
+ }
+ window_counter++;
+ break;
+
+ case 5: // read from tile graphics (for the window)
+ if ((window_counter % 2) == 1)
+ {
+ y_scroll_offset = (window_y_tile_inc) % 8;
+
+ if (((LCDC & 0x10) > 0))
+ {
+
+ tile_data[0] = VRAM[tile_byte * 16 + y_scroll_offset * 2];
+
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed uint8_t
+ if (((tile_byte & 0x80) > 0))
+ {
+ tile_byte -= 256;
+ }
+
+ tile_data[0] = VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2];
+ }
+
+ read_case = 6;
+ }
+ window_counter++;
+ break;
+
+ case 6: // read from tile graphics (for the window)
+ if ((window_counter % 2) == 1)
+ {
+ y_scroll_offset = (window_y_tile_inc) % 8;
+ if (((LCDC & 0x10) > 0))
+ {
+ // if LCDC somehow changed between the two reads, make sure we have a positive number
+ if (tile_byte < 0)
+ {
+ tile_byte += 256;
+ }
+
+ tile_data[1] = VRAM[tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed uint8_t
+ if (((tile_byte & 0x80) > 0) && tile_byte > 0)
+ {
+ tile_byte -= 256;
+ }
+
+ tile_data[1] = VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+
+ if (window_pre_render)
+ {
+ // here we set up rendering
+ // unlike for the normal background case, there is no pre-render period for the window
+ // so start shifting in data to the screen right away
+ if (window_x_latch <= 7)
+ {
+ if (render_offset == 0)
+ {
+ read_case = 4;
+ }
+ else
+ {
+ read_case = 9 + render_offset - 1;
+ }
+ render_counter = 8 - render_offset;
+
+ render_offset = 7 - window_x_latch;
+ }
+ else
+ {
+ render_offset = 0;
+ read_case = 4;
+ render_counter = 8;
+ }
+
+ latch_counter = 0;
+ latch_new_data = true;
+ window_pre_render = false;
+ }
+ else
+ {
+ read_case = 7;
+ }
+ }
+ window_counter++;
+ break;
+
+ case 7: // read from sprite data
+ if ((window_counter % 2) == 1)
+ {
+ read_case = 4;
+ latch_new_data = true;
+ }
+ window_counter++;
+ break;
+
+ case 8: // done reading, we are now in phase 0
+ pre_render = true;
+
+ STAT &= 0xFC;
+ STAT |= 0x00;
+
+ if (((STAT & 0x8) > 0)) { HBL_INT = true; }
+
+ OAM_access_read = true;
+ OAM_access_write = true;
+ VRAM_access_read = true;
+ VRAM_access_write = true;
+ break;
+
+ case 9:
+ // this is a degenerate case for starting the window at 0
+ // kevtris' timing doc indicates an additional normal BG access
+ // but this information is thrown away, so it's faster to do this then constantly check
+ // for it in read case 0
+ read_case = 4;
+ break;
+ case 10:
+ case 11:
+ case 12:
+ case 13:
+ case 14:
+ case 15:
+ case 16:
+ case 17:
+ read_case--;
+ break;
+ }
+ internal_cycle++;
+
+ if (latch_new_data)
+ {
+ latch_new_data = false;
+ tile_data_latch[0] = tile_data[0];
+ tile_data_latch[1] = tile_data[1];
+ }
+ }
+
+ // every in range sprite takes 6 cycles to process
+ // sprites located at x=0 still take 6 cycles to process even though they don't appear on screen
+ // sprites above x=168 do not take any cycles to process however
+ if (fetch_sprite)
+ {
+ if (going_to_fetch)
+ {
+ going_to_fetch = false;
+
+ last_eval = 0;
+
+ // at this time it is unknown what each cycle does, but we only need to accurately keep track of cycles
+ for (uint32_t i = 0; i < SL_sprites_index; i++)
+ {
+ if ((pixel_counter >= (int32_t)(SL_sprites[i * 4 + 1] - 8)) &&
+ (pixel_counter < (int32_t)(SL_sprites[i * 4 + 1])) &&
+ !((evaled_sprites & (1 << i)) > 0))
+ {
+ sprite_fetch_counter += 6;
+ evaled_sprites |= (1 << i);
+ last_eval = SL_sprites[i * 4 + 1];
+ }
+ }
+
+ // x scroll offsets the penalty table
+ // there is no penalty if the next sprites to be fetched are within the currentfetch block (8 pixels)
+ if (first_fetch || ((int32_t)last_eval >= consecutive_sprite))
+ {
+ if (((last_eval + render_offset) % 8) == 0) { sprite_fetch_counter += 5; }
+ else if (((last_eval + render_offset) % 8) == 1) { sprite_fetch_counter += 4; }
+ else if (((last_eval + render_offset) % 8) == 2) { sprite_fetch_counter += 3; }
+ else if (((last_eval + render_offset) % 8) == 3) { sprite_fetch_counter += 2; }
+ else if (((last_eval + render_offset) % 8) == 4) { sprite_fetch_counter += 1; }
+ else if (((last_eval + render_offset) % 8) == 5) { sprite_fetch_counter += 0; }
+ else if (((last_eval + render_offset) % 8) == 6) { sprite_fetch_counter += 0; }
+ else if (((last_eval + render_offset) % 8) == 7) { sprite_fetch_counter += 0; }
+
+ consecutive_sprite = (uint32_t)floor((double)((uint32_t)last_eval + render_offset) / 8.0) * 8 + 8 - render_offset;
+
+ // special case exists here for sprites at zero with non-zero x-scroll. Not sure exactly the reason for it.
+ if (last_eval == 0 && render_offset != 0)
+ {
+ sprite_fetch_counter += render_offset;
+ }
+ }
+
+ total_counter += sprite_fetch_counter;
+
+ first_fetch = false;
+ }
+ else
+ {
+ sprite_fetch_counter--;
+ if (sprite_fetch_counter == 0)
+ {
+ fetch_sprite = false;
+ }
+ }
+ }
+ }
+
+ void process_sprite()
+ {
+ uint32_t y;
+
+ if ((SL_sprites[sl_use_index * 4 + 3] & 0x40) > 0)
+ {
+ if (((LCDC & 0x4) > 0))
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ y = 15 - y;
+ sprite_sel[0] = VRAM[(SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
+ sprite_sel[1] = VRAM[(SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
+ }
+ else
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ y = 7 - y;
+ sprite_sel[0] = VRAM[SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
+ sprite_sel[1] = VRAM[SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
+ }
+ }
+ else
+ {
+ if (((LCDC & 0x4) > 0))
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ sprite_sel[0] = VRAM[(SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
+ sprite_sel[1] = VRAM[(SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
+ }
+ else
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ sprite_sel[0] = VRAM[SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
+ sprite_sel[1] = VRAM[SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
+ }
+ }
+
+ if ((SL_sprites[sl_use_index * 4 + 3] & 0x20) > 0)
+ {
+ uint32_t b0, b1, b2, b3, b4, b5, b6, b7 = 0;
+ for (uint32_t i = 0; i < 2; i++)
+ {
+ b0 = (sprite_sel[i] & 0x01) << 7;
+ b1 = (sprite_sel[i] & 0x02) << 5;
+ b2 = (sprite_sel[i] & 0x04) << 3;
+ b3 = (sprite_sel[i] & 0x08) << 1;
+ b4 = (sprite_sel[i] & 0x10) >> 1;
+ b5 = (sprite_sel[i] & 0x20) >> 3;
+ b6 = (sprite_sel[i] & 0x40) >> 5;
+ b7 = (sprite_sel[i] & 0x80) >> 7;
+
+ sprite_sel[i] = (uint8_t)(b0 | b1 | b2 | b3 | b4 | b5 | b6 | b7);
+ }
+ }
+ }
+
+ // normal DMA moves twice as fast in double speed mode on GBC
+ // So give it it's own function so we can seperate it from PPU tick
+ void DMA_tick()
+ {
+ // Note that DMA is halted when the CPU is halted
+ if (DMA_start && !cpu_halted[0])
+ {
+ if (DMA_clock >= 4)
+ {
+ DMA_OAM_access = false;
+ if ((DMA_clock % 4) == 1)
+ {
+ // the cpu can't access memory during this time, but we still need the ppu to be able to.
+ DMA_start = false;
+ // Gekkio reports that A14 being high on DMA transfers always represent WRAM accesses
+ // So transfers nominally from higher memory areas are actually still from there (i.e. FF -> DF)
+ uint8_t DMA_actual = DMA_addr;
+ if (DMA_addr > 0xDF) { DMA_actual &= 0xDF; }
+ DMA_byte = ReadMemory((uint32_t)((DMA_actual << 8) + DMA_inc));
+ DMA_start = true;
+ }
+ else if ((DMA_clock % 4) == 3)
+ {
+ OAM[DMA_inc] = DMA_byte;
+
+ if (DMA_inc < (0xA0 - 1)) { DMA_inc++; }
+ }
+ }
+
+ DMA_clock++;
+
+ if (DMA_clock == 648)
+ {
+ DMA_start = false;
+ DMA_OAM_access = true;
+ }
+ }
+ }
+
+ // order sprites according to x coordinate
+ // note that for sprites of equal x coordinate, priority goes to first on the list
+ void reorder_and_assemble_sprites()
+ {
+ sprite_ordered_index = 0;
+
+ for (uint32_t i = 0; i < 256; i++)
+ {
+ for (uint32_t j = 0; j < SL_sprites_index; j++)
+ {
+ if (SL_sprites[j * 4 + 1] == i)
+ {
+ sl_use_index = j;
+ process_sprite();
+ SL_sprites_ordered[sprite_ordered_index * 4] = SL_sprites[j * 4 + 1];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 1] = sprite_sel[0];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 2] = sprite_sel[1];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 3] = SL_sprites[j * 4 + 3];
+ sprite_ordered_index++;
+ }
+ }
+ }
+
+ bool have_pixel = false;
+ uint8_t s_pixel = 0;
+ uint8_t sprite_attr = 0;
+
+ for (uint32_t i = 0; i < 160; i++)
+ {
+ have_pixel = false;
+ for (uint32_t j = 0; j < SL_sprites_index; j++)
+ {
+ if ((i >= (SL_sprites_ordered[j * 4] - 8)) &&
+ (i < SL_sprites_ordered[j * 4]) &&
+ !have_pixel)
+ {
+ // we can use the current sprite, so pick out a pixel for it
+ uint32_t t_index = i - (SL_sprites_ordered[j * 4] - 8);
+
+ t_index = 7 - t_index;
+
+ sprite_data[0] = (uint8_t)((SL_sprites_ordered[j * 4 + 1] >> t_index) & 1);
+ sprite_data[1] = (uint8_t)(((SL_sprites_ordered[j * 4 + 2] >> t_index) & 1) << 1);
+
+ s_pixel = (uint8_t)(sprite_data[0] + sprite_data[1]);
+ sprite_attr = (uint8_t)SL_sprites_ordered[j * 4 + 3];
+
+ // pixel color of 0 is transparent, so if this is the case we don't have a pixel
+ if (s_pixel != 0)
+ {
+ have_pixel = true;
+ }
+ }
+ }
+
+ if (have_pixel)
+ {
+ sprite_present_list[i] = 1;
+ sprite_pixel_list[i] = s_pixel;
+ sprite_attr_list[i] = sprite_attr;
+ }
+ else
+ {
+ sprite_present_list[i] = 0;
+ }
+ }
+ }
+
+ void OAM_scan(uint32_t OAM_cycle)
+ {
+ // we are now in STAT mode 2
+ // TODO: maybe stat mode 2 flags are set at cycle 0 on visible scanlines?
+ if (OAM_cycle == 0)
+ {
+ OAM_access_read = false;
+
+ OAM_scan_index = 0;
+ SL_sprites_index = 0;
+ write_sprite = 0;
+ }
+
+ // the gameboy has 80 cycles to scan through 40 sprites, picking out the first 10 it finds to draw
+ // the following is a guessed at implmenentation based on how NES does it, it's probably pretty close
+ if (OAM_cycle < 10)
+ {
+ // start by clearing the sprite table (probably just clears X on hardware, but let's be safe here.)
+ SL_sprites[OAM_cycle * 4] = 0;
+ SL_sprites[OAM_cycle * 4 + 1] = 0;
+ SL_sprites[OAM_cycle * 4 + 2] = 0;
+ SL_sprites[OAM_cycle * 4 + 3] = 0;
+ }
+ else
+ {
+ if (write_sprite == 0)
+ {
+ if (OAM_scan_index < 40)
+ {
+ uint32_t temp = DMA_OAM_access ? OAM[OAM_scan_index * 4] : (uint32_t)0xFF;
+ // (sprite Y - 16) equals LY, we have a sprite
+ if ((temp - 16) <= LY &&
+ ((temp - 16) + 8 + (((LCDC & 0x4) > 0) ? 8 : 0)) > LY)
+ {
+ // always pick the first 10 in range sprites
+ if (SL_sprites_index < 10)
+ {
+ SL_sprites[SL_sprites_index * 4] = temp;
+
+ write_sprite = 1;
+ }
+ else
+ {
+ // if we already have 10 sprites, there's nothing to do, increment the index
+ OAM_scan_index++;
+ }
+ }
+ else
+ {
+ OAM_scan_index++;
+ }
+ }
+ }
+ else
+ {
+ uint32_t temp2 = DMA_OAM_access ? OAM[OAM_scan_index * 4 + write_sprite] : (uint32_t)0xFF;
+ SL_sprites[SL_sprites_index * 4 + write_sprite] = temp2;
+ write_sprite++;
+
+ if (write_sprite == 4)
+ {
+ write_sprite = 0;
+ SL_sprites_index++;
+ OAM_scan_index++;
+ }
+ }
+ }
+ }
+
+ void Reset()
+ {
+ LCDC = 0;
+ STAT = 0x80;
+ scroll_y = 0;
+ scroll_x = 0;
+ LY = 0;
+ LYC = 0;
+ DMA_addr = 0xFF;
+ BGP = 0xFF;
+ obj_pal_0 = 0xFF;
+ obj_pal_1 = 0xFF;
+ window_y = 0x0;
+ window_x = 0x0;
+ window_x_latch = 0xFF;
+ window_y_latch = 0xFF;
+ LY_inc = 1;
+ no_scan = false;
+ OAM_access_read = true;
+ VRAM_access_read = true;
+ OAM_access_write = true;
+ VRAM_access_write = true;
+ DMA_OAM_access = true;
+
+ cycle = 0;
+ LYC_INT = false;
+ HBL_INT = false;
+ VBL_INT = false;
+ OAM_INT = false;
+
+ stat_line = false;
+ stat_line_old = false;
+
+ window_counter = 0;
+ window_pre_render = false;
+ window_started = false;
+ window_is_reset = true;
+ window_tile_inc = 0;
+ window_y_tile = 0;
+ window_x_tile = 0;
+ window_y_tile_inc = 0;
+ }
+ };
+
+ #pragma endregion
+
+ #pragma region GBC_PPU
+
+ class GBC_PPU : public PPU
+ {
+ public:
+
+ uint8_t ReadReg(uint32_t addr)
+ {
+ uint8_t ret = 0;
+ //Console.WriteLine(Core.cpu.TotalExecutedCycles);
+ switch (addr)
+ {
+ case 0xFF40: ret = LCDC; break; // LCDC
+ case 0xFF41: ret = STAT; break; // STAT
+ case 0xFF42: ret = scroll_y; break; // SCY
+ case 0xFF43: ret = scroll_x; break; // SCX
+ case 0xFF44: ret = LY; break; // LY
+ case 0xFF45: ret = LYC; break; // LYC
+ case 0xFF46: ret = DMA_addr; break; // DMA
+ case 0xFF47: ret = BGP; break; // BGP
+ case 0xFF48: ret = obj_pal_0; break; // OBP0
+ case 0xFF49: ret = obj_pal_1; break; // OBP1
+ case 0xFF4A: ret = window_y; break; // WY
+ case 0xFF4B: ret = window_x; break; // WX
+
+ // These are GBC specific Regs
+ case 0xFF51: ret = HDMA_src_hi; break; // HDMA1
+ case 0xFF52: ret = HDMA_src_lo; break; // HDMA2
+ case 0xFF53: ret = HDMA_dest_hi; break; // HDMA3
+ case 0xFF54: ret = HDMA_dest_lo; break; // HDMA4
+ case 0xFF55: ret = HDMA_ctrl(); break; // HDMA5
+ case 0xFF68: ret = BG_pal_ret(); break; // BGPI
+ case 0xFF69: ret = BG_PAL_read(); break; // BGPD
+ case 0xFF6A: ret = OBJ_pal_ret(); break; // OBPI
+ case 0xFF6B: ret = OBJ_bytes[OBJ_bytes_index]; break; // OBPD
+ }
+
+ return ret;
+ }
+
+ uint8_t BG_PAL_read()
+ {
+ if (VRAM_access_read)
+ {
+ return BG_bytes[BG_bytes_index];
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ void WriteReg(uint32_t addr, uint8_t value)
+ {
+ switch (addr)
+ {
+ case 0xFF40: // LCDC
+ if (((LCDC & 0x80) > 0) && !((value & 0x80) > 0))
+ {
+ VRAM_access_read = true;
+ VRAM_access_write = true;
+ OAM_access_read = true;
+ OAM_access_write = true;
+ }
+
+ if (!((LCDC & 0x80) > 0) && ((value & 0x80) > 0))
+ {
+ // don't draw for one frame after turning on
+ blank_frame = true;
+ }
+
+ LCDC = value;
+ break;
+ case 0xFF41: // STAT
+ // note that their is no stat interrupt bug in GBC
+ STAT = (uint8_t)((value & 0xF8) | (STAT & 7) | 0x80);
+
+ if (((STAT & 3) == 0) && ((STAT & 0x8) > 0)) { HBL_INT = true; }
+ else { HBL_INT = false; }
+ if (((STAT & 3) == 1) && ((STAT & 0x10) > 0)) { VBL_INT = true; }
+ else { VBL_INT = false; }
+ // OAM not triggered?
+ // if (((STAT & 3) == 2) && STAT.Bit(5)) { OAM_INT = true; } else { OAM_INT = false; }
+
+ if (((value & 0x40) > 0) && ((LCDC & 0x80) > 0))
+ {
+ if (LY == LYC) { LYC_INT = true; }
+ else { LYC_INT = false; }
+ }
+ if (!((STAT & 0x40) > 0)) { LYC_INT = false; }
+ break;
+ case 0xFF42: // SCY
+ scroll_y = value;
+ break;
+ case 0xFF43: // SCX
+ scroll_x = value;
+ break;
+ case 0xFF44: // LY
+ LY = 0; /*reset*/
+ break;
+ case 0xFF45: // LYC
+ // tests indicate that latching writes to LYC should take place 4 cycles after the write
+ // otherwise tests around LY boundaries will fail
+ LYC_t = value;
+ LYC_cd = 4;
+ break;
+ case 0xFF46: // DMA
+ DMA_addr = value;
+ DMA_start = true;
+ DMA_OAM_access = true;
+ DMA_clock = 0;
+ DMA_inc = 0;
+ break;
+ case 0xFF47: // BGP
+ BGP = value;
+ break;
+ case 0xFF48: // OBP0
+ obj_pal_0 = value;
+ break;
+ case 0xFF49: // OBP1
+ obj_pal_1 = value;
+ break;
+ case 0xFF4A: // WY
+ window_y = value;
+ break;
+ case 0xFF4B: // WX
+ window_x = value;
+ break;
+
+ // These are GBC specific Regs
+ case 0xFF51: // HDMA1
+ HDMA_src_hi = value;
+ cur_DMA_src = (uint32_t)(((HDMA_src_hi & 0xFF) << 8) | (cur_DMA_src & 0xF0));
+ break;
+ case 0xFF52: // HDMA2
+ HDMA_src_lo = value;
+ cur_DMA_src = (uint32_t)((cur_DMA_src & 0xFF00) | (HDMA_src_lo & 0xF0));
+ break;
+ case 0xFF53: // HDMA3
+ HDMA_dest_hi = value;
+ cur_DMA_dest = (uint32_t)(((HDMA_dest_hi & 0x1F) << 8) | (cur_DMA_dest & 0xF0));
+ break;
+ case 0xFF54: // HDMA4
+ HDMA_dest_lo = value;
+ cur_DMA_dest = (uint32_t)((cur_DMA_dest & 0xFF00) | (HDMA_dest_lo & 0xF0));
+ break;
+ case 0xFF55: // HDMA5
+ if (!HDMA_active)
+ {
+ HDMA_mode = ((value & 0x80) > 0);
+ HDMA_countdown = 4;
+ HDMA_tick = 0;
+ if (((value & 0x80) > 0))
+ {
+ // HDMA during HBlank only, but only if screen is on, otherwise DMA immediately one block of data
+ // worms armaggedon requires HDMA to fire in hblank mode even if the screen is off.
+ HDMA_active = true;
+ HBL_HDMA_count = 0x10;
+
+ last_HBL = LY - 1;
+
+ HBL_test = true;
+ HBL_HDMA_go = false;
+
+ if (!((LCDC & 0x80) > 0))
+ {
+ HDMA_run_once = true;
+ }
+ }
+ else
+ {
+ // HDMA immediately
+ HDMA_active = true;
+ HDMA_transfer[0] = true;
+ }
+
+ HDMA_length = ((value & 0x7F) + 1) * 16;
+ }
+ else
+ {
+ //terminate the transfer
+ if (!((value & 0x80) > 0))
+ {
+ HDMA_active = false;
+ }
+ }
+
+ break;
+ case 0xFF68: // BGPI
+ BG_bytes_index = (uint8_t)(value & 0x3F);
+ BG_bytes_inc = ((value & 0x80) == 0x80);
+ break;
+ case 0xFF69: // BGPD
+ if (VRAM_access_write)
+ {
+ BG_transfer_byte = value;
+ BG_bytes[BG_bytes_index] = value;
+ }
+
+ // change the appropriate palette color
+ color_compute_BG();
+ if (BG_bytes_inc) { BG_bytes_index++; BG_bytes_index &= 0x3F; }
+ break;
+ case 0xFF6A: // OBPI
+ OBJ_bytes_index = (uint8_t)(value & 0x3F);
+ OBJ_bytes_inc = ((value & 0x80) == 0x80);
+ break;
+ case 0xFF6B: // OBPD
+ OBJ_transfer_byte = value;
+ OBJ_bytes[OBJ_bytes_index] = value;
+
+ // change the appropriate palette color
+ color_compute_OBJ();
+
+ if (OBJ_bytes_inc) { OBJ_bytes_index++; OBJ_bytes_index &= 0x3F; }
+ break;
+ }
+ }
+
+ void tick()
+ {
+ // Do HDMA ticks
+ if (HDMA_active)
+ {
+ if (HDMA_length > 0)
+ {
+ if (!HDMA_mode)
+ {
+ if (HDMA_countdown > 0)
+ {
+ HDMA_countdown--;
+ }
+ else
+ {
+ // immediately transfer bytes, 2 bytes per cycles
+ if ((HDMA_tick % 2) == 0)
+ {
+ HDMA_byte = ReadMemory(cur_DMA_src);
+ }
+ else
+ {
+ VRAM[(VRAM_Bank[0] * 0x2000) + cur_DMA_dest] = HDMA_byte;
+ cur_DMA_dest = (uint8_t)((cur_DMA_dest + 1) & 0x1FFF);
+ cur_DMA_src = (uint8_t)((cur_DMA_src + 1) & 0xFFFF);
+ HDMA_length--;
+ }
+
+ HDMA_tick++;
+ }
+ }
+ else
+ {
+ // only transfer during mode 0, and only 16 bytes at a time
+ if (((STAT & 3) == 0) && (LY != last_HBL) && HBL_test && (LY_inc == 1) && (cycle > 4))
+ {
+ HBL_HDMA_go = true;
+ HBL_test = false;
+ }
+ else if (HDMA_run_once)
+ {
+ HBL_HDMA_go = true;
+ HBL_test = false;
+ HDMA_run_once = false;
+ }
+
+ if (HBL_HDMA_go && (HBL_HDMA_count > 0))
+ {
+ HDMA_transfer[0] = true;
+
+ if (HDMA_countdown > 0)
+ {
+ HDMA_countdown--;
+ }
+ else
+ {
+ if ((HDMA_tick % 2) == 0)
+ {
+ HDMA_byte = ReadMemory(cur_DMA_src);
+ }
+ else
+ {
+ VRAM[(VRAM_Bank[0] * 0x2000) + cur_DMA_dest] = HDMA_byte;
+ cur_DMA_dest = (uint32_t)((cur_DMA_dest + 1) & 0x1FFF);
+ cur_DMA_src = (uint32_t)((cur_DMA_src + 1) & 0xFFFF);
+ HDMA_length--;
+ HBL_HDMA_count--;
+ }
+
+ if ((HBL_HDMA_count == 0) && (HDMA_length != 0))
+ {
+
+ HBL_test = true;
+ last_HBL = LY;
+ HBL_HDMA_count = 0x10;
+ HBL_HDMA_go = false;
+ HDMA_countdown = 4;
+ }
+
+ HDMA_tick++;
+ }
+ }
+ else
+ {
+ HDMA_transfer[0] = false;
+ }
+ }
+ }
+ else
+ {
+ HDMA_active = false;
+ HDMA_transfer[0] = false;
+ }
+ }
+
+ // the ppu only does anything if it is turned on via bit 7 of LCDC
+ if (((LCDC & 0x80) > 0))
+ {
+ // start the next scanline
+ if (cycle == 456)
+ {
+ // scanline callback
+ if ((LY + LY_inc) == _scanlineCallbackLine[0])
+ {
+ //if (Core._scanlineCallback != null)
+ //{
+ // Core.GetGPU();
+ // Core._scanlineCallback(LCDC);
+ //}
+ }
+
+ cycle = 0;
+ LY += LY_inc;
+ cpu_LY[0] = LY;
+
+ no_scan = false;
+
+ if (LY == 0 && LY_inc == 0)
+ {
+ LY_inc = 1;
+ in_vblank[0] = false;
+
+ //STAT &= 0xFC;
+
+ // special note here, the y coordiate of the window is kept if the window is deactivated
+ // meaning it will pick up where it left off if re-enabled later
+ // so we don't reset it in the scanline loop
+ window_y_tile = 0;
+ window_y_latch = window_y;
+ window_y_tile_inc = 0;
+ window_started = false;
+ if (!((LCDC & 0x20) > 0)) { window_is_reset = true; }
+ }
+
+ // Automatically restore access to VRAM at this time (force end drawing)
+ // Who Framed Roger Rabbit seems to run into this.
+ VRAM_access_write = true;
+ VRAM_access_read = true;
+
+ if (LY == 144)
+ {
+ in_vblank[0] = true;
+ }
+ }
+
+ // exit vblank if LCD went from off to on
+ if (LCD_was_off)
+ {
+ //VBL_INT = false;
+ in_vblank[0] = false;
+ LCD_was_off = false;
+
+ // we exit vblank into mode 0 for 4 cycles
+ // but no hblank interrupt, presumably this only happens transitioning from mode 3 to 0
+ STAT &= 0xFC;
+
+ // also the LCD doesn't turn on right away
+ // also, the LCD does not enter mode 2 on scanline 0 when first turned on
+ no_scan = true;
+ cycle = 8;
+ }
+
+ // the VBL stat is continuously asserted
+ if (LY >= 144)
+ {
+ if (((STAT & 0x10) > 0))
+ {
+ if ((cycle >= 4) && (LY == 144))
+ {
+ VBL_INT = true;
+ }
+ else if (LY > 144)
+ {
+ VBL_INT = true;
+ }
+ }
+
+ if ((cycle == 2) && (LY == 144))
+ {
+ // there is an edge case where a VBL INT is triggered if STAT bit 5 is set
+ if (((STAT & 0x20) > 0)) { VBL_INT = true; }
+ }
+
+ if ((cycle == 4) && (LY == 144))
+ {
+ HBL_INT = false;
+
+ // set STAT mode to 1 (VBlank) and interrupt flag if it is enabled
+ STAT &= 0xFC;
+ STAT |= 0x01;
+
+ if ((REG_FFFF[0] & 1) > 0) { FlagI[0] = true; }
+ REG_FF0F[0] |= 0x01;
+ }
+
+ if ((cycle == 4) && (LY == 144))
+ {
+ if (((STAT & 0x20) > 0)) { VBL_INT = false; }
+ }
+
+ if ((cycle == 8) && (LY == 153))
+ {
+ LY = 0;
+ LY_inc = 0;
+ cpu_LY[0] = LY;
+ }
+ }
+
+ if (!in_vblank[0])
+ {
+ if (no_scan)
+ {
+ // timings are slightly different if we just turned on the LCD
+ // there is no mode 2 (presumably it missed the trigger)
+ if (cycle < 85)
+ {
+ if (cycle == 8)
+ {
+ // clear the sprite table
+ for (uint32_t k = 0; k < 10; k++)
+ {
+ SL_sprites[k * 4] = 0;
+ SL_sprites[k * 4 + 1] = 0;
+ SL_sprites[k * 4 + 2] = 0;
+ SL_sprites[k * 4 + 3] = 0;
+ }
+
+ if (LY != LYC)
+ {
+ LYC_INT = false;
+ STAT &= 0xFB;
+ }
+
+ if ((LY == LYC) && !((STAT & 0x4) > 0))
+ {
+ // set STAT coincidence FLAG and interrupt flag if it is enabled
+ STAT |= 0x04;
+ if (((STAT & 0x40) > 0)) { LYC_INT = true; }
+ }
+ }
+
+ if (cycle == 84)
+ {
+ STAT &= 0xFC;
+ STAT |= 0x03;
+ OAM_INT = false;
+
+ OAM_access_read = false;
+ OAM_access_write = false;
+ VRAM_access_read = false;
+ VRAM_access_write = false;
+ }
+ }
+ else
+ {
+ if (cycle >= 85)
+ {
+ if (cycle == 85)
+ {
+ // x-scroll is expected to be latched one cycle later
+ // this is fine since nothing has started in the rendering until the second cycle
+ // calculate the column number of the tile to start with
+ x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
+ render_offset = scroll_x % 8;
+ }
+
+ // render the screen and handle hblank
+ render(cycle - 85);
+ }
+ }
+ }
+ else
+ {
+ if (cycle <= 80)
+ {
+ if (cycle == 2)
+ {
+ if (LY != 0)
+ {
+ HBL_INT = false;
+
+ if (((STAT & 0x20) > 0)) { OAM_INT = true; }
+ }
+ }
+ else if (cycle == 4)
+ {
+ // here mode 2 will be set to true and interrupts fired if enabled
+ STAT &= 0xFC;
+ STAT |= 0x2;
+
+ if (LY == 0)
+ {
+ VBL_INT = false;
+ if (((STAT & 0x20) > 0)) { OAM_INT = true; }
+ }
+ }
+
+ if (cycle == 80)
+ {
+ OAM_access_read = false;
+ OAM_access_write = true;
+ VRAM_access_read = false;
+ }
+ else
+ {
+ // here OAM scanning is performed
+ OAM_scan(cycle);
+ }
+ }
+ else if (cycle >= 83)
+ {
+ if (cycle == 84)
+ {
+ STAT &= 0xFC;
+ STAT |= 0x03;
+ OAM_INT = false;
+ OAM_access_write = false;
+ VRAM_access_write = false;
+
+ // x-scroll is expected to be latched one cycle later
+ // this is fine since nothing has started in the rendering until the second cycle
+ // calculate the column number of the tile to start with
+ x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
+ render_offset = scroll_x % 8;
+ }
+
+ // render the screen and handle hblank
+ render(cycle - 83);
+ }
+ }
+ }
+
+ if (LY_inc == 0)
+ {
+ if (cycle == 12)
+ {
+ LYC_INT = false;
+ STAT &= 0xFB;
+ }
+ else if (cycle == 14)
+ {
+ // Special case of LY = LYC
+ if ((LY == LYC) && !((STAT & 0x4) > 0))
+ {
+ // set STAT coincidence FLAG and interrupt flag if it is enabled
+ STAT |= 0x04;
+ if (((STAT & 0x40) > 0)) { LYC_INT = true; }
+ }
+ }
+ }
+
+ // here LY=LYC will be asserted or cleared (but only if LY isnt 0 as that's a special case)
+ if ((cycle == 4) && (LY != 0))
+ {
+ if (LY_inc == 1)
+ {
+ LYC_INT = false;
+ STAT &= 0xFB;
+ }
+ }
+ else if ((cycle == 6) && (LY != 0))
+ {
+ if ((LY == LYC) && !((STAT & 0x4) > 0))
+ {
+ // set STAT coincidence FLAG and interrupt flag if it is enabled
+ STAT |= 0x04;
+ if (((STAT & 0x40) > 0)) { LYC_INT = true; }
+ }
+ }
+
+ cycle++;
+ }
+ else
+ {
+ STAT &= 0xFC;
+
+ VBL_INT = LYC_INT = HBL_INT = OAM_INT = false;
+
+ in_vblank[0] = true;
+
+ LCD_was_off = true;
+
+ LY = 0;
+ cpu_LY[0] = LY;
+
+ cycle = 0;
+ }
+
+ // assert the STAT IRQ line if the line went from zero to 1
+ stat_line = VBL_INT | LYC_INT | HBL_INT | OAM_INT;
+
+ if (stat_line && !stat_line_old)
+ {
+ if ((REG_FFFF[0] & 0x2) > 0) { FlagI[0] = true; }
+ REG_FF0F[0] |= 0x02;
+
+ //if (LY == 46)
+ //{
+ //Console.Write(VBL_INT + " " + LYC_INT + " " + HBL_INT + " " + OAM_INT + " " + LY + " ");
+ //Console.Write(render_offset + " " + scroll_x + " " + total_counter + " ");
+ //Console.WriteLine(STAT + " " + cycle + " " + Core.cpu.TotalExecutedCycles);
+ //}
+ }
+
+ stat_line_old = stat_line;
+
+ // process latch delays
+ //latch_delay();
+
+ if (LYC_cd > 0)
+ {
+ LYC_cd--;
+ if (LYC_cd == 0)
+ {
+ LYC = LYC_t;
+
+ if (((LCDC & 0x80) > 0))
+ {
+ if (LY != LYC) { STAT &= 0xFB; LYC_INT = false; }
+ else { STAT |= 0x4; LYC_INT = true; }
+ }
+ }
+ }
+ }
+
+ // might be needed, not sure yet
+ void latch_delay()
+ {
+ //BGP_l = BGP;
+ }
+
+ void render(uint32_t render_cycle)
+ {
+ // we are now in STAT mode 3
+ // NOTE: presumably the first necessary sprite is fetched at sprite evaulation
+ // i.e. just keeping track of the lowest x-value sprite
+ if (render_cycle == 0)
+ {
+ /*
+ OAM_access_read = false;
+ OAM_access_write = true;
+ VRAM_access_read = false;
+ */
+ // window X is latched for the scanline, mid-line changes have no effect
+ window_x_latch = window_x;
+
+ OAM_scan_index = 0;
+ read_case = 0;
+ internal_cycle = 0;
+ pre_render = true;
+ pre_render_2 = true;
+ tile_inc = 0;
+ pixel_counter = -8;
+ sl_use_index = 0;
+ fetch_sprite = false;
+ going_to_fetch = false;
+ first_fetch = true;
+ consecutive_sprite = (int32_t)(render_offset * (-1)) + 8;
+ no_sprites = false;
+ evaled_sprites = 0;
+ window_pre_render = false;
+ window_latch = ((LCDC & 0x20) > 0);
+
+ total_counter = 0;
+
+ // TODO: If Window is turned on midscanline what happens? When is this check done exactly?
+ if ((window_started && window_latch) || (window_is_reset && !window_latch && (LY >= window_y_latch)))
+ {
+ window_y_tile_inc++;
+ if (window_y_tile_inc == 8)
+ {
+ window_y_tile_inc = 0;
+ window_y_tile++;
+ window_y_tile %= 32;
+ }
+ }
+ window_started = false;
+
+ if (SL_sprites_index == 0) { no_sprites = true; }
+ // it is much easier to process sprites if we order them according to the rules of sprite priority first
+ if (!no_sprites) { reorder_and_assemble_sprites(); }
+ }
+
+ // before anything else, we have to check if windowing is in effect
+ if (window_latch && !window_started && (LY >= window_y_latch) && (pixel_counter >= (int32_t)(window_x_latch - 7)) && (window_x_latch < 167))
+ {
+ /*
+ Console.Write(LY);
+ Console.Write(" ");
+ Console.Write(cycle);
+ Console.Write(" ");
+ Console.Write(window_y_tile);
+ Console.Write(" ");
+ Console.Write(render_offset);
+ Console.Write(" ");
+ Console.Write(window_x_latch);
+ Console.Write(" ");
+ Console.WriteLine(pixel_counter);
+ */
+
+ if (window_x_latch == 0)
+ {
+ // if the window starts at zero, we still do the first access to the BG
+ // but then restart all over again at the window
+ if ((render_offset % 7) <= 6)
+ {
+ read_case = 9;
+ }
+ else
+ {
+ read_case = 10;
+ }
+ }
+ else
+ {
+ read_case = 4;
+ }
+
+ window_pre_render = true;
+
+ window_counter = 0;
+ render_counter = 0;
+
+ window_x_tile = (uint32_t)floor((float)(pixel_counter - (window_x_latch - 7)) / 8.0);
+
+ window_tile_inc = 0;
+ window_started = true;
+ window_is_reset = false;
+ }
+
+ if (!pre_render && !fetch_sprite)
+ {
+ // start shifting data into the LCD
+ if (render_counter >= (render_offset + 8))
+ {
+ if ((tile_data_latch[2] & 0x20) > 0)
+ {
+ pixel = (tile_data_latch[0] & (1 << (render_counter % 8))) > 0 ? 1 : 0;
+ pixel |= (tile_data_latch[1] & (1 << (render_counter % 8))) > 0 ? 2 : 0;
+ }
+ else
+ {
+ pixel = (tile_data_latch[0] & (1 << (7 - (render_counter % 8)))) > 0 ? 1 : 0;
+ pixel |= (tile_data_latch[1] & (1 << (7 - (render_counter % 8)))) > 0 ? 2 : 0;
+ }
+
+ uint32_t ref_pixel = pixel;
+
+ uint32_t pal_num = tile_data_latch[2] & 0x7;
+
+ bool use_sprite = false;
+
+ uint32_t s_pixel = 0;
+
+ // now we have the BG pixel, we next need the sprite pixel
+ if (!no_sprites)
+ {
+ bool have_sprite = false;
+ uint32_t sprite_attr = 0;
+
+ if (sprite_present_list[pixel_counter] == 1)
+ {
+ have_sprite = true;
+ s_pixel = sprite_pixel_list[pixel_counter];
+ sprite_attr = sprite_attr_list[pixel_counter];
+ }
+
+ if (have_sprite)
+ {
+ if (((LCDC & 0x2) > 0))
+ {
+ if (!((sprite_attr & 0x80) > 0))
+ {
+ use_sprite = true;
+ }
+ else if (ref_pixel == 0)
+ {
+ use_sprite = true;
+ }
+
+ if (!((LCDC & 0x1) > 0))
+ {
+ use_sprite = true;
+ }
+
+ // There is another priority bit in GBC, that can still override sprite priority
+ if (((LCDC & 0x1) > 0) && ((tile_data_latch[2] & 0x80) > 0) && (ref_pixel != 0))
+ {
+ use_sprite = false;
+ }
+ }
+
+ if (use_sprite)
+ {
+ pal_num = sprite_attr & 7;
+ }
+ }
+ }
+
+ // based on sprite priority and pixel values, pick a final pixel color
+ if (use_sprite)
+ {
+ _vidbuffer[LY * 160 + pixel_counter] = (int)OBJ_palette[pal_num * 4 + s_pixel];
+ }
+ else
+ {
+ _vidbuffer[LY * 160 + pixel_counter] = (int)BG_palette[pal_num * 4 + pixel];
+ }
+
+ pixel_counter++;
+
+ if (pixel_counter == 160)
+ {
+ read_case = 8;
+ hbl_countdown = 2;
+ }
+ }
+ else if (pixel_counter < 0)
+ {
+ pixel_counter++;
+ }
+ render_counter++;
+ }
+
+ if (!fetch_sprite)
+ {
+ if (!pre_render_2)
+ {
+ // before we go on to read case 3, we need to know if we stall there or not
+ // Gekkio's tests show that if sprites are at position 0 or 1 (mod 8)
+ // then it takes an extra cycle (1 or 2 more t-states) to process them
+
+ if (!no_sprites && (pixel_counter < 160))
+ {
+ for (uint32_t i = 0; i < SL_sprites_index; i++)
+ {
+ if ((pixel_counter >= (int32_t)(SL_sprites[i * 4 + 1] - 8)) &&
+ (pixel_counter < (int32_t)(SL_sprites[i * 4 + 1])) &&
+ !((evaled_sprites & (1 << i)) > 0))
+ {
+ going_to_fetch = true;
+ fetch_sprite = true;
+ }
+ }
+ }
+ }
+
+ switch (read_case)
+ {
+ case 0: // read a background tile
+ if ((internal_cycle % 2) == 1)
+ {
+ // calculate the row number of the tiles to be fetched
+ y_tile = ((uint32_t)floor((float)((uint32_t)scroll_y + LY) / 8.0)) % 32;
+
+ temp_fetch = y_tile * 32 + (x_tile + tile_inc) % 32;
+ tile_byte = VRAM[0x1800 + (((LCDC & 0x4) > 0) ? 1 : 0) * 0x400 + temp_fetch];
+ tile_data[2] = VRAM[0x3800 + (((LCDC & 0x4) > 0) ? 1 : 0) * 0x400 + temp_fetch];
+ VRAM_sel = ((tile_data[2] & 0x8) > 0) ? 1 : 0;
+
+ BG_V_flip = ((tile_data[2] & 0x40) > 0);
+
+ read_case = 1;
+ if (!pre_render)
+ {
+ tile_inc++;
+ }
+ }
+ break;
+
+ case 1: // read from tile graphics (0)
+ if ((internal_cycle % 2) == 1)
+ {
+ y_scroll_offset = (scroll_y + LY) % 8;
+
+ if (BG_V_flip)
+ {
+ y_scroll_offset = 7 - y_scroll_offset;
+ }
+
+ if (((LCDC & 0x10) > 0))
+ {
+ tile_data[0] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed byte
+ if (((tile_byte & 0x80) > 0))
+ {
+ tile_byte -= 256;
+ }
+ tile_data[0] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
+ }
+
+ read_case = 2;
+ }
+ break;
+
+ case 2: // read from tile graphics (1)
+ if ((internal_cycle % 2) == 0)
+ {
+ pre_render_2 = false;
+ }
+ else
+ {
+ y_scroll_offset = (scroll_y + LY) % 8;
+
+ if (BG_V_flip)
+ {
+ y_scroll_offset = 7 - y_scroll_offset;
+ }
+
+ if (((LCDC & 0x10) > 0))
+ {
+ // if LCDC somehow changed between the two reads, make sure we have a positive number
+ if (tile_byte < 0)
+ {
+ tile_byte += 256;
+ }
+ tile_data[1] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed byte
+ if (((tile_byte & 0x80) > 0) && tile_byte > 0)
+ {
+ tile_byte -= 256;
+ }
+ tile_data[1] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+
+ if (pre_render)
+ {
+ // here we set up rendering
+ pre_render = false;
+
+ render_counter = 0;
+ latch_counter = 0;
+ read_case = 0;
+ }
+ else
+ {
+ read_case = 3;
+ }
+ }
+ break;
+
+ case 3: // read from sprite data
+ if ((internal_cycle % 2) == 1)
+ {
+ read_case = 0;
+ latch_new_data = true;
+ }
+ break;
+
+ case 4: // read from window data
+ if ((window_counter % 2) == 1)
+ {
+ temp_fetch = window_y_tile * 32 + (window_x_tile + window_tile_inc) % 32;
+ tile_byte = VRAM[0x1800 + (((LCDC & 0x40) > 0) ? 1 : 0) * 0x400 + temp_fetch];
+ tile_data[2] = VRAM[0x3800 + (((LCDC & 0x40) > 0) ? 1 : 0) * 0x400 + temp_fetch];
+ VRAM_sel = ((tile_data[2] & 0x8) > 0) ? 1 : 0;
+ BG_V_flip = ((tile_data[2] & 0x40) > 0);
+
+ window_tile_inc++;
+ read_case = 5;
+ }
+ window_counter++;
+ break;
+
+ case 5: // read from tile graphics (for the window)
+ if ((window_counter % 2) == 1)
+ {
+ y_scroll_offset = (window_y_tile_inc) % 8;
+
+ if (BG_V_flip)
+ {
+ y_scroll_offset = 7 - y_scroll_offset;
+ }
+
+ if (((LCDC & 0x10) > 0))
+ {
+ tile_data[0] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed byte
+ if (((tile_byte & 0x80) > 0))
+ {
+ tile_byte -= 256;
+ }
+ tile_data[0] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
+ }
+
+ read_case = 6;
+ }
+ window_counter++;
+ break;
+
+ case 6: // read from tile graphics (for the window)
+ if ((window_counter % 2) == 1)
+ {
+ y_scroll_offset = (window_y_tile_inc) % 8;
+
+ if (BG_V_flip)
+ {
+ y_scroll_offset = 7 - y_scroll_offset;
+ }
+
+ if (((LCDC & 0x10) > 0))
+ {
+ // if LCDC somehow changed between the two reads, make sure we have a positive number
+ if (tile_byte < 0)
+ {
+ tile_byte += 256;
+ }
+ tile_data[1] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed byte
+ if (((tile_byte & 0x80) > 0) && tile_byte > 0)
+ {
+ tile_byte -= 256;
+ }
+ tile_data[1] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+
+ if (window_pre_render)
+ {
+ // here we set up rendering
+ // unlike for the normal background case, there is no pre-render period for the window
+ // so start shifting in data to the screen right away
+ if (window_x_latch <= 7)
+ {
+ if (render_offset == 0)
+ {
+ read_case = 4;
+ }
+ else
+ {
+ read_case = 9 + render_offset - 1;
+ }
+ render_counter = 8 - render_offset;
+
+ render_offset = 7 - window_x_latch;
+ }
+ else
+ {
+ render_offset = 0;
+ read_case = 4;
+ render_counter = 8;
+ }
+
+ latch_counter = 0;
+ latch_new_data = true;
+ window_pre_render = false;
+ }
+ else
+ {
+ read_case = 7;
+ }
+ }
+ window_counter++;
+ break;
+
+ case 7: // read from sprite data
+ if ((window_counter % 2) == 1)
+ {
+ read_case = 4;
+ latch_new_data = true;
+ }
+ window_counter++;
+ break;
+
+ case 8: // done reading, we are now in phase 0
+ pre_render = true;
+
+ // the other interrupts appear to be delayed by 1 CPU cycle, so do the same here
+ if (hbl_countdown > 0)
+ {
+ hbl_countdown--;
+
+ if (hbl_countdown == 0)
+ {
+ OAM_access_read = true;
+ OAM_access_write = true;
+ VRAM_access_read = true;
+ VRAM_access_write = true;
+ }
+ else
+ {
+ STAT &= 0xFC;
+ STAT |= 0x00;
+
+ if (((STAT & 0x8) > 0)) { HBL_INT = true; }
+ }
+ }
+ break;
+
+ case 9:
+ // this is a degenerate case for starting the window at 0
+ // kevtris' timing doc indicates an additional normal BG access
+ // but this information is thrown away, so it's faster to do this then constantly check
+ // for it in read case 0
+ read_case = 4;
+ break;
+ case 10:
+ case 11:
+ case 12:
+ case 13:
+ case 14:
+ case 15:
+ case 16:
+ case 17:
+ read_case--;
+ break;
+ }
+ internal_cycle++;
+
+ if (latch_new_data)
+ {
+ latch_new_data = false;
+ tile_data_latch[0] = tile_data[0];
+ tile_data_latch[1] = tile_data[1];
+ tile_data_latch[2] = tile_data[2];
+ }
+ }
+
+ // every in range sprite takes 6 cycles to process
+ // sprites located at x=0 still take 6 cycles to process even though they don't appear on screen
+ // sprites above x=168 do not take any cycles to process however
+ if (fetch_sprite)
+ {
+ if (going_to_fetch)
+ {
+ going_to_fetch = false;
+
+ last_eval = 0;
+
+ // at this time it is unknown what each cycle does, but we only need to accurately keep track of cycles
+ for (uint32_t i = 0; i < SL_sprites_index; i++)
+ {
+ if ((pixel_counter >= (int32_t)(SL_sprites[i * 4 + 1] - 8)) &&
+ (pixel_counter < (int32_t)(SL_sprites[i * 4 + 1])) &&
+ !((evaled_sprites & (1 << i)) > 0))
+ {
+ sprite_fetch_counter += 6;
+ evaled_sprites |= (1 << i);
+ last_eval = SL_sprites[i * 4 + 1];
+ }
+ }
+
+ // x scroll offsets the penalty table
+ // there is no penalty if the next sprites to be fetched are within the currentfetch block (8 pixels)
+ if (first_fetch || ((int32_t)last_eval >= consecutive_sprite))
+ {
+ if (((last_eval + render_offset) % 8) == 0) { sprite_fetch_counter += 5; }
+ else if (((last_eval + render_offset) % 8) == 1) { sprite_fetch_counter += 4; }
+ else if (((last_eval + render_offset) % 8) == 2) { sprite_fetch_counter += 3; }
+ else if (((last_eval + render_offset) % 8) == 3) { sprite_fetch_counter += 2; }
+ else if (((last_eval + render_offset) % 8) == 4) { sprite_fetch_counter += 1; }
+ else if (((last_eval + render_offset) % 8) == 5) { sprite_fetch_counter += 0; }
+ else if (((last_eval + render_offset) % 8) == 6) { sprite_fetch_counter += 0; }
+ else if (((last_eval + render_offset) % 8) == 7) { sprite_fetch_counter += 0; }
+
+ consecutive_sprite = (uint32_t)floor((double)(last_eval + render_offset) / 8.0) * 8 + 8 - render_offset;
+
+ // special case exists here for sprites at zero with non-zero x-scroll. Not sure exactly the reason for it.
+ if (last_eval == 0 && render_offset != 0)
+ {
+ sprite_fetch_counter += render_offset;
+ }
+ }
+
+ total_counter += sprite_fetch_counter;
+
+ first_fetch = false;
+ }
+ else
+ {
+ sprite_fetch_counter--;
+ if (sprite_fetch_counter == 0)
+ {
+ fetch_sprite = false;
+ }
+ }
+ }
+
+ }
+
+ void process_sprite()
+ {
+ uint32_t y;
+ uint32_t VRAM_temp = ((SL_sprites[sl_use_index * 4 + 3] & 0x8) > 0) ? 1 : 0;
+
+ if ((SL_sprites[sl_use_index * 4 + 3] & 0x40) > 0)
+ {
+ if (((LCDC & 0x4) > 0))
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ y = 15 - y;
+ sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
+ sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
+ }
+ else
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ y = 7 - y;
+ sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
+ sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
+ }
+ }
+ else
+ {
+ if (((LCDC & 0x4) > 0))
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
+ sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
+ }
+ else
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
+ sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
+ }
+ }
+
+ if ((SL_sprites[sl_use_index * 4 + 3] & 0x20) > 0)
+ {
+ uint32_t b0, b1, b2, b3, b4, b5, b6, b7 = 0;
+ for (uint32_t i = 0; i < 2; i++)
+ {
+ b0 = (sprite_sel[i] & 0x01) << 7;
+ b1 = (sprite_sel[i] & 0x02) << 5;
+ b2 = (sprite_sel[i] & 0x04) << 3;
+ b3 = (sprite_sel[i] & 0x08) << 1;
+ b4 = (sprite_sel[i] & 0x10) >> 1;
+ b5 = (sprite_sel[i] & 0x20) >> 3;
+ b6 = (sprite_sel[i] & 0x40) >> 5;
+ b7 = (sprite_sel[i] & 0x80) >> 7;
+
+ sprite_sel[i] = (uint8_t)(b0 | b1 | b2 | b3 | b4 | b5 | b6 | b7);
+ }
+ }
+ }
+
+ // normal DMA moves twice as fast in double speed mode on GBC
+ // So give it it's own function so we can seperate it from PPU tick
+ void DMA_tick()
+ {
+ // Note that DMA is halted when the CPU is halted
+ if (DMA_start && !cpu_halted)
+ {
+ if (DMA_clock >= 4)
+ {
+ DMA_OAM_access = false;
+ if ((DMA_clock % 4) == 1)
+ {
+ // the cpu can't access memory during this time, but we still need the ppu to be able to.
+ DMA_start = false;
+ // Gekkio reports that A14 being high on DMA transfers always represent WRAM accesses
+ // So transfers nominally from higher memory areas are actually still from there (i.e. FF -> DF)
+ uint8_t DMA_actual = DMA_addr;
+ if (DMA_addr > 0xDF) { DMA_actual &= 0xDF; }
+ DMA_byte = ReadMemory((uint32_t)((DMA_actual << 8) + DMA_inc));
+ DMA_start = true;
+ }
+ else if ((DMA_clock % 4) == 3)
+ {
+ OAM[DMA_inc] = DMA_byte;
+
+ if (DMA_inc < (0xA0 - 1)) { DMA_inc++; }
+ }
+ }
+
+ DMA_clock++;
+
+ if (DMA_clock == 648)
+ {
+ DMA_start = false;
+ DMA_OAM_access = true;
+ }
+ }
+ }
+
+ // order sprites according to x coordinate
+ // note that for sprites of equal x coordinate, priority goes to first on the list
+ void reorder_and_assemble_sprites()
+ {
+ sprite_ordered_index = 0;
+
+ // In CGB mode, sprites are ordered solely based on their position in OAM, so they are already ordered
+ for (uint32_t j = 0; j < SL_sprites_index; j++)
+ {
+ sl_use_index = j;
+ process_sprite();
+ SL_sprites_ordered[sprite_ordered_index * 4] = SL_sprites[j * 4 + 1];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 1] = sprite_sel[0];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 2] = sprite_sel[1];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 3] = SL_sprites[j * 4 + 3];
+ sprite_ordered_index++;
+ }
+
+ bool have_pixel = false;
+ uint8_t s_pixel = 0;
+ uint8_t sprite_attr = 0;
+
+ for (uint32_t i = 0; i < 160; i++)
+ {
+ have_pixel = false;
+ for (uint32_t j = 0; j < SL_sprites_index; j++)
+ {
+ if ((i >= (SL_sprites_ordered[j * 4] - 8)) &&
+ (i < SL_sprites_ordered[j * 4]) &&
+ !have_pixel)
+ {
+ // we can use the current sprite, so pick out a pixel for it
+ uint32_t t_index = i - (SL_sprites_ordered[j * 4] - 8);
+
+ t_index = 7 - t_index;
+
+ sprite_data[0] = (uint8_t)((SL_sprites_ordered[j * 4 + 1] >> t_index) & 1);
+ sprite_data[1] = (uint8_t)(((SL_sprites_ordered[j * 4 + 2] >> t_index) & 1) << 1);
+
+ s_pixel = (uint8_t)(sprite_data[0] + sprite_data[1]);
+ sprite_attr = (uint8_t)SL_sprites_ordered[j * 4 + 3];
+
+ // pixel color of 0 is transparent, so if this is the case we don't have a pixel
+ if (s_pixel != 0)
+ {
+ have_pixel = true;
+ }
+ }
+ }
+
+ if (have_pixel)
+ {
+ sprite_present_list[i] = 1;
+ sprite_pixel_list[i] = s_pixel;
+ sprite_attr_list[i] = sprite_attr;
+ }
+ else
+ {
+ sprite_present_list[i] = 0;
+ }
+ }
+ }
+
+ void OAM_scan(uint32_t OAM_cycle)
+ {
+ // we are now in STAT mode 2
+ // TODO: maybe stat mode 2 flags are set at cycle 0 on visible scanlines?
+ if (OAM_cycle == 0)
+ {
+ OAM_access_read = false;
+ OAM_access_write = false;
+
+ OAM_scan_index = 0;
+ SL_sprites_index = 0;
+ write_sprite = 0;
+ }
+
+ // the gameboy has 80 cycles to scan through 40 sprites, picking out the first 10 it finds to draw
+ // the following is a guessed at implmenentation based on how NES does it, it's probably pretty close
+ if (OAM_cycle < 10)
+ {
+ // start by clearing the sprite table (probably just clears X on hardware, but let's be safe here.)
+ SL_sprites[OAM_cycle * 4] = 0;
+ SL_sprites[OAM_cycle * 4 + 1] = 0;
+ SL_sprites[OAM_cycle * 4 + 2] = 0;
+ SL_sprites[OAM_cycle * 4 + 3] = 0;
+ }
+ else
+ {
+ if (write_sprite == 0)
+ {
+ if (OAM_scan_index < 40)
+ {
+ uint32_t temp = DMA_OAM_access ? OAM[OAM_scan_index * 4] : (uint32_t)0xFF;
+ // (sprite Y - 16) equals LY, we have a sprite
+ if ((temp - 16) <= LY &&
+ ((temp - 16) + 8 + (((LCDC & 0x4) > 0) ? 8 : 0)) > LY)
+ {
+ // always pick the first 10 in range sprites
+ if (SL_sprites_index < 10)
+ {
+ SL_sprites[SL_sprites_index * 4] = temp;
+
+ write_sprite = 1;
+ }
+ else
+ {
+ // if we already have 10 sprites, there's nothing to do, increment the index
+ OAM_scan_index++;
+ }
+ }
+ else
+ {
+ OAM_scan_index++;
+ }
+ }
+ }
+ else
+ {
+ uint32_t temp2 = DMA_OAM_access ? OAM[OAM_scan_index * 4 + write_sprite] : (uint32_t)0xFF;
+ SL_sprites[SL_sprites_index * 4 + write_sprite] = temp2;
+ write_sprite++;
+
+ if (write_sprite == 4)
+ {
+ write_sprite = 0;
+ SL_sprites_index++;
+ OAM_scan_index++;
+ }
+ }
+ }
+ }
+
+ void color_compute_BG()
+ {
+ uint32_t R;
+ uint32_t G;
+ uint32_t B;
+
+ if ((BG_bytes_index % 2) == 0)
+ {
+ R = (uint32_t)(BG_bytes[BG_bytes_index] & 0x1F);
+ G = (uint32_t)(((BG_bytes[BG_bytes_index] & 0xE0) | ((BG_bytes[BG_bytes_index + 1] & 0x03) << 8)) >> 5);
+ B = (uint32_t)((BG_bytes[BG_bytes_index + 1] & 0x7C) >> 2);
+ }
+ else
+ {
+ R = (uint32_t)(BG_bytes[BG_bytes_index - 1] & 0x1F);
+ G = (uint32_t)(((BG_bytes[BG_bytes_index - 1] & 0xE0) | ((BG_bytes[BG_bytes_index] & 0x03) << 8)) >> 5);
+ B = (uint32_t)((BG_bytes[BG_bytes_index] & 0x7C) >> 2);
+ }
+
+ uint32_t retR = ((R * 13 + G * 2 + B) >> 1) & 0xFF;
+ uint32_t retG = ((G * 3 + B) << 1) & 0xFF;
+ uint32_t retB = ((R * 3 + G * 2 + B * 11) >> 1) & 0xFF;
+
+ BG_palette[BG_bytes_index >> 1] = (uint32_t)(0xFF000000 | (retR << 16) | (retG << 8) | retB);
+ }
+
+ void color_compute_OBJ()
+ {
+ uint32_t R;
+ uint32_t G;
+ uint32_t B;
+
+ if ((OBJ_bytes_index % 2) == 0)
+ {
+ R = (uint32_t)(OBJ_bytes[OBJ_bytes_index] & 0x1F);
+ G = (uint32_t)(((OBJ_bytes[OBJ_bytes_index] & 0xE0) | ((OBJ_bytes[OBJ_bytes_index + 1] & 0x03) << 8)) >> 5);
+ B = (uint32_t)((OBJ_bytes[OBJ_bytes_index + 1] & 0x7C) >> 2);
+ }
+ else
+ {
+ R = (uint32_t)(OBJ_bytes[OBJ_bytes_index - 1] & 0x1F);
+ G = (uint32_t)(((OBJ_bytes[OBJ_bytes_index - 1] & 0xE0) | ((OBJ_bytes[OBJ_bytes_index] & 0x03) << 8)) >> 5);
+ B = (uint32_t)((OBJ_bytes[OBJ_bytes_index] & 0x7C) >> 2);
+ }
+
+ uint32_t retR = ((R * 13 + G * 2 + B) >> 1) & 0xFF;
+ uint32_t retG = ((G * 3 + B) << 1) & 0xFF;
+ uint32_t retB = ((R * 3 + G * 2 + B * 11) >> 1) & 0xFF;
+
+ OBJ_palette[OBJ_bytes_index >> 1] = (uint32_t)(0xFF000000 | (retR << 16) | (retG << 8) | retB);
+ }
+
+ void Reset()
+ {
+ LCDC = 0;
+ STAT = 0x80;
+ scroll_y = 0;
+ scroll_x = 0;
+ LY = 0;
+ LYC = 0;
+ DMA_addr = 0;
+ BGP = 0xFF;
+ obj_pal_0 = 0;
+ obj_pal_1 = 0;
+ window_y = 0x0;
+ window_x = 0x0;
+ window_x_latch = 0xFF;
+ window_y_latch = 0xFF;
+ LY_inc = 1;
+ no_scan = false;
+ OAM_access_read = true;
+ VRAM_access_read = true;
+ OAM_access_write = true;
+ VRAM_access_write = true;
+ DMA_OAM_access = true;
+
+ cycle = 0;
+ LYC_INT = false;
+ HBL_INT = false;
+ VBL_INT = false;
+ OAM_INT = false;
+
+ stat_line = false;
+ stat_line_old = false;
+
+ window_counter = 0;
+ window_pre_render = false;
+ window_started = false;
+ window_tile_inc = 0;
+ window_y_tile = 0;
+ window_x_tile = 0;
+ window_y_tile_inc = 0;
+
+ BG_bytes_inc = false;
+ OBJ_bytes_inc = false;
+ BG_bytes_index = 0;
+ OBJ_bytes_index = 0;
+ BG_transfer_byte = 0;
+ OBJ_transfer_byte = 0;
+
+ HDMA_src_hi = 0;
+ HDMA_src_lo = 0;
+ HDMA_dest_hi = 0;
+ HDMA_dest_lo = 0;
+
+ VRAM_sel = 0;
+ BG_V_flip = false;
+ HDMA_active = false;
+ HDMA_mode = false;
+ cur_DMA_src = 0;
+ cur_DMA_dest = 0;
+ HDMA_length = 0;
+ HDMA_countdown = 0;
+ HBL_HDMA_count = 0;
+ last_HBL = 0;
+ HBL_HDMA_go = false;
+ HBL_test = false;
+ }
+
+ };
+
+ #pragma endregion
+
+ #pragma region GBC_GB_PPU
+
+ class GBC_GB_PPU : public PPU
+ {
+ public:
+ uint8_t ReadReg(uint32_t addr)
+ {
+ uint8_t ret = 0;
+ //Console.WriteLine(Core.cpu.TotalExecutedCycles);
+ switch (addr)
+ {
+ case 0xFF40: ret = LCDC; break; // LCDC
+ case 0xFF41: ret = STAT; break; // STAT
+ case 0xFF42: ret = scroll_y; break; // SCY
+ case 0xFF43: ret = scroll_x; break; // SCX
+ case 0xFF44: ret = LY; break; // LY
+ case 0xFF45: ret = LYC; break; // LYC
+ case 0xFF46: ret = DMA_addr; break; // DMA
+ case 0xFF47: ret = BGP; break; // BGP
+ case 0xFF48: ret = obj_pal_0; break; // OBP0
+ case 0xFF49: ret = obj_pal_1; break; // OBP1
+ case 0xFF4A: ret = window_y; break; // WY
+ case 0xFF4B: ret = window_x; break; // WX
+
+ // These are GBC specific Regs
+ case 0xFF51: ret = HDMA_src_hi; break; // HDMA1
+ case 0xFF52: ret = HDMA_src_lo; break; // HDMA2
+ case 0xFF53: ret = HDMA_dest_hi; break; // HDMA3
+ case 0xFF54: ret = HDMA_dest_lo; break; // HDMA4
+ case 0xFF55: ret = HDMA_ctrl(); break; // HDMA5
+ case 0xFF68: ret = BG_pal_ret(); break; // BGPI
+ case 0xFF69: ret = BG_PAL_read(); break; // BGPD
+ case 0xFF6A: ret = OBJ_pal_ret(); break; // OBPI
+ case 0xFF6B: ret = OBJ_bytes[OBJ_bytes_index]; break; // OBPD
+ }
+
+ return ret;
+ }
+
+ uint8_t BG_PAL_read()
+ {
+ if (VRAM_access_read)
+ {
+ return BG_bytes[BG_bytes_index];
+ }
+ else
+ {
+ return 0xFF;
+ }
+ }
+
+ void WriteReg(uint32_t addr, uint8_t value)
+ {
+ switch (addr)
+ {
+ case 0xFF40: // LCDC
+ if (((LCDC & 0x80) > 0) && !((value & 0x80) > 0))
+ {
+ VRAM_access_read = true;
+ VRAM_access_write = true;
+ OAM_access_read = true;
+ OAM_access_write = true;
+ }
+
+ if (!((LCDC & 0x80) > 0) && ((value & 0x80) > 0))
+ {
+ // don't draw for one frame after turning on
+ blank_frame = true;
+ }
+
+ LCDC = value;
+ break;
+ case 0xFF41: // STAT
+ // note that their is no stat interrupt bug in GBC
+ STAT = (uint8_t)((value & 0xF8) | (STAT & 7) | 0x80);
+
+ if (((STAT & 3) == 0) && ((STAT & 0x8) > 0)) { HBL_INT = true; }
+ else { HBL_INT = false; }
+ if (((STAT & 3) == 1) && ((STAT & 0x10) > 0)) { VBL_INT = true; }
+ else { VBL_INT = false; }
+ // OAM not triggered?
+ // if (((STAT & 3) == 2) && STAT.Bit(5)) { OAM_INT = true; } else { OAM_INT = false; }
+
+ if (((value & 0x40) > 0) && ((LCDC & 0x80) > 0))
+ {
+ if (LY == LYC) { LYC_INT = true; }
+ else { LYC_INT = false; }
+ }
+ if (!((STAT & 0x40) > 0)) { LYC_INT = false; }
+ break;
+ case 0xFF42: // SCY
+ scroll_y = value;
+ break;
+ case 0xFF43: // SCX
+ scroll_x = value;
+ break;
+ case 0xFF44: // LY
+ LY = 0; /*reset*/
+ break;
+ case 0xFF45: // LYC
+ // tests indicate that latching writes to LYC should take place 4 cycles after the write
+ // otherwise tests around LY boundaries will fail
+ LYC_t = value;
+ LYC_cd = 4;
+ break;
+ case 0xFF46: // DMA
+ DMA_addr = value;
+ DMA_start = true;
+ DMA_OAM_access = true;
+ DMA_clock = 0;
+ DMA_inc = 0;
+ break;
+ case 0xFF47: // BGP
+ BGP = value;
+ break;
+ case 0xFF48: // OBP0
+ obj_pal_0 = value;
+ break;
+ case 0xFF49: // OBP1
+ obj_pal_1 = value;
+ break;
+ case 0xFF4A: // WY
+ window_y = value;
+ break;
+ case 0xFF4B: // WX
+ window_x = value;
+ break;
+
+ // These are GBC specific Regs
+ case 0xFF51: // HDMA1
+ HDMA_src_hi = value;
+ cur_DMA_src = (uint32_t)(((HDMA_src_hi & 0xFF) << 8) | (cur_DMA_src & 0xF0));
+ break;
+ case 0xFF52: // HDMA2
+ HDMA_src_lo = value;
+ cur_DMA_src = (uint32_t)((cur_DMA_src & 0xFF00) | (HDMA_src_lo & 0xF0));
+ break;
+ case 0xFF53: // HDMA3
+ HDMA_dest_hi = value;
+ cur_DMA_dest = (uint32_t)(((HDMA_dest_hi & 0x1F) << 8) | (cur_DMA_dest & 0xF0));
+ break;
+ case 0xFF54: // HDMA4
+ HDMA_dest_lo = value;
+ cur_DMA_dest = (uint32_t)((cur_DMA_dest & 0xFF00) | (HDMA_dest_lo & 0xF0));
+ break;
+ case 0xFF55: // HDMA5
+ if (!HDMA_active)
+ {
+ HDMA_mode = ((value & 0x80) > 0);
+ HDMA_countdown = 4;
+ HDMA_tick = 0;
+ if (((value & 0x80) > 0))
+ {
+ // HDMA during HBlank only, but only if screen is on, otherwise DMA immediately one block of data
+ // worms armaggedon requires HDMA to fire in hblank mode even if the screen is off.
+ HDMA_active = true;
+ HBL_HDMA_count = 0x10;
+
+ last_HBL = LY - 1;
+
+ HBL_test = true;
+ HBL_HDMA_go = false;
+
+ if (!((LCDC & 0x80) > 0))
+ {
+ HDMA_run_once = true;
+ }
+ }
+ else
+ {
+ // HDMA immediately
+ HDMA_active = true;
+ HDMA_transfer[0] = true;
+ }
+
+ HDMA_length = ((value & 0x7F) + 1) * 16;
+ }
+ else
+ {
+ //terminate the transfer
+ if (!((value & 0x80) > 0))
+ {
+ HDMA_active = false;
+ }
+ }
+
+ break;
+ case 0xFF68: // BGPI
+ BG_bytes_index = (uint8_t)(value & 0x3F);
+ BG_bytes_inc = ((value & 0x80) == 0x80);
+ break;
+ case 0xFF69: // BGPD
+ if (VRAM_access_write)
+ {
+ BG_transfer_byte = value;
+ BG_bytes[BG_bytes_index] = value;
+ }
+
+ // change the appropriate palette color
+ color_compute_BG();
+ if (BG_bytes_inc) { BG_bytes_index++; BG_bytes_index &= 0x3F; }
+ break;
+ case 0xFF6A: // OBPI
+ OBJ_bytes_index = (uint8_t)(value & 0x3F);
+ OBJ_bytes_inc = ((value & 0x80) == 0x80);
+ break;
+ case 0xFF6B: // OBPD
+ OBJ_transfer_byte = value;
+ OBJ_bytes[OBJ_bytes_index] = value;
+
+ // change the appropriate palette color
+ color_compute_OBJ();
+
+ if (OBJ_bytes_inc) { OBJ_bytes_index++; OBJ_bytes_index &= 0x3F; }
+ break;
+ }
+ }
+
+ void tick()
+ {
+ // Do HDMA ticks
+ if (HDMA_active)
+ {
+ if (HDMA_length > 0)
+ {
+ if (!HDMA_mode)
+ {
+ if (HDMA_countdown > 0)
+ {
+ HDMA_countdown--;
+ }
+ else
+ {
+ // immediately transfer bytes, 2 bytes per cycles
+ if ((HDMA_tick % 2) == 0)
+ {
+ HDMA_byte = ReadMemory(cur_DMA_src);
+ }
+ else
+ {
+ VRAM[(VRAM_Bank[0] * 0x2000) + cur_DMA_dest] = HDMA_byte;
+ cur_DMA_dest = (uint8_t)((cur_DMA_dest + 1) & 0x1FFF);
+ cur_DMA_src = (uint8_t)((cur_DMA_src + 1) & 0xFFFF);
+ HDMA_length--;
+ }
+
+ HDMA_tick++;
+ }
+ }
+ else
+ {
+ // only transfer during mode 0, and only 16 bytes at a time
+ if (((STAT & 3) == 0) && (LY != last_HBL) && HBL_test && (LY_inc == 1) && (cycle > 4))
+ {
+ HBL_HDMA_go = true;
+ HBL_test = false;
+ }
+ else if (HDMA_run_once)
+ {
+ HBL_HDMA_go = true;
+ HBL_test = false;
+ HDMA_run_once = false;
+ }
+
+ if (HBL_HDMA_go && (HBL_HDMA_count > 0))
+ {
+ HDMA_transfer[0] = true;
+
+ if (HDMA_countdown > 0)
+ {
+ HDMA_countdown--;
+ }
+ else
+ {
+ if ((HDMA_tick % 2) == 0)
+ {
+ HDMA_byte = ReadMemory(cur_DMA_src);
+ }
+ else
+ {
+ VRAM[(VRAM_Bank[0] * 0x2000) + cur_DMA_dest] = HDMA_byte;
+ cur_DMA_dest = (uint32_t)((cur_DMA_dest + 1) & 0x1FFF);
+ cur_DMA_src = (uint32_t)((cur_DMA_src + 1) & 0xFFFF);
+ HDMA_length--;
+ HBL_HDMA_count--;
+ }
+
+ if ((HBL_HDMA_count == 0) && (HDMA_length != 0))
+ {
+
+ HBL_test = true;
+ last_HBL = LY;
+ HBL_HDMA_count = 0x10;
+ HBL_HDMA_go = false;
+ HDMA_countdown = 4;
+ }
+
+ HDMA_tick++;
+ }
+ }
+ else
+ {
+ HDMA_transfer[0] = false;
+ }
+ }
+ }
+ else
+ {
+ HDMA_active = false;
+ HDMA_transfer[0] = false;
+ }
+ }
+
+ // the ppu only does anything if it is turned on via bit 7 of LCDC
+ if (((LCDC & 0x80) > 0))
+ {
+ // start the next scanline
+ if (cycle == 456)
+ {
+ // scanline callback
+ if ((LY + LY_inc) == _scanlineCallbackLine[0])
+ {
+ //if (Core._scanlineCallback != null)
+ //{
+ // Core.GetGPU();
+ // Core._scanlineCallback(LCDC);
+ //}
+ }
+
+ cycle = 0;
+ LY += LY_inc;
+ cpu_LY[0] = LY;
+
+ no_scan = false;
+
+ if (LY == 0 && LY_inc == 0)
+ {
+ LY_inc = 1;
+ in_vblank[0] = false;
+
+ //STAT &= 0xFC;
+
+ // special note here, the y coordiate of the window is kept if the window is deactivated
+ // meaning it will pick up where it left off if re-enabled later
+ // so we don't reset it in the scanline loop
+ window_y_tile = 0;
+ window_y_latch = window_y;
+ window_y_tile_inc = 0;
+ window_started = false;
+ if (!((LCDC & 0x20) > 0)) { window_is_reset = true; }
+ }
+
+ // Automatically restore access to VRAM at this time (force end drawing)
+ // Who Framed Roger Rabbit seems to run into this.
+ VRAM_access_write = true;
+ VRAM_access_read = true;
+
+ if (LY == 144)
+ {
+ in_vblank[0] = true;
+ }
+ }
+
+ // exit vblank if LCD went from off to on
+ if (LCD_was_off)
+ {
+ //VBL_INT = false;
+ in_vblank[0] = false;
+ LCD_was_off = false;
+
+ // we exit vblank into mode 0 for 4 cycles
+ // but no hblank interrupt, presumably this only happens transitioning from mode 3 to 0
+ STAT &= 0xFC;
+
+ // also the LCD doesn't turn on right away
+ // also, the LCD does not enter mode 2 on scanline 0 when first turned on
+ no_scan = true;
+ cycle = 8;
+ }
+
+ // the VBL stat is continuously asserted
+ if (LY >= 144)
+ {
+ if (((STAT & 0x10) > 0))
+ {
+ if ((cycle >= 4) && (LY == 144))
+ {
+ VBL_INT = true;
+ }
+ else if (LY > 144)
+ {
+ VBL_INT = true;
+ }
+ }
+
+ if ((cycle == 2) && (LY == 144))
+ {
+ // there is an edge case where a VBL INT is triggered if STAT bit 5 is set
+ if (((STAT & 0x20) > 0)) { VBL_INT = true; }
+ }
+
+ if ((cycle == 4) && (LY == 144))
+ {
+ HBL_INT = false;
+
+ // set STAT mode to 1 (VBlank) and interrupt flag if it is enabled
+ STAT &= 0xFC;
+ STAT |= 0x01;
+
+ if ((REG_FFFF[0] & 1) > 0) { FlagI[0] = true; }
+ REG_FF0F[0] |= 0x01;
+ }
+
+ if ((cycle == 4) && (LY == 144))
+ {
+ if (((STAT & 0x20) > 0)) { VBL_INT = false; }
+ }
+
+ if ((cycle == 8) && (LY == 153))
+ {
+ LY = 0;
+ LY_inc = 0;
+ cpu_LY[0] = LY;
+ }
+ }
+
+ if (!in_vblank[0])
+ {
+ if (no_scan)
+ {
+ // timings are slightly different if we just turned on the LCD
+ // there is no mode 2 (presumably it missed the trigger)
+ if (cycle < 85)
+ {
+ if (cycle == 8)
+ {
+ // clear the sprite table
+ for (uint32_t k = 0; k < 10; k++)
+ {
+ SL_sprites[k * 4] = 0;
+ SL_sprites[k * 4 + 1] = 0;
+ SL_sprites[k * 4 + 2] = 0;
+ SL_sprites[k * 4 + 3] = 0;
+ }
+
+ if (LY != LYC)
+ {
+ LYC_INT = false;
+ STAT &= 0xFB;
+ }
+
+ if ((LY == LYC) && !((STAT & 0x4) > 0))
+ {
+ // set STAT coincidence FLAG and interrupt flag if it is enabled
+ STAT |= 0x04;
+ if (((STAT & 0x40) > 0)) { LYC_INT = true; }
+ }
+ }
+
+ if (cycle == 84)
+ {
+ STAT &= 0xFC;
+ STAT |= 0x03;
+ OAM_INT = false;
+
+ OAM_access_read = false;
+ OAM_access_write = false;
+ VRAM_access_read = false;
+ VRAM_access_write = false;
+ }
+ }
+ else
+ {
+ if (cycle >= 85)
+ {
+ if (cycle == 85)
+ {
+ // x-scroll is expected to be latched one cycle later
+ // this is fine since nothing has started in the rendering until the second cycle
+ // calculate the column number of the tile to start with
+ x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
+ render_offset = scroll_x % 8;
+ }
+
+ // render the screen and handle hblank
+ render(cycle - 85);
+ }
+ }
+ }
+ else
+ {
+ if (cycle <= 80)
+ {
+ if (cycle == 2)
+ {
+ if (LY != 0)
+ {
+ HBL_INT = false;
+
+ if (((STAT & 0x20) > 0)) { OAM_INT = true; }
+ }
+ }
+ else if (cycle == 4)
+ {
+ // here mode 2 will be set to true and interrupts fired if enabled
+ STAT &= 0xFC;
+ STAT |= 0x2;
+
+ if (LY == 0)
+ {
+ VBL_INT = false;
+ if (((STAT & 0x20) > 0)) { OAM_INT = true; }
+ }
+ }
+
+ if (cycle == 80)
+ {
+ OAM_access_read = false;
+ OAM_access_write = true;
+ VRAM_access_read = false;
+ }
+ else
+ {
+ // here OAM scanning is performed
+ OAM_scan(cycle);
+ }
+ }
+ else if (cycle >= 83)
+ {
+ if (cycle == 84)
+ {
+ STAT &= 0xFC;
+ STAT |= 0x03;
+ OAM_INT = false;
+ OAM_access_write = false;
+ VRAM_access_write = false;
+
+ // x-scroll is expected to be latched one cycle later
+ // this is fine since nothing has started in the rendering until the second cycle
+ // calculate the column number of the tile to start with
+ x_tile = (uint32_t)floor((float)(scroll_x) / 8.0);
+ render_offset = scroll_x % 8;
+ }
+
+ // render the screen and handle hblank
+ render(cycle - 83);
+ }
+ }
+ }
+
+ if (LY_inc == 0)
+ {
+ if (cycle == 12)
+ {
+ LYC_INT = false;
+ STAT &= 0xFB;
+ }
+ else if (cycle == 14)
+ {
+ // Special case of LY = LYC
+ if ((LY == LYC) && !((STAT & 0x4) > 0))
+ {
+ // set STAT coincidence FLAG and interrupt flag if it is enabled
+ STAT |= 0x04;
+ if (((STAT & 0x40) > 0)) { LYC_INT = true; }
+ }
+ }
+ }
+
+ // here LY=LYC will be asserted or cleared (but only if LY isnt 0 as that's a special case)
+ if ((cycle == 4) && (LY != 0))
+ {
+ if (LY_inc == 1)
+ {
+ LYC_INT = false;
+ STAT &= 0xFB;
+ }
+ }
+ else if ((cycle == 6) && (LY != 0))
+ {
+ if ((LY == LYC) && !((STAT & 0x4) > 0))
+ {
+ // set STAT coincidence FLAG and interrupt flag if it is enabled
+ STAT |= 0x04;
+ if (((STAT & 0x40) > 0)) { LYC_INT = true; }
+ }
+ }
+
+ cycle++;
+ }
+ else
+ {
+ STAT &= 0xFC;
+
+ VBL_INT = LYC_INT = HBL_INT = OAM_INT = false;
+
+ in_vblank[0] = true;
+
+ LCD_was_off = true;
+
+ LY = 0;
+ cpu_LY[0] = LY;
+
+ cycle = 0;
+ }
+
+ // assert the STAT IRQ line if the line went from zero to 1
+ stat_line = VBL_INT | LYC_INT | HBL_INT | OAM_INT;
+
+ if (stat_line && !stat_line_old)
+ {
+ if ((REG_FFFF[0] & 0x2) > 0) { FlagI[0] = true; }
+ REG_FF0F[0] |= 0x02;
+ }
+
+ stat_line_old = stat_line;
+
+ // process latch delays
+ //latch_delay();
+
+ if (LYC_cd > 0)
+ {
+ LYC_cd--;
+ if (LYC_cd == 0)
+ {
+ LYC = LYC_t;
+
+ if (((LCDC & 0x80) > 0))
+ {
+ if (LY != LYC) { STAT &= 0xFB; LYC_INT = false; }
+ else { STAT |= 0x4; LYC_INT = true; }
+ }
+ }
+ }
+ }
+
+ // might be needed, not sure yet
+ void latch_delay()
+ {
+ //BGP_l = BGP;
+ }
+
+ void render(uint32_t render_cycle)
+ {
+ // we are now in STAT mode 3
+ // NOTE: presumably the first necessary sprite is fetched at sprite evaulation
+ // i.e. just keeping track of the lowest x-value sprite
+ if (render_cycle == 0)
+ {
+ /*
+ OAM_access_read = false;
+ OAM_access_write = true;
+ VRAM_access_read = false;
+ */
+ // window X is latched for the scanline, mid-line changes have no effect
+ window_x_latch = window_x;
+
+ OAM_scan_index = 0;
+ read_case = 0;
+ internal_cycle = 0;
+ pre_render = true;
+ pre_render_2 = true;
+ tile_inc = 0;
+ pixel_counter = -8;
+ sl_use_index = 0;
+ fetch_sprite = false;
+ going_to_fetch = false;
+ first_fetch = true;
+ consecutive_sprite = (int32_t)(render_offset * (-1)) + 8;
+ no_sprites = false;
+ evaled_sprites = 0;
+ window_pre_render = false;
+ window_latch = ((LCDC & 0x20) > 0);
+
+ total_counter = 0;
+
+ // TODO: If Window is turned on midscanline what happens? When is this check done exactly?
+ if ((window_started && window_latch) || (window_is_reset && !window_latch && (LY >= window_y_latch)))
+ {
+ window_y_tile_inc++;
+ if (window_y_tile_inc == 8)
+ {
+ window_y_tile_inc = 0;
+ window_y_tile++;
+ window_y_tile %= 32;
+ }
+ }
+ window_started = false;
+
+ if (SL_sprites_index == 0) { no_sprites = true; }
+ // it is much easier to process sprites if we order them according to the rules of sprite priority first
+ if (!no_sprites) { reorder_and_assemble_sprites(); }
+ }
+
+ // before anything else, we have to check if windowing is in effect
+ if (window_latch && !window_started && (LY >= window_y_latch) && (pixel_counter >= (int32_t)(window_x_latch - 7)) && (window_x_latch < 167))
+ {
+ /*
+ Console.Write(LY);
+ Console.Write(" ");
+ Console.Write(cycle);
+ Console.Write(" ");
+ Console.Write(window_y_tile);
+ Console.Write(" ");
+ Console.Write(render_offset);
+ Console.Write(" ");
+ Console.Write(window_x_latch);
+ Console.Write(" ");
+ Console.WriteLine(pixel_counter);
+ */
+
+ if (window_x_latch == 0)
+ {
+ // if the window starts at zero, we still do the first access to the BG
+ // but then restart all over again at the window
+ if ((render_offset % 7) <= 6)
+ {
+ read_case = 9;
+ }
+ else
+ {
+ read_case = 10;
+ }
+ }
+ else
+ {
+ read_case = 4;
+ }
+
+ window_pre_render = true;
+
+ window_counter = 0;
+ render_counter = 0;
+
+ window_x_tile = (uint32_t)floor((float)(pixel_counter - (window_x_latch - 7)) / 8.0);
+
+ window_tile_inc = 0;
+ window_started = true;
+ window_is_reset = false;
+ }
+
+ if (!pre_render && !fetch_sprite)
+ {
+ // start shifting data into the LCD
+ if (render_counter >= (render_offset + 8))
+ {
+ if (((tile_data_latch[2] & 0x20) > 0) && GBC_compat[0])
+ {
+ pixel = (tile_data_latch[0] & (1 << (render_counter % 8))) > 0 ? 1 : 0;
+ pixel |= (tile_data_latch[1] & (1 << (render_counter % 8))) > 0 ? 2 : 0;
+ }
+ else
+ {
+ pixel = (tile_data_latch[0] & (1 << (7 - (render_counter % 8)))) > 0 ? 1 : 0;
+ pixel |= (tile_data_latch[1] & (1 << (7 - (render_counter % 8)))) > 0 ? 2 : 0;
+ }
+
+ uint32_t ref_pixel = pixel;
+
+ if (!GBC_compat[0])
+ {
+ if (((LCDC & 0x1) > 0))
+ {
+ pixel = (BGP >> (pixel * 2)) & 3;
+ }
+ else
+ {
+ pixel = 0;
+ }
+ }
+
+ uint32_t pal_num = tile_data_latch[2] & 0x7;
+
+ bool use_sprite = false;
+
+ uint32_t s_pixel = 0;
+
+ // now we have the BG pixel, we next need the sprite pixel
+ if (!no_sprites)
+ {
+ bool have_sprite = false;
+ uint32_t sprite_attr = 0;
+
+ if (sprite_present_list[pixel_counter] == 1)
+ {
+ have_sprite = true;
+ s_pixel = sprite_pixel_list[pixel_counter];
+ sprite_attr = sprite_attr_list[pixel_counter];
+ }
+
+ if (have_sprite)
+ {
+ if (((LCDC & 0x2) > 0))
+ {
+ if (!((sprite_attr & 0x80) > 0))
+ {
+ use_sprite = true;
+ }
+ else if (ref_pixel == 0)
+ {
+ use_sprite = true;
+ }
+
+ if (!((LCDC & 0x1) > 0))
+ {
+ use_sprite = true;
+ }
+
+ // There is another priority bit in GBC, that can still override sprite priority
+ if (((LCDC & 0x1) > 0) && ((tile_data_latch[2] & 0x80) > 0) && (ref_pixel != 0) && GBC_compat[0])
+ {
+ use_sprite = false;
+ }
+ }
+
+ if (use_sprite)
+ {
+ pal_num = sprite_attr & 7;
+
+ if (!GBC_compat[0])
+ {
+ pal_num = ((sprite_attr & 0x10) > 0) ? 1 : 0;
+
+ if (((sprite_attr & 0x10) > 0))
+ {
+ pixel = (obj_pal_1 >> (s_pixel * 2)) & 3;
+ }
+ else
+ {
+ pixel = (obj_pal_0 >> (s_pixel * 2)) & 3;
+ }
+ }
+ }
+ }
+ }
+
+ // based on sprite priority and pixel values, pick a final pixel color
+ if (GBC_compat[0])
+ {
+ if (use_sprite)
+ {
+ _vidbuffer[LY * 160 + pixel_counter] = (uint32_t)OBJ_palette[pal_num * 4 + s_pixel];
+ }
+ else
+ {
+ _vidbuffer[LY * 160 + pixel_counter] = (uint32_t)BG_palette[pal_num * 4 + pixel];
+ }
+ }
+ else
+ {
+ if (use_sprite)
+ {
+ _vidbuffer[LY * 160 + pixel_counter] = (uint32_t)OBJ_palette[pal_num * 4 + pixel];
+ }
+ else
+ {
+ _vidbuffer[LY * 160 + pixel_counter] = (uint32_t)BG_palette[pixel];
+ }
+ }
+
+ pixel_counter++;
+
+ if (pixel_counter == 160)
+ {
+ read_case = 8;
+ hbl_countdown = 2;
+ }
+ }
+ else if (pixel_counter < 0)
+ {
+ pixel_counter++;
+ }
+ render_counter++;
+ }
+
+ if (!fetch_sprite)
+ {
+ if (!pre_render_2)
+ {
+ // before we go on to read case 3, we need to know if we stall there or not
+ // Gekkio's tests show that if sprites are at position 0 or 1 (mod 8)
+ // then it takes an extra cycle (1 or 2 more t-states) to process them
+
+ if (!no_sprites && (pixel_counter < 160))
+ {
+ for (uint32_t i = 0; i < SL_sprites_index; i++)
+ {
+ if ((pixel_counter >= (int32_t)(SL_sprites[i * 4 + 1] - 8)) &&
+ (pixel_counter < (int32_t)(SL_sprites[i * 4 + 1])) &&
+ !((evaled_sprites & (1 << i)) > 0))
+ {
+ going_to_fetch = true;
+ fetch_sprite = true;
+ }
+ }
+ }
+ }
+
+ switch (read_case)
+ {
+ case 0: // read a background tile
+ if ((internal_cycle % 2) == 1)
+ {
+ // calculate the row number of the tiles to be fetched
+ y_tile = ((uint32_t)floor((float)((uint32_t)scroll_y + LY) / 8.0)) % 32;
+
+ temp_fetch = y_tile * 32 + (x_tile + tile_inc) % 32;
+ tile_byte = VRAM[0x1800 + (((LCDC & 0x4) > 0) ? 1 : 0) * 0x400 + temp_fetch];
+ tile_data[2] = VRAM[0x3800 + (((LCDC & 0x4) > 0) ? 1 : 0) * 0x400 + temp_fetch];
+ VRAM_sel = ((tile_data[2] & 0x8) > 0) ? 1 : 0;
+
+ BG_V_flip = ((tile_data[2] & 0x40) > 0)& GBC_compat[0];
+
+ read_case = 1;
+ if (!pre_render)
+ {
+ tile_inc++;
+ }
+ }
+ break;
+
+ case 1: // read from tile graphics (0)
+ if ((internal_cycle % 2) == 1)
+ {
+ y_scroll_offset = (scroll_y + LY) % 8;
+
+ if (BG_V_flip)
+ {
+ y_scroll_offset = 7 - y_scroll_offset;
+ }
+
+ if (((LCDC & 0x10) > 0))
+ {
+ tile_data[0] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed byte
+ if (((tile_byte & 0x80) > 0))
+ {
+ tile_byte -= 256;
+ }
+ tile_data[0] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
+ }
+
+ read_case = 2;
+ }
+ break;
+
+ case 2: // read from tile graphics (1)
+ if ((internal_cycle % 2) == 0)
+ {
+ pre_render_2 = false;
+ }
+ else
+ {
+ y_scroll_offset = (scroll_y + LY) % 8;
+
+ if (BG_V_flip)
+ {
+ y_scroll_offset = 7 - y_scroll_offset;
+ }
+
+ if (((LCDC & 0x10) > 0))
+ {
+ // if LCDC somehow changed between the two reads, make sure we have a positive number
+ if (tile_byte < 0)
+ {
+ tile_byte += 256;
+ }
+ tile_data[1] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed byte
+ if (((tile_byte & 0x80) > 0) && tile_byte > 0)
+ {
+ tile_byte -= 256;
+ }
+ tile_data[1] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+
+ if (pre_render)
+ {
+ // here we set up rendering
+ pre_render = false;
+
+ render_counter = 0;
+ latch_counter = 0;
+ read_case = 0;
+ }
+ else
+ {
+ read_case = 3;
+ }
+ }
+ break;
+
+ case 3: // read from sprite data
+ if ((internal_cycle % 2) == 1)
+ {
+ read_case = 0;
+ latch_new_data = true;
+ }
+ break;
+
+ case 4: // read from window data
+ if ((window_counter % 2) == 1)
+ {
+ temp_fetch = window_y_tile * 32 + (window_x_tile + window_tile_inc) % 32;
+ tile_byte = VRAM[0x1800 + (((LCDC & 0x40) > 0) ? 1 : 0) * 0x400 + temp_fetch];
+ tile_data[2] = VRAM[0x3800 + (((LCDC & 0x40) > 0) ? 1 : 0) * 0x400 + temp_fetch];
+ VRAM_sel = ((tile_data[2] & 0x8) > 0) ? 1 : 0;
+ BG_V_flip = ((tile_data[2] & 0x40) > 0)& GBC_compat[0];
+
+ window_tile_inc++;
+ read_case = 5;
+ }
+ window_counter++;
+ break;
+
+ case 5: // read from tile graphics (for the window)
+ if ((window_counter % 2) == 1)
+ {
+ y_scroll_offset = (window_y_tile_inc) % 8;
+
+ if (BG_V_flip)
+ {
+ y_scroll_offset = 7 - y_scroll_offset;
+ }
+
+ if (((LCDC & 0x10) > 0))
+ {
+ tile_data[0] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed byte
+ if (((tile_byte & 0x80) > 0))
+ {
+ tile_byte -= 256;
+ }
+ tile_data[0] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
+ }
+
+ read_case = 6;
+ }
+ window_counter++;
+ break;
+
+ case 6: // read from tile graphics (for the window)
+ if ((window_counter % 2) == 1)
+ {
+ y_scroll_offset = (window_y_tile_inc) % 8;
+
+ if (BG_V_flip)
+ {
+ y_scroll_offset = 7 - y_scroll_offset;
+ }
+
+ if (((LCDC & 0x10) > 0))
+ {
+ // if LCDC somehow changed between the two reads, make sure we have a positive number
+ if (tile_byte < 0)
+ {
+ tile_byte += 256;
+ }
+ tile_data[1] = VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+ else
+ {
+ // same as before except now tile uint8_t represents a signed byte
+ if (((tile_byte & 0x80) > 0) && tile_byte > 0)
+ {
+ tile_byte -= 256;
+ }
+ tile_data[1] = VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
+ }
+
+ if (window_pre_render)
+ {
+ // here we set up rendering
+ // unlike for the normal background case, there is no pre-render period for the window
+ // so start shifting in data to the screen right away
+ if (window_x_latch <= 7)
+ {
+ if (render_offset == 0)
+ {
+ read_case = 4;
+ }
+ else
+ {
+ read_case = 9 + render_offset - 1;
+ }
+ render_counter = 8 - render_offset;
+
+ render_offset = 7 - window_x_latch;
+ }
+ else
+ {
+ render_offset = 0;
+ read_case = 4;
+ render_counter = 8;
+ }
+
+ latch_counter = 0;
+ latch_new_data = true;
+ window_pre_render = false;
+ }
+ else
+ {
+ read_case = 7;
+ }
+ }
+ window_counter++;
+ break;
+
+ case 7: // read from sprite data
+ if ((window_counter % 2) == 1)
+ {
+ read_case = 4;
+ latch_new_data = true;
+ }
+ window_counter++;
+ break;
+
+ case 8: // done reading, we are now in phase 0
+ pre_render = true;
+
+ // the other interrupts appear to be delayed by 1 CPU cycle, so do the same here
+ if (hbl_countdown > 0)
+ {
+ hbl_countdown--;
+
+ if (hbl_countdown == 0)
+ {
+ OAM_access_read = true;
+ OAM_access_write = true;
+ VRAM_access_read = true;
+ VRAM_access_write = true;
+ }
+ else
+ {
+ STAT &= 0xFC;
+ STAT |= 0x00;
+
+ if (((STAT & 0x8) > 0)) { HBL_INT = true; }
+ }
+ }
+ break;
+
+ case 9:
+ // this is a degenerate case for starting the window at 0
+ // kevtris' timing doc indicates an additional normal BG access
+ // but this information is thrown away, so it's faster to do this then constantly check
+ // for it in read case 0
+ read_case = 4;
+ break;
+ case 10:
+ case 11:
+ case 12:
+ case 13:
+ case 14:
+ case 15:
+ case 16:
+ case 17:
+ read_case--;
+ break;
+ }
+ internal_cycle++;
+
+ if (latch_new_data)
+ {
+ latch_new_data = false;
+ tile_data_latch[0] = tile_data[0];
+ tile_data_latch[1] = tile_data[1];
+ tile_data_latch[2] = tile_data[2];
+ }
+ }
+
+ // every in range sprite takes 6 cycles to process
+ // sprites located at x=0 still take 6 cycles to process even though they don't appear on screen
+ // sprites above x=168 do not take any cycles to process however
+ if (fetch_sprite)
+ {
+ if (going_to_fetch)
+ {
+ going_to_fetch = false;
+
+ last_eval = 0;
+
+ // at this time it is unknown what each cycle does, but we only need to accurately keep track of cycles
+ for (uint32_t i = 0; i < SL_sprites_index; i++)
+ {
+ if ((pixel_counter >= (int32_t)(SL_sprites[i * 4 + 1] - 8)) &&
+ (pixel_counter < (int32_t)(SL_sprites[i * 4 + 1])) &&
+ !((evaled_sprites & (1 << i)) > 0))
+ {
+ sprite_fetch_counter += 6;
+ evaled_sprites |= (1 << i);
+ last_eval = SL_sprites[i * 4 + 1];
+ }
+ }
+
+ // x scroll offsets the penalty table
+ // there is no penalty if the next sprites to be fetched are within the currentfetch block (8 pixels)
+ if (first_fetch || ((int32_t)last_eval >= consecutive_sprite))
+ {
+ if (((last_eval + render_offset) % 8) == 0) { sprite_fetch_counter += 5; }
+ else if (((last_eval + render_offset) % 8) == 1) { sprite_fetch_counter += 4; }
+ else if (((last_eval + render_offset) % 8) == 2) { sprite_fetch_counter += 3; }
+ else if (((last_eval + render_offset) % 8) == 3) { sprite_fetch_counter += 2; }
+ else if (((last_eval + render_offset) % 8) == 4) { sprite_fetch_counter += 1; }
+ else if (((last_eval + render_offset) % 8) == 5) { sprite_fetch_counter += 0; }
+ else if (((last_eval + render_offset) % 8) == 6) { sprite_fetch_counter += 0; }
+ else if (((last_eval + render_offset) % 8) == 7) { sprite_fetch_counter += 0; }
+
+ consecutive_sprite = (uint32_t)floor((double)(last_eval + render_offset) / 8.0) * 8 + 8 - render_offset;
+
+ // special case exists here for sprites at zero with non-zero x-scroll. Not sure exactly the reason for it.
+ if (last_eval == 0 && render_offset != 0)
+ {
+ sprite_fetch_counter += render_offset;
+ }
+ }
+
+ total_counter += sprite_fetch_counter;
+
+ first_fetch = false;
+ }
+ else
+ {
+ sprite_fetch_counter--;
+ if (sprite_fetch_counter == 0)
+ {
+ fetch_sprite = false;
+ }
+ }
+ }
+
+ }
+
+ void process_sprite()
+ {
+ uint32_t y;
+ uint32_t VRAM_temp = (((SL_sprites[sl_use_index * 4 + 3] & 0x8) > 0) && GBC_compat[0]) ? 1 : 0;
+
+ if ((SL_sprites[sl_use_index * 4 + 3] & 0x40) > 0)
+ {
+ if (((LCDC & 0x4) > 0))
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ y = 15 - y;
+ sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
+ sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
+ }
+ else
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ y = 7 - y;
+ sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
+ sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
+ }
+ }
+ else
+ {
+ if (((LCDC & 0x4) > 0))
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2];
+ sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + (SL_sprites[sl_use_index * 4 + 2] & 0xFE) * 16 + y * 2 + 1];
+ }
+ else
+ {
+ y = LY - (SL_sprites[sl_use_index * 4] - 16);
+ sprite_sel[0] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2];
+ sprite_sel[1] = VRAM[(VRAM_temp * 0x2000) + SL_sprites[sl_use_index * 4 + 2] * 16 + y * 2 + 1];
+ }
+ }
+
+ if ((SL_sprites[sl_use_index * 4 + 3] & 0x20) > 0)
+ {
+ uint32_t b0, b1, b2, b3, b4, b5, b6, b7 = 0;
+ for (uint32_t i = 0; i < 2; i++)
+ {
+ b0 = (sprite_sel[i] & 0x01) << 7;
+ b1 = (sprite_sel[i] & 0x02) << 5;
+ b2 = (sprite_sel[i] & 0x04) << 3;
+ b3 = (sprite_sel[i] & 0x08) << 1;
+ b4 = (sprite_sel[i] & 0x10) >> 1;
+ b5 = (sprite_sel[i] & 0x20) >> 3;
+ b6 = (sprite_sel[i] & 0x40) >> 5;
+ b7 = (sprite_sel[i] & 0x80) >> 7;
+
+ sprite_sel[i] = (uint8_t)(b0 | b1 | b2 | b3 | b4 | b5 | b6 | b7);
+ }
+ }
+ }
+
+ // normal DMA moves twice as fast in double speed mode on GBC
+ // So give it it's own function so we can seperate it from PPU tick
+ void DMA_tick()
+ {
+ // Note that DMA is halted when the CPU is halted
+ if (DMA_start && !cpu_halted)
+ {
+ if (DMA_clock >= 4)
+ {
+ DMA_OAM_access = false;
+ if ((DMA_clock % 4) == 1)
+ {
+ // the cpu can't access memory during this time, but we still need the ppu to be able to.
+ DMA_start = false;
+ // Gekkio reports that A14 being high on DMA transfers always represent WRAM accesses
+ // So transfers nominally from higher memory areas are actually still from there (i.e. FF -> DF)
+ uint8_t DMA_actual = DMA_addr;
+ if (DMA_addr > 0xDF) { DMA_actual &= 0xDF; }
+ DMA_byte = ReadMemory((uint32_t)((DMA_actual << 8) + DMA_inc));
+ DMA_start = true;
+ }
+ else if ((DMA_clock % 4) == 3)
+ {
+ OAM[DMA_inc] = DMA_byte;
+
+ if (DMA_inc < (0xA0 - 1)) { DMA_inc++; }
+ }
+ }
+
+ DMA_clock++;
+
+ if (DMA_clock == 648)
+ {
+ DMA_start = false;
+ DMA_OAM_access = true;
+ }
+ }
+ }
+
+ // order sprites according to x coordinate
+ // note that for sprites of equal x coordinate, priority goes to first on the list
+ void reorder_and_assemble_sprites()
+ {
+ sprite_ordered_index = 0;
+
+ // In CGB mode, sprites are ordered solely based on their position in OAM, so they are already ordered
+
+ if (GBC_compat[0])
+ {
+ for (uint32_t j = 0; j < SL_sprites_index; j++)
+ {
+ sl_use_index = j;
+ process_sprite();
+ SL_sprites_ordered[sprite_ordered_index * 4] = SL_sprites[j * 4 + 1];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 1] = sprite_sel[0];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 2] = sprite_sel[1];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 3] = SL_sprites[j * 4 + 3];
+ sprite_ordered_index++;
+ }
+ }
+ else
+ {
+ for (int i = 0; i < 256; i++)
+ {
+ for (int j = 0; j < SL_sprites_index; j++)
+ {
+ if (SL_sprites[j * 4 + 1] == i)
+ {
+ sl_use_index = j;
+ process_sprite();
+ SL_sprites_ordered[sprite_ordered_index * 4] = SL_sprites[j * 4 + 1];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 1] = sprite_sel[0];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 2] = sprite_sel[1];
+ SL_sprites_ordered[sprite_ordered_index * 4 + 3] = SL_sprites[j * 4 + 3];
+ sprite_ordered_index++;
+ }
+ }
+ }
+ }
+
+ bool have_pixel = false;
+ uint8_t s_pixel = 0;
+ uint8_t sprite_attr = 0;
+
+ for (uint32_t i = 0; i < 160; i++)
+ {
+ have_pixel = false;
+ for (uint32_t j = 0; j < SL_sprites_index; j++)
+ {
+ if ((i >= (SL_sprites_ordered[j * 4] - 8)) &&
+ (i < SL_sprites_ordered[j * 4]) &&
+ !have_pixel)
+ {
+ // we can use the current sprite, so pick out a pixel for it
+ uint32_t t_index = i - (SL_sprites_ordered[j * 4] - 8);
+
+ t_index = 7 - t_index;
+
+ sprite_data[0] = (uint8_t)((SL_sprites_ordered[j * 4 + 1] >> t_index) & 1);
+ sprite_data[1] = (uint8_t)(((SL_sprites_ordered[j * 4 + 2] >> t_index) & 1) << 1);
+
+ s_pixel = (uint8_t)(sprite_data[0] + sprite_data[1]);
+ sprite_attr = (uint8_t)SL_sprites_ordered[j * 4 + 3];
+
+ // pixel color of 0 is transparent, so if this is the case we don't have a pixel
+ if (s_pixel != 0)
+ {
+ have_pixel = true;
+ }
+ }
+ }
+
+ if (have_pixel)
+ {
+ sprite_present_list[i] = 1;
+ sprite_pixel_list[i] = s_pixel;
+ sprite_attr_list[i] = sprite_attr;
+ }
+ else
+ {
+ sprite_present_list[i] = 0;
+ }
+ }
+ }
+
+ void OAM_scan(uint32_t OAM_cycle)
+ {
+ // we are now in STAT mode 2
+ // TODO: maybe stat mode 2 flags are set at cycle 0 on visible scanlines?
+ if (OAM_cycle == 0)
+ {
+ OAM_access_read = false;
+ OAM_access_write = false;
+
+ OAM_scan_index = 0;
+ SL_sprites_index = 0;
+ write_sprite = 0;
+ }
+
+ // the gameboy has 80 cycles to scan through 40 sprites, picking out the first 10 it finds to draw
+ // the following is a guessed at implmenentation based on how NES does it, it's probably pretty close
+ if (OAM_cycle < 10)
+ {
+ // start by clearing the sprite table (probably just clears X on hardware, but let's be safe here.)
+ SL_sprites[OAM_cycle * 4] = 0;
+ SL_sprites[OAM_cycle * 4 + 1] = 0;
+ SL_sprites[OAM_cycle * 4 + 2] = 0;
+ SL_sprites[OAM_cycle * 4 + 3] = 0;
+ }
+ else
+ {
+ if (write_sprite == 0)
+ {
+ if (OAM_scan_index < 40)
+ {
+ uint32_t temp = DMA_OAM_access ? OAM[OAM_scan_index * 4] : (uint32_t)0xFF;
+ // (sprite Y - 16) equals LY, we have a sprite
+ if ((temp - 16) <= LY &&
+ ((temp - 16) + 8 + (((LCDC & 0x4) > 0) ? 8 : 0)) > LY)
+ {
+ // always pick the first 10 in range sprites
+ if (SL_sprites_index < 10)
+ {
+ SL_sprites[SL_sprites_index * 4] = temp;
+
+ write_sprite = 1;
+ }
+ else
+ {
+ // if we already have 10 sprites, there's nothing to do, increment the index
+ OAM_scan_index++;
+ }
+ }
+ else
+ {
+ OAM_scan_index++;
+ }
+ }
+ }
+ else
+ {
+ uint32_t temp2 = DMA_OAM_access ? OAM[OAM_scan_index * 4 + write_sprite] : (uint32_t)0xFF;
+ SL_sprites[SL_sprites_index * 4 + write_sprite] = temp2;
+ write_sprite++;
+
+ if (write_sprite == 4)
+ {
+ write_sprite = 0;
+ SL_sprites_index++;
+ OAM_scan_index++;
+ }
+ }
+ }
+ }
+
+ void color_compute_BG()
+ {
+ uint32_t R;
+ uint32_t G;
+ uint32_t B;
+
+ if ((BG_bytes_index % 2) == 0)
+ {
+ R = (uint32_t)(BG_bytes[BG_bytes_index] & 0x1F);
+ G = (uint32_t)(((BG_bytes[BG_bytes_index] & 0xE0) | ((BG_bytes[BG_bytes_index + 1] & 0x03) << 8)) >> 5);
+ B = (uint32_t)((BG_bytes[BG_bytes_index + 1] & 0x7C) >> 2);
+ }
+ else
+ {
+ R = (uint32_t)(BG_bytes[BG_bytes_index - 1] & 0x1F);
+ G = (uint32_t)(((BG_bytes[BG_bytes_index - 1] & 0xE0) | ((BG_bytes[BG_bytes_index] & 0x03) << 8)) >> 5);
+ B = (uint32_t)((BG_bytes[BG_bytes_index] & 0x7C) >> 2);
+ }
+
+ uint32_t retR = ((R * 13 + G * 2 + B) >> 1) & 0xFF;
+ uint32_t retG = ((G * 3 + B) << 1) & 0xFF;
+ uint32_t retB = ((R * 3 + G * 2 + B * 11) >> 1) & 0xFF;
+
+ BG_palette[BG_bytes_index >> 1] = (uint32_t)(0xFF000000 | (retR << 16) | (retG << 8) | retB);
+ }
+
+ void color_compute_OBJ()
+ {
+ uint32_t R;
+ uint32_t G;
+ uint32_t B;
+
+ if ((OBJ_bytes_index % 2) == 0)
+ {
+ R = (uint32_t)(OBJ_bytes[OBJ_bytes_index] & 0x1F);
+ G = (uint32_t)(((OBJ_bytes[OBJ_bytes_index] & 0xE0) | ((OBJ_bytes[OBJ_bytes_index + 1] & 0x03) << 8)) >> 5);
+ B = (uint32_t)((OBJ_bytes[OBJ_bytes_index + 1] & 0x7C) >> 2);
+ }
+ else
+ {
+ R = (uint32_t)(OBJ_bytes[OBJ_bytes_index - 1] & 0x1F);
+ G = (uint32_t)(((OBJ_bytes[OBJ_bytes_index - 1] & 0xE0) | ((OBJ_bytes[OBJ_bytes_index] & 0x03) << 8)) >> 5);
+ B = (uint32_t)((OBJ_bytes[OBJ_bytes_index] & 0x7C) >> 2);
+ }
+
+ uint32_t retR = ((R * 13 + G * 2 + B) >> 1) & 0xFF;
+ uint32_t retG = ((G * 3 + B) << 1) & 0xFF;
+ uint32_t retB = ((R * 3 + G * 2 + B * 11) >> 1) & 0xFF;
+
+ OBJ_palette[OBJ_bytes_index >> 1] = (uint32_t)(0xFF000000 | (retR << 16) | (retG << 8) | retB);
+ }
+
+ void Reset()
+ {
+ LCDC = 0;
+ STAT = 0x80;
+ scroll_y = 0;
+ scroll_x = 0;
+ LY = 0;
+ LYC = 0;
+ DMA_addr = 0;
+ BGP = 0xFF;
+ obj_pal_0 = 0;
+ obj_pal_1 = 0;
+ window_y = 0x0;
+ window_x = 0x0;
+ window_x_latch = 0xFF;
+ window_y_latch = 0xFF;
+ LY_inc = 1;
+ no_scan = false;
+ OAM_access_read = true;
+ VRAM_access_read = true;
+ OAM_access_write = true;
+ VRAM_access_write = true;
+ DMA_OAM_access = true;
+
+ cycle = 0;
+ LYC_INT = false;
+ HBL_INT = false;
+ VBL_INT = false;
+ OAM_INT = false;
+
+ stat_line = false;
+ stat_line_old = false;
+
+ window_counter = 0;
+ window_pre_render = false;
+ window_started = false;
+ window_tile_inc = 0;
+ window_y_tile = 0;
+ window_x_tile = 0;
+ window_y_tile_inc = 0;
+
+ BG_bytes_inc = false;
+ OBJ_bytes_inc = false;
+ BG_bytes_index = 0;
+ OBJ_bytes_index = 0;
+ BG_transfer_byte = 0;
+ OBJ_transfer_byte = 0;
+
+ HDMA_src_hi = 0;
+ HDMA_src_lo = 0;
+ HDMA_dest_hi = 0;
+ HDMA_dest_lo = 0;
+
+ VRAM_sel = 0;
+ BG_V_flip = false;
+ HDMA_active = false;
+ HDMA_mode = false;
+ cur_DMA_src = 0;
+ cur_DMA_dest = 0;
+ HDMA_length = 0;
+ HDMA_countdown = 0;
+ HBL_HDMA_count = 0;
+ last_HBL = 0;
+ HBL_HDMA_go = false;
+ HBL_test = false;
+ }
+ };
+
+ #pragma endregion
+}
diff --git a/libHawk/GBHawk/GBHawk/PPU_Base.h b/libHawk/GBHawk/GBHawk/PPU_Base.h
deleted file mode 100644
index b5a7ea831a..0000000000
--- a/libHawk/GBHawk/GBHawk/PPU_Base.h
+++ /dev/null
@@ -1,577 +0,0 @@
-#include
-#include
-#include
-#include
-
-using namespace std;
-
-namespace GBHawk
-{
- class MemoryManager;
-
- class PPU
- {
- public:
- #pragma region PPU
-
- PPU()
- {
-
- }
-
- uint8_t ReadMemory(uint32_t);
-
- MemoryManager* mem_ctrl;
-
- // pointers not stated
- bool* FlagI = nullptr;
- bool* in_vblank = nullptr;
- bool* cpu_halted = nullptr;
- bool* HDMA_transfer = nullptr;
- bool* GBC_compat = nullptr;
-
- uint8_t* cpu_LY = nullptr;
- uint8_t* REG_FFFF = nullptr;
- uint8_t* REG_FF0F = nullptr;
- uint8_t* _scanlineCallbackLine = nullptr;
- uint8_t* OAM = nullptr;
- uint8_t* VRAM = nullptr;
- uint32_t* VRAM_Bank = nullptr;
- uint32_t* _vidbuffer = nullptr;
- uint32_t* color_palette = nullptr;
-
- uint32_t BG_palette[32] = {};
- uint32_t OBJ_palette[32] = {};
-
- bool HDMA_active;
- bool clear_screen;
-
- // register variables
- uint8_t LCDC;
- uint8_t STAT;
- uint8_t scroll_y;
- uint8_t scroll_x;
- uint8_t LY;
- uint8_t LY_actual;
- uint8_t LY_inc;
- uint8_t LYC;
- uint8_t DMA_addr;
- uint8_t BGP;
- uint8_t obj_pal_0;
- uint8_t obj_pal_1;
- uint8_t window_y;
- uint8_t window_x;
- bool DMA_start;
- uint32_t DMA_clock;
- uint32_t DMA_inc;
- uint8_t DMA_byte;
-
- // state variables
- uint32_t cycle;
- bool LYC_INT;
- bool HBL_INT;
- bool VBL_INT;
- bool OAM_INT;
- bool LCD_was_off;
- bool stat_line;
- bool stat_line_old;
- // OAM scan
- bool DMA_OAM_access;
- bool OAM_access_read;
- bool OAM_access_write;
- uint32_t OAM_scan_index;
- uint32_t SL_sprites_index;
- uint32_t SL_sprites[40] = {};
- uint32_t write_sprite;
- bool no_scan;
- // render
- bool VRAM_access_read;
- bool VRAM_access_write;
- uint32_t read_case;
- uint32_t internal_cycle;
- uint32_t y_tile;
- uint32_t y_scroll_offset;
- uint32_t x_tile;
- uint32_t x_scroll_offset;
- uint32_t tile_byte;
- uint32_t sprite_fetch_cycles;
- bool fetch_sprite;
- bool going_to_fetch;
- bool first_fetch;
- uint32_t sprite_fetch_counter;
- uint8_t sprite_attr_list[160] = {};
- uint8_t sprite_pixel_list[160] = {};
- uint8_t sprite_present_list[160] = {};
- uint32_t temp_fetch;
- uint32_t tile_inc;
- bool pre_render;
- bool pre_render_2;
- uint8_t tile_data[3] = {};
- uint8_t tile_data_latch[3] = {};
- uint32_t latch_counter;
- bool latch_new_data;
- uint32_t render_counter;
- uint32_t render_offset;
- uint32_t pixel_counter;
- uint32_t pixel;
- uint8_t sprite_data[2] = {};
- uint8_t sprite_sel[2] = {};
- uint32_t sl_use_index;
- bool no_sprites;
- uint32_t SL_sprites_ordered[40] = {}; // (x_end, data_low, data_high, attr)
- uint32_t evaled_sprites;
- uint32_t sprite_ordered_index;
- bool blank_frame;
- bool window_latch;
- uint32_t consecutive_sprite;
- uint32_t last_eval;
-
- uint32_t total_counter;
- // windowing state
- uint32_t window_counter;
- bool window_pre_render;
- bool window_started;
- bool window_is_reset;
- uint32_t window_tile_inc;
- uint32_t window_y_tile;
- uint32_t window_x_tile;
- uint32_t window_y_tile_inc;
- uint32_t window_x_latch;
- uint32_t window_y_latch;
-
- uint32_t hbl_countdown;
-
- // The following are GBC specific variables
- // individual uint8_t used in palette colors
- uint8_t BG_bytes[64] = {};
- uint8_t OBJ_bytes[64] = {};
- bool BG_bytes_inc;
- bool OBJ_bytes_inc;
- uint8_t BG_bytes_index;
- uint8_t OBJ_bytes_index;
- uint8_t BG_transfer_byte;
- uint8_t OBJ_transfer_byte;
-
- // HDMA is unique to GBC, do it as part of the PPU tick
- uint8_t HDMA_src_hi;
- uint8_t HDMA_src_lo;
- uint8_t HDMA_dest_hi;
- uint8_t HDMA_dest_lo;
- uint32_t HDMA_tick;
- uint8_t HDMA_byte;
-
- // controls for tile attributes
- uint32_t VRAM_sel;
- bool BG_V_flip;
- bool HDMA_mode;
- bool HDMA_run_once;
- uint32_t cur_DMA_src;
- uint32_t cur_DMA_dest;
- uint32_t HDMA_length;
- uint32_t HDMA_countdown;
- uint32_t HBL_HDMA_count;
- uint32_t last_HBL;
- bool HBL_HDMA_go;
- bool HBL_test;
- uint8_t LYC_t;
- uint32_t LYC_cd;
-
- // accessors for derived values (GBC only)
- uint8_t BG_pal_ret() { return (uint8_t)(((BG_bytes_inc ? 1 : 0) << 7) | (BG_bytes_index & 0x3F)); }
-
- uint8_t OBJ_pal_ret() { return (uint8_t)(((OBJ_bytes_inc ? 1 : 0) << 7) | (OBJ_bytes_index & 0x3F)); }
-
- uint8_t HDMA_ctrl() { return (uint8_t)(((HDMA_active ? 0 : 1) << 7) | ((HDMA_length >> 4) - 1)); }
-
- virtual uint8_t ReadReg(uint32_t addr)
- {
- return 0;
- }
-
- virtual void WriteReg(uint32_t addr, uint8_t value)
- {
-
- }
-
- virtual void tick()
- {
-
- }
-
- // might be needed, not sure yet
- virtual void latch_delay()
- {
-
- }
-
- virtual void render(uint32_t render_cycle)
- {
-
- }
-
- virtual void process_sprite()
- {
-
- }
-
- // normal DMA moves twice as fast in double speed mode on GBC
- // So give it it's own function so we can seperate it from PPU tick
- virtual void DMA_tick()
- {
-
- }
-
- virtual void OAM_scan(uint32_t OAM_cycle)
- {
-
- }
-
- virtual void Reset()
- {
-
- }
-
- // order sprites according to x coordinate
- // note that for sprites of equal x coordinate, priority goes to first on the list
- virtual void reorder_and_assemble_sprites()
- {
-
- }
-
- virtual void color_compute_BG()
- {
-
- }
-
- void color_compute_OBJ()
- {
-
- }
-
- #pragma endregion
-
- #pragma region State Save / Load
-
- uint8_t* SaveState(uint8_t* saver)
- {
- for (int i = 0; i < 32; i++) { saver = int_saver(BG_palette[i], saver); }
- for (int i = 0; i < 32; i++) { saver = int_saver(OBJ_palette[i], saver); }
- for (int i = 0; i < 40; i++) { saver = int_saver(SL_sprites[i], saver); }
-
- for (int i = 0; i < 160; i++) { saver = byte_saver(sprite_attr_list[i], saver); }
- for (int i = 0; i < 160; i++) { saver = byte_saver(sprite_pixel_list[i], saver); }
- for (int i = 0; i < 160; i++) { saver = byte_saver(sprite_present_list[i], saver); }
- for (int i = 0; i < 3; i++) { saver = byte_saver(tile_data[i], saver); }
- for (int i = 0; i < 3; i++) { saver = byte_saver(tile_data_latch[i], saver); }
- for (int i = 0; i < 2; i++) { saver = byte_saver(sprite_data[i], saver); }
- for (int i = 0; i < 2; i++) { saver = byte_saver(sprite_sel[i], saver); }
- for (int i = 0; i < 40; i++) { saver = int_saver(SL_sprites_ordered[i], saver); }
-
- saver = bool_saver(HDMA_active, saver);
- saver = bool_saver(clear_screen, saver);
-
- saver = byte_saver(LCDC, saver);
- saver = byte_saver(STAT, saver);
- saver = byte_saver(scroll_y, saver);
- saver = byte_saver(scroll_x, saver);
- saver = byte_saver(LY, saver);
- saver = byte_saver(LY_actual, saver);
- saver = byte_saver(LY_inc, saver);
- saver = byte_saver(LYC, saver);
- saver = byte_saver(DMA_addr, saver);
- saver = byte_saver(BGP, saver);
- saver = byte_saver(obj_pal_0, saver);
- saver = byte_saver(obj_pal_1, saver);
- saver = byte_saver(window_y, saver);
- saver = byte_saver(window_x, saver);
- saver = bool_saver(DMA_start, saver);
- saver = int_saver(DMA_clock, saver);
- saver = int_saver(DMA_inc, saver);
- saver = byte_saver(DMA_byte, saver);
-
- saver = int_saver(cycle, saver);
- saver = bool_saver(LYC_INT, saver);
- saver = bool_saver(HBL_INT, saver);
- saver = bool_saver(VBL_INT, saver);
- saver = bool_saver(OAM_INT, saver);
- saver = bool_saver(stat_line, saver);
- saver = bool_saver(stat_line_old, saver);
- saver = bool_saver(LCD_was_off, saver);
- saver = int_saver(OAM_scan_index, saver);
- saver = int_saver(SL_sprites_index, saver);
- saver = int_saver(write_sprite, saver);
- saver = bool_saver(no_scan, saver);
-
- saver = bool_saver(DMA_OAM_access, saver);
- saver = bool_saver(OAM_access_read, saver);
- saver = bool_saver(OAM_access_write, saver);
- saver = bool_saver(VRAM_access_read, saver);
- saver = bool_saver(VRAM_access_write, saver);
-
- saver = int_saver(read_case, saver);
- saver = int_saver(internal_cycle, saver);
- saver = int_saver(y_tile, saver);
- saver = int_saver(y_scroll_offset, saver);
- saver = int_saver(x_tile, saver);
- saver = int_saver(x_scroll_offset, saver);
- saver = int_saver(tile_byte, saver);
- saver = int_saver(sprite_fetch_cycles, saver);
- saver = bool_saver(fetch_sprite, saver);
- saver = bool_saver(going_to_fetch, saver);
- saver = bool_saver(first_fetch, saver);
- saver = int_saver(sprite_fetch_counter, saver);
-
- saver = int_saver(temp_fetch, saver);
- saver = int_saver(tile_inc, saver);
- saver = bool_saver(pre_render, saver);
- saver = bool_saver(pre_render_2, saver);
- saver = int_saver(latch_counter, saver);
- saver = bool_saver(latch_new_data, saver);
- saver = int_saver(render_counter, saver);
- saver = int_saver(render_offset, saver);
- saver = int_saver(pixel_counter, saver);
- saver = int_saver(pixel, saver);
- saver = int_saver(sl_use_index, saver);
- saver = bool_saver(no_sprites, saver);
- saver = int_saver(evaled_sprites, saver);
- saver = int_saver(sprite_ordered_index, saver);
- saver = bool_saver(blank_frame, saver);
- saver = bool_saver(window_latch, saver);
- saver = int_saver(consecutive_sprite, saver);
- saver = int_saver(last_eval, saver);
-
- saver = int_saver(window_counter, saver);
- saver = bool_saver(window_pre_render, saver);
- saver = bool_saver(window_started, saver);
- saver = bool_saver(window_is_reset, saver);
- saver = int_saver(window_tile_inc, saver);
- saver = int_saver(window_y_tile, saver);
- saver = int_saver(window_x_tile, saver);
- saver = int_saver(window_y_tile_inc, saver);
- saver = int_saver(window_x_latch, saver);
- saver = int_saver(window_y_latch, saver);
-
- saver = int_saver(hbl_countdown, saver);
-
- // The following are GBC specific variables
- for (int i = 0; i < 64; i++) { saver = byte_saver(BG_bytes[i], saver); }
- for (int i = 0; i < 64; i++) { saver = byte_saver(OBJ_bytes[i], saver); }
-
- saver = byte_saver(BG_transfer_byte, saver);
- saver = byte_saver(OBJ_transfer_byte, saver);
- saver = byte_saver(HDMA_src_hi, saver);
- saver = byte_saver(HDMA_src_lo, saver);
- saver = byte_saver(HDMA_dest_hi, saver);
- saver = byte_saver(HDMA_dest_lo, saver);
- saver = int_saver(HDMA_tick, saver);
- saver = byte_saver(HDMA_byte, saver);
-
- saver = int_saver(VRAM_sel, saver);
- saver = bool_saver(BG_V_flip, saver);
- saver = bool_saver(HDMA_mode, saver);
- saver = bool_saver(HDMA_run_once, saver);
- saver = int_saver(cur_DMA_src, saver);
- saver = int_saver(cur_DMA_dest, saver);
- saver = int_saver(HDMA_length, saver);
- saver = int_saver(HDMA_countdown, saver);
- saver = int_saver(HBL_HDMA_count, saver);
- saver = int_saver(last_HBL, saver);
- saver = bool_saver(HBL_HDMA_go, saver);
- saver = bool_saver(HBL_test, saver);
-
- saver = bool_saver(BG_bytes_inc, saver);
- saver = bool_saver(OBJ_bytes_inc, saver);
- saver = byte_saver(BG_bytes_index, saver);
- saver = byte_saver(OBJ_bytes_index, saver);
-
- saver = byte_saver(LYC_t, saver);
- saver = int_saver(LYC_cd, saver);
-
- return saver;
- }
-
- uint8_t* LoadState(uint8_t* loader)
- {
- for (int i = 0; i < 32; i++) { loader = int_loader(&BG_palette[i], loader); }
- for (int i = 0; i < 32; i++) { loader = int_loader(&OBJ_palette[i], loader); }
- for (int i = 0; i < 40; i++) { loader = int_loader(&SL_sprites[i], loader); }
-
- for (int i = 0; i < 160; i++) { loader = byte_loader(&sprite_attr_list[i], loader); }
- for (int i = 0; i < 160; i++) { loader = byte_loader(&sprite_pixel_list[i], loader); }
- for (int i = 0; i < 160; i++) { loader = byte_loader(&sprite_present_list[i], loader); }
- for (int i = 0; i < 3; i++) { loader = byte_loader(&tile_data[i], loader); }
- for (int i = 0; i < 3; i++) { loader = byte_loader(&tile_data_latch[i], loader); }
- for (int i = 0; i < 2; i++) { loader = byte_loader(&sprite_data[i], loader); }
- for (int i = 0; i < 2; i++) { loader = byte_loader(&sprite_sel[i], loader); }
- for (int i = 0; i < 40; i++) { loader = int_loader(&SL_sprites_ordered[i], loader); }
-
- loader = bool_loader(&HDMA_active, loader);
- loader = bool_loader(&clear_screen, loader);
-
- loader = byte_loader(&LCDC, loader);
- loader = byte_loader(&STAT, loader);
- loader = byte_loader(&scroll_y, loader);
- loader = byte_loader(&scroll_x, loader);
- loader = byte_loader(&LY, loader);
- loader = byte_loader(&LY_actual, loader);
- loader = byte_loader(&LY_inc, loader);
- loader = byte_loader(&LYC, loader);
- loader = byte_loader(&DMA_addr, loader);
- loader = byte_loader(&BGP, loader);
- loader = byte_loader(&obj_pal_0, loader);
- loader = byte_loader(&obj_pal_1, loader);
- loader = byte_loader(&window_y, loader);
- loader = byte_loader(&window_x, loader);
- loader = bool_loader(&DMA_start, loader);
- loader = int_loader(&DMA_clock, loader);
- loader = int_loader(&DMA_inc, loader);
- loader = byte_loader(&DMA_byte, loader);
-
- loader = int_loader(&cycle, loader);
- loader = bool_loader(&LYC_INT, loader);
- loader = bool_loader(&HBL_INT, loader);
- loader = bool_loader(&VBL_INT, loader);
- loader = bool_loader(&OAM_INT, loader);
- loader = bool_loader(&stat_line, loader);
- loader = bool_loader(&stat_line_old, loader);
- loader = bool_loader(&LCD_was_off, loader);
- loader = int_loader(&OAM_scan_index, loader);
- loader = int_loader(&SL_sprites_index, loader);
- loader = int_loader(&write_sprite, loader);
- loader = bool_loader(&no_scan, loader);
-
- loader = bool_loader(&DMA_OAM_access, loader);
- loader = bool_loader(&OAM_access_read, loader);
- loader = bool_loader(&OAM_access_write, loader);
- loader = bool_loader(&VRAM_access_read, loader);
- loader = bool_loader(&VRAM_access_write, loader);
-
- loader = int_loader(&read_case, loader);
- loader = int_loader(&internal_cycle, loader);
- loader = int_loader(&y_tile, loader);
- loader = int_loader(&y_scroll_offset, loader);
- loader = int_loader(&x_tile, loader);
- loader = int_loader(&x_scroll_offset, loader);
- loader = int_loader(&tile_byte, loader);
- loader = int_loader(&sprite_fetch_cycles, loader);
- loader = bool_loader(&fetch_sprite, loader);
- loader = bool_loader(&going_to_fetch, loader);
- loader = bool_loader(&first_fetch, loader);
- loader = int_loader(&sprite_fetch_counter, loader);
-
- loader = int_loader(&temp_fetch, loader);
- loader = int_loader(&tile_inc, loader);
- loader = bool_loader(&pre_render, loader);
- loader = bool_loader(&pre_render_2, loader);
- loader = int_loader(&latch_counter, loader);
- loader = bool_loader(&latch_new_data, loader);
- loader = int_loader(&render_counter, loader);
- loader = int_loader(&render_offset, loader);
- loader = int_loader(&pixel_counter, loader);
- loader = int_loader(&pixel, loader);
- loader = int_loader(&sl_use_index, loader);
- loader = bool_loader(&no_sprites, loader);
- loader = int_loader(&evaled_sprites, loader);
- loader = int_loader(&sprite_ordered_index, loader);
- loader = bool_loader(&blank_frame, loader);
- loader = bool_loader(&window_latch, loader);
- loader = int_loader(&consecutive_sprite, loader);
- loader = int_loader(&last_eval, loader);
-
- loader = int_loader(&window_counter, loader);
- loader = bool_loader(&window_pre_render, loader);
- loader = bool_loader(&window_started, loader);
- loader = bool_loader(&window_is_reset, loader);
- loader = int_loader(&window_tile_inc, loader);
- loader = int_loader(&window_y_tile, loader);
- loader = int_loader(&window_x_tile, loader);
- loader = int_loader(&window_y_tile_inc, loader);
- loader = int_loader(&window_x_latch, loader);
- loader = int_loader(&window_y_latch, loader);
-
- loader = int_loader(&hbl_countdown, loader);
-
- // The following are GBC specific variables
- for (int i = 0; i < 64; i++) { loader = byte_loader(&BG_bytes[i], loader); }
- for (int i = 0; i < 64; i++) { loader = byte_loader(&OBJ_bytes[i], loader); }
-
- loader = byte_loader(&BG_transfer_byte, loader);
- loader = byte_loader(&OBJ_transfer_byte, loader);
- loader = byte_loader(&HDMA_src_hi, loader);
- loader = byte_loader(&HDMA_src_lo, loader);
- loader = byte_loader(&HDMA_dest_hi, loader);
- loader = byte_loader(&HDMA_dest_lo, loader);
- loader = int_loader(&HDMA_tick, loader);
- loader = byte_loader(&HDMA_byte, loader);
-
- loader = int_loader(&VRAM_sel, loader);
- loader = bool_loader(&BG_V_flip, loader);
- loader = bool_loader(&HDMA_mode, loader);
- loader = bool_loader(&HDMA_run_once, loader);
- loader = int_loader(&cur_DMA_src, loader);
- loader = int_loader(&cur_DMA_dest, loader);
- loader = int_loader(&HDMA_length, loader);
- loader = int_loader(&HDMA_countdown, loader);
- loader = int_loader(&HBL_HDMA_count, loader);
- loader = int_loader(&last_HBL, loader);
- loader = bool_loader(&HBL_HDMA_go, loader);
- loader = bool_loader(&HBL_test, loader);
-
- loader = bool_loader(&BG_bytes_inc, loader);
- loader = bool_loader(&OBJ_bytes_inc, loader);
- loader = byte_loader(&BG_bytes_index, loader);
- loader = byte_loader(&OBJ_bytes_index, loader);
-
- loader = byte_loader(&LYC_t, loader);
- loader = int_loader(&LYC_cd, loader);
-
- return loader;
- }
-
- uint8_t* bool_saver(bool to_save, uint8_t* saver)
- {
- *saver = (uint8_t)(to_save ? 1 : 0); saver++;
-
- return saver;
- }
-
- uint8_t* byte_saver(uint8_t to_save, uint8_t* saver)
- {
- *saver = to_save; saver++;
-
- return saver;
- }
-
- uint8_t* int_saver(uint32_t to_save, uint8_t* saver)
- {
- *saver = (uint8_t)(to_save & 0xFF); saver++; *saver = (uint8_t)((to_save >> 8) & 0xFF); saver++;
- *saver = (uint8_t)((to_save >> 16) & 0xFF); saver++; *saver = (uint8_t)((to_save >> 24) & 0xFF); saver++;
-
- return saver;
- }
-
- uint8_t* bool_loader(bool* to_load, uint8_t* loader)
- {
- to_load[0] = *to_load == 1; loader++;
-
- return loader;
- }
-
- uint8_t* byte_loader(uint8_t* to_load, uint8_t* loader)
- {
- to_load[0] = *loader; loader++;
-
- return loader;
- }
-
- uint8_t* int_loader(uint32_t* to_load, uint8_t* loader)
- {
- to_load[0] = *loader; loader++; to_load[0] |= (*loader << 8); loader++;
- to_load[0] |= (*loader << 16); loader++; to_load[0] |= (*loader << 24); loader++;
-
- return loader;
- }
-
- #pragma endregion
- };
-}