diff --git a/Makefile b/Makefile index 3fc53828..67ce3a77 100644 --- a/Makefile +++ b/Makefile @@ -30,6 +30,7 @@ SRCS = \ flashcart/64drive/64drive.c \ flashcart/flashcart_utils.c \ flashcart/ed64/ed64_vseries.c \ + flashcart/ed64/ed64_xseries.c \ flashcart/flashcart.c \ flashcart/sc64/sc64_ll.c \ flashcart/sc64/sc64.c \ diff --git a/README.md b/README.md index 4336211a..4d00ac5a 100644 --- a/README.md +++ b/README.md @@ -126,7 +126,8 @@ The aim is to reach feature parity with [ED64-UnofficialOS](https://github.com/n Download the `OS64.v64` ROM from the latest [action run - assets] and place it in the `/ED64` folder. #### ED64 (X series) -X Series support is currently awaiting fixes. Please use the official [OS](https://krikzz.com/pub/support/everdrive-64/x-series/OS/) for now. +The aim is to reach feature parity with [OS](https://krikzz.com/pub/support/everdrive-64/x-series/OS/) for now. +Download the `OS64.v64` ROM from the latest [action run - assets] and place it in the `/ED64` folder. #### ED64 (P clone) Download the `OS64P.v64` ROM from the latest [action run - assets] and place it in the `/ED64P` folder. diff --git a/src/flashcart/ed64/ed64_vseries.c b/src/flashcart/ed64/ed64_vseries.c index 13ce9de6..3eca82f1 100644 --- a/src/flashcart/ed64/ed64_vseries.c +++ b/src/flashcart/ed64/ed64_vseries.c @@ -9,6 +9,7 @@ #include "utils/utils.h" #include "../flashcart_utils.h" +#include "ed64_vseries_ll.h" #include "ed64_vseries.h" typedef enum { diff --git a/src/flashcart/ed64/ed64_vseries_ll.h b/src/flashcart/ed64/ed64_vseries_ll.h new file mode 100644 index 00000000..34614bf8 --- /dev/null +++ b/src/flashcart/ed64/ed64_vseries_ll.h @@ -0,0 +1,14 @@ +/** + * @file ed64_vseries_ll.h + * @brief ed64v flashcart low level access + * @ingroup flashcart + */ + +#ifndef FLASHCART_ED64_VSERIES_LL_H__ +#define FLASHCART_ED64_VSERIES_LL_H__ + + +/** @} */ /* ed64_vseries_ll */ + + +#endif diff --git a/src/flashcart/ed64/ed64_xseries.c b/src/flashcart/ed64/ed64_xseries.c new file mode 100644 index 00000000..1f1f4e07 --- /dev/null +++ b/src/flashcart/ed64/ed64_xseries.c @@ -0,0 +1,175 @@ +#include +#include +#include + +#include +#include + +#include "utils/fs.h" +#include "utils/utils.h" + +#include "../flashcart_utils.h" +#include "ed64_xseries_ll.h" +#include "ed64_xseries.h" + +typedef enum { + // potentially handle if the firmware supports it... + ED64_X5_0 = 550, + ED64_X7_0 = 570, + ED64_UKNOWN = 0, +} ed64_xseries_device_variant_t; + +/* ED64 save location base address */ +#define SRAM_ADDRESS (0xA8000000) +/* ED64 ROM location base address */ +#define ROM_ADDRESS (0xB0000000) + +static flashcart_firmware_version_t ed64_xseries_get_firmware_version (void) { + flashcart_firmware_version_t version_info; + // FIXME: get version from ll + version_info.major = 1; + version_info.minor = 1; + version_info.revision = 0; + + //ed64_ll_get_version(&version_info.major, &version_info.minor, &version_info.revision); + + return version_info; +} + +static flashcart_err_t ed64_xseries_init (void) { + + return FLASHCART_OK; +} + +static flashcart_err_t ed64_xseries_deinit (void) { + + return FLASHCART_OK; +} + +static ed64_xseries_device_variant_t get_cart_model() { + ed64_xseries_device_variant_t variant = ED64_X7_0; // FIXME: check cart model from ll for better feature handling. + return variant; +} + +static bool ed64_xseries_has_feature (flashcart_features_t feature) { + bool is_model_x7 = (get_cart_model() == ED64_X7_0); + switch (feature) { + case FLASHCART_FEATURE_RTC: return is_model_x7 ? true : false; + case FLASHCART_FEATURE_USB: return is_model_x7 ? true : false; + case FLASHCART_FEATURE_64DD: return false; + case FLASHCART_FEATURE_AUTO_CIC: return true; + case FLASHCART_FEATURE_AUTO_REGION: return true; + default: return false; + } +} + +static flashcart_err_t ed64_xseries_load_rom (char *rom_path, flashcart_progress_callback_t *progress) { + FIL fil; + UINT br; + + if (f_open(&fil, strip_fs_prefix(rom_path), FA_READ) != FR_OK) { + return FLASHCART_ERR_LOAD; + } + + fatfs_fix_file_size(&fil); + + size_t rom_size = f_size(&fil); + + if (rom_size > MiB(64)) { + f_close(&fil); + return FLASHCART_ERR_LOAD; + } + + size_t sdram_size = MiB(64); + + size_t chunk_size = KiB(128); + for (int offset = 0; offset < sdram_size; offset += chunk_size) { + size_t block_size = MIN(sdram_size - offset, chunk_size); + if (f_read(&fil, (void *) (ROM_ADDRESS + offset), block_size, &br) != FR_OK) { + f_close(&fil); + return FLASHCART_ERR_LOAD; + } + if (progress) { + progress(f_tell(&fil) / (float) (f_size(&fil))); + } + } + if (f_tell(&fil) != rom_size) { + f_close(&fil); + return FLASHCART_ERR_LOAD; + } + + if (f_close(&fil) != FR_OK) { + return FLASHCART_ERR_LOAD; + } + + return FLASHCART_OK; +} + +static flashcart_err_t ed64_xseries_load_file (char *file_path, uint32_t rom_offset, uint32_t file_offset) { + FIL fil; + UINT br; + + if (f_open(&fil, strip_fs_prefix(file_path), FA_READ) != FR_OK) { + return FLASHCART_ERR_LOAD; + } + + fatfs_fix_file_size(&fil); + + size_t file_size = f_size(&fil) - file_offset; + + if (file_size > (MiB(64) - rom_offset)) { + f_close(&fil); + return FLASHCART_ERR_ARGS; + } + + if (f_lseek(&fil, file_offset) != FR_OK) { + f_close(&fil); + return FLASHCART_ERR_LOAD; + } + + if (f_read(&fil, (void *) (ROM_ADDRESS + rom_offset), file_size, &br) != FR_OK) { + f_close(&fil); + return FLASHCART_ERR_LOAD; + } + if (br != file_size) { + f_close(&fil); + return FLASHCART_ERR_LOAD; + } + + if (f_close(&fil) != FR_OK) { + return FLASHCART_ERR_LOAD; + } + + return FLASHCART_OK; +} + +static flashcart_err_t ed64_xseries_load_save (char *save_path) { + + + return FLASHCART_OK; +} + +static flashcart_err_t ed64_xseries_set_save_type (flashcart_save_type_t save_type) { + + + return FLASHCART_OK; +} + +static flashcart_t flashcart_ed64_xseries = { + .init = ed64_xseries_init, + .deinit = ed64_xseries_deinit, + .has_feature = ed64_xseries_has_feature, + .get_firmware_version = ed64_xseries_get_firmware_version, + .load_rom = ed64_xseries_load_rom, + .load_file = ed64_xseries_load_file, + .load_save = ed64_xseries_load_save, + .load_64dd_ipl = NULL, + .load_64dd_disk = NULL, + .set_save_type = ed64_xseries_set_save_type, + .set_save_writeback = NULL, +}; + + +flashcart_t *ed64_xseries_get_flashcart (void) { + return &flashcart_ed64_xseries; +} diff --git a/src/flashcart/ed64/ed64_xseries.h b/src/flashcart/ed64/ed64_xseries.h index a6bf497c..4dfad0c4 100644 --- a/src/flashcart/ed64/ed64_xseries.h +++ b/src/flashcart/ed64/ed64_xseries.h @@ -16,7 +16,7 @@ * @{ */ -flashcart_t *ed64xseries_get_flashcart (void); +flashcart_t *ed64_xseries_get_flashcart (void); /** @} */ /* ED64_Xseries */ diff --git a/src/flashcart/ed64/ed64_xseries_ll.h b/src/flashcart/ed64/ed64_xseries_ll.h new file mode 100644 index 00000000..eb3bb717 --- /dev/null +++ b/src/flashcart/ed64/ed64_xseries_ll.h @@ -0,0 +1,14 @@ +/** + * @file ed64_xseries_ll.h + * @brief ed64x flashcart low level access + * @ingroup flashcart + */ + +#ifndef FLASHCART_ED64_XSERIES_LL_H__ +#define FLASHCART_ED64_XSERIES_LL_H__ + + +/** @} */ /* ed64_xseries_ll */ + + +#endif diff --git a/src/flashcart/flashcart.c b/src/flashcart/flashcart.c index 6b634cac..8f670aad 100644 --- a/src/flashcart/flashcart.c +++ b/src/flashcart/flashcart.c @@ -11,6 +11,7 @@ #include "flashcart_utils.h" #include "ed64/ed64_vseries.h" +#include "ed64/ed64_xseries.h" #include "64drive/64drive.h" #include "sc64/sc64.h" @@ -109,10 +110,9 @@ flashcart_err_t flashcart_init (const char **storage_prefix) { flashcart = d64_get_flashcart(); break; - // FIXME: this is commented out awaiting a fix from libcart. - // case CART_EDX: // Series X EverDrive-64 - // flashcart = ed64_xseries_get_flashcart(); - // break; + case CART_EDX: // Official EverDrive 64 Series X + flashcart = ed64_xseries_get_flashcart(); + break; case CART_ED: // Series V EverDrive-64 or clone flashcart = ed64_vseries_get_flashcart();