Path: blob/master/nyx/nyx_gui/frontend/gui_emummc_tools.c
3711 views
/*1* Copyright (c) 2019-2026 CTCaer2*3* This program is free software; you can redistribute it and/or modify it4* under the terms and conditions of the GNU General Public License,5* version 2, as published by the Free Software Foundation.6*7* This program is distributed in the hope it will be useful, but WITHOUT8* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or9* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for10* more details.11*12* You should have received a copy of the GNU General Public License13* along with this program. If not, see <http://www.gnu.org/licenses/>.14*/1516#include <stdlib.h>1718#include <bdk.h>1920#include "gui.h"21#include "fe_emummc_tools.h"22#include "gui_tools_partition_manager.h"23#include <libs/fatfs/ff.h>2425extern char *emmcsn_path_impl(char *path, char *sub_dir, char *filename, sdmmc_storage_t *storage);2627typedef struct _mbr_ctxt_t28{29u32 available;30u32 sector[3];31u32 resized_cnt[3];3233int part_idx;34u32 sector_start;35} mbr_ctxt_t;3637static bool emummc_backup;38static mbr_ctxt_t mbr_ctx;39static lv_obj_t *emummc_manage_window;40static lv_res_t (*emummc_tools)(lv_obj_t *btn);4142static lv_res_t _action_emummc_window_close(lv_obj_t *btn)43{44nyx_win_close_action(btn);4546// Delete and relaunch main emuMMC window.47lv_obj_del(emummc_manage_window);48(*emummc_tools)(NULL);4950return LV_RES_INV;51}5253static void _create_window_emummc()54{55emmc_tool_gui_t emmc_tool_gui_ctxt;5657lv_obj_t *win;58if (!mbr_ctx.part_idx)59win = nyx_create_standard_window(SYMBOL_DRIVE" Create SD File emuMMC", _action_emummc_window_close);60else61win = nyx_create_standard_window(SYMBOL_DRIVE" Create SD Partition emuMMC", _action_emummc_window_close);6263//Disable buttons.64nyx_window_toggle_buttons(win, true);6566// Create important info container.67lv_obj_t *h1 = lv_cont_create(win, NULL);68lv_cont_set_fit(h1, false, true);69lv_obj_set_width(h1, (LV_HOR_RES / 9) * 5);70lv_obj_set_click(h1, false);71lv_cont_set_layout(h1, LV_LAYOUT_OFF);7273static lv_style_t h_style;74lv_style_copy(&h_style, lv_cont_get_style(h1));75h_style.body.main_color = LV_COLOR_HEX(0x1d1d1d);76h_style.body.grad_color = h_style.body.main_color;77h_style.body.opa = LV_OPA_COVER;7879// Chreate log container.80lv_obj_t *h2 = lv_cont_create(win, h1);81lv_cont_set_style(h2, &h_style);82lv_cont_set_fit(h2, false, false);83lv_obj_set_size(h2, (LV_HOR_RES / 11) * 4, LV_DPI * 5);84lv_cont_set_layout(h2, LV_LAYOUT_OFF);85lv_obj_align(h2, h1, LV_ALIGN_OUT_RIGHT_TOP, 0, LV_DPI / 5);8687lv_obj_t *label_log = lv_label_create(h2, NULL);88lv_label_set_recolor(label_log, true);89lv_obj_set_style(label_log, &monospace_text);90lv_label_set_long_mode(label_log, LV_LABEL_LONG_BREAK);91lv_label_set_static_text(label_log, "");92lv_obj_set_width(label_log, lv_obj_get_width(h2));93lv_obj_align(label_log, h2, LV_ALIGN_IN_TOP_LEFT, LV_DPI / 10, LV_DPI / 10);94emmc_tool_gui_ctxt.label_log = label_log;9596// Create elements for info container.97lv_obj_t *label_sep = lv_label_create(h1, NULL);98lv_label_set_static_text(label_sep, "");99100lv_obj_t *label_info = lv_label_create(h1, NULL);101lv_label_set_recolor(label_info, true);102lv_obj_set_width(label_info, lv_obj_get_width(h1));103lv_label_set_static_text(label_info, "\n\n\n\n\n\n\n\n\n");104lv_obj_set_style(label_info, lv_theme_get_current()->label.prim);105lv_obj_align(label_info, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 10);106emmc_tool_gui_ctxt.label_info = label_info;107108lv_obj_t *bar = lv_bar_create(h1, NULL);109lv_obj_set_size(bar, LV_DPI * 38 / 10, LV_DPI / 5);110lv_bar_set_range(bar, 0, 100);111lv_bar_set_value(bar, 0);112lv_obj_align(bar, label_info, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 8);113lv_obj_set_opa_scale(bar, LV_OPA_0);114lv_obj_set_opa_scale_enable(bar, true);115emmc_tool_gui_ctxt.bar = bar;116117lv_obj_t *label_pct= lv_label_create(h1, NULL);118lv_label_set_recolor(label_pct, true);119lv_label_set_static_text(label_pct, " "SYMBOL_DOT" 0%");120lv_obj_set_style(label_pct, lv_theme_get_current()->label.prim);121lv_obj_align(label_pct, bar, LV_ALIGN_OUT_RIGHT_MID, LV_DPI / 20, 0);122lv_obj_set_opa_scale(label_pct, LV_OPA_0);123lv_obj_set_opa_scale_enable(label_pct, true);124emmc_tool_gui_ctxt.label_pct = label_pct;125126lv_obj_t *label_finish = lv_label_create(h1, NULL);127lv_label_set_recolor(label_finish, true);128lv_label_set_static_text(label_finish, "");129lv_obj_set_style(label_finish, lv_theme_get_current()->label.prim);130lv_obj_align(label_finish, bar, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI * 9 / 20);131emmc_tool_gui_ctxt.label_finish = label_finish;132133if (!mbr_ctx.part_idx)134dump_emummc_file(&emmc_tool_gui_ctxt);135else136dump_emummc_raw(&emmc_tool_gui_ctxt, mbr_ctx.part_idx, mbr_ctx.sector_start, mbr_ctx.resized_cnt[mbr_ctx.part_idx - 1]);137138nyx_window_toggle_buttons(win, false);139}140141static lv_res_t _create_emummc_raw_format(lv_obj_t * btns, const char * txt)142{143int btn_idx = lv_btnm_get_pressed(btns);144145// Delete parent mbox.146nyx_mbox_action(btns, txt);147148// Create partition window.149if (!btn_idx)150create_window_sd_partition_manager(btns);151152mbr_ctx.part_idx = 0;153mbr_ctx.sector_start = 0;154155return LV_RES_INV;156}157158static lv_res_t _create_emummc_raw_action(lv_obj_t * btns, const char * txt)159{160int btn_idx = lv_btnm_get_pressed(btns);161lv_obj_t *bg = lv_obj_get_parent(lv_obj_get_parent(btns));162163mbr_ctx.sector_start = 0x8000; // Protective offset.164165switch (btn_idx)166{167case 0:168mbr_ctx.part_idx = 1;169mbr_ctx.sector_start += mbr_ctx.sector[0];170break;171case 1:172mbr_ctx.part_idx = 2;173mbr_ctx.sector_start += mbr_ctx.sector[1];174break;175case 2:176mbr_ctx.part_idx = 3;177mbr_ctx.sector_start += mbr_ctx.sector[2];178break;179default:180break;181}182183if (btn_idx < 3)184{185lv_obj_set_style(bg, &lv_style_transp);186_create_window_emummc();187}188189mbr_ctx.part_idx = 0;190mbr_ctx.sector_start = 0;191192nyx_mbox_action(btns, txt);193194return LV_RES_INV;195}196197static void _create_mbox_emummc_raw()198{199lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);200lv_obj_set_style(dark_bg, &mbox_darken);201lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);202203static const char *mbox_btn_format[] = { "\222Continue", "\222Cancel", "" };204static char *mbox_btn_parts[] = { "\262Part 1", "\262Part 2", "\262Part 3", "\222Cancel", "" };205lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);206lv_mbox_set_recolor_text(mbox, true);207lv_obj_set_width(mbox, LV_HOR_RES / 9 * 6);208209char *txt_buf = (char *)malloc(SZ_16K);210mbr_t *mbr = (mbr_t *)malloc(sizeof(mbr_t));211212memset(&mbr_ctx, 0, sizeof(mbr_ctxt_t));213214sd_mount();215sdmmc_storage_read(&sd_storage, 0, 1, mbr);216sd_unmount();217218emmc_initialize(false);219220u32 emmc_size_safe = emmc_storage.sec_cnt + 0xC000; // eMMC GPP size + BOOT0/1.221222emmc_end();223224for (int i = 1; i < 4; i++)225{226u32 part_size = mbr->partitions[i].size_sct;227u32 part_start = mbr->partitions[i].start_sct;228u8 part_type = mbr->partitions[i].type;229230// Skip Linux, GPT (Android) and SFD partitions.231bool valid_part = (part_type != 0x83) && (part_type != 0xEE) && (part_type != 0xFF);232233// Check if at least 4GB and start above 16MB.234if ((part_size >= 0x80F000) && part_start > 0x8000 && valid_part)235{236mbr_ctx.available |= BIT(i - 1);237mbr_ctx.sector[i - 1] = part_start;238239// Only allow up to 28GB resized emuMMC.240if (part_size <= 0x3810000)241mbr_ctx.resized_cnt[i - 1] = part_size - 0xC000; // Save sectors count without protective size and BOOT0/1.242else if (part_size >= emmc_size_safe)243mbr_ctx.resized_cnt[i - 1] = 0;244else245{246mbr_ctx.available &= ~BIT(i - 1);247mbr_ctx.sector[i - 1] = 0;248}249}250}251252if (mbr_ctx.available)253{254s_printf(txt_buf,255"#C7EA46 Found applicable partition(s)!#\n"256"#FF8000 Choose a partition to continue:#\n\n");257}258else259s_printf(txt_buf, "#FFDD00 Failed to find applicable partition!#\n\n");260261s_printf(txt_buf + strlen(txt_buf),262"Partition table:\n"263"#C0C0C0 Part 0: Type: %02x, Start: %08x, Size: %08x#\n"264"#%s Part 1: Type: %02x, Start: %08x, Size: %08x#\n"265"#%s Part 2: Type: %02x, Start: %08x, Size: %08x#\n"266"#%s Part 3: Type: %02x, Start: %08x, Size: %08x#",267mbr->partitions[0].type, mbr->partitions[0].start_sct, mbr->partitions[0].size_sct,268(mbr_ctx.available & BIT(0)) ? (mbr_ctx.resized_cnt[0] ? "FFDD00" : "C7EA46") : "C0C0C0",269mbr->partitions[1].type, mbr->partitions[1].start_sct, mbr->partitions[1].size_sct,270(mbr_ctx.available & BIT(1)) ? (mbr_ctx.resized_cnt[1] ? "FFDD00" : "C7EA46") : "C0C0C0",271mbr->partitions[2].type, mbr->partitions[2].start_sct, mbr->partitions[2].size_sct,272(mbr_ctx.available & BIT(2)) ? (mbr_ctx.resized_cnt[2] ? "FFDD00" : "C7EA46") : "C0C0C0",273mbr->partitions[3].type, mbr->partitions[3].start_sct, mbr->partitions[3].size_sct);274275if (mbr_ctx.resized_cnt[0] || mbr_ctx.resized_cnt[1] || mbr_ctx.resized_cnt[2])276strcat(txt_buf, "\n\n#FFDD00 Note:# Yellow entries have USER partition resized.");277278if (!mbr_ctx.available)279strcat(txt_buf, "\n#FF8000 Do you want to partition the SD card?#\n"280"#FF8000 (You will be asked on how to proceed)#");281282lv_mbox_set_text(mbox, txt_buf);283free(txt_buf);284free(mbr);285286if (mbr_ctx.available)287{288// Check available partitions and enable the corresponding buttons.289if (mbr_ctx.available & 1)290mbox_btn_parts[0][0] = '\222';291else292mbox_btn_parts[0][0] = '\262';293if (mbr_ctx.available & 2)294mbox_btn_parts[1][0] = '\222';295else296mbox_btn_parts[1][0] = '\262';297if (mbr_ctx.available & 4)298mbox_btn_parts[2][0] = '\222';299else300mbox_btn_parts[2][0] = '\262';301302lv_mbox_add_btns(mbox, (const char **)mbox_btn_parts, _create_emummc_raw_action);303}304else305lv_mbox_add_btns(mbox, mbox_btn_format, _create_emummc_raw_format);306307lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);308lv_obj_set_top(mbox, true);309}310311static lv_res_t _create_emummc_action(lv_obj_t * btns, const char * txt)312{313int btn_idx = lv_btnm_get_pressed(btns);314lv_obj_t *bg = lv_obj_get_parent(lv_obj_get_parent(btns));315316mbr_ctx.part_idx = 0;317mbr_ctx.sector_start = 0;318319switch (btn_idx)320{321case 0:322lv_obj_set_style(bg, &lv_style_transp);323_create_window_emummc();324break;325case 1:326_create_mbox_emummc_raw();327break;328}329330nyx_mbox_action(btns, txt);331332return LV_RES_INV;333}334335static lv_res_t _create_mbox_emummc_create(lv_obj_t *btn)336{337if (!nyx_emmc_check_battery_enough())338return LV_RES_OK;339340lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);341lv_obj_set_style(dark_bg, &mbox_darken);342lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);343344static const char * mbox_btn_map[] = { "\222SD File", "\222SD Partition", "\222Cancel", "" };345lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);346lv_mbox_set_recolor_text(mbox, true);347lv_obj_set_width(mbox, LV_HOR_RES / 9 * 6);348349lv_mbox_set_text(mbox,350"Welcome to #C7EA46 emuMMC# creation tool!\n\n"351"Please choose what type of emuMMC you want to create.\n"352"#FF8000 SD File# is saved as files in the FAT partition.\n"353"#FF8000 SD Partition# is saved as raw image in an available partition.");354355lv_mbox_add_btns(mbox, mbox_btn_map, _create_emummc_action);356357lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);358lv_obj_set_top(mbox, true);359360return LV_RES_OK;361}362363static void _change_raw_emummc_part_type()364{365mbr_t *mbr = (mbr_t *)malloc(sizeof(mbr_t));366sdmmc_storage_read(&sd_storage, 0, 1, mbr);367mbr->partitions[mbr_ctx.part_idx].type = 0xE0;368sdmmc_storage_write(&sd_storage, 0, 1, mbr);369free(mbr);370}371372static lv_res_t _save_emummc_cfg_mig_mbox_action(lv_obj_t *btns, const char *txt)373{374// Delete main emuMMC and popup windows and relaunch main emuMMC window.375lv_obj_del(emummc_manage_window);376nyx_mbox_action(btns, txt);377378(*emummc_tools)(NULL);379380return LV_RES_INV;381}382383static void _create_emummc_migrated_mbox()384{385lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);386lv_obj_set_style(dark_bg, &mbox_darken);387lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);388389static const char *mbox_btn_map[] = { "\251", "OK", "\251", "" };390lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);391lv_mbox_set_recolor_text(mbox, true);392lv_obj_set_width(mbox, LV_HOR_RES / 9 * 4);393394lv_mbox_set_text(mbox,395"#FF8000 emuMMC Configuration#\n\n"396"#96FF00 The emuMMC configuration#\n#96FF00 was saved to sd card!#");397398lv_mbox_add_btns(mbox, mbox_btn_map, _save_emummc_cfg_mig_mbox_action);399400lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);401lv_obj_set_top(mbox, true);402}403404static void _migrate_sd_raw_emummc_based()405{406char *tmp = (char *)malloc(0x80);407s_printf(tmp, "emuMMC/RAW%d", mbr_ctx.part_idx);408409sd_mount();410f_mkdir("emuMMC");411f_mkdir(tmp);412strcat(tmp, "/raw_based");413414FIL fp;415if (!f_open(&fp, tmp, FA_CREATE_ALWAYS | FA_WRITE))416{417f_write(&fp, &mbr_ctx.sector_start, 4, NULL);418f_close(&fp);419}420421s_printf(tmp, "emuMMC/RAW%d", mbr_ctx.part_idx);422423_change_raw_emummc_part_type();424425save_emummc_cfg(mbr_ctx.part_idx, mbr_ctx.sector_start, tmp);426_create_emummc_migrated_mbox();427free(tmp);428429sd_unmount();430}431432static void _migrate_sd_backup_file_based()433{434char *emu_path = (char *)malloc(128);435char *parts_path = (char *)malloc(128);436char *backup_path = (char *)malloc(128);437char *backup_file_path = (char *)malloc(128);438439sd_mount();440f_mkdir("emuMMC");441442strcpy(emu_path, "emuMMC/BK");443u32 base_len = strlen(emu_path);444445for (int j = 0; j < 100; j++)446{447update_emummc_base_folder(emu_path, base_len, j);448if (f_stat(emu_path, NULL) == FR_NO_FILE)449break;450}451base_len = strlen(emu_path);452453f_mkdir(emu_path);454strcat(emu_path, "/eMMC");455f_mkdir(emu_path);456457FIL fp;458// Create file based flag.459strcpy(emu_path + base_len, "/file_based");460f_open(&fp, "emuMMC/BK00/file_based", FA_CREATE_ALWAYS | FA_WRITE);461f_close(&fp);462463if (!emummc_backup)464emmcsn_path_impl(backup_path, "", "", NULL);465else466emmcsn_path_impl(backup_path, "/emummc", "", NULL);467468// Move BOOT0.469s_printf(backup_file_path, "%s/BOOT0", backup_path);470strcpy(emu_path + base_len, "/eMMC/BOOT0");471f_rename(backup_file_path, emu_path);472473// Move BOOT1.474s_printf(backup_file_path, "%s/BOOT1", backup_path);475strcpy(emu_path + base_len, "/eMMC/BOOT1");476f_rename(backup_file_path, emu_path);477478// Move raw GPP.479bool multipart = false;480s_printf(backup_file_path, "%s/rawnand.bin", backup_path);481482if (f_stat(backup_file_path, NULL))483multipart = true;484485if (!multipart)486{487strcpy(emu_path + base_len, "/eMMC/00");488f_rename(backup_file_path, emu_path);489}490else491{492emu_path[base_len] = 0;493for (int i = 0; i < 32; i++)494{495s_printf(backup_file_path, "%s/rawnand.bin.%02d", backup_path, i);496s_printf(parts_path, "%s/eMMC/%02d", emu_path, i);497if (f_rename(backup_file_path, parts_path))498break;499}500}501502free(emu_path);503free(parts_path);504free(backup_path);505free(backup_file_path);506507save_emummc_cfg(0, 0, "emuMMC/BK00");508_create_emummc_migrated_mbox();509sd_unmount();510}511512static lv_res_t _create_emummc_mig0_action(lv_obj_t * btns, const char * txt)513{514switch (lv_btnm_get_pressed(btns))515{516case 0:517_migrate_sd_raw_emummc_based();518break;519}520521mbr_ctx.part_idx = 0;522mbr_ctx.sector_start = 0;523524nyx_mbox_action(btns, txt);525526return LV_RES_INV;527}528529static lv_res_t _create_emummc_mig1_action(lv_obj_t * btns, const char * txt)530{531switch (lv_btnm_get_pressed(btns))532{533case 0:534_migrate_sd_backup_file_based();535break;536}537538mbr_ctx.part_idx = 0;539mbr_ctx.sector_start = 0;540541nyx_mbox_action(btns, txt);542543return LV_RES_INV;544}545546static lv_res_t _create_emummc_migrate_action(lv_obj_t * btns, const char * txt)547{548bool backup = false;549bool emummc = false;550551switch (lv_btnm_get_pressed(btns))552{553case 0:554backup = true;555break;556case 1:557emummc = true;558break;559case 2:560nyx_mbox_action(btns, txt);561return LV_RES_INV;562}563564lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);565lv_obj_set_style(dark_bg, &mbox_darken);566lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);567568static const char *mbox_btn_map[] = { "\222Continue", "\222Cancel", "" };569static const char *mbox_btn_map1[] = { "\251", "OK", "\251", "" };570lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);571lv_mbox_set_recolor_text(mbox, true);572lv_obj_set_width(mbox, LV_HOR_RES / 9 * 6);573574char *txt_buf = (char *)malloc(SZ_16K);575576if (backup)577{578if (!emummc_backup)579s_printf(txt_buf,580"#C7EA46 Found suitable eMMC backup!#\n\n"581"#FF8000 Do you want to migrate it?#\n");582else583s_printf(txt_buf,584"#C7EA46 Found suitable emuMMC backup!#\n\n"585"#FF8000 Do you want to migrate it?#\n");586lv_mbox_add_btns(mbox, mbox_btn_map, _create_emummc_mig1_action);587}588else if (emummc)589{590s_printf(txt_buf,591"#C7EA46 Found SD Partition based emuMMC!#\n\n"592"#FF8000 Do you want to repair the config and partition type for it?#\n");593lv_mbox_add_btns(mbox, mbox_btn_map, _create_emummc_mig0_action);594}595else596{597s_printf(txt_buf, "No emuMMC found!\n");598lv_mbox_add_btns(mbox, mbox_btn_map1, nyx_mbox_action);599}600601lv_mbox_set_text(mbox, txt_buf);602free(txt_buf);603604lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);605lv_obj_set_top(mbox, true);606607nyx_mbox_action(btns, txt);608609return LV_RES_INV;610}611612typedef struct _emummc_images_t613{614dirlist_t *dirlist;615u32 part_sector[3];616u32 part_type[3];617u32 part_end[3];618char part_path[3 * 128];619lv_obj_t *win;620} emummc_images_t;621622static lv_res_t _create_mbox_emummc_migrate(lv_obj_t *btn)623{624lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);625lv_obj_set_style(dark_bg, &mbox_darken);626lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);627628static char *mbox_btn_map[] = { "\262Backup", "\262Fix RAW", "\222Cancel", "" };629lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);630lv_mbox_set_recolor_text(mbox, true);631lv_obj_set_width(mbox, LV_HOR_RES / 9 * 6);632633lv_mbox_set_text(mbox,634"Welcome to #C7EA46 emuMMC# migration tool!\n\n"635"Please choose what type of migration you want to do.\n"636"Anything that was not found will have the button disabled.");637638char *path_buf = (char *)malloc(0x512);639mbr_t *mbr = (mbr_t *)malloc(sizeof(mbr_t));640u8 *efi_part = (u8 *)malloc(0x200);641642sd_mount();643sdmmc_storage_read(&sd_storage, 0, 1, mbr);644645emmc_initialize(false);646647bool backup = false;648bool emummc = false;649bool rawnand_backup = false;650651mbr_ctx.sector_start = 0;652mbr_ctx.part_idx = 0;653654// Try to find a partition based emuMMC.655for (int i = 1; i < 4; i++)656{657mbr_ctx.sector_start = mbr->partitions[i].start_sct;658659if (!mbr_ctx.sector_start)660continue;661662sdmmc_storage_read(&sd_storage, mbr_ctx.sector_start + 0xC001, 1, efi_part);663if (!memcmp(efi_part, "EFI PART", 8))664{665mbr_ctx.sector_start += 0x8000;666emummc = true;667mbr_ctx.part_idx = i;668break;669}670else671{672sdmmc_storage_read(&sd_storage, mbr_ctx.sector_start + 0x4001, 1, efi_part);673if (!memcmp(efi_part, "EFI PART", 8))674{675emummc = true;676mbr_ctx.part_idx = i;677break;678}679}680}681682emummc_backup = false;683684emmcsn_path_impl(path_buf, "", "BOOT0", &emmc_storage);685if (!f_stat(path_buf, NULL))686backup = true;687688emmcsn_path_impl(path_buf, "", "rawnand.bin", &emmc_storage);689if (!f_stat(path_buf, NULL))690rawnand_backup = true;691692emmcsn_path_impl(path_buf, "", "rawnand.bin.00", &emmc_storage);693if (!f_stat(path_buf, NULL))694rawnand_backup = true;695696backup = backup && rawnand_backup;697698if (!backup)699{700rawnand_backup = false;701emummc_backup = true;702703emmcsn_path_impl(path_buf, "/emummc", "BOOT0", &emmc_storage);704if (!f_stat(path_buf, NULL))705backup = true;706707emmcsn_path_impl(path_buf, "/emummc", "rawnand.bin", &emmc_storage);708if (!f_stat(path_buf, NULL))709rawnand_backup = true;710711emmcsn_path_impl(path_buf, "/emummc", "rawnand.bin.00", &emmc_storage);712if (!f_stat(path_buf, NULL))713rawnand_backup = true;714715backup = backup && rawnand_backup;716}717718sd_unmount();719emmc_end();720721// Check available types and enable the corresponding buttons.722if (backup)723mbox_btn_map[0][0] = '\222';724else725mbox_btn_map[0][0] = '\262';726if (emummc)727mbox_btn_map[1][0] = '\222';728else729mbox_btn_map[1][0] = '\262';730731free(path_buf);732free(mbr);733free(efi_part);734735lv_mbox_add_btns(mbox, (const char **)mbox_btn_map, _create_emummc_migrate_action);736737lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);738lv_obj_set_top(mbox, true);739740return LV_RES_OK;741}742743static emummc_images_t *emummc_img;744745static lv_res_t _save_emummc_cfg_mbox_action(lv_obj_t *btns, const char *txt)746{747// Free components, delete main emuMMC and popup windows and relaunch main emuMMC window.748lv_obj_del(emummc_img->win);749lv_obj_del(emummc_manage_window);750free(emummc_img->dirlist);751free(emummc_img);752753nyx_mbox_action(btns, txt);754755(*emummc_tools)(NULL);756757return LV_RES_INV;758}759760static void _create_emummc_saved_mbox()761{762lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);763lv_obj_set_style(dark_bg, &mbox_darken);764lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);765766static const char *mbox_btn_map[] = { "\251", "OK", "\251", "" };767lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);768lv_mbox_set_recolor_text(mbox, true);769lv_obj_set_width(mbox, LV_HOR_RES / 9 * 4);770771lv_mbox_set_text(mbox,772"#FF8000 emuMMC Configuration#\n\n"773"#96FF00 The emuMMC configuration#\n#96FF00 was saved to sd card!#");774775lv_mbox_add_btns(mbox, mbox_btn_map, _save_emummc_cfg_mbox_action);776777lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);778lv_obj_set_top(mbox, true);779}780781static lv_res_t _save_raw_emummc_cfg_action(lv_obj_t * btn)782{783lv_btn_ext_t *ext = lv_obj_get_ext_attr(btn);784switch (ext->idx)785{786case 0:787save_emummc_cfg(1, emummc_img->part_sector[0], &emummc_img->part_path[0]);788break;789case 1:790save_emummc_cfg(2, emummc_img->part_sector[1], &emummc_img->part_path[128]);791break;792case 2:793save_emummc_cfg(3, emummc_img->part_sector[2], &emummc_img->part_path[256]);794break;795}796797_create_emummc_saved_mbox();798sd_unmount();799800return LV_RES_INV;801}802803static lv_res_t _save_disable_emummc_cfg_action(lv_obj_t * btn)804{805save_emummc_cfg(0, 0, NULL);806_create_emummc_saved_mbox();807sd_unmount();808809return LV_RES_INV;810}811812static lv_res_t _save_file_emummc_cfg_action(lv_obj_t *btn)813{814save_emummc_cfg(0, 0, lv_list_get_btn_text(btn));815_create_emummc_saved_mbox();816sd_unmount();817818return LV_RES_INV;819}820821static lv_res_t _action_win_change_emummc_close(lv_obj_t *btn)822{823free(emummc_img->dirlist);824free(emummc_img);825826return nyx_win_close_action(btn);827}828829static lv_res_t _create_change_emummc_window(lv_obj_t *btn_caller)830{831lv_obj_t *win = nyx_create_standard_window(SYMBOL_SETTINGS" Change emuMMC", _action_win_change_emummc_close);832lv_win_add_btn(win, NULL, SYMBOL_POWER" Disable", _save_disable_emummc_cfg_action);833834sd_mount();835836emummc_img = malloc(sizeof(emummc_images_t));837emummc_img->win = win;838839mbr_t *mbr = (mbr_t *)malloc(sizeof(mbr_t));840char *path = malloc(512);841842sdmmc_storage_read(&sd_storage, 0, 1, mbr);843844memset(emummc_img->part_path, 0, 3 * 128);845846for (int i = 1; i < 4; i++)847{848emummc_img->part_sector[i - 1] = mbr->partitions[i].start_sct;849emummc_img->part_end[i - 1] = emummc_img->part_sector[i - 1] + mbr->partitions[i].size_sct - 1;850emummc_img->part_type[i - 1] = mbr->partitions[i].type;851}852free(mbr);853854emummc_img->dirlist = dirlist("emuMMC", NULL, DIR_SHOW_DIRS);855856if (!emummc_img->dirlist)857goto out0;858859u32 emummc_idx = 0;860861FIL fp;862863// Check for sd raw partitions, based on the folders in /emuMMC.864while (emummc_img->dirlist->name[emummc_idx])865{866s_printf(path, "emuMMC/%s/raw_based", emummc_img->dirlist->name[emummc_idx]);867868if (!f_stat(path, NULL))869{870f_open(&fp, path, FA_READ);871u32 curr_list_sector = 0;872f_read(&fp, &curr_list_sector, 4, NULL);873f_close(&fp);874875// Check if there's a HOS image there.876if ((curr_list_sector == 2) || (emummc_img->part_sector[0] && curr_list_sector >= emummc_img->part_sector[0] &&877curr_list_sector < emummc_img->part_end[0] && emummc_img->part_type[0] != 0x83))878{879s_printf(&emummc_img->part_path[0], "emuMMC/%s", emummc_img->dirlist->name[emummc_idx]);880emummc_img->part_sector[0] = curr_list_sector;881emummc_img->part_end[0] = 0;882}883else if (emummc_img->part_sector[1] && curr_list_sector >= emummc_img->part_sector[1] &&884curr_list_sector < emummc_img->part_end[1] && emummc_img->part_type[1] != 0x83)885{886s_printf(&emummc_img->part_path[1 * 128], "emuMMC/%s", emummc_img->dirlist->name[emummc_idx]);887emummc_img->part_sector[1] = curr_list_sector;888emummc_img->part_end[1] = 0;889}890else if (emummc_img->part_sector[2] && curr_list_sector >= emummc_img->part_sector[2] &&891curr_list_sector < emummc_img->part_end[2] && emummc_img->part_type[2] != 0x83)892{893s_printf(&emummc_img->part_path[2 * 128], "emuMMC/%s", emummc_img->dirlist->name[emummc_idx]);894emummc_img->part_sector[2] = curr_list_sector;895emummc_img->part_end[2] = 0;896}897}898emummc_idx++;899}900901emummc_idx = 0;902u32 file_based_idx = 0;903904// Sanitize the directory list with sd file based ones.905while (emummc_img->dirlist->name[emummc_idx])906{907s_printf(path, "emuMMC/%s/file_based", emummc_img->dirlist->name[emummc_idx]);908909if (!f_stat(path, NULL))910{911char *tmp = emummc_img->dirlist->name[emummc_idx];912memcpy(emummc_img->dirlist->name[file_based_idx], tmp, strlen(tmp) + 1);913file_based_idx++;914}915emummc_idx++;916}917emummc_img->dirlist->name[file_based_idx] = NULL;918919out0:;920static lv_style_t h_style;921lv_style_copy(&h_style, &lv_style_transp);922h_style.body.padding.inner = 0;923h_style.body.padding.hor = LV_DPI - (LV_DPI / 4);924h_style.body.padding.ver = LV_DPI / 6;925926// Create SD Raw Partitions container.927lv_obj_t *h1 = lv_cont_create(win, NULL);928lv_cont_set_style(h1, &h_style);929lv_cont_set_fit(h1, false, true);930lv_obj_set_width(h1, (LV_HOR_RES / 9) * 4);931lv_obj_set_click(h1, false);932lv_cont_set_layout(h1, LV_LAYOUT_OFF);933934lv_obj_t *label_sep = lv_label_create(h1, NULL);935lv_label_set_static_text(label_sep, "");936937lv_obj_t *label_txt = lv_label_create(h1, NULL);938lv_label_set_static_text(label_txt, "SD Raw Partitions");939lv_obj_set_style(label_txt, lv_theme_get_current()->label.prim);940lv_obj_align(label_txt, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, -(LV_DPI / 2));941942lv_obj_t *line_sep = lv_line_create(h1, NULL);943static const lv_point_t line_pp[] = { {0, 0}, { LV_HOR_RES - (LV_DPI - (LV_DPI / 4)) * 2, 0} };944lv_line_set_points(line_sep, line_pp, 2);945lv_line_set_style(line_sep, lv_theme_get_current()->line.decor);946lv_obj_align(line_sep, label_txt, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 8);947948lv_obj_t *btn = NULL;949lv_btn_ext_t *ext;950lv_obj_t *btn_label = NULL;951lv_obj_t *lv_desc = NULL;952char *txt_buf = malloc(SZ_16K);953954// Create RAW buttons.955for (u32 raw_btn_idx = 0; raw_btn_idx < 3; raw_btn_idx++)956{957btn = lv_btn_create(h1, btn);958ext = lv_obj_get_ext_attr(btn);959ext->idx = raw_btn_idx;960btn_label = lv_label_create(btn, btn_label);961962lv_btn_set_state(btn, LV_BTN_STATE_REL);963lv_obj_set_click(btn, true);964965if (emummc_img->part_type[raw_btn_idx] != 0x83)966{967s_printf(txt_buf, "SD RAW %d", raw_btn_idx + 1);968lv_label_set_text(btn_label, txt_buf);969}970971if (!emummc_img->part_sector[raw_btn_idx] || emummc_img->part_type[raw_btn_idx] == 0x83 || !emummc_img->part_path[raw_btn_idx * 128])972{973lv_btn_set_state(btn, LV_BTN_STATE_INA);974lv_obj_set_click(btn, false);975976if (emummc_img->part_type[raw_btn_idx] == 0x83)977lv_label_set_static_text(btn_label, "Linux");978}979980if (!raw_btn_idx)981{982lv_btn_set_fit(btn, false, true);983lv_obj_set_width(btn, LV_DPI * 3);984lv_obj_align(btn, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 2, LV_DPI / 5);985}986else987lv_obj_align(btn, lv_desc, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3);988989lv_btn_set_action(btn, LV_BTN_ACTION_CLICK, _save_raw_emummc_cfg_action);990991lv_desc = lv_label_create(h1, lv_desc);992lv_label_set_recolor(lv_desc, true);993lv_obj_set_style(lv_desc, &hint_small_style);994995s_printf(txt_buf, "Sector start: 0x%08X\nFolder: %s", emummc_img->part_sector[raw_btn_idx], &emummc_img->part_path[raw_btn_idx * 128]);996lv_label_set_text(lv_desc, txt_buf);997lv_obj_align(lv_desc, btn, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 5);998}999free(txt_buf);10001001// Create SD File Based container.1002lv_obj_t *h2 = lv_cont_create(win, NULL);1003lv_cont_set_style(h2, &h_style);1004lv_cont_set_fit(h2, false, true);1005lv_obj_set_width(h2, (LV_HOR_RES / 9) * 4);1006lv_obj_set_click(h2, false);1007lv_cont_set_layout(h2, LV_LAYOUT_OFF);1008lv_obj_align(h2, h1, LV_ALIGN_OUT_RIGHT_TOP, LV_DPI * 17 / 29, 0);10091010label_sep = lv_label_create(h2, NULL);1011lv_label_set_static_text(label_sep, "");10121013lv_obj_t *label_txt3 = lv_label_create(h2, NULL);1014lv_label_set_static_text(label_txt3, "SD File Based");1015lv_obj_set_style(label_txt3, lv_theme_get_current()->label.prim);1016lv_obj_align(label_txt3, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, -LV_DPI / 7);10171018line_sep = lv_line_create(h2, line_sep);1019lv_obj_align(line_sep, label_txt3, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 2), LV_DPI / 8);1020lv_line_set_style(line_sep, lv_theme_get_current()->line.decor);10211022lv_obj_t *list_sd_based = lv_list_create(h2, NULL);1023lv_obj_align(list_sd_based, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 2, LV_DPI / 4);10241025lv_obj_set_size(list_sd_based, LV_HOR_RES * 4 / 10, LV_VER_RES * 6 / 10);1026lv_list_set_single_mode(list_sd_based, true);10271028if (!emummc_img->dirlist)1029goto out1;10301031emummc_idx = 0;10321033// Add file based to the list.1034while (emummc_img->dirlist->name[emummc_idx])1035{1036s_printf(path, "emuMMC/%s", emummc_img->dirlist->name[emummc_idx]);10371038lv_list_add(list_sd_based, NULL, path, _save_file_emummc_cfg_action);10391040emummc_idx++;1041}10421043out1:1044free(path);1045sd_unmount();10461047return LV_RES_OK;1048}10491050lv_res_t create_win_emummc_tools(lv_obj_t *btn)1051{1052lv_obj_t *win = nyx_create_standard_window(SYMBOL_EDIT" emuMMC Manage", NULL);10531054// Set resources to be managed by other windows.1055emummc_manage_window = win;1056emummc_tools = (void *)create_win_emummc_tools;10571058sd_mount();10591060emummc_cfg_t emu_info;1061load_emummc_cfg(&emu_info);10621063sd_unmount();10641065static lv_style_t h_style;1066lv_style_copy(&h_style, &lv_style_transp);1067h_style.body.padding.inner = 0;1068h_style.body.padding.hor = LV_DPI - (LV_DPI / 4);1069h_style.body.padding.ver = LV_DPI / 9;10701071// Create emuMMC Info & Selection container.1072lv_obj_t *h1 = lv_cont_create(win, NULL);1073lv_cont_set_style(h1, &h_style);1074lv_cont_set_fit(h1, false, true);1075lv_obj_set_width(h1, (LV_HOR_RES / 9) * 4);1076lv_obj_set_click(h1, false);1077lv_cont_set_layout(h1, LV_LAYOUT_OFF);10781079lv_obj_t *label_sep = lv_label_create(h1, NULL);1080lv_label_set_static_text(label_sep, "");10811082lv_obj_t *label_txt = lv_label_create(h1, NULL);1083lv_label_set_static_text(label_txt, "emuMMC Info & Selection");1084lv_obj_set_style(label_txt, lv_theme_get_current()->label.prim);1085lv_obj_align(label_txt, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 11);10861087lv_obj_t *line_sep = lv_line_create(h1, NULL);1088static const lv_point_t line_pp[] = { {0, 0}, { LV_HOR_RES - (LV_DPI - (LV_DPI / 4)) * 2, 0} };1089lv_line_set_points(line_sep, line_pp, 2);1090lv_line_set_style(line_sep, lv_theme_get_current()->line.decor);1091lv_obj_align(line_sep, label_txt, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 8);10921093// Create emuMMC info labels.1094lv_obj_t *label_btn = lv_label_create(h1, NULL);1095lv_label_set_recolor(label_btn, true);1096lv_label_set_static_text(label_btn, emu_info.enabled ? "#96FF00 "SYMBOL_OK" Enabled!#" : "#FF8000 "SYMBOL_CLOSE" Disabled!#");1097lv_obj_align(label_btn, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 4);10981099lv_obj_t *label_txt2 = lv_label_create(h1, NULL);1100lv_label_set_recolor(label_txt2, true);1101char *txt_buf = (char *)malloc(SZ_16K);11021103if (emu_info.enabled)1104{1105if (emu_info.sector)1106s_printf(txt_buf, "#00DDFF Type:# SD Raw Partition\n#00DDFF Sector:# 0x%08X\n#00DDFF Nintendo folder:# %s\n",1107emu_info.sector, emu_info.nintendo_path ? emu_info.nintendo_path : "");1108else1109s_printf(txt_buf, "#00DDFF Type:# SD File\n#00DDFF Base folder:# %s\n#00DDFF Nintendo folder:# %s\n",1110emu_info.path ? emu_info.path : "", emu_info.nintendo_path ? emu_info.nintendo_path : "");11111112lv_label_set_text(label_txt2, txt_buf);1113}1114else1115{1116lv_label_set_static_text(label_txt2, "emuMMC is disabled and eMMC will be used for boot.\n\n\n");1117}11181119if (emu_info.path)1120free(emu_info.path);1121if (emu_info.nintendo_path)1122free(emu_info.nintendo_path);1123free(txt_buf);11241125lv_obj_set_style(label_txt2, &hint_small_style);1126lv_obj_align(label_txt2, label_btn, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3);11271128// Create Change emuMMC button.1129lv_obj_t *btn2 = lv_btn_create(h1, NULL);1130lv_btn_set_fit(btn2, true, true);1131label_btn = lv_label_create(btn2, NULL);1132lv_label_set_static_text(label_btn, SYMBOL_SETTINGS" Change emuMMC");1133lv_obj_align(btn2, label_txt2, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI * 6 / 10);1134lv_btn_set_action(btn2, LV_BTN_ACTION_CLICK, _create_change_emummc_window);11351136label_txt2 = lv_label_create(h1, NULL);1137lv_label_set_recolor(label_txt2, true);1138lv_label_set_static_text(label_txt2,1139"Choose between images created in the emuMMC folder\n"1140"or in SD card partitions. You can have at most 3 partition\n"1141"based and countless file based.");11421143lv_obj_set_style(label_txt2, &hint_small_style);1144lv_obj_align(label_txt2, btn2, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3);11451146// Create emuMMC Tools container.1147lv_obj_t *h2 = lv_cont_create(win, NULL);1148lv_cont_set_style(h2, &h_style);1149lv_cont_set_fit(h2, false, true);1150lv_obj_set_width(h2, (LV_HOR_RES / 9) * 4);1151lv_obj_set_click(h2, false);1152lv_cont_set_layout(h2, LV_LAYOUT_OFF);1153lv_obj_align(h2, h1, LV_ALIGN_OUT_RIGHT_TOP, LV_DPI * 17 / 29, 0);11541155label_sep = lv_label_create(h2, NULL);1156lv_label_set_static_text(label_sep, "");11571158lv_obj_t *label_txt3 = lv_label_create(h2, NULL);1159lv_label_set_static_text(label_txt3, "emuMMC Tools");1160lv_obj_set_style(label_txt3, lv_theme_get_current()->label.prim);1161lv_obj_align(label_txt3, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 5);11621163line_sep = lv_line_create(h2, line_sep);1164lv_obj_align(line_sep, label_txt3, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 8);11651166// Create Create emuMMC button.1167lv_obj_t *btn3 = lv_btn_create(h2, btn2);1168label_btn = lv_label_create(btn3, NULL);1169lv_btn_set_fit(btn3, true, true);1170lv_label_set_static_text(label_btn, SYMBOL_DRIVE" Create emuMMC");1171lv_obj_align(btn3, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 4);1172lv_btn_set_action(btn3, LV_BTN_ACTION_CLICK, _create_mbox_emummc_create);11731174lv_obj_t *label_txt4 = lv_label_create(h2, NULL);1175lv_label_set_recolor(label_txt4, true);1176lv_label_set_static_text(label_txt4,1177"Allows you to create a new #C7EA46 SD File# or #C7EA46 SD Raw Partition#\n"1178"emuMMC. You can create it from eMMC or a eMMC Backup.\n");11791180lv_obj_set_style(label_txt4, &hint_small_style);1181lv_obj_align(label_txt4, btn3, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3);11821183// Create Migrate emuMMC button.1184lv_obj_t *btn4 = lv_btn_create(h2, btn2);1185label_btn = lv_label_create(btn4, NULL);1186lv_label_set_static_text(label_btn, SYMBOL_SHUFFLE" Migrate emuMMC");1187lv_obj_align(btn4, label_txt4, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 2);1188lv_btn_set_action(btn4, LV_BTN_ACTION_CLICK, NULL);1189lv_btn_set_action(btn4, LV_BTN_ACTION_CLICK, _create_mbox_emummc_migrate);11901191label_txt4 = lv_label_create(h2, NULL);1192lv_label_set_recolor(label_txt4, true);1193lv_label_set_static_text(label_txt4,1194"Migrate a backup to a #C7EA46 SD File emuMMC# or repair an existing\n#C7EA46 SD Raw Partition emuMMC#.\n");1195lv_obj_set_style(label_txt4, &hint_small_style);1196lv_obj_align(label_txt4, btn4, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3);11971198return LV_RES_OK;1199}120012011202