#include #include #include #include "common.h" #include "libco.h" #include "libretro.h" #include "memmap.h" #if defined(VITA) && defined(HAVE_DYNAREC) #include static int translation_caches_inited = 0; static inline int align(int x, int n) { return (((x >> n) + 1) << n ); } #define FOUR_KB_ALIGN(x) align(x, 12) #define MB_ALIGN(x) align(x, 20) int _newlib_vm_size_user = ROM_TRANSLATION_CACHE_SIZE + RAM_TRANSLATION_CACHE_SIZE + BIOS_TRANSLATION_CACHE_SIZE; int getVMBlock(); #endif #if defined(_3DS) void* linearMemAlign(size_t size, size_t alignment); void linearFree(void* mem); #if defined(HAVE_DYNAREC) #include #include "3ds/3ds_utils.h" #define MEMOP_PROT 6 #define MEMOP_MAP 4 #define MEMOP_UNMAP 5 int32_t svcDuplicateHandle(uint32_t* out, uint32_t original); int32_t svcCloseHandle(uint32_t handle); int32_t svcControlProcessMemory(uint32_t process, void* addr0, void* addr1, uint32_t size, uint32_t type, uint32_t perm); static int translation_caches_inited = 0; #endif #endif #ifndef MAX_PATH #define MAX_PATH (512) #endif static retro_log_printf_t log_cb; static retro_video_refresh_t video_cb; static retro_input_poll_t input_poll_cb; static retro_environment_t environ_cb; struct retro_perf_callback perf_cb; static cothread_t main_thread; static cothread_t cpu_thread; int dynarec_enable; u32 idle_loop_target_pc = 0xFFFFFFFF; u32 iwram_stack_optimize = 1; u32 translation_gate_target_pc[MAX_TRANSLATION_GATES]; u32 translation_gate_targets = 0; void switch_to_main_thread(void) { co_switch(main_thread); } static inline void switch_to_cpu_thread(void) { co_switch(cpu_thread); } static void cpu_thread_entry(void) { #ifdef HAVE_DYNAREC if (dynarec_enable) execute_arm_translate(execute_cycles); #endif execute_arm(execute_cycles); } static inline void init_context_switch(void) { main_thread = co_active(); cpu_thread = co_create(0x20000, cpu_thread_entry); } static inline void deinit_context_switch(void) { co_delete(cpu_thread); } #ifdef PERF_TEST extern struct retro_perf_callback perf_cb; #define RETRO_PERFORMANCE_INIT(X) \ static struct retro_perf_counter X = {#X}; \ do { \ if (!(X).registered) \ perf_cb.perf_register(&(X)); \ } while(0) #define RETRO_PERFORMANCE_START(X) perf_cb.perf_start(&(X)) #define RETRO_PERFORMANCE_STOP(X) perf_cb.perf_stop(&(X)) #else #define RETRO_PERFORMANCE_INIT(X) #define RETRO_PERFORMANCE_START(X) #define RETRO_PERFORMANCE_STOP(X) #endif void retro_get_system_info(struct retro_system_info* info) { info->library_name = "gpSP"; #ifndef GIT_VERSION #define GIT_VERSION "" #endif info->library_version = "v0.91" GIT_VERSION; info->need_fullpath = true; info->block_extract = false; info->valid_extensions = "gba|bin|agb|gbz" ; } void retro_get_system_av_info(struct retro_system_av_info* info) { info->geometry.base_width = GBA_SCREEN_WIDTH; info->geometry.base_height = GBA_SCREEN_HEIGHT; info->geometry.max_width = GBA_SCREEN_WIDTH; info->geometry.max_height = GBA_SCREEN_HEIGHT; info->geometry.aspect_ratio = 0; // 59.72750057 hz info->timing.fps = ((float) GBC_BASE_RATE) / (308 * 228 * 4); info->timing.sample_rate = GBA_SOUND_FREQUENCY; } void retro_init(void) { #if defined(_3DS) && defined(HAVE_DYNAREC) if (__ctr_svchax && !translation_caches_inited) { uint32_t currentHandle; rom_translation_cache_ptr = memalign(0x1000, ROM_TRANSLATION_CACHE_SIZE); ram_translation_cache_ptr = memalign(0x1000, RAM_TRANSLATION_CACHE_SIZE); bios_translation_cache_ptr = memalign(0x1000, BIOS_TRANSLATION_CACHE_SIZE); svcDuplicateHandle(¤tHandle, 0xFFFF8001); svcControlProcessMemory(currentHandle, rom_translation_cache, rom_translation_cache_ptr, ROM_TRANSLATION_CACHE_SIZE, MEMOP_MAP, 0b111); svcControlProcessMemory(currentHandle, ram_translation_cache, ram_translation_cache_ptr, RAM_TRANSLATION_CACHE_SIZE, MEMOP_MAP, 0b111); svcControlProcessMemory(currentHandle, bios_translation_cache, bios_translation_cache_ptr, BIOS_TRANSLATION_CACHE_SIZE, MEMOP_MAP, 0b111); svcCloseHandle(currentHandle); rom_translation_ptr = rom_translation_cache; ram_translation_ptr = ram_translation_cache; bios_translation_ptr = bios_translation_cache; ctr_flush_invalidate_cache(); translation_caches_inited = 1; } #endif #if defined(VITA) && defined(HAVE_DYNAREC) if(!translation_caches_inited){ void* currentHandle; sceBlock = getVMBlock(); if (sceBlock < 0) { return; } // get base address int ret = sceKernelGetMemBlockBase(sceBlock, ¤tHandle); if (ret < 0) { return; } rom_translation_cache = (u8*)currentHandle; ram_translation_cache = rom_translation_cache + ROM_TRANSLATION_CACHE_SIZE; bios_translation_cache = ram_translation_cache + RAM_TRANSLATION_CACHE_SIZE; rom_translation_ptr = rom_translation_cache; ram_translation_ptr = ram_translation_cache; bios_translation_ptr = bios_translation_cache; sceKernelOpenVMDomain(); translation_caches_inited = 1; } #endif if (!gamepak_rom) init_gamepak_buffer(); init_sound(1); if(!gba_screen_pixels) #ifdef _3DS gba_screen_pixels = (uint16_t*)linearMemAlign(GBA_SCREEN_PITCH * GBA_SCREEN_HEIGHT * sizeof(uint16_t), 128); #else gba_screen_pixels = (uint16_t*)malloc(GBA_SCREEN_PITCH * GBA_SCREEN_HEIGHT * sizeof(uint16_t)); #endif } void retro_deinit(void) { perf_cb.perf_log(); memory_term(); #if defined(HAVE_MMAP) && defined(HAVE_DYNAREC) munmap(rom_translation_cache, ROM_TRANSLATION_CACHE_SIZE); munmap(ram_translation_cache, RAM_TRANSLATION_CACHE_SIZE); munmap(bios_translation_cache, BIOS_TRANSLATION_CACHE_SIZE); #endif #if defined(_3DS) && defined(HAVE_DYNAREC) if (__ctr_svchax && translation_caches_inited) { uint32_t currentHandle; svcDuplicateHandle(¤tHandle, 0xFFFF8001); svcControlProcessMemory(currentHandle, rom_translation_cache, rom_translation_cache_ptr, ROM_TRANSLATION_CACHE_SIZE, MEMOP_UNMAP, 0b111); svcControlProcessMemory(currentHandle, ram_translation_cache, ram_translation_cache_ptr, RAM_TRANSLATION_CACHE_SIZE, MEMOP_UNMAP, 0b111); svcControlProcessMemory(currentHandle, bios_translation_cache, bios_translation_cache_ptr, BIOS_TRANSLATION_CACHE_SIZE, MEMOP_UNMAP, 0b111); svcCloseHandle(currentHandle); free(rom_translation_cache_ptr); free(ram_translation_cache_ptr); free(bios_translation_cache_ptr); translation_caches_inited = 0; } #endif #if defined(VITA) && defined(HAVE_DYNAREC) if(translation_caches_inited){ translation_caches_inited = 0; } #endif #ifdef _3DS linearFree(gba_screen_pixels); #else free(gba_screen_pixels); #endif gba_screen_pixels = NULL; } static retro_time_t retro_perf_dummy_get_time_usec() { return 0; } static retro_perf_tick_t retro_perf_dummy_get_counter() { return 0; } static uint64_t retro_perf_dummy_get_cpu_features() { return 0; } static void retro_perf_dummy_log() {} static void retro_perf_dummy_counter(struct retro_perf_counter *counter) {}; void retro_set_environment(retro_environment_t cb) { struct retro_log_callback log; static struct retro_variable vars[] = { #ifdef HAVE_DYNAREC { "gpsp_drc", "Dynamic recompiler (restart); enabled|disabled" }, #endif { NULL, NULL }, }; #if defined(_3DS) && (HAVE_DYNAREC) if(!__ctr_svchax) { vars[0].key = 0; vars[0].value = NULL; } #endif environ_cb = cb; if (environ_cb(RETRO_ENVIRONMENT_GET_LOG_INTERFACE, &log)) log_cb = log.log; else log_cb = NULL; perf_cb = (struct retro_perf_callback){ retro_perf_dummy_get_time_usec, retro_perf_dummy_get_counter, retro_perf_dummy_get_cpu_features, retro_perf_dummy_counter, retro_perf_dummy_counter, retro_perf_dummy_counter, retro_perf_dummy_log, }; environ_cb(RETRO_ENVIRONMENT_GET_PERF_INTERFACE, &perf_cb); environ_cb(RETRO_ENVIRONMENT_SET_VARIABLES, (void*)vars); } void retro_set_video_refresh(retro_video_refresh_t cb) { video_cb = cb; } void retro_set_input_poll(retro_input_poll_t cb) { input_poll_cb = cb; } void retro_set_controller_port_device(unsigned port, unsigned device) {} void retro_reset(void) { deinit_context_switch(); update_backup(); reset_gba(); init_context_switch(); } size_t retro_serialize_size(void) { return GBA_STATE_MEM_SIZE; } bool retro_serialize(void* data, size_t size) { if (size != GBA_STATE_MEM_SIZE) return false; memset (data,0, GBA_STATE_MEM_SIZE); gba_save_state(data); return true; } bool retro_unserialize(const void* data, size_t size) { if (size != GBA_STATE_MEM_SIZE) return false; gba_load_state(data); return true; } void retro_cheat_reset(void) { } void retro_cheat_set(unsigned index, bool enabled, const char* code) {} void error_msg(const char* text) { if (log_cb) log_cb(RETRO_LOG_ERROR, "[gpSP]: %s\n", text); } void info_msg(const char* text) { if (log_cb) log_cb(RETRO_LOG_INFO, "[gpSP]: %s\n", text); } static void extract_directory(char* buf, const char* path, size_t size) { strncpy(buf, path, size - 1); buf[size - 1] = '\0'; char* base = strrchr(buf, '/'); if (base) *base = '\0'; else strncpy(buf, ".", size); } static void check_variables(int started_from_load) { #ifdef HAVE_DYNAREC struct retro_variable var; var.key = "gpsp_drc"; var.value = NULL; if (environ_cb(RETRO_ENVIRONMENT_GET_VARIABLE, &var) && var.value) { if (started_from_load) { if (strcmp(var.value, "disabled") == 0) dynarec_enable = 0; else if (strcmp(var.value, "enabled") == 0) dynarec_enable = 1; } } else dynarec_enable = 1; #endif } static void set_input_descriptors() { struct retro_input_descriptor descriptors[] = { { 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_LEFT, "D-Pad Left" }, { 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_UP, "D-Pad Up" }, { 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_DOWN, "D-Pad Down" }, { 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_RIGHT, "D-Pad Right" }, { 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_B, "B" }, { 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_A, "A" }, { 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_SELECT, "Select" }, { 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_START, "Start" }, { 0 }, }; environ_cb(RETRO_ENVIRONMENT_SET_INPUT_DESCRIPTORS, descriptors); } bool retro_load_game(const struct retro_game_info* info) { if (!info) return false; check_variables(1); set_input_descriptors(); #if defined(HAVE_DYNAREC) if (dynarec_enable) { #if defined(HAVE_MMAP) rom_translation_cache = mmap(NULL, ROM_TRANSLATION_CACHE_SIZE, PROT_READ | PROT_WRITE | PROT_EXEC, MAP_ANON | MAP_PRIVATE, -1, 0); ram_translation_cache = mmap(NULL, RAM_TRANSLATION_CACHE_SIZE, PROT_READ | PROT_WRITE | PROT_EXEC, MAP_ANON | MAP_PRIVATE, -1, 0); bios_translation_cache = mmap(NULL, BIOS_TRANSLATION_CACHE_SIZE, PROT_READ | PROT_WRITE | PROT_EXEC, MAP_ANON | MAP_PRIVATE, -1, 0); rom_translation_ptr = rom_translation_cache; ram_translation_ptr = ram_translation_cache; bios_translation_ptr = bios_translation_cache; #elif defined(_3DS) dynarec_enable = __ctr_svchax; rom_translation_ptr = rom_translation_cache; ram_translation_ptr = ram_translation_cache; bios_translation_ptr = bios_translation_cache; #elif defined(VITA) dynarec_enable = 1; rom_translation_ptr = rom_translation_cache; ram_translation_ptr = ram_translation_cache; bios_translation_ptr = bios_translation_cache; #endif } else dynarec_enable = 0; #else dynarec_enable = 0; #endif char filename_bios[MAX_PATH]; const char* dir = NULL; enum retro_pixel_format fmt = RETRO_PIXEL_FORMAT_RGB565; if (!environ_cb(RETRO_ENVIRONMENT_SET_PIXEL_FORMAT, &fmt)) info_msg("RGB565 is not supported."); extract_directory(main_path, info->path, sizeof(main_path)); if (environ_cb(RETRO_ENVIRONMENT_GET_SYSTEM_DIRECTORY, &dir) && dir) strncpy(filename_bios, dir, sizeof(filename_bios)); else strncpy(filename_bios, main_path, sizeof(filename_bios)); strncat(filename_bios, "/gba_bios.bin", sizeof(filename_bios)); if (environ_cb(RETRO_ENVIRONMENT_GET_SAVE_DIRECTORY, &dir) && dir) strncpy(save_path, dir, sizeof(save_path)); else strncpy(save_path, main_path, sizeof(save_path)); if (load_bios(filename_bios) != 0) { error_msg("Could not load BIOS image file."); return false; } if (bios_rom[0] != 0x18) { info_msg("You have an incorrect BIOS image."); info_msg("While many games will work fine, some will not."); info_msg("It is strongly recommended that you obtain the correct BIOS file."); } gamepak_filename[0] = 0; if (load_gamepak(info, info->path) != 0) { error_msg("Could not load the game file."); return false; } reset_gba(); init_context_switch(); return true; } bool retro_load_game_special(unsigned game_type, const struct retro_game_info* info, size_t num_info) { return false; } void retro_unload_game(void) { deinit_context_switch(); update_backup(); } unsigned retro_get_region(void) { return RETRO_REGION_NTSC; } void* retro_get_memory_data(unsigned id) { // switch (id) // { // case RETRO_MEMORY_SAVE_RAM: // return gamepak_backup; // } return 0; } size_t retro_get_memory_size(unsigned id) { // switch (id) // { // case RETRO_MEMORY_SAVE_RAM: // switch(backup_type) // { // case BACKUP_SRAM: // return sram_size; // case BACKUP_FLASH: // return flash_size; // case BACKUP_EEPROM: // return eeprom_size; // case BACKUP_NONE: // return 0x0; // default: // return 0x8000; // } // } return 0; } void retro_run(void) { bool updated = false; input_poll_cb(); switch_to_cpu_thread(); update_input(); render_audio(); video_cb(gba_screen_pixels, GBA_SCREEN_WIDTH, GBA_SCREEN_HEIGHT, GBA_SCREEN_PITCH * 2); if (environ_cb(RETRO_ENVIRONMENT_GET_VARIABLE_UPDATE, &updated) && updated) check_variables(0); } unsigned retro_api_version(void) { return RETRO_API_VERSION; }