-
Notifications
You must be signed in to change notification settings - Fork 59
/
sb_types.h
348 lines (330 loc) · 9.98 KB
/
sb_types.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
/*****************************************************************************
*
* SkyBoy GB Emulator
*
* Copyright (c) 2021 Skyler "Sky" Saleh
*
**/
#ifndef SB_TYPES_H
#define SB_TYPES_H 1
#include <stdint.h>
#include <stdbool.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <ctype.h>
#if defined(__GNUC__) || defined(__clang__)
#define FORCE_INLINE inline __attribute__((always_inline))
#elif defined(_MSC_VER)
#define FORCE_INLINE __forceinline
#else
#define FORCE_INLINE inline
#endif
// Macro for hinting that an expression is likely to be false.
#if defined(__GNUC__) || defined(__clang__)
#define SB_UNLIKELY(x) __builtin_expect(!!(x), 0)
#else
#define SB_UNLIKELY(x) (x)
#endif // defined(COMPILER_GCC)
#if defined(COMPILER_GCC) || defined(__clang__)
#define SB_LIKELY(x) __builtin_expect(!!(x), 1)
#else
#define SB_LIKELY(x) (x)
#endif // defined(COMPILER_GCC)
#define SB_FILE_PATH_SIZE 1024
#define MAX_CARTRIDGE_SIZE 8 * 1024 * 1024
#define MAX_CARTRIDGE_RAM 128 * 1024
#define SB_U16_LO(A) ((A)&0xff)
#define SB_U16_LO_SET(A,VAL) A = (((A)&0xff00)|(((int)(VAL))&0xff))
#define SB_U16_HI(A) ((A >> 8) & 0xff)
#define SB_U16_HI_SET(A,VAL) A = (((A)&0x00ff)|((((int)(VAL))&0xff)<<8))
// Extract bits from a bitfield
#define SB_BFE(VALUE, BITOFFSET, SIZE) \
(((VALUE) >> (BITOFFSET)) & ((1llu << (SIZE)) - 1))
#define SB_BIT_TEST(VALUE,BITOFFSET) ((VALUE)&(1u<<(BITOFFSET)))
#define SB_MODE_PAUSE 0
#define SB_MODE_RESET 1
#define SB_MODE_RUN 2
#define SB_MODE_STEP 3
#define SB_MODE_REWIND 4
#define SB_LCD_W 160
#define SB_LCD_H 144
#define SB_PPU_BG_COLOR_PALETTES 64
#define SB_PPU_SPRITE_COLOR_PALETTES 64
#define SB_VRAM_BANK_SIZE 8192
#define SB_VRAM_NUM_BANKS 2
#define SB_WRAM_BANK_SIZE 4096
#define SB_WRAM_NUM_BANKS 8
#define SB_GB 0
#define SB_GBC 1
#define SB_GBC_GB_BACK_COMPAT 2
#define SE_BIND_KEYBOARD 0
#define SE_BIND_KEY 1
#define SE_BIND_ANALOG 2
#define SE_KEY_A 0
#define SE_KEY_B 1
#define SE_KEY_X 2
#define SE_KEY_Y 3
#define SE_KEY_UP 4
#define SE_KEY_DOWN 5
#define SE_KEY_LEFT 6
#define SE_KEY_RIGHT 7
#define SE_KEY_L 8
#define SE_KEY_R 9
#define SE_KEY_START 10
#define SE_KEY_SELECT 11
#define SE_KEY_FOLD_SCREEN 12
#define SE_KEY_PEN_DOWN 13
#define SE_KEY_EMU_PAUSE 14
#define SE_KEY_EMU_REWIND 15
#define SE_KEY_EMU_FF_2X 16
#define SE_KEY_EMU_FF_MAX 17
#define SE_KEY_CAPTURE_STATE(A) (18+(A)*2)
#define SE_KEY_RESTORE_STATE(A) (18+(A)*2+1)
#define SE_KEY_RESET_GAME 26
#define SE_KEY_TURBO_A 27
#define SE_KEY_TURBO_B 28
#define SE_KEY_TURBO_X 29
#define SE_KEY_TURBO_Y 30
#define SE_KEY_TURBO_L 31
#define SE_KEY_TURBO_R 32
#define SE_KEY_SOLAR_P 33
#define SE_KEY_SOLAR_M 34
#define SE_KEY_TOGGLE_FULLSCREEN 35
#define SE_NUM_KEYBINDS 36
//Should be power of 2 for perf, 8192 samples gives ~85ms maximal latency for 48kHz
#define SB_AUDIO_RING_BUFFER_SIZE (2048*8)
#define SYSTEM_UNKNOWN 0
#define SYSTEM_GB 1
#define SYSTEM_GBA 2
#define SYSTEM_NDS 3
#define SE_RPT4 for(int r=0;r<4;++r)
#define SE_RPT3 for(int r=0;r<3;++r)
#define SE_RPT2 for(int r=0;r<2;++r)
typedef struct{
float inputs[SE_NUM_KEYBINDS];
float touch_pos[2];
float rumble;
float solar_sensor;
} sb_joy_t;
typedef struct{
int16_t data[SB_AUDIO_RING_BUFFER_SIZE];
uint32_t read_ptr;
uint32_t write_ptr;
}sb_ring_buffer_t;
static FORCE_INLINE uint32_t sb_ring_buffer_size(sb_ring_buffer_t* buff){
if(buff->read_ptr>SB_AUDIO_RING_BUFFER_SIZE){
buff->write_ptr-=SB_AUDIO_RING_BUFFER_SIZE;
buff->read_ptr-=SB_AUDIO_RING_BUFFER_SIZE;
}
uint32_t v = (buff->write_ptr-buff->read_ptr);
v= v%SB_AUDIO_RING_BUFFER_SIZE;
return v;
}
typedef struct {
int run_mode; // [0: Reset, 1: Pause, 2: Run, 3: Step ]
int step_instructions; // Number of instructions to advance while stepping
int step_frames;
int pc_breakpoint; // PC to run until
bool rom_loaded;
int system; // Enum to emulated system Ex. SYSTEM_GB, SYSTEM_GBA
sb_joy_t joy;
sb_joy_t prev_frame_joy; //Used for tracking button press changes in a frame
int frame;
bool render_frame;
sb_ring_buffer_t audio_ring_buff;
float audio_channel_output[16];
float mix_l_volume, mix_r_volume;
float master_volume;
int cmd_line_arg_count;
char** cmd_line_args;
//Temporary storage for use by cores that persists across frames but not in save states
//or rewind buffers
uint32_t frames_since_rewind_push;
char save_data_base_path[SB_FILE_PATH_SIZE];
char save_file_path[SB_FILE_PATH_SIZE];
float screen_ghosting_strength; //0 = off 1 = full strength
size_t rom_size;
uint8_t *rom_data;
char rom_path[SB_FILE_PATH_SIZE];
bool force_dmg_mode;
} sb_emu_state_t;
typedef struct{
bool read_since_reset;
bool read_in_tick;
bool write_since_reset;
bool write_in_tick;
bool trigger_breakpoint;
}sb_debug_mmio_access_t;
typedef struct{
uint32_t addr;
const char * name;
struct{
uint8_t start;
uint8_t size;
const char* name;
} bits[32];
}mmio_reg_t;
static inline float sb_random_float(float min, float max){
float v = rand()/(float)RAND_MAX;
return min + v*(max-min);
}
static inline bool sb_path_has_file_ext(const char * path, const char * ext){
if(ext[0]=='*')ext++;
if(ext[0]=='.')ext++;
if(ext[0]=='*')return true;
int ext_len = strlen(ext);
int path_len = strlen(path);
if(path_len<ext_len)return false;
for(int i=0;i<ext_len;++i){
if(tolower(path[path_len-ext_len+i])!=tolower(ext[i]))return false;
}
return true;
}
static bool sb_file_exists(const char * path){
FILE * f = fopen(path,"r");
if(f){fclose(f);return true;}
printf("%s does not exist\n",path);
return false;
}
static bool sb_load_file_data_into_buffer(const char* path, void* buffer, size_t buffer_size){
FILE *f = fopen(path, "rb");
if(f){
size_t size = 0;
fseek(f, 0,SEEK_END);
size = ftell(f);
fseek(f, 0,SEEK_SET);
if(size!=buffer_size){
printf("%s is the wrong size. Expected: %zu got: %zu\n",path,buffer_size,size);
return false;
}
size =fread(buffer, 1, size, f);
printf("Loaded file %s file_size %zu\n",path,size);
fclose(f);
return true;
}else{
printf("Failed to open file %s\n",path);
}
return false;
}
static uint8_t* sb_load_file_data(const char* path, size_t *file_size){
FILE *f = fopen(path, "rb");
if(file_size)*file_size = 0;
if(f){
size_t size = 0;
fseek(f, 0,SEEK_END);
size = ftell(f);
fseek(f, 0,SEEK_SET);
uint8_t *data = (uint8_t*)malloc(size);
if(!data)return NULL;
size =fread(data, 1, size, f);
if(size==EOF){size = 0; free(data);}
if(file_size)*file_size = size;
printf("Loaded file %s file_size %zu\n",path,*file_size);
return data;
}else{
printf("Failed to open file %s\n",path);
}
return NULL;
}
static bool sb_save_file_data(const char* path, uint8_t* data, size_t file_size){
FILE *f = fopen(path, "wb");
size_t written = -1;
if(f){
written = fwrite(data,1,file_size, f);
fclose(f);
}
if(written!=file_size){
printf("Error failed to save: %s (wrote: %zu out of %zu)\n",path,written,file_size);
}else{
printf("Saved: %s (size: %zu)\n",path,written);
}
return written ==file_size;
}
static void sb_free_file_data(uint8_t* data){
if(data)free(data);
}
static const char* sb_parent_path(const char* path){
static char tmp_path[SB_FILE_PATH_SIZE];
strncpy(tmp_path,path,SB_FILE_PATH_SIZE-1);
tmp_path[SB_FILE_PATH_SIZE-1]='\0';
size_t sz = strlen(tmp_path);
while(sz>1){
char c = tmp_path[sz-1];
bool is_slash = c=='\\'||c=='/';
if(!is_slash){break;}
tmp_path[--sz]='\0';
}
if(sz){
bool found_dir = false;
while(sz--){
char c = tmp_path[sz];
bool is_slash = c=='\\'||c=='/';
if(found_dir&&!is_slash)break;
if(is_slash)found_dir = true;
tmp_path[sz]='\0';
}
}
return tmp_path;
}
static const char *sb_get_home_path(){
static char homedir[SB_FILE_PATH_SIZE];
#ifdef PLATFORM_ANDROID
return "/sdcard/";
#elif defined(_WIN32)
snprintf(homedir, SB_FILE_PATH_SIZE, "%s%s", getenv("HOMEDRIVE"), getenv("HOMEPATH"));
#else
snprintf(homedir, SB_FILE_PATH_SIZE, "%s", getenv("HOME"));
#endif
return homedir;
}
static void sb_breakup_path(const char* path, const char** base_path, const char** file_name, const char** ext){
static char tmp_path[SB_FILE_PATH_SIZE];
strncpy(tmp_path,path,SB_FILE_PATH_SIZE-1);
tmp_path[SB_FILE_PATH_SIZE-1]='\0';
size_t sz = strlen(tmp_path);
*base_path = "";
*file_name = tmp_path;
*ext = "";
// Search for end of extension or start of base path
bool found_ext = false;
while(sz--){
if(tmp_path[sz]=='.'&&!found_ext){
tmp_path[sz] = '\0';
found_ext = true;
*ext = tmp_path + sz+1;
*file_name = tmp_path;
}
if(tmp_path[sz]=='\\'||tmp_path[sz]=='/'){
tmp_path[sz]='\0';
*file_name = tmp_path+sz+1;
*base_path = tmp_path;
break;
}
}
}
static void se_join_path(char * dest_path, int dest_size, const char * base_path, const char* file_name, const char* add_extension){
char * seperator = base_path[0]==0? "" : "/";
if(strlen(base_path)!=0){
char last_base_char = base_path[strlen(base_path)-1];
if(last_base_char=='/'||last_base_char=='\\')seperator="";
}
if(add_extension){
const char * ext_sep = add_extension[0]=='.' ? "": ".";
snprintf(dest_path,dest_size,"%s%s%s%s%s",base_path, seperator, file_name,ext_sep,add_extension);
}else snprintf(dest_path,dest_size,"%s%s%s",base_path, seperator, file_name);
dest_path[dest_size-1]=0;
}
bool se_load_bios_file(const char* name, const char* base_path, const char* file_name, uint8_t * data, size_t data_size);
static FILE * se_load_log_file(const char* rom_path, const char* log_name){
bool loaded_bios=false;
const char* base, *file, *ext;
sb_breakup_path(rom_path, &base,&file, &ext);
static char log_path[SB_FILE_PATH_SIZE];
se_join_path(log_path,SB_FILE_PATH_SIZE,base,file,log_name);
log_path[SB_FILE_PATH_SIZE-1]=0;
FILE * f = fopen(log_path, "rb");
if(f)printf("Loaded log file:%s\n",log_path);
return f;
}
#endif