2023-10-10 21:12:53 +02:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include <libdragon.h>
|
|
|
|
|
2023-08-18 22:19:01 +02:00
|
|
|
#include "cart_load.h"
|
|
|
|
#include "path.h"
|
|
|
|
#include "utils/fs.h"
|
2023-10-10 21:12:53 +02:00
|
|
|
#include "utils/utils.h"
|
2023-08-18 22:19:01 +02:00
|
|
|
|
2023-10-10 21:12:53 +02:00
|
|
|
#ifndef SAVES_SUBDIRECTORY
|
2023-08-18 22:19:01 +02:00
|
|
|
#define SAVES_SUBDIRECTORY "saves"
|
2023-10-10 21:12:53 +02:00
|
|
|
#endif
|
|
|
|
#ifndef DDIPL_LOCATION
|
|
|
|
#define DDIPL_LOCATION "/menu/64ddipl"
|
|
|
|
#endif
|
|
|
|
#ifndef EMU_LOCATION
|
|
|
|
#define EMU_LOCATION "/menu/emulators"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
static bool is_64dd_connected (void) {
|
2024-04-26 23:04:19 +02:00
|
|
|
bool is_64dd_io_present = ((io_read(0x05000540) & 0x0000FFFF) == 0x0000);
|
|
|
|
bool is_64dd_ipl_present = (io_read(0x06001010) == 0x2129FFF8);
|
|
|
|
return (is_64dd_io_present || is_64dd_ipl_present);
|
2023-10-10 21:12:53 +02:00
|
|
|
}
|
2023-08-18 22:19:01 +02:00
|
|
|
|
2023-10-10 21:12:53 +02:00
|
|
|
static bool create_saves_subdirectory (path_t *path) {
|
|
|
|
path_t *save_folder_path = path_clone(path);
|
|
|
|
path_pop(save_folder_path);
|
|
|
|
path_push(save_folder_path, SAVES_SUBDIRECTORY);
|
2023-08-18 22:19:01 +02:00
|
|
|
bool error = directory_create(path_get(save_folder_path));
|
|
|
|
path_free(save_folder_path);
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2023-12-21 19:58:30 +01:00
|
|
|
static flashcart_save_type_t convert_save_type (rom_save_type_t save_type) {
|
|
|
|
switch (save_type) {
|
2023-10-10 21:12:53 +02:00
|
|
|
case SAVE_TYPE_EEPROM_4K: return FLASHCART_SAVE_TYPE_EEPROM_4K;
|
|
|
|
case SAVE_TYPE_EEPROM_16K: return FLASHCART_SAVE_TYPE_EEPROM_16K;
|
|
|
|
case SAVE_TYPE_SRAM: return FLASHCART_SAVE_TYPE_SRAM;
|
|
|
|
case SAVE_TYPE_SRAM_BANKED: return FLASHCART_SAVE_TYPE_SRAM_BANKED;
|
|
|
|
case SAVE_TYPE_SRAM_128K: return FLASHCART_SAVE_TYPE_SRAM_128K;
|
|
|
|
case SAVE_TYPE_FLASHRAM: return FLASHCART_SAVE_TYPE_FLASHRAM;
|
|
|
|
case SAVE_TYPE_FLASHRAM_PKST2: return FLASHCART_SAVE_TYPE_FLASHRAM_PKST2;
|
|
|
|
default: return FLASHCART_SAVE_TYPE_NONE;
|
2023-08-18 22:19:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
char *cart_load_convert_error_message (cart_load_err_t err) {
|
|
|
|
switch (err) {
|
2023-10-10 21:12:53 +02:00
|
|
|
case CART_LOAD_OK: return "Cart load OK";
|
|
|
|
case CART_LOAD_ERR_ROM_LOAD_FAIL: return "Error occured during ROM loading";
|
|
|
|
case CART_LOAD_ERR_SAVE_LOAD_FAIL: return "Error occured during save loading";
|
|
|
|
case CART_LOAD_ERR_64DD_PRESENT: return "64DD accessory is connected to the N64";
|
|
|
|
case CART_LOAD_ERR_64DD_IPL_NOT_FOUND: return "Required 64DD IPL file was not found";
|
|
|
|
case CART_LOAD_ERR_64DD_IPL_LOAD_FAIL: return "Error occured during 64DD IPL loading";
|
|
|
|
case CART_LOAD_ERR_64DD_DISK_LOAD_FAIL: return "Error occured during 64DD disk loading";
|
|
|
|
case CART_LOAD_ERR_EMU_NOT_FOUND: return "Required emulator file was not found";
|
|
|
|
case CART_LOAD_ERR_EMU_LOAD_FAIL: return "Error occured during emulator ROM loading";
|
|
|
|
case CART_LOAD_ERR_EMU_ROM_LOAD_FAIL: return "Error occured during emulated ROM loading";
|
|
|
|
case CART_LOAD_ERR_CREATE_SAVES_SUBDIR_FAIL: return "Couldn't create saves subdirectory";
|
|
|
|
case CART_LOAD_ERR_EXP_PAK_NOT_FOUND: return "Mandatory Expansion Pak accessory was not found";
|
|
|
|
case CART_LOAD_ERR_FUNCTION_NOT_SUPPORTED: return "Your flashcart doesn't support required functionality";
|
|
|
|
default: return "Unknown error [CART_LOAD]";
|
2023-08-18 22:19:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-10 21:12:53 +02:00
|
|
|
cart_load_err_t cart_load_n64_rom_and_save (menu_t *menu, flashcart_progress_callback_t progress) {
|
|
|
|
path_t *path = path_clone(menu->load.rom_path);
|
|
|
|
|
|
|
|
bool byte_swap = (menu->load.rom_info.endianness == ENDIANNESS_BYTE_SWAP);
|
2023-12-21 19:58:30 +01:00
|
|
|
flashcart_save_type_t save_type = convert_save_type(rom_info_get_save_type(&menu->load.rom_info));
|
2023-08-20 14:11:17 +02:00
|
|
|
|
2023-10-10 21:12:53 +02:00
|
|
|
menu->flashcart_err = flashcart_load_rom(path_get(path), byte_swap, progress);
|
|
|
|
if (menu->flashcart_err != FLASHCART_OK) {
|
2023-08-18 22:19:01 +02:00
|
|
|
path_free(path);
|
2023-10-10 21:12:53 +02:00
|
|
|
return CART_LOAD_ERR_ROM_LOAD_FAIL;
|
2023-08-18 22:19:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
path_ext_replace(path, "sav");
|
|
|
|
if (menu->settings.use_saves_folder) {
|
2023-10-10 21:12:53 +02:00
|
|
|
if ((save_type != FLASHCART_SAVE_TYPE_NONE) && create_saves_subdirectory(path)) {
|
2023-08-20 14:11:17 +02:00
|
|
|
path_free(path);
|
2023-10-10 21:12:53 +02:00
|
|
|
return CART_LOAD_ERR_CREATE_SAVES_SUBDIR_FAIL;
|
2023-08-20 14:11:17 +02:00
|
|
|
}
|
2023-10-10 21:12:53 +02:00
|
|
|
path_push_subdir(path, SAVES_SUBDIRECTORY);
|
2023-08-18 22:19:01 +02:00
|
|
|
}
|
|
|
|
|
2023-10-10 21:12:53 +02:00
|
|
|
menu->flashcart_err = flashcart_load_save(path_get(path), save_type);
|
|
|
|
if (menu->flashcart_err != FLASHCART_OK) {
|
2023-08-18 22:19:01 +02:00
|
|
|
path_free(path);
|
2023-10-10 21:12:53 +02:00
|
|
|
return CART_LOAD_ERR_SAVE_LOAD_FAIL;
|
2023-08-18 22:19:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
path_free(path);
|
|
|
|
|
|
|
|
return CART_LOAD_OK;
|
|
|
|
}
|
|
|
|
|
2023-10-10 21:12:53 +02:00
|
|
|
cart_load_err_t cart_load_64dd_ipl_and_disk (menu_t *menu, flashcart_progress_callback_t progress) {
|
|
|
|
if (!flashcart_has_feature(FLASHCART_FEATURE_64DD)) {
|
|
|
|
return CART_LOAD_ERR_FUNCTION_NOT_SUPPORTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_64dd_connected()) {
|
|
|
|
return CART_LOAD_ERR_64DD_PRESENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_memory_expanded()) {
|
|
|
|
return CART_LOAD_ERR_EXP_PAK_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
2024-04-24 02:45:09 +02:00
|
|
|
path_t *path = path_init(menu->storage_prefix, DDIPL_LOCATION);
|
2023-10-10 21:12:53 +02:00
|
|
|
flashcart_disk_parameters_t disk_parameters;
|
|
|
|
|
|
|
|
disk_parameters.development_drive = (menu->load.disk_info.region == DISK_REGION_DEVELOPMENT);
|
|
|
|
disk_parameters.disk_type = menu->load.disk_info.disk_type;
|
|
|
|
memcpy(disk_parameters.bad_system_area_lbas, menu->load.disk_info.bad_system_area_lbas, sizeof(disk_parameters.bad_system_area_lbas));
|
|
|
|
memcpy(disk_parameters.defect_tracks, menu->load.disk_info.defect_tracks, sizeof(disk_parameters.defect_tracks));
|
|
|
|
|
|
|
|
switch (menu->load.disk_info.region) {
|
|
|
|
case DISK_REGION_DEVELOPMENT:
|
|
|
|
path_push(path, "NDXJ0.n64");
|
|
|
|
break;
|
|
|
|
case DISK_REGION_JAPANESE:
|
|
|
|
path_push(path, "NDDJ2.n64");
|
|
|
|
break;
|
|
|
|
case DISK_REGION_USA:
|
|
|
|
path_push(path, "NDDE0.n64");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!file_exists(path_get(path))) {
|
|
|
|
path_free(path);
|
|
|
|
return CART_LOAD_ERR_64DD_IPL_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
menu->flashcart_err = flashcart_load_64dd_ipl(path_get(path), progress);
|
|
|
|
if (menu->flashcart_err != FLASHCART_OK) {
|
|
|
|
path_free(path);
|
|
|
|
return CART_LOAD_ERR_64DD_IPL_LOAD_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
path_free(path);
|
|
|
|
|
|
|
|
menu->flashcart_err = flashcart_load_64dd_disk(path_get(menu->load.disk_path), &disk_parameters);
|
|
|
|
if (menu->flashcart_err != FLASHCART_OK) {
|
|
|
|
return CART_LOAD_ERR_64DD_DISK_LOAD_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return CART_LOAD_OK;
|
|
|
|
}
|
|
|
|
|
2023-08-18 22:19:01 +02:00
|
|
|
cart_load_err_t cart_load_emulator (menu_t *menu, cart_load_emu_type_t emu_type, flashcart_progress_callback_t progress) {
|
2024-04-24 02:45:09 +02:00
|
|
|
path_t *path = path_init(menu->storage_prefix, EMU_LOCATION);
|
2023-10-10 21:12:53 +02:00
|
|
|
|
2023-08-18 22:19:01 +02:00
|
|
|
flashcart_save_type_t save_type = FLASHCART_SAVE_TYPE_NONE;
|
|
|
|
uint32_t emulated_rom_offset = 0x200000;
|
2023-08-20 20:26:48 +02:00
|
|
|
uint32_t emulated_file_offset = 0;
|
2023-08-18 22:19:01 +02:00
|
|
|
|
|
|
|
switch (emu_type) {
|
|
|
|
case CART_LOAD_EMU_TYPE_NES:
|
2023-08-20 20:58:34 +02:00
|
|
|
path_push(path, "neon64bu.rom");
|
2023-08-18 22:19:01 +02:00
|
|
|
save_type = FLASHCART_SAVE_TYPE_SRAM_BANKED;
|
|
|
|
break;
|
2023-08-20 20:58:34 +02:00
|
|
|
case CART_LOAD_EMU_TYPE_SNES:
|
|
|
|
path_push(path, "sodium64.z64");
|
|
|
|
save_type = FLASHCART_SAVE_TYPE_SRAM;
|
|
|
|
break;
|
2023-08-18 22:19:01 +02:00
|
|
|
case CART_LOAD_EMU_TYPE_GAMEBOY:
|
2023-08-20 20:58:34 +02:00
|
|
|
path_push(path, "gb.v64");
|
2023-08-18 22:19:01 +02:00
|
|
|
save_type = FLASHCART_SAVE_TYPE_FLASHRAM;
|
|
|
|
break;
|
|
|
|
case CART_LOAD_EMU_TYPE_GAMEBOY_COLOR:
|
2023-08-20 20:58:34 +02:00
|
|
|
path_push(path, "gbc.v64");
|
2023-08-18 22:19:01 +02:00
|
|
|
save_type = FLASHCART_SAVE_TYPE_FLASHRAM;
|
|
|
|
break;
|
Update emulator load display (#46)
<!--- Provide a general summary of your changes in the Title above -->
## Description
<!--- Describe your changes in detail -->
Update emulator load display to be, well, better.
Add TotalSMS (not tested, however, shows that the emulator is not
found.)
Fix rolling release for new forks.
## Motivation and Context
<!--- What does this sample do? What problem does it solve? -->
<!--- If it fixes/closes/resolves an open issue, please link to the
issue here -->
## How Has This Been Tested?
<!-- (if applicable) -->
<!--- Please describe in detail how you tested your sample/changes. -->
<!--- Include details of your testing environment, and the tests you ran
to -->
<!--- see how your change affects other areas of the code, etc. -->
## Screenshots
<!-- (if appropriate): -->
## Types of changes
<!--- What types of changes does your code introduce? Put an `x` in all
the boxes that apply: -->
- [ ] Improvement (non-breaking change that adds a new feature)
- [ ] Bug fix (fixes an issue)
- [ ] Breaking change (breaking change)
- [ ] Config and build (change in the configuration and build system,
has no impact on code or features)
## Checklist:
<!--- Go over all the following points, and put an `x` in all the boxes
that apply. -->
<!--- If you're unsure about any of these, don't hesitate to ask. We're
here to help! -->
- [ ] My code follows the code style of this project.
- [ ] My change requires a change to the documentation.
- [ ] I have updated the documentation accordingly.
- [ ] I have added tests to cover my changes.
- [ ] All new and existing tests passed.
<!--- It would be nice if you could sign off your contribution by
replacing the name with your GitHub user name and GitHub email contact.
-->
Signed-off-by: GITHUB_USER <GITHUB_USER_EMAIL>
2023-08-28 17:19:51 +02:00
|
|
|
case CART_LOAD_EMU_TYPE_SEGA_GENERIC_8BIT:
|
|
|
|
path_push(path, "TotalSMS.z64");
|
|
|
|
save_type = FLASHCART_SAVE_TYPE_SRAM;
|
|
|
|
break;
|
2023-08-18 22:19:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!file_exists(path_get(path))) {
|
|
|
|
path_free(path);
|
|
|
|
return CART_LOAD_ERR_EMU_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
2023-10-10 21:12:53 +02:00
|
|
|
menu->flashcart_err = flashcart_load_rom(path_get(path), false, progress);
|
|
|
|
if (menu->flashcart_err != FLASHCART_OK) {
|
2023-08-18 22:19:01 +02:00
|
|
|
path_free(path);
|
2023-10-10 21:12:53 +02:00
|
|
|
return CART_LOAD_ERR_EMU_LOAD_FAIL;
|
2023-08-18 22:19:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
path_free(path);
|
|
|
|
|
|
|
|
path = path_clone_push(menu->browser.directory, menu->browser.entry->name);
|
|
|
|
|
2023-08-20 20:58:34 +02:00
|
|
|
switch (emu_type) {
|
|
|
|
case CART_LOAD_EMU_TYPE_SNES:
|
2023-10-10 21:12:53 +02:00
|
|
|
// NOTE: The emulator expects the header to be removed from the ROM being uploaded.
|
2023-08-20 20:58:34 +02:00
|
|
|
emulated_file_offset = ((file_get_size(path_get(path)) & 0x3FF) == 0x200) ? 0x200 : 0;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2023-10-10 21:12:53 +02:00
|
|
|
menu->flashcart_err = flashcart_load_file(path_get(path), emulated_rom_offset, emulated_file_offset);
|
|
|
|
if (menu->flashcart_err != FLASHCART_OK) {
|
2023-08-18 22:19:01 +02:00
|
|
|
path_free(path);
|
2023-10-10 21:12:53 +02:00
|
|
|
return CART_LOAD_ERR_EMU_ROM_LOAD_FAIL;
|
2023-08-18 22:19:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
path_ext_replace(path, "sav");
|
|
|
|
if (menu->settings.use_saves_folder) {
|
2023-10-10 21:12:53 +02:00
|
|
|
if ((save_type != FLASHCART_SAVE_TYPE_NONE) && create_saves_subdirectory(path)) {
|
2023-08-20 14:11:17 +02:00
|
|
|
path_free(path);
|
2023-10-10 21:12:53 +02:00
|
|
|
return CART_LOAD_ERR_CREATE_SAVES_SUBDIR_FAIL;
|
2023-08-20 14:11:17 +02:00
|
|
|
}
|
2023-10-10 21:12:53 +02:00
|
|
|
path_push_subdir(path, SAVES_SUBDIRECTORY);
|
2023-08-18 22:19:01 +02:00
|
|
|
}
|
|
|
|
|
2023-10-10 21:12:53 +02:00
|
|
|
menu->flashcart_err = flashcart_load_save(path_get(path), save_type);
|
|
|
|
if (menu->flashcart_err != FLASHCART_OK) {
|
2023-08-18 22:19:01 +02:00
|
|
|
path_free(path);
|
2023-10-10 21:12:53 +02:00
|
|
|
return CART_LOAD_ERR_SAVE_LOAD_FAIL;
|
2023-08-18 22:19:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
path_free(path);
|
|
|
|
|
|
|
|
return CART_LOAD_OK;
|
|
|
|
}
|