2016-06-18 20:29:11 +03:00
|
|
|
|
#ifndef GB_h
|
|
|
|
|
#define GB_h
|
2019-08-16 17:38:43 +03:00
|
|
|
|
#define typeof __typeof__
|
2016-03-30 23:07:55 +03:00
|
|
|
|
#include <stdbool.h>
|
|
|
|
|
#include <stdint.h>
|
2021-12-20 18:59:51 +02:00
|
|
|
|
#include <stdalign.h>
|
2016-03-30 23:07:55 +03:00
|
|
|
|
#include <time.h>
|
2017-04-17 20:16:17 +03:00
|
|
|
|
|
2021-11-07 13:39:18 +02:00
|
|
|
|
#include "defs.h"
|
2017-06-14 00:23:34 +03:00
|
|
|
|
#include "save_state.h"
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
2017-04-17 20:16:17 +03:00
|
|
|
|
#include "apu.h"
|
|
|
|
|
#include "camera.h"
|
|
|
|
|
#include "debugger.h"
|
|
|
|
|
#include "display.h"
|
|
|
|
|
#include "joypad.h"
|
|
|
|
|
#include "mbc.h"
|
2017-10-12 19:42:30 +03:00
|
|
|
|
#include "memory.h"
|
2017-04-17 20:16:17 +03:00
|
|
|
|
#include "printer.h"
|
|
|
|
|
#include "timing.h"
|
2018-02-10 14:42:14 +02:00
|
|
|
|
#include "rewind.h"
|
2019-01-12 00:42:16 +02:00
|
|
|
|
#include "sm83_cpu.h"
|
2017-04-17 20:16:17 +03:00
|
|
|
|
#include "symbol_hash.h"
|
2018-11-11 01:16:32 +02:00
|
|
|
|
#include "sgb.h"
|
2020-04-09 20:11:55 +03:00
|
|
|
|
#include "cheats.h"
|
2020-04-29 16:06:11 +03:00
|
|
|
|
#include "rumble.h"
|
2020-09-19 19:31:24 +03:00
|
|
|
|
#include "workboy.h"
|
2021-10-30 16:03:13 +03:00
|
|
|
|
#include "random.h"
|
2016-06-11 14:52:09 +03:00
|
|
|
|
|
2021-11-07 01:10:58 +02:00
|
|
|
|
#define GB_STRUCT_VERSION 14
|
2018-06-16 13:59:33 +03:00
|
|
|
|
|
2018-07-14 21:52:54 +03:00
|
|
|
|
#define GB_MODEL_FAMILY_MASK 0xF00
|
|
|
|
|
#define GB_MODEL_DMG_FAMILY 0x000
|
|
|
|
|
#define GB_MODEL_MGB_FAMILY 0x100
|
|
|
|
|
#define GB_MODEL_CGB_FAMILY 0x200
|
2021-01-16 14:51:06 +02:00
|
|
|
|
#define GB_MODEL_PAL_BIT 0x40
|
|
|
|
|
#define GB_MODEL_NO_SFC_BIT 0x80
|
|
|
|
|
|
2022-01-14 08:29:41 -08:00
|
|
|
|
#define GB_REWIND_FRAMES_PER_KEY 255
|
|
|
|
|
|
2018-11-03 01:31:14 +02:00
|
|
|
|
#if __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__
|
|
|
|
|
#define GB_BIG_ENDIAN
|
|
|
|
|
#elif __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
|
|
|
|
|
#define GB_LITTLE_ENDIAN
|
|
|
|
|
#else
|
|
|
|
|
#error Unable to detect endianess
|
|
|
|
|
#endif
|
|
|
|
|
|
2022-01-14 08:26:15 -08:00
|
|
|
|
#ifdef GB_BIG_ENDIAN
|
2022-01-14 22:26:23 +02:00
|
|
|
|
#define GB_REGISTER_ORDER a, f, \
|
|
|
|
|
b, c, \
|
|
|
|
|
d, e, \
|
|
|
|
|
h, l
|
2022-01-14 08:26:15 -08:00
|
|
|
|
#else
|
2022-01-14 22:26:23 +02:00
|
|
|
|
#define GB_REGISTER_ORDER f, a, \
|
|
|
|
|
c, b, \
|
|
|
|
|
e, d, \
|
|
|
|
|
l, h
|
2022-01-14 08:26:15 -08:00
|
|
|
|
#endif
|
2020-06-22 20:14:32 +00:00
|
|
|
|
|
2020-01-29 14:19:11 +02:00
|
|
|
|
typedef struct {
|
2021-11-05 19:07:27 +02:00
|
|
|
|
struct GB_color_s {
|
2020-04-24 20:37:57 +03:00
|
|
|
|
uint8_t r, g, b;
|
2020-01-29 14:19:11 +02:00
|
|
|
|
} colors[5];
|
|
|
|
|
} GB_palette_t;
|
|
|
|
|
|
|
|
|
|
extern const GB_palette_t GB_PALETTE_GREY;
|
|
|
|
|
extern const GB_palette_t GB_PALETTE_DMG;
|
|
|
|
|
extern const GB_palette_t GB_PALETTE_MGB;
|
|
|
|
|
extern const GB_palette_t GB_PALETTE_GBL;
|
|
|
|
|
|
2018-11-03 01:31:14 +02:00
|
|
|
|
typedef union {
|
|
|
|
|
struct {
|
|
|
|
|
uint8_t seconds;
|
|
|
|
|
uint8_t minutes;
|
|
|
|
|
uint8_t hours;
|
|
|
|
|
uint8_t days;
|
|
|
|
|
uint8_t high;
|
|
|
|
|
};
|
2021-04-16 00:35:54 +03:00
|
|
|
|
struct {
|
|
|
|
|
uint8_t seconds;
|
|
|
|
|
uint8_t minutes;
|
|
|
|
|
uint8_t hours:5;
|
|
|
|
|
uint8_t weekday:3;
|
|
|
|
|
uint8_t weeks;
|
|
|
|
|
} tpp1;
|
2018-11-03 01:31:14 +02:00
|
|
|
|
uint8_t data[5];
|
|
|
|
|
} GB_rtc_time_t;
|
|
|
|
|
|
2021-04-12 23:36:35 +03:00
|
|
|
|
typedef struct __attribute__((packed)) {
|
|
|
|
|
uint64_t last_rtc_second;
|
|
|
|
|
uint16_t minutes;
|
|
|
|
|
uint16_t days;
|
|
|
|
|
uint16_t alarm_minutes, alarm_days;
|
|
|
|
|
uint8_t alarm_enabled;
|
|
|
|
|
} GB_huc3_rtc_time_t;
|
|
|
|
|
|
2018-07-14 21:52:54 +03:00
|
|
|
|
typedef enum {
|
2018-06-16 13:59:33 +03:00
|
|
|
|
// GB_MODEL_DMG_0 = 0x000,
|
|
|
|
|
// GB_MODEL_DMG_A = 0x001,
|
|
|
|
|
GB_MODEL_DMG_B = 0x002,
|
|
|
|
|
// GB_MODEL_DMG_C = 0x003,
|
2018-11-11 01:16:32 +02:00
|
|
|
|
GB_MODEL_SGB = 0x004,
|
|
|
|
|
GB_MODEL_SGB_NTSC = GB_MODEL_SGB,
|
2019-07-15 20:47:16 +03:00
|
|
|
|
GB_MODEL_SGB_PAL = GB_MODEL_SGB | GB_MODEL_PAL_BIT,
|
2019-10-08 15:10:24 +03:00
|
|
|
|
GB_MODEL_SGB_NTSC_NO_SFC = GB_MODEL_SGB | GB_MODEL_NO_SFC_BIT,
|
|
|
|
|
GB_MODEL_SGB_NO_SFC = GB_MODEL_SGB_NTSC_NO_SFC,
|
|
|
|
|
GB_MODEL_SGB_PAL_NO_SFC = GB_MODEL_SGB | GB_MODEL_NO_SFC_BIT | GB_MODEL_PAL_BIT,
|
2021-10-23 23:28:54 +03:00
|
|
|
|
GB_MODEL_MGB = 0x100,
|
2018-11-11 01:16:32 +02:00
|
|
|
|
GB_MODEL_SGB2 = 0x101,
|
2019-07-15 20:47:16 +03:00
|
|
|
|
GB_MODEL_SGB2_NO_SFC = GB_MODEL_SGB2 | GB_MODEL_NO_SFC_BIT,
|
2021-11-04 00:32:15 +02:00
|
|
|
|
GB_MODEL_CGB_0 = 0x200,
|
2022-01-05 21:55:46 +02:00
|
|
|
|
GB_MODEL_CGB_A = 0x201,
|
2021-10-30 20:58:57 +03:00
|
|
|
|
GB_MODEL_CGB_B = 0x202,
|
2018-07-03 21:43:46 +03:00
|
|
|
|
GB_MODEL_CGB_C = 0x203,
|
2021-10-27 01:40:28 +03:00
|
|
|
|
GB_MODEL_CGB_D = 0x204,
|
2018-06-16 13:59:33 +03:00
|
|
|
|
GB_MODEL_CGB_E = 0x205,
|
2022-01-30 18:11:35 +02:00
|
|
|
|
// GB_MODEL_AGB_0 = 0x206,
|
|
|
|
|
GB_MODEL_AGB_A = 0x207,
|
|
|
|
|
GB_MODEL_AGB = GB_MODEL_AGB_A,
|
|
|
|
|
//GB_MODEL_AGB_B = 0x208
|
2022-03-09 00:32:50 +02:00
|
|
|
|
//GB_MODEL_AGB_E = 0x209
|
2018-06-16 13:59:33 +03:00
|
|
|
|
} GB_model_t;
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
GB_REGISTER_AF,
|
|
|
|
|
GB_REGISTER_BC,
|
|
|
|
|
GB_REGISTER_DE,
|
|
|
|
|
GB_REGISTER_HL,
|
|
|
|
|
GB_REGISTER_SP,
|
2021-12-30 23:53:24 +02:00
|
|
|
|
GB_REGISTER_PC,
|
2016-03-30 23:07:55 +03:00
|
|
|
|
GB_REGISTERS_16_BIT /* Count */
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* Todo: Actually use these! */
|
|
|
|
|
enum {
|
|
|
|
|
GB_CARRY_FLAG = 16,
|
|
|
|
|
GB_HALF_CARRY_FLAG = 32,
|
2020-01-03 21:11:45 +02:00
|
|
|
|
GB_SUBTRACT_FLAG = 64,
|
2016-03-30 23:07:55 +03:00
|
|
|
|
GB_ZERO_FLAG = 128,
|
|
|
|
|
};
|
|
|
|
|
|
2020-02-08 13:28:46 +02:00
|
|
|
|
typedef enum {
|
|
|
|
|
GB_BORDER_SGB,
|
|
|
|
|
GB_BORDER_NEVER,
|
|
|
|
|
GB_BORDER_ALWAYS,
|
|
|
|
|
} GB_border_mode_t;
|
|
|
|
|
|
2016-03-30 23:07:55 +03:00
|
|
|
|
enum {
|
|
|
|
|
/* Joypad and Serial */
|
|
|
|
|
GB_IO_JOYP = 0x00, // Joypad (R/W)
|
|
|
|
|
GB_IO_SB = 0x01, // Serial transfer data (R/W)
|
|
|
|
|
GB_IO_SC = 0x02, // Serial Transfer Control (R/W)
|
|
|
|
|
|
|
|
|
|
/* Missing */
|
|
|
|
|
|
|
|
|
|
/* Timers */
|
|
|
|
|
GB_IO_DIV = 0x04, // Divider Register (R/W)
|
|
|
|
|
GB_IO_TIMA = 0x05, // Timer counter (R/W)
|
|
|
|
|
GB_IO_TMA = 0x06, // Timer Modulo (R/W)
|
|
|
|
|
GB_IO_TAC = 0x07, // Timer Control (R/W)
|
|
|
|
|
|
|
|
|
|
/* Missing */
|
|
|
|
|
|
2022-02-13 16:58:44 +02:00
|
|
|
|
GB_IO_IF = 0x0F, // Interrupt Flag (R/W)
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
|
|
/* Sound */
|
|
|
|
|
GB_IO_NR10 = 0x10, // Channel 1 Sweep register (R/W)
|
|
|
|
|
GB_IO_NR11 = 0x11, // Channel 1 Sound length/Wave pattern duty (R/W)
|
|
|
|
|
GB_IO_NR12 = 0x12, // Channel 1 Volume Envelope (R/W)
|
|
|
|
|
GB_IO_NR13 = 0x13, // Channel 1 Frequency lo (Write Only)
|
|
|
|
|
GB_IO_NR14 = 0x14, // Channel 1 Frequency hi (R/W)
|
2017-07-27 23:11:33 +03:00
|
|
|
|
/* NR20 does not exist */
|
2016-03-30 23:07:55 +03:00
|
|
|
|
GB_IO_NR21 = 0x16, // Channel 2 Sound Length/Wave Pattern Duty (R/W)
|
|
|
|
|
GB_IO_NR22 = 0x17, // Channel 2 Volume Envelope (R/W)
|
|
|
|
|
GB_IO_NR23 = 0x18, // Channel 2 Frequency lo data (W)
|
|
|
|
|
GB_IO_NR24 = 0x19, // Channel 2 Frequency hi data (R/W)
|
2022-02-13 16:58:44 +02:00
|
|
|
|
GB_IO_NR30 = 0x1A, // Channel 3 Sound on/off (R/W)
|
|
|
|
|
GB_IO_NR31 = 0x1B, // Channel 3 Sound Length
|
|
|
|
|
GB_IO_NR32 = 0x1C, // Channel 3 Select output level (R/W)
|
|
|
|
|
GB_IO_NR33 = 0x1D, // Channel 3 Frequency's lower data (W)
|
|
|
|
|
GB_IO_NR34 = 0x1E, // Channel 3 Frequency's higher data (R/W)
|
2017-07-27 23:11:33 +03:00
|
|
|
|
/* NR40 does not exist */
|
2016-03-30 23:07:55 +03:00
|
|
|
|
GB_IO_NR41 = 0x20, // Channel 4 Sound Length (R/W)
|
|
|
|
|
GB_IO_NR42 = 0x21, // Channel 4 Volume Envelope (R/W)
|
|
|
|
|
GB_IO_NR43 = 0x22, // Channel 4 Polynomial Counter (R/W)
|
|
|
|
|
GB_IO_NR44 = 0x23, // Channel 4 Counter/consecutive, Inital (R/W)
|
|
|
|
|
GB_IO_NR50 = 0x24, // Channel control / ON-OFF / Volume (R/W)
|
|
|
|
|
GB_IO_NR51 = 0x25, // Selection of Sound output terminal (R/W)
|
|
|
|
|
GB_IO_NR52 = 0x26, // Sound on/off
|
|
|
|
|
|
|
|
|
|
/* Missing */
|
|
|
|
|
|
|
|
|
|
GB_IO_WAV_START = 0x30, // Wave pattern start
|
2022-02-13 16:58:44 +02:00
|
|
|
|
GB_IO_WAV_END = 0x3F, // Wave pattern end
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
|
|
/* Graphics */
|
|
|
|
|
GB_IO_LCDC = 0x40, // LCD Control (R/W)
|
|
|
|
|
GB_IO_STAT = 0x41, // LCDC Status (R/W)
|
|
|
|
|
GB_IO_SCY = 0x42, // Scroll Y (R/W)
|
|
|
|
|
GB_IO_SCX = 0x43, // Scroll X (R/W)
|
|
|
|
|
GB_IO_LY = 0x44, // LCDC Y-Coordinate (R)
|
|
|
|
|
GB_IO_LYC = 0x45, // LY Compare (R/W)
|
|
|
|
|
GB_IO_DMA = 0x46, // DMA Transfer and Start Address (W)
|
|
|
|
|
GB_IO_BGP = 0x47, // BG Palette Data (R/W) - Non CGB Mode Only
|
|
|
|
|
GB_IO_OBP0 = 0x48, // Object Palette 0 Data (R/W) - Non CGB Mode Only
|
|
|
|
|
GB_IO_OBP1 = 0x49, // Object Palette 1 Data (R/W) - Non CGB Mode Only
|
2022-02-13 16:58:44 +02:00
|
|
|
|
GB_IO_WY = 0x4A, // Window Y Position (R/W)
|
|
|
|
|
GB_IO_WX = 0x4B, // Window X Position minus 7 (R/W)
|
2021-11-06 13:34:34 +02:00
|
|
|
|
// Controls DMG mode and PGB mode
|
2022-02-13 16:58:44 +02:00
|
|
|
|
GB_IO_KEY0 = 0x4C,
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
|
|
/* General CGB features */
|
2022-02-13 16:58:44 +02:00
|
|
|
|
GB_IO_KEY1 = 0x4D, // CGB Mode Only - Prepare Speed Switch
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
|
|
/* Missing */
|
|
|
|
|
|
2022-02-13 16:58:44 +02:00
|
|
|
|
GB_IO_VBK = 0x4F, // CGB Mode Only - VRAM Bank
|
2020-04-09 02:36:27 +03:00
|
|
|
|
GB_IO_BANK = 0x50, // Write to disable the BIOS mapping
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
|
|
/* CGB DMA */
|
|
|
|
|
GB_IO_HDMA1 = 0x51, // CGB Mode Only - New DMA Source, High
|
|
|
|
|
GB_IO_HDMA2 = 0x52, // CGB Mode Only - New DMA Source, Low
|
|
|
|
|
GB_IO_HDMA3 = 0x53, // CGB Mode Only - New DMA Destination, High
|
|
|
|
|
GB_IO_HDMA4 = 0x54, // CGB Mode Only - New DMA Destination, Low
|
|
|
|
|
GB_IO_HDMA5 = 0x55, // CGB Mode Only - New DMA Length/Mode/Start
|
|
|
|
|
|
|
|
|
|
/* IR */
|
|
|
|
|
GB_IO_RP = 0x56, // CGB Mode Only - Infrared Communications Port
|
|
|
|
|
|
|
|
|
|
/* Missing */
|
|
|
|
|
|
2022-02-05 18:50:33 +02:00
|
|
|
|
/* CGB Palettes */
|
2016-03-30 23:07:55 +03:00
|
|
|
|
GB_IO_BGPI = 0x68, // CGB Mode Only - Background Palette Index
|
|
|
|
|
GB_IO_BGPD = 0x69, // CGB Mode Only - Background Palette Data
|
2022-02-13 16:58:44 +02:00
|
|
|
|
GB_IO_OBPI = 0x6A, // CGB Mode Only - Object Palette Index
|
|
|
|
|
GB_IO_OBPD = 0x6B, // CGB Mode Only - Object Palette Data
|
|
|
|
|
GB_IO_OPRI = 0x6C, // Affects object priority (X based or index based)
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
|
|
/* Missing */
|
|
|
|
|
|
|
|
|
|
GB_IO_SVBK = 0x70, // CGB Mode Only - WRAM Bank
|
2022-02-14 22:59:05 +02:00
|
|
|
|
GB_IO_PSM = 0x71, // Palette Selection Mode, controls the PSW and key combo
|
2021-11-06 13:34:34 +02:00
|
|
|
|
GB_IO_PSWX = 0x72, // X position of the palette switching window
|
|
|
|
|
GB_IO_PSWY = 0x73, // Y position of the palette switching window
|
|
|
|
|
GB_IO_PSW = 0x74, // Key combo to trigger the palette switching window
|
2016-03-30 23:07:55 +03:00
|
|
|
|
GB_IO_UNKNOWN5 = 0x75, // (8Fh) - Bit 4-6 (Read/Write)
|
2021-11-06 13:34:34 +02:00
|
|
|
|
GB_IO_PCM12 = 0x76, // Channels 1 and 2 amplitudes
|
|
|
|
|
GB_IO_PCM34 = 0x77, // Channels 3 and 4 amplitudes
|
2016-03-30 23:07:55 +03:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
|
GB_LOG_BOLD = 1,
|
|
|
|
|
GB_LOG_DASHED_UNDERLINE = 2,
|
|
|
|
|
GB_LOG_UNDERLINE = 4,
|
|
|
|
|
GB_LOG_UNDERLINE_MASK = GB_LOG_DASHED_UNDERLINE | GB_LOG_UNDERLINE
|
2016-06-18 20:29:11 +03:00
|
|
|
|
} GB_log_attributes;
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
2020-01-29 20:29:30 +02:00
|
|
|
|
typedef enum {
|
2021-11-04 00:32:15 +02:00
|
|
|
|
GB_BOOT_ROM_DMG_0,
|
2020-01-29 20:29:30 +02:00
|
|
|
|
GB_BOOT_ROM_DMG,
|
|
|
|
|
GB_BOOT_ROM_MGB,
|
|
|
|
|
GB_BOOT_ROM_SGB,
|
|
|
|
|
GB_BOOT_ROM_SGB2,
|
2021-11-04 00:32:15 +02:00
|
|
|
|
GB_BOOT_ROM_CGB_0,
|
2020-01-29 20:29:30 +02:00
|
|
|
|
GB_BOOT_ROM_CGB,
|
|
|
|
|
GB_BOOT_ROM_AGB,
|
|
|
|
|
} GB_boot_rom_t;
|
|
|
|
|
|
2017-04-17 20:16:17 +03:00
|
|
|
|
#ifdef GB_INTERNAL
|
|
|
|
|
#define LCDC_PERIOD 70224
|
|
|
|
|
#define CPU_FREQUENCY 0x400000
|
2018-11-11 01:16:32 +02:00
|
|
|
|
#define SGB_NTSC_FREQUENCY (21477272 / 5)
|
|
|
|
|
#define SGB_PAL_FREQUENCY (21281370 / 5)
|
2017-04-17 20:16:17 +03:00
|
|
|
|
#define DIV_CYCLES (0x100)
|
2017-10-12 17:22:22 +03:00
|
|
|
|
|
|
|
|
|
#if !defined(MIN)
|
2020-04-24 20:37:57 +03:00
|
|
|
|
#define MIN(A, B) ({ __typeof__(A) __a = (A); __typeof__(B) __b = (B); __a < __b ? __a : __b; })
|
2017-10-12 17:22:22 +03:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#if !defined(MAX)
|
2020-04-24 20:37:57 +03:00
|
|
|
|
#define MAX(A, B) ({ __typeof__(A) __a = (A); __typeof__(B) __b = (B); __a < __b ? __b : __a; })
|
2017-10-12 17:22:22 +03:00
|
|
|
|
#endif
|
2017-04-17 20:16:17 +03:00
|
|
|
|
#endif
|
|
|
|
|
|
2022-06-25 01:59:51 +03:00
|
|
|
|
typedef void (*GB_vblank_callback_t)(GB_gameboy_t *gb, GB_vblank_type_t type);
|
2016-06-18 20:29:11 +03:00
|
|
|
|
typedef void (*GB_log_callback_t)(GB_gameboy_t *gb, const char *string, GB_log_attributes attributes);
|
2016-03-30 23:07:55 +03:00
|
|
|
|
typedef char *(*GB_input_callback_t)(GB_gameboy_t *gb);
|
2016-06-18 20:29:11 +03:00
|
|
|
|
typedef uint32_t (*GB_rgb_encode_callback_t)(GB_gameboy_t *gb, uint8_t r, uint8_t g, uint8_t b);
|
2020-11-21 00:52:54 +02:00
|
|
|
|
typedef void (*GB_infrared_callback_t)(GB_gameboy_t *gb, bool on);
|
2019-10-19 19:26:04 +03:00
|
|
|
|
typedef void (*GB_rumble_callback_t)(GB_gameboy_t *gb, double rumble_amplitude);
|
2019-03-15 14:36:10 +02:00
|
|
|
|
typedef void (*GB_serial_transfer_bit_start_callback_t)(GB_gameboy_t *gb, bool bit_to_send);
|
|
|
|
|
typedef bool (*GB_serial_transfer_bit_end_callback_t)(GB_gameboy_t *gb);
|
2019-06-07 18:27:25 +03:00
|
|
|
|
typedef void (*GB_update_input_hint_callback_t)(GB_gameboy_t *gb);
|
2019-07-15 22:01:41 +03:00
|
|
|
|
typedef void (*GB_joyp_write_callback_t)(GB_gameboy_t *gb, uint8_t value);
|
2019-07-16 20:44:27 +03:00
|
|
|
|
typedef void (*GB_icd_pixel_callback_t)(GB_gameboy_t *gb, uint8_t row);
|
|
|
|
|
typedef void (*GB_icd_hreset_callback_t)(GB_gameboy_t *gb);
|
2019-07-15 23:02:58 +03:00
|
|
|
|
typedef void (*GB_icd_vreset_callback_t)(GB_gameboy_t *gb);
|
2020-01-29 20:29:30 +02:00
|
|
|
|
typedef void (*GB_boot_rom_load_callback_t)(GB_gameboy_t *gb, GB_boot_rom_t type);
|
2016-11-12 01:58:53 +02:00
|
|
|
|
|
2021-12-10 19:42:47 +02:00
|
|
|
|
typedef void (*GB_execution_callback_t)(GB_gameboy_t *gb, uint16_t address, uint8_t opcode);
|
2021-12-10 19:49:52 +02:00
|
|
|
|
typedef void (*GB_lcd_line_callback_t)(GB_gameboy_t *gb, uint8_t line);
|
2021-12-10 19:42:47 +02:00
|
|
|
|
|
2016-07-07 00:29:25 +03:00
|
|
|
|
struct GB_breakpoint_s;
|
|
|
|
|
struct GB_watchpoint_s;
|
|
|
|
|
|
2018-03-03 15:47:36 +02:00
|
|
|
|
typedef struct {
|
|
|
|
|
uint8_t pixel; // Color, 0-3
|
|
|
|
|
uint8_t palette; // Palette, 0 - 7 (CGB); 0-1 in DMG (or just 0 for BG)
|
2021-12-17 21:16:23 +02:00
|
|
|
|
uint8_t priority; // Object priority – 0 in DMG, OAM index in CGB
|
|
|
|
|
bool bg_priority; // For object FIFO – the BG priority bit. For the BG FIFO – the CGB attributes priority bit
|
2018-03-03 15:47:36 +02:00
|
|
|
|
} GB_fifo_item_t;
|
|
|
|
|
|
2022-05-07 19:27:25 +03:00
|
|
|
|
#define GB_FIFO_LENGTH 8
|
2018-03-03 15:47:36 +02:00
|
|
|
|
typedef struct {
|
2018-04-06 18:26:04 +03:00
|
|
|
|
GB_fifo_item_t fifo[GB_FIFO_LENGTH];
|
2018-03-03 15:47:36 +02:00
|
|
|
|
uint8_t read_end;
|
2022-05-07 19:27:25 +03:00
|
|
|
|
uint8_t size;
|
2018-03-03 15:47:36 +02:00
|
|
|
|
} GB_fifo_t;
|
|
|
|
|
|
2021-04-02 19:08:03 +03:00
|
|
|
|
typedef struct {
|
|
|
|
|
uint32_t magic;
|
|
|
|
|
uint8_t track_count;
|
|
|
|
|
uint8_t first_track;
|
|
|
|
|
uint16_t load_address;
|
|
|
|
|
uint16_t init_address;
|
|
|
|
|
uint16_t play_address;
|
|
|
|
|
uint16_t sp;
|
|
|
|
|
uint8_t TMA;
|
|
|
|
|
uint8_t TAC;
|
|
|
|
|
char title[32];
|
|
|
|
|
char author[32];
|
|
|
|
|
char copyright[32];
|
|
|
|
|
} GB_gbs_header_t;
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
|
uint8_t track_count;
|
|
|
|
|
uint8_t first_track;
|
|
|
|
|
char title[33];
|
|
|
|
|
char author[33];
|
|
|
|
|
char copyright[33];
|
|
|
|
|
} GB_gbs_info_t;
|
|
|
|
|
|
2021-12-18 14:51:14 +02:00
|
|
|
|
/* Duplicated so it can remain anonymous in GB_gameboy_t */
|
|
|
|
|
typedef union {
|
|
|
|
|
uint16_t registers[GB_REGISTERS_16_BIT];
|
|
|
|
|
struct {
|
|
|
|
|
uint16_t af,
|
2021-12-30 23:53:24 +02:00
|
|
|
|
bc,
|
|
|
|
|
de,
|
|
|
|
|
hl,
|
|
|
|
|
sp,
|
|
|
|
|
pc;
|
2021-12-18 14:51:14 +02:00
|
|
|
|
};
|
|
|
|
|
struct {
|
2022-01-14 22:26:23 +02:00
|
|
|
|
uint8_t GB_REGISTER_ORDER;
|
2021-12-18 14:51:14 +02:00
|
|
|
|
};
|
|
|
|
|
} GB_registers_t;
|
|
|
|
|
|
2016-06-11 14:52:09 +03:00
|
|
|
|
/* When state saving, each section is dumped independently of other sections.
|
|
|
|
|
This allows adding data to the end of the section without worrying about future compatibility.
|
2016-06-18 20:29:11 +03:00
|
|
|
|
Some other changes might be "safe" as well.
|
|
|
|
|
This struct is not packed, but dumped sections exclusively use types that have the same alignment in both 32 and 64
|
|
|
|
|
bit platforms. */
|
|
|
|
|
|
2017-04-17 20:16:17 +03:00
|
|
|
|
#ifdef GB_INTERNAL
|
|
|
|
|
struct GB_gameboy_s {
|
|
|
|
|
#else
|
|
|
|
|
struct GB_gameboy_internal_s {
|
|
|
|
|
#endif
|
2016-06-11 14:52:09 +03:00
|
|
|
|
GB_SECTION(header,
|
2016-06-18 20:29:11 +03:00
|
|
|
|
/* The magic makes sure a state file is:
|
|
|
|
|
- Indeed a SameBoy state file.
|
|
|
|
|
- Has the same endianess has the current platform. */
|
2017-02-24 15:14:00 +02:00
|
|
|
|
volatile uint32_t magic;
|
2016-06-18 20:29:11 +03:00
|
|
|
|
/* The version field makes sure we don't load save state files with a completely different structure.
|
|
|
|
|
This happens when struct fields are removed/resized in an backward incompatible manner. */
|
|
|
|
|
uint32_t version;
|
2022-01-13 19:58:07 -08:00
|
|
|
|
)
|
2016-06-11 14:52:09 +03:00
|
|
|
|
|
|
|
|
|
GB_SECTION(core_state,
|
|
|
|
|
/* Registers */
|
2021-12-18 14:51:14 +02:00
|
|
|
|
union {
|
|
|
|
|
uint16_t registers[GB_REGISTERS_16_BIT];
|
|
|
|
|
struct {
|
|
|
|
|
uint16_t af,
|
|
|
|
|
bc,
|
|
|
|
|
de,
|
|
|
|
|
hl,
|
2021-12-30 23:53:24 +02:00
|
|
|
|
sp,
|
|
|
|
|
pc;
|
2021-12-18 14:51:14 +02:00
|
|
|
|
};
|
|
|
|
|
struct {
|
2022-01-14 22:26:23 +02:00
|
|
|
|
uint8_t GB_REGISTER_ORDER;
|
2021-12-18 14:51:14 +02:00
|
|
|
|
};
|
|
|
|
|
};
|
2016-08-05 16:36:38 +03:00
|
|
|
|
uint8_t ime;
|
2016-06-18 20:29:11 +03:00
|
|
|
|
uint8_t interrupt_enable;
|
|
|
|
|
uint8_t cgb_ram_bank;
|
2016-06-11 14:52:09 +03:00
|
|
|
|
|
|
|
|
|
/* CPU and General Hardware Flags*/
|
2018-06-16 13:59:33 +03:00
|
|
|
|
GB_model_t model;
|
2016-06-11 14:52:09 +03:00
|
|
|
|
bool cgb_mode;
|
|
|
|
|
bool cgb_double_speed;
|
|
|
|
|
bool halted;
|
|
|
|
|
bool stopped;
|
2016-06-18 20:29:11 +03:00
|
|
|
|
bool boot_rom_finished;
|
2018-02-25 22:32:41 +02:00
|
|
|
|
bool ime_toggle; /* ei has delayed a effect.*/
|
2016-09-22 01:51:09 +03:00
|
|
|
|
bool halt_bug;
|
2018-05-12 22:13:52 +03:00
|
|
|
|
bool just_halted;
|
2016-07-18 22:05:11 +03:00
|
|
|
|
|
2017-01-13 21:27:37 +02:00
|
|
|
|
/* Misc state */
|
2016-07-18 22:05:11 +03:00
|
|
|
|
bool infrared_input;
|
2017-01-13 21:27:37 +02:00
|
|
|
|
GB_printer_t printer;
|
2022-02-13 16:58:44 +02:00
|
|
|
|
uint8_t extra_oam[0xFF00 - 0xFEA0];
|
2019-06-07 19:18:07 +03:00
|
|
|
|
uint32_t ram_size; // Different between CGB and DMG
|
2020-09-19 19:31:24 +03:00
|
|
|
|
GB_workboy_t workboy;
|
2020-11-21 00:52:54 +02:00
|
|
|
|
|
|
|
|
|
int32_t ir_sensor;
|
|
|
|
|
bool effective_ir_input;
|
2021-10-19 01:53:24 +03:00
|
|
|
|
uint16_t address_bus;
|
2022-01-13 19:58:07 -08:00
|
|
|
|
)
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
2016-08-03 23:31:10 +03:00
|
|
|
|
/* DMA and HDMA */
|
|
|
|
|
GB_SECTION(dma,
|
2016-06-11 14:52:09 +03:00
|
|
|
|
bool hdma_on;
|
|
|
|
|
bool hdma_on_hblank;
|
2016-06-18 20:29:11 +03:00
|
|
|
|
uint8_t hdma_steps_left;
|
|
|
|
|
uint16_t hdma_current_src, hdma_current_dest;
|
2016-08-03 23:31:10 +03:00
|
|
|
|
|
|
|
|
|
uint8_t dma_current_dest;
|
2022-01-12 00:26:18 +02:00
|
|
|
|
uint8_t last_dma_read;
|
2016-08-03 23:31:10 +03:00
|
|
|
|
uint16_t dma_current_src;
|
2022-01-16 13:50:59 +02:00
|
|
|
|
uint16_t dma_cycles;
|
|
|
|
|
int8_t dma_cycles_modulo;
|
2022-01-17 22:07:24 +02:00
|
|
|
|
bool dma_ppu_vram_conflict;
|
|
|
|
|
uint16_t dma_ppu_vram_conflict_addr;
|
2022-01-28 02:49:16 +02:00
|
|
|
|
uint8_t hdma_open_bus; /* Required to emulate HDMA reads from Exxx */
|
2022-06-17 18:17:29 +03:00
|
|
|
|
bool allow_hdma_on_wake;
|
2022-01-13 19:58:07 -08:00
|
|
|
|
)
|
2016-06-11 14:52:09 +03:00
|
|
|
|
|
|
|
|
|
/* MBC */
|
|
|
|
|
GB_SECTION(mbc,
|
2016-06-18 20:29:11 +03:00
|
|
|
|
uint16_t mbc_rom_bank;
|
2022-03-05 16:35:03 +02:00
|
|
|
|
uint16_t mbc_rom0_bank; /* For multicart mappings . */
|
2016-06-18 20:29:11 +03:00
|
|
|
|
uint8_t mbc_ram_bank;
|
|
|
|
|
uint32_t mbc_ram_size;
|
2016-06-11 14:52:09 +03:00
|
|
|
|
bool mbc_ram_enable;
|
2016-07-09 17:34:55 +03:00
|
|
|
|
union {
|
|
|
|
|
struct {
|
|
|
|
|
uint8_t bank_low:5;
|
2017-06-21 01:07:11 +03:00
|
|
|
|
uint8_t bank_high:2;
|
2016-07-09 17:34:55 +03:00
|
|
|
|
uint8_t mode:1;
|
|
|
|
|
} mbc1;
|
|
|
|
|
|
|
|
|
|
struct {
|
|
|
|
|
uint8_t rom_bank:4;
|
|
|
|
|
} mbc2;
|
|
|
|
|
|
|
|
|
|
struct {
|
2020-04-08 19:07:29 +03:00
|
|
|
|
uint8_t rom_bank:8;
|
|
|
|
|
uint8_t ram_bank:3;
|
2021-11-07 01:10:58 +02:00
|
|
|
|
bool rtc_mapped:1;
|
2016-07-09 17:34:55 +03:00
|
|
|
|
} mbc3;
|
|
|
|
|
|
|
|
|
|
struct {
|
|
|
|
|
uint8_t rom_bank_low;
|
|
|
|
|
uint8_t rom_bank_high:1;
|
|
|
|
|
uint8_t ram_bank:4;
|
2022-06-11 14:44:06 +03:00
|
|
|
|
} mbc5; // Also used for GB_CAMERA
|
2021-11-12 17:44:51 +02:00
|
|
|
|
|
2022-03-05 16:35:03 +02:00
|
|
|
|
struct {
|
|
|
|
|
uint8_t rom_bank;
|
|
|
|
|
uint16_t x_latch;
|
|
|
|
|
uint16_t y_latch;
|
|
|
|
|
bool latch_ready:1;
|
|
|
|
|
bool eeprom_do:1;
|
|
|
|
|
bool eeprom_di:1;
|
|
|
|
|
bool eeprom_clk:1;
|
|
|
|
|
bool eeprom_cs:1;
|
|
|
|
|
uint16_t eeprom_command:11;
|
|
|
|
|
uint16_t read_bits;
|
2022-06-05 14:09:33 +03:00
|
|
|
|
uint8_t argument_bits_left:5;
|
2022-03-05 16:35:03 +02:00
|
|
|
|
bool secondary_ram_enable:1;
|
|
|
|
|
bool eeprom_write_enabled:1;
|
|
|
|
|
} mbc7;
|
|
|
|
|
|
|
|
|
|
struct {
|
|
|
|
|
uint8_t rom_bank_low:5;
|
|
|
|
|
uint8_t rom_bank_mid:2;
|
|
|
|
|
bool mbc1_mode:1;
|
|
|
|
|
|
|
|
|
|
uint8_t rom_bank_mask:4;
|
|
|
|
|
uint8_t rom_bank_high:2;
|
|
|
|
|
uint8_t ram_bank_low:2;
|
|
|
|
|
|
|
|
|
|
uint8_t ram_bank_high:2;
|
|
|
|
|
uint8_t ram_bank_mask:2;
|
|
|
|
|
|
|
|
|
|
bool locked:1;
|
|
|
|
|
bool mbc1_mode_disable:1;
|
|
|
|
|
bool multiplex_mode:1;
|
|
|
|
|
} mmm01;
|
2017-02-16 21:07:35 +02:00
|
|
|
|
|
|
|
|
|
struct {
|
|
|
|
|
uint8_t bank_low:6;
|
|
|
|
|
uint8_t bank_high:3;
|
2022-06-11 20:57:02 +03:00
|
|
|
|
bool ir_mode;
|
2017-02-16 21:07:35 +02:00
|
|
|
|
} huc1;
|
2016-10-17 18:51:43 +03:00
|
|
|
|
|
|
|
|
|
struct {
|
2020-05-16 23:27:17 +03:00
|
|
|
|
uint8_t rom_bank:7;
|
|
|
|
|
uint8_t padding:1;
|
|
|
|
|
uint8_t ram_bank:4;
|
2021-11-07 01:10:58 +02:00
|
|
|
|
uint8_t mode;
|
|
|
|
|
uint8_t access_index;
|
|
|
|
|
uint16_t minutes, days;
|
|
|
|
|
uint16_t alarm_minutes, alarm_days;
|
|
|
|
|
bool alarm_enabled;
|
|
|
|
|
uint8_t read;
|
|
|
|
|
uint8_t access_flags;
|
2016-10-17 18:51:43 +03:00
|
|
|
|
} huc3;
|
2021-11-07 01:10:58 +02:00
|
|
|
|
|
|
|
|
|
struct {
|
|
|
|
|
uint16_t rom_bank;
|
|
|
|
|
uint8_t ram_bank;
|
|
|
|
|
uint8_t mode;
|
|
|
|
|
} tpp1;
|
2016-07-09 17:34:55 +03:00
|
|
|
|
};
|
2016-10-02 03:40:11 +03:00
|
|
|
|
bool camera_registers_mapped;
|
2016-10-02 17:14:58 +03:00
|
|
|
|
uint8_t camera_registers[0x36];
|
2021-04-10 23:56:41 +03:00
|
|
|
|
uint8_t rumble_strength;
|
2020-05-13 22:21:31 +03:00
|
|
|
|
bool cart_ir;
|
2022-01-13 19:58:07 -08:00
|
|
|
|
)
|
2016-06-11 14:52:09 +03:00
|
|
|
|
|
|
|
|
|
/* HRAM and HW Registers */
|
|
|
|
|
GB_SECTION(hram,
|
2016-06-18 20:29:11 +03:00
|
|
|
|
uint8_t hram[0xFFFF - 0xFF80];
|
|
|
|
|
uint8_t io_registers[0x80];
|
2022-01-13 19:58:07 -08:00
|
|
|
|
)
|
2016-06-11 14:52:09 +03:00
|
|
|
|
|
|
|
|
|
/* Timing */
|
|
|
|
|
GB_SECTION(timing,
|
2018-02-25 00:48:45 +02:00
|
|
|
|
GB_UNIT(display);
|
2018-02-23 13:16:05 +02:00
|
|
|
|
GB_UNIT(div);
|
2018-06-09 15:11:20 +03:00
|
|
|
|
uint16_t div_counter;
|
2016-08-06 13:56:29 +03:00
|
|
|
|
uint8_t tima_reload_state; /* After TIMA overflows, it becomes 0 for 4 cycles before actually reloading. */
|
2022-04-17 23:41:05 +03:00
|
|
|
|
bool serial_master_clock;
|
|
|
|
|
uint8_t serial_mask;
|
2019-01-13 01:09:41 +02:00
|
|
|
|
uint8_t double_speed_alignment;
|
2019-03-15 14:36:10 +02:00
|
|
|
|
uint8_t serial_count;
|
2021-07-10 15:07:23 +03:00
|
|
|
|
int32_t speed_switch_halt_countdown;
|
2021-07-11 12:25:39 +03:00
|
|
|
|
uint8_t speed_switch_countdown; // To compensate for the lack of pipeline emulation
|
2021-07-10 15:07:23 +03:00
|
|
|
|
uint8_t speed_switch_freeze; // Solely for realigning the PPU, should be removed when the odd modes are implemented
|
2021-12-04 15:04:46 +02:00
|
|
|
|
/* For timing of the vblank callback */
|
|
|
|
|
uint32_t cycles_since_vblank_callback;
|
|
|
|
|
bool lcd_disabled_outside_of_vblank;
|
2021-12-26 01:47:59 +02:00
|
|
|
|
int32_t allowed_pending_cycles;
|
|
|
|
|
uint16_t mode3_batching_length;
|
2022-01-13 19:58:07 -08:00
|
|
|
|
)
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
2016-06-11 14:52:09 +03:00
|
|
|
|
/* APU */
|
|
|
|
|
GB_SECTION(apu,
|
|
|
|
|
GB_apu_t apu;
|
2022-01-13 19:58:07 -08:00
|
|
|
|
)
|
2016-06-11 14:52:09 +03:00
|
|
|
|
|
|
|
|
|
/* RTC */
|
|
|
|
|
GB_SECTION(rtc,
|
2018-11-03 01:31:14 +02:00
|
|
|
|
GB_rtc_time_t rtc_real, rtc_latched;
|
|
|
|
|
uint64_t last_rtc_second;
|
2021-02-25 22:12:14 +02:00
|
|
|
|
uint32_t rtc_cycles;
|
2021-04-16 00:35:54 +03:00
|
|
|
|
uint8_t tpp1_mr4;
|
2022-01-13 19:58:07 -08:00
|
|
|
|
)
|
2016-06-11 14:52:09 +03:00
|
|
|
|
|
|
|
|
|
/* Video Display */
|
|
|
|
|
GB_SECTION(video,
|
2016-06-18 20:29:11 +03:00
|
|
|
|
uint32_t vram_size; // Different between CGB and DMG
|
2021-10-30 16:03:13 +03:00
|
|
|
|
bool cgb_vram_bank;
|
2016-06-18 20:29:11 +03:00
|
|
|
|
uint8_t oam[0xA0];
|
2017-04-19 23:26:39 +03:00
|
|
|
|
uint8_t background_palettes_data[0x40];
|
2021-12-17 21:16:23 +02:00
|
|
|
|
uint8_t object_palettes_data[0x40];
|
2018-02-25 00:48:45 +02:00
|
|
|
|
uint8_t position_in_line;
|
2016-06-11 14:52:09 +03:00
|
|
|
|
bool stat_interrupt_line;
|
2020-02-23 23:16:45 +02:00
|
|
|
|
uint8_t window_y;
|
2017-06-03 20:06:52 +03:00
|
|
|
|
/* The LCDC will skip the first frame it renders after turning it on.
|
|
|
|
|
On the CGB, a frame is not skipped if the previous frame was skipped as well.
|
2017-06-03 16:42:42 +03:00
|
|
|
|
See https://www.reddit.com/r/EmuDev/comments/6exyxu/ */
|
2019-07-16 20:44:27 +03:00
|
|
|
|
|
|
|
|
|
/* TODO: Drop this and properly emulate the dropped vreset signal*/
|
2017-06-03 16:42:42 +03:00
|
|
|
|
enum {
|
2017-06-03 20:06:52 +03:00
|
|
|
|
GB_FRAMESKIP_LCD_TURNED_ON, // On a DMG, the LCD renders a blank screen during this state,
|
|
|
|
|
// on a CGB, the previous frame is repeated (which might be
|
|
|
|
|
// blank if the LCD was off for more than a few cycles)
|
|
|
|
|
GB_FRAMESKIP_FIRST_FRAME_SKIPPED, // This state is 'skipped' when emulating a DMG
|
2017-06-03 16:42:42 +03:00
|
|
|
|
GB_FRAMESKIP_SECOND_FRAME_RENDERED,
|
|
|
|
|
} frame_skip_state;
|
2017-06-18 21:27:07 +03:00
|
|
|
|
bool oam_read_blocked;
|
|
|
|
|
bool vram_read_blocked;
|
|
|
|
|
bool oam_write_blocked;
|
|
|
|
|
bool vram_write_blocked;
|
2018-02-25 00:48:45 +02:00
|
|
|
|
uint8_t current_line;
|
|
|
|
|
uint16_t ly_for_comparison;
|
2018-03-03 15:47:36 +02:00
|
|
|
|
GB_fifo_t bg_fifo, oam_fifo;
|
2018-03-03 20:51:38 +02:00
|
|
|
|
uint8_t fetcher_y;
|
2018-03-03 15:47:36 +02:00
|
|
|
|
uint16_t cycles_for_line;
|
|
|
|
|
uint8_t current_tile;
|
2018-03-09 18:52:36 +02:00
|
|
|
|
uint8_t current_tile_attributes;
|
2018-03-03 15:47:36 +02:00
|
|
|
|
uint8_t current_tile_data[2];
|
2018-04-06 19:29:49 +03:00
|
|
|
|
uint8_t fetcher_state;
|
2020-02-29 17:06:08 +02:00
|
|
|
|
bool window_is_being_fetched;
|
2020-03-01 23:58:28 +02:00
|
|
|
|
bool wx166_glitch;
|
2020-02-23 23:16:45 +02:00
|
|
|
|
bool wx_triggered;
|
2018-03-04 22:21:56 +02:00
|
|
|
|
uint8_t visible_objs[10];
|
2022-01-14 15:07:50 +02:00
|
|
|
|
uint8_t objects_x[10];
|
|
|
|
|
uint8_t objects_y[10];
|
2022-01-16 13:50:59 +02:00
|
|
|
|
uint8_t object_tile_data[2];
|
2022-02-06 13:02:15 +02:00
|
|
|
|
uint8_t mode2_y_bus;
|
|
|
|
|
// They're the same bus
|
|
|
|
|
union {
|
|
|
|
|
uint8_t mode2_x_bus;
|
|
|
|
|
uint8_t object_flags;
|
|
|
|
|
};
|
2018-03-04 22:21:56 +02:00
|
|
|
|
uint8_t n_visible_objs;
|
2022-05-14 01:14:41 +03:00
|
|
|
|
uint8_t orig_n_visible_objs;
|
2018-03-23 19:07:14 +03:00
|
|
|
|
uint8_t oam_search_index;
|
2018-03-27 15:46:00 +03:00
|
|
|
|
uint8_t accessed_oam_row;
|
2018-06-04 02:07:38 +03:00
|
|
|
|
uint8_t mode_for_interrupt;
|
2018-04-27 13:40:39 +03:00
|
|
|
|
bool lyc_interrupt_line;
|
2019-01-19 19:32:26 +02:00
|
|
|
|
bool cgb_palettes_blocked;
|
2019-02-15 17:04:48 +02:00
|
|
|
|
uint8_t current_lcd_line; // The LCD can go out of sync since the vsync signal is skipped in some cases.
|
2020-02-15 15:32:06 +02:00
|
|
|
|
uint8_t object_priority;
|
2020-02-15 19:21:43 +02:00
|
|
|
|
bool oam_ppu_blocked;
|
|
|
|
|
bool vram_ppu_blocked;
|
|
|
|
|
bool cgb_palettes_ppu_blocked;
|
2020-02-21 15:14:33 +02:00
|
|
|
|
bool object_fetch_aborted;
|
2020-02-21 16:16:02 +02:00
|
|
|
|
bool during_object_fetch;
|
2020-02-21 16:43:51 +02:00
|
|
|
|
uint16_t object_low_line_address;
|
2020-02-23 23:16:45 +02:00
|
|
|
|
bool wy_triggered;
|
|
|
|
|
uint8_t window_tile_x;
|
2020-03-06 14:41:13 +02:00
|
|
|
|
uint8_t lcd_x; // The LCD can go out of sync since the push signal is skipped in some cases.
|
2020-03-26 20:54:18 +02:00
|
|
|
|
bool is_odd_frame;
|
2020-05-29 23:10:23 +03:00
|
|
|
|
uint16_t last_tile_data_address;
|
|
|
|
|
uint16_t last_tile_index_address;
|
2020-05-30 01:25:21 +03:00
|
|
|
|
bool cgb_repeated_a_frame;
|
2020-11-20 16:24:16 +02:00
|
|
|
|
uint8_t data_for_sel_glitch;
|
2022-05-14 01:14:41 +03:00
|
|
|
|
bool delayed_glitch_hblank_interrupt;
|
2022-01-13 19:58:07 -08:00
|
|
|
|
)
|
2016-06-11 14:52:09 +03:00
|
|
|
|
|
|
|
|
|
/* Unsaved data. This includes all pointers, as well as everything that shouldn't be on a save state */
|
2017-04-17 20:16:17 +03:00
|
|
|
|
/* This data is reserved on reset and must come last in the struct */
|
|
|
|
|
GB_SECTION(unsaved,
|
|
|
|
|
/* ROM */
|
|
|
|
|
uint8_t *rom;
|
|
|
|
|
uint32_t rom_size;
|
|
|
|
|
const GB_cartridge_t *cartridge_type;
|
|
|
|
|
enum {
|
|
|
|
|
GB_STANDARD_MBC1_WIRING,
|
|
|
|
|
GB_MBC1M_WIRING,
|
|
|
|
|
} mbc1_wiring;
|
2020-04-08 19:07:29 +03:00
|
|
|
|
bool is_mbc30;
|
2017-04-17 20:16:17 +03:00
|
|
|
|
|
2018-04-14 13:23:33 +03:00
|
|
|
|
unsigned pending_cycles;
|
|
|
|
|
|
2017-04-17 20:16:17 +03:00
|
|
|
|
/* Various RAMs */
|
|
|
|
|
uint8_t *ram;
|
|
|
|
|
uint8_t *vram;
|
|
|
|
|
uint8_t *mbc_ram;
|
|
|
|
|
|
|
|
|
|
/* I/O */
|
|
|
|
|
uint32_t *screen;
|
2017-10-12 17:22:22 +03:00
|
|
|
|
uint32_t background_palettes_rgb[0x20];
|
2021-12-17 21:16:23 +02:00
|
|
|
|
uint32_t object_palettes_rgb[0x20];
|
2020-01-29 14:19:11 +02:00
|
|
|
|
const GB_palette_t *dmg_palette;
|
2017-10-12 17:22:22 +03:00
|
|
|
|
GB_color_correction_mode_t color_correction_mode;
|
2020-12-25 14:14:17 +02:00
|
|
|
|
double light_temperature;
|
2018-11-12 00:37:06 +02:00
|
|
|
|
bool keys[4][GB_KEY_MAX];
|
2021-11-12 17:44:51 +02:00
|
|
|
|
double accelerometer_x, accelerometer_y;
|
2020-02-08 13:28:46 +02:00
|
|
|
|
GB_border_mode_t border_mode;
|
|
|
|
|
GB_sgb_border_t borrowed_border;
|
|
|
|
|
bool tried_loading_sgb_border;
|
|
|
|
|
bool has_sgb_border;
|
2021-12-17 21:12:26 +02:00
|
|
|
|
bool objects_disabled;
|
|
|
|
|
bool background_disabled;
|
2021-12-18 01:25:06 +02:00
|
|
|
|
bool joyp_accessed;
|
2021-12-19 00:28:24 +02:00
|
|
|
|
bool illegal_inputs_allowed;
|
2021-12-18 01:25:06 +02:00
|
|
|
|
|
2017-04-21 16:00:53 +03:00
|
|
|
|
/* Timing */
|
|
|
|
|
uint64_t last_sync;
|
2018-02-20 21:17:12 +02:00
|
|
|
|
uint64_t cycles_since_last_sync; // In 8MHz units
|
2021-02-25 22:12:14 +02:00
|
|
|
|
GB_rtc_mode_t rtc_mode;
|
2021-12-11 02:51:21 +02:00
|
|
|
|
uint32_t rtc_second_length;
|
2021-12-26 15:50:24 +02:00
|
|
|
|
uint32_t clock_rate;
|
|
|
|
|
uint32_t unmultiplied_clock_rate;
|
2017-04-17 20:16:17 +03:00
|
|
|
|
|
2017-04-21 16:00:53 +03:00
|
|
|
|
/* Audio */
|
2017-07-21 23:06:02 +03:00
|
|
|
|
GB_apu_output_t apu_output;
|
2017-04-17 20:16:17 +03:00
|
|
|
|
|
|
|
|
|
/* Callbacks */
|
|
|
|
|
void *user_data;
|
|
|
|
|
GB_log_callback_t log_callback;
|
|
|
|
|
GB_input_callback_t input_callback;
|
|
|
|
|
GB_input_callback_t async_input_callback;
|
|
|
|
|
GB_rgb_encode_callback_t rgb_encode_callback;
|
|
|
|
|
GB_vblank_callback_t vblank_callback;
|
|
|
|
|
GB_infrared_callback_t infrared_callback;
|
|
|
|
|
GB_camera_get_pixel_callback_t camera_get_pixel_callback;
|
|
|
|
|
GB_camera_update_request_callback_t camera_update_request_callback;
|
|
|
|
|
GB_rumble_callback_t rumble_callback;
|
2019-03-15 14:36:10 +02:00
|
|
|
|
GB_serial_transfer_bit_start_callback_t serial_transfer_bit_start_callback;
|
|
|
|
|
GB_serial_transfer_bit_end_callback_t serial_transfer_bit_end_callback;
|
2019-06-07 18:27:25 +03:00
|
|
|
|
GB_update_input_hint_callback_t update_input_hint_callback;
|
2019-07-15 22:01:41 +03:00
|
|
|
|
GB_joyp_write_callback_t joyp_write_callback;
|
2019-07-16 20:44:27 +03:00
|
|
|
|
GB_icd_pixel_callback_t icd_pixel_callback;
|
|
|
|
|
GB_icd_vreset_callback_t icd_hreset_callback;
|
2019-07-15 23:02:58 +03:00
|
|
|
|
GB_icd_vreset_callback_t icd_vreset_callback;
|
2019-08-16 17:38:43 +03:00
|
|
|
|
GB_read_memory_callback_t read_memory_callback;
|
2021-11-26 14:09:41 +02:00
|
|
|
|
GB_write_memory_callback_t write_memory_callback;
|
2020-01-29 20:29:30 +02:00
|
|
|
|
GB_boot_rom_load_callback_t boot_rom_load_callback;
|
2020-06-10 22:46:19 +03:00
|
|
|
|
GB_print_image_callback_t printer_callback;
|
2020-09-19 19:31:24 +03:00
|
|
|
|
GB_workboy_set_time_callback workboy_set_time_callback;
|
|
|
|
|
GB_workboy_get_time_callback workboy_get_time_callback;
|
2021-12-10 19:42:47 +02:00
|
|
|
|
GB_execution_callback_t execution_callback;
|
2021-12-10 19:49:52 +02:00
|
|
|
|
GB_lcd_line_callback_t lcd_line_callback;
|
2017-04-17 20:16:17 +03:00
|
|
|
|
|
|
|
|
|
/*** Debugger ***/
|
|
|
|
|
volatile bool debug_stopped, debug_disable;
|
|
|
|
|
bool debug_fin_command, debug_next_command;
|
|
|
|
|
|
|
|
|
|
/* Breakpoints */
|
|
|
|
|
uint16_t n_breakpoints;
|
|
|
|
|
struct GB_breakpoint_s *breakpoints;
|
2020-05-30 16:35:07 -04:00
|
|
|
|
bool has_jump_to_breakpoints, has_software_breakpoints;
|
2019-04-12 20:29:43 +03:00
|
|
|
|
void *nontrivial_jump_state;
|
|
|
|
|
bool non_trivial_jump_breakpoint_occured;
|
2017-04-17 20:16:17 +03:00
|
|
|
|
|
|
|
|
|
/* SLD (Todo: merge with backtrace) */
|
|
|
|
|
bool stack_leak_detection;
|
2020-04-09 14:32:52 +03:00
|
|
|
|
signed debug_call_depth;
|
2017-04-17 20:16:17 +03:00
|
|
|
|
uint16_t sp_for_call_depth[0x200]; /* Should be much more than enough */
|
|
|
|
|
uint16_t addr_for_call_depth[0x200];
|
|
|
|
|
|
|
|
|
|
/* Backtrace */
|
2019-06-15 23:22:27 +03:00
|
|
|
|
unsigned backtrace_size;
|
2017-04-17 20:16:17 +03:00
|
|
|
|
uint16_t backtrace_sps[0x200];
|
|
|
|
|
struct {
|
|
|
|
|
uint16_t bank;
|
|
|
|
|
uint16_t addr;
|
|
|
|
|
} backtrace_returns[0x200];
|
|
|
|
|
|
|
|
|
|
/* Watchpoints */
|
|
|
|
|
uint16_t n_watchpoints;
|
|
|
|
|
struct GB_watchpoint_s *watchpoints;
|
|
|
|
|
|
|
|
|
|
/* Symbol tables */
|
2022-02-28 23:29:49 +02:00
|
|
|
|
GB_symbol_map_t **bank_symbols;
|
|
|
|
|
size_t n_symbol_maps;
|
2017-04-17 20:16:17 +03:00
|
|
|
|
GB_reversed_symbol_map_t reversed_symbol_map;
|
|
|
|
|
|
|
|
|
|
/* Ticks command */
|
2020-04-09 14:32:52 +03:00
|
|
|
|
uint64_t debugger_ticks;
|
2021-07-11 12:25:39 +03:00
|
|
|
|
uint64_t absolute_debugger_ticks;
|
2018-02-10 14:42:14 +02:00
|
|
|
|
|
2020-11-21 16:19:58 +02:00
|
|
|
|
/* Undo */
|
|
|
|
|
uint8_t *undo_state;
|
|
|
|
|
const char *undo_label;
|
|
|
|
|
|
2018-02-10 14:42:14 +02:00
|
|
|
|
/* Rewind */
|
|
|
|
|
size_t rewind_buffer_length;
|
|
|
|
|
struct {
|
|
|
|
|
uint8_t *key_state;
|
|
|
|
|
uint8_t *compressed_states[GB_REWIND_FRAMES_PER_KEY];
|
|
|
|
|
unsigned pos;
|
|
|
|
|
} *rewind_sequences; // lasts about 4 seconds
|
|
|
|
|
size_t rewind_pos;
|
2018-11-16 01:53:01 +02:00
|
|
|
|
|
|
|
|
|
/* SGB - saved and allocated optionally */
|
|
|
|
|
GB_sgb_t *sgb;
|
2018-11-24 13:21:00 +02:00
|
|
|
|
|
|
|
|
|
double sgb_intro_jingle_phases[7];
|
|
|
|
|
double sgb_intro_sweep_phase;
|
2018-12-01 13:39:43 +02:00
|
|
|
|
double sgb_intro_sweep_previous_sample;
|
2020-04-09 20:11:55 +03:00
|
|
|
|
|
|
|
|
|
/* Cheats */
|
2020-04-11 18:03:10 +03:00
|
|
|
|
bool cheat_enabled;
|
2020-04-09 20:11:55 +03:00
|
|
|
|
size_t cheat_count;
|
|
|
|
|
GB_cheat_t **cheats;
|
|
|
|
|
GB_cheat_hash_t *cheat_hash[256];
|
2017-04-17 20:16:17 +03:00
|
|
|
|
|
|
|
|
|
/* Misc */
|
|
|
|
|
bool turbo;
|
|
|
|
|
bool turbo_dont_skip;
|
|
|
|
|
bool disable_rendering;
|
|
|
|
|
uint8_t boot_rom[0x900];
|
|
|
|
|
bool vblank_just_occured; // For slow operations involving syscalls; these should only run once per vblank
|
2022-01-28 02:49:16 +02:00
|
|
|
|
unsigned cycles_since_run; // How many cycles have passed since the last call to GB_run(), in 8MHz units
|
2018-02-10 23:30:30 +02:00
|
|
|
|
double clock_multiplier;
|
2020-04-29 16:50:31 +03:00
|
|
|
|
GB_rumble_mode_t rumble_mode;
|
2019-10-19 19:26:04 +03:00
|
|
|
|
uint32_t rumble_on_cycles;
|
|
|
|
|
uint32_t rumble_off_cycles;
|
2020-02-27 00:12:42 +02:00
|
|
|
|
|
|
|
|
|
/* Temporary state */
|
|
|
|
|
bool wx_just_changed;
|
2020-11-20 16:24:16 +02:00
|
|
|
|
bool tile_sel_glitch;
|
2021-11-26 13:54:28 +02:00
|
|
|
|
bool disable_oam_corruption; // For safe memory reads
|
2022-01-17 22:07:24 +02:00
|
|
|
|
bool in_dma_read;
|
2022-01-31 01:02:31 +02:00
|
|
|
|
bool hdma_in_progress;
|
2022-01-30 14:38:58 +02:00
|
|
|
|
uint16_t addr_for_hdma_conflict;
|
2021-04-02 19:08:03 +03:00
|
|
|
|
|
|
|
|
|
GB_gbs_header_t gbs_header;
|
2022-01-13 19:58:07 -08:00
|
|
|
|
)
|
2017-04-17 20:16:17 +03:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#ifndef GB_INTERNAL
|
|
|
|
|
struct GB_gameboy_s {
|
2021-12-20 18:59:51 +02:00
|
|
|
|
alignas(struct GB_gameboy_internal_s) uint8_t __internal[sizeof(struct GB_gameboy_internal_s)];
|
2017-04-17 20:16:17 +03:00
|
|
|
|
};
|
|
|
|
|
#endif
|
2016-06-11 14:52:09 +03:00
|
|
|
|
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
|
|
#ifndef __printflike
|
|
|
|
|
/* Missing from Linux headers. */
|
|
|
|
|
#define __printflike(fmtarg, firstvararg) \
|
|
|
|
|
__attribute__((__format__ (__printf__, fmtarg, firstvararg)))
|
|
|
|
|
#endif
|
|
|
|
|
|
2018-06-16 13:59:33 +03:00
|
|
|
|
void GB_init(GB_gameboy_t *gb, GB_model_t model);
|
2017-04-19 21:55:58 +03:00
|
|
|
|
bool GB_is_inited(GB_gameboy_t *gb);
|
2021-12-26 01:47:59 +02:00
|
|
|
|
bool GB_is_cgb(const GB_gameboy_t *gb);
|
2021-11-25 17:16:11 -06:00
|
|
|
|
bool GB_is_cgb_in_cgb_mode(GB_gameboy_t *gb);
|
2019-07-15 20:47:16 +03:00
|
|
|
|
bool GB_is_sgb(GB_gameboy_t *gb); // Returns true if the model is SGB or SGB2
|
|
|
|
|
bool GB_is_hle_sgb(GB_gameboy_t *gb); // Returns true if the model is SGB or SGB2 and the SFC/SNES side is HLE'd
|
2018-06-16 13:59:33 +03:00
|
|
|
|
GB_model_t GB_get_model(GB_gameboy_t *gb);
|
2016-06-18 20:29:11 +03:00
|
|
|
|
void GB_free(GB_gameboy_t *gb);
|
2017-04-17 20:16:17 +03:00
|
|
|
|
void GB_reset(GB_gameboy_t *gb);
|
2018-06-16 13:59:33 +03:00
|
|
|
|
void GB_switch_model_and_reset(GB_gameboy_t *gb, GB_model_t model);
|
2018-01-31 15:18:04 +02:00
|
|
|
|
|
2019-07-13 20:29:11 +03:00
|
|
|
|
/* Returns the time passed, in 8MHz ticks. */
|
2022-01-28 02:49:16 +02:00
|
|
|
|
unsigned GB_run(GB_gameboy_t *gb);
|
2017-04-25 00:19:10 +03:00
|
|
|
|
/* Returns the time passed since the last frame, in nanoseconds */
|
|
|
|
|
uint64_t GB_run_frame(GB_gameboy_t *gb);
|
2017-04-19 21:55:58 +03:00
|
|
|
|
|
2017-04-19 23:26:39 +03:00
|
|
|
|
typedef enum {
|
|
|
|
|
GB_DIRECT_ACCESS_ROM,
|
|
|
|
|
GB_DIRECT_ACCESS_RAM,
|
|
|
|
|
GB_DIRECT_ACCESS_CART_RAM,
|
|
|
|
|
GB_DIRECT_ACCESS_VRAM,
|
|
|
|
|
GB_DIRECT_ACCESS_HRAM,
|
|
|
|
|
GB_DIRECT_ACCESS_IO, /* Warning: Some registers can only be read/written correctly via GB_memory_read/write. */
|
|
|
|
|
GB_DIRECT_ACCESS_BOOTROM,
|
|
|
|
|
GB_DIRECT_ACCESS_OAM,
|
|
|
|
|
GB_DIRECT_ACCESS_BGP,
|
|
|
|
|
GB_DIRECT_ACCESS_OBP,
|
2017-10-12 22:49:39 +03:00
|
|
|
|
GB_DIRECT_ACCESS_IE,
|
2017-04-19 23:26:39 +03:00
|
|
|
|
} GB_direct_access_t;
|
|
|
|
|
|
|
|
|
|
/* Returns a mutable pointer to various hardware memories. If that memory is banked, the current bank
|
|
|
|
|
is returned at *bank, even if only a portion of the memory is banked. */
|
|
|
|
|
void *GB_get_direct_access(GB_gameboy_t *gb, GB_direct_access_t access, size_t *size, uint16_t *bank);
|
2021-12-18 14:51:14 +02:00
|
|
|
|
GB_registers_t *GB_get_registers(GB_gameboy_t *gb);
|
2017-04-19 23:26:39 +03:00
|
|
|
|
|
2017-04-19 21:55:58 +03:00
|
|
|
|
void *GB_get_user_data(GB_gameboy_t *gb);
|
|
|
|
|
void GB_set_user_data(GB_gameboy_t *gb, void *data);
|
|
|
|
|
|
2016-06-18 20:29:11 +03:00
|
|
|
|
int GB_load_boot_rom(GB_gameboy_t *gb, const char *path);
|
2017-10-12 22:06:01 +03:00
|
|
|
|
void GB_load_boot_rom_from_buffer(GB_gameboy_t *gb, const unsigned char *buffer, size_t size);
|
2016-06-18 20:29:11 +03:00
|
|
|
|
int GB_load_rom(GB_gameboy_t *gb, const char *path);
|
2019-07-16 21:04:29 +03:00
|
|
|
|
void GB_load_rom_from_buffer(GB_gameboy_t *gb, const uint8_t *buffer, size_t size);
|
2020-04-25 22:48:48 +03:00
|
|
|
|
int GB_load_isx(GB_gameboy_t *gb, const char *path);
|
2021-04-20 08:38:53 -04:00
|
|
|
|
int GB_load_gbs_from_buffer(GB_gameboy_t *gb, const uint8_t *buffer, size_t size, GB_gbs_info_t *info);
|
2021-04-02 19:08:03 +03:00
|
|
|
|
int GB_load_gbs(GB_gameboy_t *gb, const char *path, GB_gbs_info_t *info);
|
|
|
|
|
void GB_gbs_switch_track(GB_gameboy_t *gb, uint8_t track);
|
|
|
|
|
|
2019-08-16 17:38:43 +03:00
|
|
|
|
int GB_save_battery_size(GB_gameboy_t *gb);
|
|
|
|
|
int GB_save_battery_to_buffer(GB_gameboy_t *gb, uint8_t *buffer, size_t size);
|
2016-06-18 20:29:11 +03:00
|
|
|
|
int GB_save_battery(GB_gameboy_t *gb, const char *path);
|
2019-08-16 17:38:43 +03:00
|
|
|
|
|
|
|
|
|
void GB_load_battery_from_buffer(GB_gameboy_t *gb, const uint8_t *buffer, size_t size);
|
2016-06-18 20:29:11 +03:00
|
|
|
|
void GB_load_battery(GB_gameboy_t *gb, const char *path);
|
2017-04-19 21:55:58 +03:00
|
|
|
|
|
|
|
|
|
void GB_set_turbo_mode(GB_gameboy_t *gb, bool on, bool no_frame_skip);
|
|
|
|
|
void GB_set_rendering_disabled(GB_gameboy_t *gb, bool disabled);
|
|
|
|
|
|
|
|
|
|
void GB_log(GB_gameboy_t *gb, const char *fmt, ...) __printflike(2, 3);
|
|
|
|
|
void GB_attributed_log(GB_gameboy_t *gb, GB_log_attributes attributes, const char *fmt, ...) __printflike(3, 4);
|
|
|
|
|
|
2016-06-18 20:29:11 +03:00
|
|
|
|
void GB_set_pixels_output(GB_gameboy_t *gb, uint32_t *output);
|
2020-02-08 13:28:46 +02:00
|
|
|
|
void GB_set_border_mode(GB_gameboy_t *gb, GB_border_mode_t border_mode);
|
|
|
|
|
|
2017-04-19 21:55:58 +03:00
|
|
|
|
void GB_set_infrared_input(GB_gameboy_t *gb, bool state);
|
|
|
|
|
|
2016-06-18 20:29:11 +03:00
|
|
|
|
void GB_set_vblank_callback(GB_gameboy_t *gb, GB_vblank_callback_t callback);
|
|
|
|
|
void GB_set_log_callback(GB_gameboy_t *gb, GB_log_callback_t callback);
|
|
|
|
|
void GB_set_input_callback(GB_gameboy_t *gb, GB_input_callback_t callback);
|
2016-07-18 00:39:43 +03:00
|
|
|
|
void GB_set_async_input_callback(GB_gameboy_t *gb, GB_input_callback_t callback);
|
2016-06-18 20:29:11 +03:00
|
|
|
|
void GB_set_rgb_encode_callback(GB_gameboy_t *gb, GB_rgb_encode_callback_t callback);
|
2016-07-18 22:05:11 +03:00
|
|
|
|
void GB_set_infrared_callback(GB_gameboy_t *gb, GB_infrared_callback_t callback);
|
2016-10-22 15:37:03 +03:00
|
|
|
|
void GB_set_rumble_callback(GB_gameboy_t *gb, GB_rumble_callback_t callback);
|
2019-06-07 18:27:25 +03:00
|
|
|
|
void GB_set_update_input_hint_callback(GB_gameboy_t *gb, GB_update_input_hint_callback_t callback);
|
2020-01-29 20:29:30 +02:00
|
|
|
|
/* Called when a new boot ROM is needed. The callback should call GB_load_boot_rom or GB_load_boot_rom_from_buffer */
|
|
|
|
|
void GB_set_boot_rom_load_callback(GB_gameboy_t *gb, GB_boot_rom_load_callback_t callback);
|
2021-12-10 19:42:47 +02:00
|
|
|
|
|
|
|
|
|
void GB_set_execution_callback(GB_gameboy_t *gb, GB_execution_callback_t callback);
|
2021-12-10 19:49:52 +02:00
|
|
|
|
void GB_set_lcd_line_callback(GB_gameboy_t *gb, GB_lcd_line_callback_t callback);
|
|
|
|
|
|
2020-01-29 14:19:11 +02:00
|
|
|
|
void GB_set_palette(GB_gameboy_t *gb, const GB_palette_t *palette);
|
2021-12-29 13:08:46 +02:00
|
|
|
|
const GB_palette_t *GB_get_palette(GB_gameboy_t *gb);
|
2016-11-12 01:58:53 +02:00
|
|
|
|
|
|
|
|
|
/* These APIs are used when using internal clock */
|
2019-03-15 14:36:10 +02:00
|
|
|
|
void GB_set_serial_transfer_bit_start_callback(GB_gameboy_t *gb, GB_serial_transfer_bit_start_callback_t callback);
|
|
|
|
|
void GB_set_serial_transfer_bit_end_callback(GB_gameboy_t *gb, GB_serial_transfer_bit_end_callback_t callback);
|
2016-11-12 01:58:53 +02:00
|
|
|
|
|
|
|
|
|
/* These APIs are used when using external clock */
|
2019-03-15 14:36:10 +02:00
|
|
|
|
bool GB_serial_get_data_bit(GB_gameboy_t *gb);
|
|
|
|
|
void GB_serial_set_data_bit(GB_gameboy_t *gb, bool data);
|
2017-04-19 21:55:58 +03:00
|
|
|
|
|
2017-01-13 21:27:37 +02:00
|
|
|
|
void GB_disconnect_serial(GB_gameboy_t *gb);
|
2019-07-15 22:01:41 +03:00
|
|
|
|
|
2020-05-23 14:50:54 +03:00
|
|
|
|
/* For cartridges with an alarm clock */
|
|
|
|
|
unsigned GB_time_to_alarm(GB_gameboy_t *gb); // 0 if no alarm
|
2021-12-11 02:51:21 +02:00
|
|
|
|
|
2021-11-12 17:44:51 +02:00
|
|
|
|
/* For cartridges motion controls */
|
|
|
|
|
bool GB_has_accelerometer(GB_gameboy_t *gb);
|
2021-11-14 13:18:58 +02:00
|
|
|
|
// In units of g (gravity's acceleration).
|
|
|
|
|
// Values within ±4 recommended
|
2021-11-12 17:44:51 +02:00
|
|
|
|
void GB_set_accelerometer_values(GB_gameboy_t *gb, double x, double y);
|
|
|
|
|
|
2019-07-15 22:01:41 +03:00
|
|
|
|
/* For integration with SFC/SNES emulators */
|
|
|
|
|
void GB_set_joyp_write_callback(GB_gameboy_t *gb, GB_joyp_write_callback_t callback);
|
2019-07-16 20:44:27 +03:00
|
|
|
|
void GB_set_icd_pixel_callback(GB_gameboy_t *gb, GB_icd_pixel_callback_t callback);
|
|
|
|
|
void GB_set_icd_hreset_callback(GB_gameboy_t *gb, GB_icd_hreset_callback_t callback);
|
2019-07-15 23:02:58 +03:00
|
|
|
|
void GB_set_icd_vreset_callback(GB_gameboy_t *gb, GB_icd_vreset_callback_t callback);
|
|
|
|
|
|
2018-02-10 23:30:30 +02:00
|
|
|
|
uint32_t GB_get_clock_rate(GB_gameboy_t *gb);
|
2021-02-25 22:12:14 +02:00
|
|
|
|
uint32_t GB_get_unmultiplied_clock_rate(GB_gameboy_t *gb);
|
2018-02-10 23:30:30 +02:00
|
|
|
|
void GB_set_clock_multiplier(GB_gameboy_t *gb, double multiplier);
|
2018-11-15 00:21:21 +02:00
|
|
|
|
|
2019-05-18 18:45:31 +03:00
|
|
|
|
unsigned GB_get_screen_width(GB_gameboy_t *gb);
|
|
|
|
|
unsigned GB_get_screen_height(GB_gameboy_t *gb);
|
2019-06-19 23:49:43 +03:00
|
|
|
|
double GB_get_usual_frame_rate(GB_gameboy_t *gb);
|
2018-12-15 18:55:41 +02:00
|
|
|
|
unsigned GB_get_player_count(GB_gameboy_t *gb);
|
2021-05-30 20:55:04 +03:00
|
|
|
|
|
|
|
|
|
/* Handy ROM info APIs */
|
|
|
|
|
// `title` must be at least 17 bytes in size
|
|
|
|
|
void GB_get_rom_title(GB_gameboy_t *gb, char *title);
|
|
|
|
|
uint32_t GB_get_rom_crc32(GB_gameboy_t *gb);
|
2021-10-17 20:04:31 +03:00
|
|
|
|
|
|
|
|
|
#ifdef GB_INTERNAL
|
2021-11-07 14:13:52 +02:00
|
|
|
|
internal void GB_borrow_sgb_border(GB_gameboy_t *gb);
|
2021-12-26 15:50:24 +02:00
|
|
|
|
internal void GB_update_clock_rate(GB_gameboy_t *gb);
|
2021-10-17 20:04:31 +03:00
|
|
|
|
#endif
|
2021-04-16 00:35:54 +03:00
|
|
|
|
|
2016-06-18 20:29:11 +03:00
|
|
|
|
#endif /* GB_h */
|