Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
CTCaer
GitHub Repository: CTCaer/hekate
Path: blob/master/nyx/nyx_gui/frontend/gui_emummc_tools.c
1476 views
1
/*
2
* Copyright (c) 2019-2024 CTCaer
3
*
4
* This program is free software; you can redistribute it and/or modify it
5
* under the terms and conditions of the GNU General Public License,
6
* version 2, as published by the Free Software Foundation.
7
*
8
* This program is distributed in the hope it will be useful, but WITHOUT
9
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
10
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
11
* more details.
12
*
13
* You should have received a copy of the GNU General Public License
14
* along with this program. If not, see <http://www.gnu.org/licenses/>.
15
*/
16
17
#include <stdlib.h>
18
19
#include <bdk.h>
20
21
#include "gui.h"
22
#include "fe_emummc_tools.h"
23
#include "gui_tools_partition_manager.h"
24
#include <libs/fatfs/ff.h>
25
26
extern char *emmcsn_path_impl(char *path, char *sub_dir, char *filename, sdmmc_storage_t *storage);
27
28
typedef struct _mbr_ctxt_t
29
{
30
u32 available;
31
u32 sector[3];
32
u32 resized_cnt[3];
33
34
int part_idx;
35
u32 sector_start;
36
} mbr_ctxt_t;
37
38
static bool emummc_backup;
39
static mbr_ctxt_t mbr_ctx;
40
static lv_obj_t *emummc_manage_window;
41
static lv_res_t (*emummc_tools)(lv_obj_t *btn);
42
43
static lv_res_t _action_emummc_window_close(lv_obj_t *btn)
44
{
45
nyx_win_close_action_custom(btn);
46
47
// Delete and relaunch main emuMMC window.
48
lv_obj_del(emummc_manage_window);
49
(*emummc_tools)(NULL);
50
51
return LV_RES_INV;
52
}
53
54
static void _create_window_emummc()
55
{
56
emmc_tool_gui_t emmc_tool_gui_ctxt;
57
58
lv_obj_t *win;
59
if (!mbr_ctx.part_idx)
60
win = nyx_create_window_custom_close_btn(SYMBOL_DRIVE" Create SD File emuMMC", _action_emummc_window_close);
61
else
62
win = nyx_create_window_custom_close_btn(SYMBOL_DRIVE" Create SD Partition emuMMC", _action_emummc_window_close);
63
64
//Disable buttons.
65
nyx_window_toggle_buttons(win, true);
66
67
// Create important info container.
68
lv_obj_t *h1 = lv_cont_create(win, NULL);
69
lv_cont_set_fit(h1, false, true);
70
lv_obj_set_width(h1, (LV_HOR_RES / 9) * 5);
71
lv_obj_set_click(h1, false);
72
lv_cont_set_layout(h1, LV_LAYOUT_OFF);
73
74
static lv_style_t h_style;
75
lv_style_copy(&h_style, lv_cont_get_style(h1));
76
h_style.body.main_color = LV_COLOR_HEX(0x1d1d1d);
77
h_style.body.grad_color = h_style.body.main_color;
78
h_style.body.opa = LV_OPA_COVER;
79
80
// Chreate log container.
81
lv_obj_t *h2 = lv_cont_create(win, h1);
82
lv_cont_set_style(h2, &h_style);
83
lv_cont_set_fit(h2, false, false);
84
lv_obj_set_size(h2, (LV_HOR_RES / 11) * 4, LV_DPI * 5);
85
lv_cont_set_layout(h2, LV_LAYOUT_OFF);
86
lv_obj_align(h2, h1, LV_ALIGN_OUT_RIGHT_TOP, 0, LV_DPI / 5);
87
88
lv_obj_t *label_log = lv_label_create(h2, NULL);
89
lv_label_set_recolor(label_log, true);
90
lv_obj_set_style(label_log, &monospace_text);
91
lv_label_set_long_mode(label_log, LV_LABEL_LONG_BREAK);
92
lv_label_set_static_text(label_log, "");
93
lv_obj_set_width(label_log, lv_obj_get_width(h2));
94
lv_obj_align(label_log, h2, LV_ALIGN_IN_TOP_LEFT, LV_DPI / 10, LV_DPI / 10);
95
emmc_tool_gui_ctxt.label_log = label_log;
96
97
// Create elements for info container.
98
lv_obj_t *label_sep = lv_label_create(h1, NULL);
99
lv_label_set_static_text(label_sep, "");
100
101
lv_obj_t *label_info = lv_label_create(h1, NULL);
102
lv_label_set_recolor(label_info, true);
103
lv_obj_set_width(label_info, lv_obj_get_width(h1));
104
lv_label_set_static_text(label_info, "\n\n\n\n\n\n\n\n\n");
105
lv_obj_set_style(label_info, lv_theme_get_current()->label.prim);
106
lv_obj_align(label_info, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 10);
107
emmc_tool_gui_ctxt.label_info = label_info;
108
109
lv_obj_t *bar = lv_bar_create(h1, NULL);
110
lv_obj_set_size(bar, LV_DPI * 38 / 10, LV_DPI / 5);
111
lv_bar_set_range(bar, 0, 100);
112
lv_bar_set_value(bar, 0);
113
lv_obj_align(bar, label_info, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 8);
114
lv_obj_set_opa_scale(bar, LV_OPA_0);
115
lv_obj_set_opa_scale_enable(bar, true);
116
emmc_tool_gui_ctxt.bar = bar;
117
118
lv_obj_t *label_pct= lv_label_create(h1, NULL);
119
lv_label_set_recolor(label_pct, true);
120
lv_label_set_static_text(label_pct, " "SYMBOL_DOT" 0%");
121
lv_obj_set_style(label_pct, lv_theme_get_current()->label.prim);
122
lv_obj_align(label_pct, bar, LV_ALIGN_OUT_RIGHT_MID, LV_DPI / 20, 0);
123
lv_obj_set_opa_scale(label_pct, LV_OPA_0);
124
lv_obj_set_opa_scale_enable(label_pct, true);
125
emmc_tool_gui_ctxt.label_pct = label_pct;
126
127
lv_obj_t *label_finish = lv_label_create(h1, NULL);
128
lv_label_set_recolor(label_finish, true);
129
lv_label_set_static_text(label_finish, "");
130
lv_obj_set_style(label_finish, lv_theme_get_current()->label.prim);
131
lv_obj_align(label_finish, bar, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI * 9 / 20);
132
emmc_tool_gui_ctxt.label_finish = label_finish;
133
134
if (!mbr_ctx.part_idx)
135
dump_emummc_file(&emmc_tool_gui_ctxt);
136
else
137
dump_emummc_raw(&emmc_tool_gui_ctxt, mbr_ctx.part_idx, mbr_ctx.sector_start, mbr_ctx.resized_cnt[mbr_ctx.part_idx - 1]);
138
139
nyx_window_toggle_buttons(win, false);
140
}
141
142
static lv_res_t _create_emummc_raw_format(lv_obj_t * btns, const char * txt)
143
{
144
int btn_idx = lv_btnm_get_pressed(btns);
145
146
// Delete parent mbox.
147
mbox_action(btns, txt);
148
149
// Create partition window.
150
if (!btn_idx)
151
create_window_partition_manager(btns);
152
153
mbr_ctx.part_idx = 0;
154
mbr_ctx.sector_start = 0;
155
156
return LV_RES_INV;
157
}
158
159
static lv_res_t _create_emummc_raw_action(lv_obj_t * btns, const char * txt)
160
{
161
int btn_idx = lv_btnm_get_pressed(btns);
162
lv_obj_t *bg = lv_obj_get_parent(lv_obj_get_parent(btns));
163
164
mbr_ctx.sector_start = 0x8000; // Protective offset.
165
166
switch (btn_idx)
167
{
168
case 0:
169
mbr_ctx.part_idx = 1;
170
mbr_ctx.sector_start += mbr_ctx.sector[0];
171
break;
172
case 1:
173
mbr_ctx.part_idx = 2;
174
mbr_ctx.sector_start += mbr_ctx.sector[1];
175
break;
176
case 2:
177
mbr_ctx.part_idx = 3;
178
mbr_ctx.sector_start += mbr_ctx.sector[2];
179
break;
180
default:
181
break;
182
}
183
184
if (btn_idx < 3)
185
{
186
lv_obj_set_style(bg, &lv_style_transp);
187
_create_window_emummc();
188
}
189
190
mbr_ctx.part_idx = 0;
191
mbr_ctx.sector_start = 0;
192
193
mbox_action(btns, txt);
194
195
return LV_RES_INV;
196
}
197
198
static void _create_mbox_emummc_raw()
199
{
200
lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);
201
lv_obj_set_style(dark_bg, &mbox_darken);
202
lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);
203
204
static const char *mbox_btn_format[] = { "\222Continue", "\222Cancel", "" };
205
static char *mbox_btn_parts[] = { "\262Part 1", "\262Part 2", "\262Part 3", "\222Cancel", "" };
206
lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);
207
lv_mbox_set_recolor_text(mbox, true);
208
lv_obj_set_width(mbox, LV_HOR_RES / 9 * 6);
209
210
char *txt_buf = (char *)malloc(SZ_16K);
211
mbr_t *mbr = (mbr_t *)malloc(sizeof(mbr_t));
212
213
memset(&mbr_ctx, 0, sizeof(mbr_ctxt_t));
214
215
sd_mount();
216
sdmmc_storage_read(&sd_storage, 0, 1, mbr);
217
sd_unmount();
218
219
emmc_initialize(false);
220
221
u32 emmc_size_safe = emmc_storage.sec_cnt + 0xC000; // eMMC GPP size + BOOT0/1.
222
223
emmc_end();
224
225
for (int i = 1; i < 4; i++)
226
{
227
u32 part_size = mbr->partitions[i].size_sct;
228
u32 part_start = mbr->partitions[i].start_sct;
229
u8 part_type = mbr->partitions[i].type;
230
231
// Skip Linux, GPT (Android) and SFD partitions.
232
bool valid_part = (part_type != 0x83) && (part_type != 0xEE) && (part_type != 0xFF);
233
234
// Check if at least 4GB and start above 16MB.
235
if ((part_size >= 0x80F000) && part_start > 0x8000 && valid_part)
236
{
237
mbr_ctx.available |= BIT(i - 1);
238
mbr_ctx.sector[i - 1] = part_start;
239
240
// Only allow up to 16GB resized emuMMC.
241
if (part_size <= 0x2010000)
242
mbr_ctx.resized_cnt[i - 1] = part_size - 0xC000; // Save sectors count without protective size and BOOT0/1.
243
else if (part_size >= emmc_size_safe)
244
mbr_ctx.resized_cnt[i - 1] = 0;
245
else
246
{
247
mbr_ctx.available &= ~BIT(i - 1);
248
mbr_ctx.sector[i - 1] = 0;
249
}
250
}
251
}
252
253
if (mbr_ctx.available)
254
{
255
s_printf(txt_buf,
256
"#C7EA46 Found applicable partition(s)!#\n"
257
"#FF8000 Choose a partition to continue:#\n\n");
258
}
259
else
260
s_printf(txt_buf, "#FFDD00 Failed to find applicable partition!#\n\n");
261
262
s_printf(txt_buf + strlen(txt_buf),
263
"Partition table:\n"
264
"#C0C0C0 Part 0: Type: %02x, Start: %08x, Size: %08x#\n"
265
"#%s Part 1: Type: %02x, Start: %08x, Size: %08x#\n"
266
"#%s Part 2: Type: %02x, Start: %08x, Size: %08x#\n"
267
"#%s Part 3: Type: %02x, Start: %08x, Size: %08x#",
268
mbr->partitions[0].type, mbr->partitions[0].start_sct, mbr->partitions[0].size_sct,
269
(mbr_ctx.available & BIT(0)) ? (mbr_ctx.resized_cnt[0] ? "FFDD00" : "C7EA46") : "C0C0C0",
270
mbr->partitions[1].type, mbr->partitions[1].start_sct, mbr->partitions[1].size_sct,
271
(mbr_ctx.available & BIT(1)) ? (mbr_ctx.resized_cnt[1] ? "FFDD00" : "C7EA46") : "C0C0C0",
272
mbr->partitions[2].type, mbr->partitions[2].start_sct, mbr->partitions[2].size_sct,
273
(mbr_ctx.available & BIT(2)) ? (mbr_ctx.resized_cnt[2] ? "FFDD00" : "C7EA46") : "C0C0C0",
274
mbr->partitions[3].type, mbr->partitions[3].start_sct, mbr->partitions[3].size_sct);
275
276
if (mbr_ctx.resized_cnt[0] || mbr_ctx.resized_cnt[1] || mbr_ctx.resized_cnt[2])
277
strcat(txt_buf, "\n\n#FFDD00 Note:# Yellow entries have USER partition resized.");
278
279
if (!mbr_ctx.available)
280
strcat(txt_buf, "\n#FF8000 Do you want to partition the SD card?#\n"
281
"#FF8000 (You will be asked on how to proceed)#");
282
283
lv_mbox_set_text(mbox, txt_buf);
284
free(txt_buf);
285
free(mbr);
286
287
if (mbr_ctx.available)
288
{
289
// Check available partitions and enable the corresponding buttons.
290
if (mbr_ctx.available & 1)
291
mbox_btn_parts[0][0] = '\222';
292
else
293
mbox_btn_parts[0][0] = '\262';
294
if (mbr_ctx.available & 2)
295
mbox_btn_parts[1][0] = '\222';
296
else
297
mbox_btn_parts[1][0] = '\262';
298
if (mbr_ctx.available & 4)
299
mbox_btn_parts[2][0] = '\222';
300
else
301
mbox_btn_parts[2][0] = '\262';
302
303
lv_mbox_add_btns(mbox, (const char **)mbox_btn_parts, _create_emummc_raw_action);
304
}
305
else
306
lv_mbox_add_btns(mbox, mbox_btn_format, _create_emummc_raw_format);
307
308
lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);
309
lv_obj_set_top(mbox, true);
310
}
311
312
static lv_res_t _create_emummc_action(lv_obj_t * btns, const char * txt)
313
{
314
int btn_idx = lv_btnm_get_pressed(btns);
315
lv_obj_t *bg = lv_obj_get_parent(lv_obj_get_parent(btns));
316
317
mbr_ctx.part_idx = 0;
318
mbr_ctx.sector_start = 0;
319
320
switch (btn_idx)
321
{
322
case 0:
323
lv_obj_set_style(bg, &lv_style_transp);
324
_create_window_emummc();
325
break;
326
case 1:
327
_create_mbox_emummc_raw();
328
break;
329
}
330
331
mbox_action(btns, txt);
332
333
return LV_RES_INV;
334
}
335
336
static lv_res_t _create_mbox_emummc_create(lv_obj_t *btn)
337
{
338
if (!nyx_emmc_check_battery_enough())
339
return LV_RES_OK;
340
341
lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);
342
lv_obj_set_style(dark_bg, &mbox_darken);
343
lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);
344
345
static const char * mbox_btn_map[] = { "\222SD File", "\222SD Partition", "\222Cancel", "" };
346
lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);
347
lv_mbox_set_recolor_text(mbox, true);
348
lv_obj_set_width(mbox, LV_HOR_RES / 9 * 6);
349
350
lv_mbox_set_text(mbox,
351
"Welcome to #C7EA46 emuMMC# creation tool!\n\n"
352
"Please choose what type of emuMMC you want to create.\n"
353
"#FF8000 SD File# is saved as files in the FAT partition.\n"
354
"#FF8000 SD Partition# is saved as raw image in an available partition.");
355
356
lv_mbox_add_btns(mbox, mbox_btn_map, _create_emummc_action);
357
358
lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);
359
lv_obj_set_top(mbox, true);
360
361
return LV_RES_OK;
362
}
363
364
static void _change_raw_emummc_part_type()
365
{
366
mbr_t *mbr = (mbr_t *)malloc(sizeof(mbr_t));
367
sdmmc_storage_read(&sd_storage, 0, 1, mbr);
368
mbr->partitions[mbr_ctx.part_idx].type = 0xE0;
369
sdmmc_storage_write(&sd_storage, 0, 1, mbr);
370
free(mbr);
371
}
372
373
static lv_res_t _save_emummc_cfg_mig_mbox_action(lv_obj_t *btns, const char *txt)
374
{
375
// Delete main emuMMC and popup windows and relaunch main emuMMC window.
376
lv_obj_del(emummc_manage_window);
377
mbox_action(btns, txt);
378
379
(*emummc_tools)(NULL);
380
381
return LV_RES_INV;
382
}
383
384
static void _create_emummc_migrated_mbox()
385
{
386
lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);
387
lv_obj_set_style(dark_bg, &mbox_darken);
388
lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);
389
390
static const char *mbox_btn_map[] = { "\251", "OK", "\251", "" };
391
lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);
392
lv_mbox_set_recolor_text(mbox, true);
393
lv_obj_set_width(mbox, LV_HOR_RES / 9 * 4);
394
395
lv_mbox_set_text(mbox,
396
"#FF8000 emuMMC Configuration#\n\n"
397
"#96FF00 The emuMMC configuration#\n#96FF00 was saved to sd card!#");
398
399
lv_mbox_add_btns(mbox, mbox_btn_map, _save_emummc_cfg_mig_mbox_action);
400
401
lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);
402
lv_obj_set_top(mbox, true);
403
}
404
405
static void _migrate_sd_raw_based()
406
{
407
mbr_ctx.sector_start = 2;
408
409
sd_mount();
410
f_mkdir("emuMMC");
411
f_mkdir("emuMMC/ER00");
412
413
f_rename("Emutendo", "emuMMC/ER00/Nintendo");
414
FIL fp;
415
f_open(&fp, "emuMMC/ER00/raw_based", FA_CREATE_ALWAYS | FA_WRITE);
416
f_write(&fp, &mbr_ctx.sector_start, 4, NULL);
417
f_close(&fp);
418
419
save_emummc_cfg(1, mbr_ctx.sector_start, "emuMMC/ER00");
420
_create_emummc_migrated_mbox();
421
sd_unmount();
422
}
423
424
static void _migrate_sd_raw_emummc_based()
425
{
426
char *tmp = (char *)malloc(0x80);
427
s_printf(tmp, "emuMMC/RAW%d", mbr_ctx.part_idx);
428
429
sd_mount();
430
f_mkdir("emuMMC");
431
f_mkdir(tmp);
432
strcat(tmp, "/raw_based");
433
434
FIL fp;
435
if (!f_open(&fp, tmp, FA_CREATE_ALWAYS | FA_WRITE))
436
{
437
f_write(&fp, &mbr_ctx.sector_start, 4, NULL);
438
f_close(&fp);
439
}
440
441
s_printf(tmp, "emuMMC/RAW%d", mbr_ctx.part_idx);
442
443
_change_raw_emummc_part_type();
444
445
save_emummc_cfg(mbr_ctx.part_idx, mbr_ctx.sector_start, tmp);
446
_create_emummc_migrated_mbox();
447
free(tmp);
448
449
sd_unmount();
450
}
451
452
static void _migrate_sd_file_based()
453
{
454
sd_mount();
455
f_mkdir("emuMMC");
456
f_mkdir("emuMMC/EF00");
457
458
f_rename("Emutendo", "emuMMC/EF00/Nintendo");
459
FIL fp;
460
f_open(&fp, "emuMMC/EF00/file_based", FA_CREATE_ALWAYS | FA_WRITE);
461
f_close(&fp);
462
463
char *path = (char *)malloc(128);
464
char *path2 = (char *)malloc(128);
465
s_printf(path, "%c%c%c%c%s", 's', 'x', 'o', 's', "/emunand");
466
f_rename(path, "emuMMC/EF00/eMMC");
467
468
for (int i = 0; i < 2; i++)
469
{
470
s_printf(path, "emuMMC/EF00/eMMC/boot%d.bin", i);
471
s_printf(path2, "emuMMC/EF00/eMMC/BOOT%d", i);
472
f_rename(path, path2);
473
}
474
for (int i = 0; i < 8; i++)
475
{
476
s_printf(path, "emuMMC/EF00/eMMC/full.%02d.bin", i);
477
s_printf(path2, "emuMMC/EF00/eMMC/%02d", i);
478
f_rename(path, path2);
479
}
480
481
free(path);
482
free(path2);
483
484
save_emummc_cfg(0, 0, "emuMMC/EF00");
485
_create_emummc_migrated_mbox();
486
sd_unmount();
487
}
488
489
static void _migrate_sd_backup_file_based()
490
{
491
char *emu_path = (char *)malloc(128);
492
char *parts_path = (char *)malloc(128);
493
char *backup_path = (char *)malloc(128);
494
char *backup_file_path = (char *)malloc(128);
495
496
sd_mount();
497
f_mkdir("emuMMC");
498
499
strcpy(emu_path, "emuMMC/BK");
500
u32 base_len = strlen(emu_path);
501
502
for (int j = 0; j < 100; j++)
503
{
504
update_emummc_base_folder(emu_path, base_len, j);
505
if (f_stat(emu_path, NULL) == FR_NO_FILE)
506
break;
507
}
508
base_len = strlen(emu_path);
509
510
f_mkdir(emu_path);
511
strcat(emu_path, "/eMMC");
512
f_mkdir(emu_path);
513
514
FIL fp;
515
// Create file based flag.
516
strcpy(emu_path + base_len, "/file_based");
517
f_open(&fp, "emuMMC/BK00/file_based", FA_CREATE_ALWAYS | FA_WRITE);
518
f_close(&fp);
519
520
if (!emummc_backup)
521
emmcsn_path_impl(backup_path, "", "", NULL);
522
else
523
emmcsn_path_impl(backup_path, "/emummc", "", NULL);
524
525
// Move BOOT0.
526
s_printf(backup_file_path, "%s/BOOT0", backup_path);
527
strcpy(emu_path + base_len, "/eMMC/BOOT0");
528
f_rename(backup_file_path, emu_path);
529
530
// Move BOOT1.
531
s_printf(backup_file_path, "%s/BOOT1", backup_path);
532
strcpy(emu_path + base_len, "/eMMC/BOOT1");
533
f_rename(backup_file_path, emu_path);
534
535
// Move raw GPP.
536
bool multipart = false;
537
s_printf(backup_file_path, "%s/rawnand.bin", backup_path);
538
539
if (f_stat(backup_file_path, NULL))
540
multipart = true;
541
542
if (!multipart)
543
{
544
strcpy(emu_path + base_len, "/eMMC/00");
545
f_rename(backup_file_path, emu_path);
546
}
547
else
548
{
549
emu_path[base_len] = 0;
550
for (int i = 0; i < 32; i++)
551
{
552
s_printf(backup_file_path, "%s/rawnand.bin.%02d", backup_path, i);
553
s_printf(parts_path, "%s/eMMC/%02d", emu_path, i);
554
if (f_rename(backup_file_path, parts_path))
555
break;
556
}
557
}
558
559
free(emu_path);
560
free(parts_path);
561
free(backup_path);
562
free(backup_file_path);
563
564
save_emummc_cfg(0, 0, "emuMMC/BK00");
565
_create_emummc_migrated_mbox();
566
sd_unmount();
567
}
568
569
static lv_res_t _create_emummc_mig1_action(lv_obj_t * btns, const char * txt)
570
{
571
switch (lv_btnm_get_pressed(btns))
572
{
573
case 0:
574
_migrate_sd_file_based();
575
break;
576
case 1:
577
_migrate_sd_raw_based();
578
break;
579
}
580
581
mbr_ctx.part_idx = 0;
582
mbr_ctx.sector_start = 0;
583
584
mbox_action(btns, txt);
585
586
return LV_RES_INV;
587
}
588
589
static lv_res_t _create_emummc_mig0_action(lv_obj_t * btns, const char * txt)
590
{
591
switch (lv_btnm_get_pressed(btns))
592
{
593
case 0:
594
_migrate_sd_file_based();
595
break;
596
}
597
598
mbr_ctx.part_idx = 0;
599
mbr_ctx.sector_start = 0;
600
601
mbox_action(btns, txt);
602
603
return LV_RES_INV;
604
}
605
606
static lv_res_t _create_emummc_mig2_action(lv_obj_t * btns, const char * txt)
607
{
608
switch (lv_btnm_get_pressed(btns))
609
{
610
case 0:
611
_migrate_sd_raw_based();
612
break;
613
}
614
615
mbr_ctx.part_idx = 0;
616
mbr_ctx.sector_start = 0;
617
618
mbox_action(btns, txt);
619
620
return LV_RES_INV;
621
}
622
623
static lv_res_t _create_emummc_mig3_action(lv_obj_t * btns, const char * txt)
624
{
625
switch (lv_btnm_get_pressed(btns))
626
{
627
case 0:
628
_migrate_sd_raw_emummc_based();
629
break;
630
}
631
632
mbr_ctx.part_idx = 0;
633
mbr_ctx.sector_start = 0;
634
635
mbox_action(btns, txt);
636
637
return LV_RES_INV;
638
}
639
640
static lv_res_t _create_emummc_mig4_action(lv_obj_t * btns, const char * txt)
641
{
642
switch (lv_btnm_get_pressed(btns))
643
{
644
case 0:
645
_migrate_sd_backup_file_based();
646
break;
647
}
648
649
mbr_ctx.part_idx = 0;
650
mbr_ctx.sector_start = 0;
651
652
mbox_action(btns, txt);
653
654
return LV_RES_INV;
655
}
656
657
bool em_raw;
658
bool em_file;
659
static lv_res_t _create_emummc_migrate_action(lv_obj_t * btns, const char * txt)
660
{
661
bool backup = false;
662
bool emummc = false;
663
664
switch (lv_btnm_get_pressed(btns))
665
{
666
case 0:
667
backup = true;
668
break;
669
case 1:
670
emummc = true;
671
break;
672
case 2:
673
break;
674
case 3:
675
mbox_action(btns, txt);
676
return LV_RES_INV;
677
}
678
679
lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);
680
lv_obj_set_style(dark_bg, &mbox_darken);
681
lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);
682
683
static const char *mbox_btn_map[] = { "\222Continue", "\222Cancel", "" };
684
static const char *mbox_btn_map1[] = { "\222SD File", "\222SD Partition", "\222Cancel", "" };
685
static const char *mbox_btn_map3[] = { "\251", "OK", "\251", "" };
686
lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);
687
lv_mbox_set_recolor_text(mbox, true);
688
lv_obj_set_width(mbox, LV_HOR_RES / 9 * 6);
689
690
char *txt_buf = (char *)malloc(SZ_16K);
691
692
if (backup)
693
{
694
if (!emummc_backup)
695
s_printf(txt_buf,
696
"#C7EA46 Found suitable eMMC backup!#\n\n"
697
"#FF8000 Do you want to migrate it?#\n");
698
else
699
s_printf(txt_buf,
700
"#C7EA46 Found suitable emuMMC backup!#\n\n"
701
"#FF8000 Do you want to migrate it?#\n");
702
lv_mbox_add_btns(mbox, mbox_btn_map, _create_emummc_mig4_action);
703
}
704
else if (emummc)
705
{
706
s_printf(txt_buf,
707
"#C7EA46 Found SD Partition based emuMMC!#\n\n"
708
"#FF8000 Do you want to repair the config and partition type for it?#\n");
709
lv_mbox_add_btns(mbox, mbox_btn_map, _create_emummc_mig3_action);
710
}
711
else if (em_raw && em_file)
712
{
713
s_printf(txt_buf,
714
"#C7EA46 Found both foreign SD File and Partition emunand!#\n\n"
715
"#FF8000 Choose what to migrate:#\n");
716
lv_mbox_add_btns(mbox, mbox_btn_map1, _create_emummc_mig1_action);
717
}
718
else if (em_raw)
719
{
720
s_printf(txt_buf,
721
"#C7EA46 Found foreign SD Partition emunand!#\n\n"
722
"#FF8000 Do you want to migrate it?#\n");
723
lv_mbox_add_btns(mbox, mbox_btn_map, _create_emummc_mig2_action);
724
}
725
else if (em_file)
726
{
727
s_printf(txt_buf,
728
"#C7EA46 Found foreign SD File emunand!#\n\n"
729
"#FF8000 Do you want to migrate it?#\n");
730
lv_mbox_add_btns(mbox, mbox_btn_map, _create_emummc_mig0_action);
731
}
732
else
733
{
734
s_printf(txt_buf, "No emuMMC or foreign emunand found!\n");
735
lv_mbox_add_btns(mbox, mbox_btn_map3, mbox_action);
736
}
737
738
lv_mbox_set_text(mbox, txt_buf);
739
free(txt_buf);
740
741
lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);
742
lv_obj_set_top(mbox, true);
743
744
mbox_action(btns, txt);
745
746
return LV_RES_INV;
747
}
748
749
typedef struct _emummc_images_t
750
{
751
dirlist_t *dirlist;
752
u32 part_sector[3];
753
u32 part_type[3];
754
u32 part_end[3];
755
char part_path[3 * 128];
756
lv_obj_t *win;
757
} emummc_images_t;
758
759
static lv_res_t _create_mbox_emummc_migrate(lv_obj_t *btn)
760
{
761
lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);
762
lv_obj_set_style(dark_bg, &mbox_darken);
763
lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);
764
765
static char *mbox_btn_map[] = { "\262Backup", "\262Fix RAW", "\262Emunand", "\222Cancel", "" };
766
lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);
767
lv_mbox_set_recolor_text(mbox, true);
768
lv_obj_set_width(mbox, LV_HOR_RES / 9 * 6);
769
770
lv_mbox_set_text(mbox,
771
"Welcome to #C7EA46 emuMMC# migration tool!\n\n"
772
"Please choose what type of migration you want to do.\n"
773
"Anything that was not found will have the button disabled.");
774
775
char *path_buf = (char *)malloc(0x512);
776
mbr_t *mbr = (mbr_t *)malloc(sizeof(mbr_t));
777
u8 *efi_part = (u8 *)malloc(0x200);
778
779
sd_mount();
780
sdmmc_storage_read(&sd_storage, 0, 1, mbr);
781
782
emmc_initialize(false);
783
784
em_raw = false;
785
em_file = false;
786
bool backup = false;
787
bool emummc = false;
788
bool rawnand_backup = false;
789
790
mbr_ctx.sector_start = 0;
791
mbr_ctx.part_idx = 0;
792
793
// Try to find a partition based emuMMC.
794
for (int i = 1; i < 4; i++)
795
{
796
mbr_ctx.sector_start = mbr->partitions[i].start_sct;
797
798
if (!mbr_ctx.sector_start)
799
continue;
800
801
sdmmc_storage_read(&sd_storage, mbr_ctx.sector_start + 0xC001, 1, efi_part);
802
if (!memcmp(efi_part, "EFI PART", 8))
803
{
804
mbr_ctx.sector_start += 0x8000;
805
emummc = true;
806
mbr_ctx.part_idx = i;
807
break;
808
}
809
else
810
{
811
sdmmc_storage_read(&sd_storage, mbr_ctx.sector_start + 0x4001, 1, efi_part);
812
if (!memcmp(efi_part, "EFI PART", 8))
813
{
814
emummc = true;
815
mbr_ctx.part_idx = i;
816
break;
817
}
818
}
819
}
820
821
if (!mbr_ctx.part_idx)
822
{
823
sdmmc_storage_read(&sd_storage, 0x4003, 1, efi_part);
824
if (!memcmp(efi_part, "EFI PART", 8))
825
em_raw = true;
826
}
827
828
s_printf(path_buf, "%c%c%c%c%s", 's', 'x', 'o','s', "/emunand/boot0.bin");
829
830
if (!f_stat(path_buf, NULL))
831
em_file = true;
832
833
emummc_backup = false;
834
835
emmcsn_path_impl(path_buf, "", "BOOT0", &emmc_storage);
836
if (!f_stat(path_buf, NULL))
837
backup = true;
838
839
emmcsn_path_impl(path_buf, "", "rawnand.bin", &emmc_storage);
840
if (!f_stat(path_buf, NULL))
841
rawnand_backup = true;
842
843
emmcsn_path_impl(path_buf, "", "rawnand.bin.00", &emmc_storage);
844
if (!f_stat(path_buf, NULL))
845
rawnand_backup = true;
846
847
backup = backup && rawnand_backup;
848
849
if (!backup)
850
{
851
rawnand_backup = false;
852
emummc_backup = true;
853
854
emmcsn_path_impl(path_buf, "/emummc", "BOOT0", &emmc_storage);
855
if (!f_stat(path_buf, NULL))
856
backup = true;
857
858
emmcsn_path_impl(path_buf, "/emummc", "rawnand.bin", &emmc_storage);
859
if (!f_stat(path_buf, NULL))
860
rawnand_backup = true;
861
862
emmcsn_path_impl(path_buf, "/emummc", "rawnand.bin.00", &emmc_storage);
863
if (!f_stat(path_buf, NULL))
864
rawnand_backup = true;
865
866
backup = backup && rawnand_backup;
867
}
868
869
sd_unmount();
870
emmc_end();
871
872
// Check available types and enable the corresponding buttons.
873
if (backup)
874
mbox_btn_map[0][0] = '\222';
875
else
876
mbox_btn_map[0][0] = '\262';
877
if (emummc)
878
mbox_btn_map[1][0] = '\222';
879
else
880
mbox_btn_map[1][0] = '\262';
881
if (em_raw || em_file)
882
mbox_btn_map[2][0] = '\222';
883
else
884
mbox_btn_map[2][0] = '\262';
885
886
free(path_buf);
887
free(mbr);
888
free(efi_part);
889
890
lv_mbox_add_btns(mbox, (const char **)mbox_btn_map, _create_emummc_migrate_action);
891
892
lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);
893
lv_obj_set_top(mbox, true);
894
895
return LV_RES_OK;
896
}
897
898
static emummc_images_t *emummc_img;
899
900
static lv_res_t _save_emummc_cfg_mbox_action(lv_obj_t *btns, const char *txt)
901
{
902
// Free components, delete main emuMMC and popup windows and relaunch main emuMMC window.
903
free(emummc_img->dirlist);
904
lv_obj_del(emummc_img->win);
905
lv_obj_del(emummc_manage_window);
906
free(emummc_img);
907
908
mbox_action(btns, txt);
909
910
(*emummc_tools)(NULL);
911
912
return LV_RES_INV;
913
}
914
915
static void _create_emummc_saved_mbox()
916
{
917
lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL);
918
lv_obj_set_style(dark_bg, &mbox_darken);
919
lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES);
920
921
static const char *mbox_btn_map[] = { "\251", "OK", "\251", "" };
922
lv_obj_t * mbox = lv_mbox_create(dark_bg, NULL);
923
lv_mbox_set_recolor_text(mbox, true);
924
lv_obj_set_width(mbox, LV_HOR_RES / 9 * 4);
925
926
lv_mbox_set_text(mbox,
927
"#FF8000 emuMMC Configuration#\n\n"
928
"#96FF00 The emuMMC configuration#\n#96FF00 was saved to sd card!#");
929
930
lv_mbox_add_btns(mbox, mbox_btn_map, _save_emummc_cfg_mbox_action);
931
932
lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0);
933
lv_obj_set_top(mbox, true);
934
}
935
936
static lv_res_t _save_raw_emummc_cfg_action(lv_obj_t * btn)
937
{
938
lv_btn_ext_t *ext = lv_obj_get_ext_attr(btn);
939
switch (ext->idx)
940
{
941
case 0:
942
save_emummc_cfg(1, emummc_img->part_sector[0], &emummc_img->part_path[0]);
943
break;
944
case 1:
945
save_emummc_cfg(2, emummc_img->part_sector[1], &emummc_img->part_path[128]);
946
break;
947
case 2:
948
save_emummc_cfg(3, emummc_img->part_sector[2], &emummc_img->part_path[256]);
949
break;
950
}
951
952
_create_emummc_saved_mbox();
953
sd_unmount();
954
955
return LV_RES_INV;
956
}
957
958
static lv_res_t _save_disable_emummc_cfg_action(lv_obj_t * btn)
959
{
960
save_emummc_cfg(0, 0, NULL);
961
_create_emummc_saved_mbox();
962
sd_unmount();
963
964
return LV_RES_INV;
965
}
966
967
static lv_res_t _save_file_emummc_cfg_action(lv_obj_t *btn)
968
{
969
save_emummc_cfg(0, 0, lv_list_get_btn_text(btn));
970
_create_emummc_saved_mbox();
971
sd_unmount();
972
973
return LV_RES_INV;
974
}
975
976
static lv_res_t _create_change_emummc_window(lv_obj_t *btn_caller)
977
{
978
lv_obj_t *win = nyx_create_standard_window(SYMBOL_SETTINGS" Change emuMMC");
979
lv_win_add_btn(win, NULL, SYMBOL_POWER" Disable", _save_disable_emummc_cfg_action);
980
981
sd_mount();
982
983
emummc_img = malloc(sizeof(emummc_images_t));
984
emummc_img->win = win;
985
986
mbr_t *mbr = (mbr_t *)malloc(sizeof(mbr_t));
987
char *path = malloc(512);
988
989
sdmmc_storage_read(&sd_storage, 0, 1, mbr);
990
991
memset(emummc_img->part_path, 0, 3 * 128);
992
993
for (int i = 1; i < 4; i++)
994
{
995
emummc_img->part_sector[i - 1] = mbr->partitions[i].start_sct;
996
emummc_img->part_end[i - 1] = emummc_img->part_sector[i - 1] + mbr->partitions[i].size_sct - 1;
997
emummc_img->part_type[i - 1] = mbr->partitions[i].type;
998
}
999
free(mbr);
1000
1001
emummc_img->dirlist = dirlist("emuMMC", NULL, false, true);
1002
1003
if (!emummc_img->dirlist)
1004
goto out0;
1005
1006
u32 emummc_idx = 0;
1007
1008
FIL fp;
1009
1010
// Check for sd raw partitions, based on the folders in /emuMMC.
1011
while (emummc_img->dirlist->name[emummc_idx])
1012
{
1013
s_printf(path, "emuMMC/%s/raw_based", emummc_img->dirlist->name[emummc_idx]);
1014
1015
if (!f_stat(path, NULL))
1016
{
1017
f_open(&fp, path, FA_READ);
1018
u32 curr_list_sector = 0;
1019
f_read(&fp, &curr_list_sector, 4, NULL);
1020
f_close(&fp);
1021
1022
// Check if there's a HOS image there.
1023
if ((curr_list_sector == 2) || (emummc_img->part_sector[0] && curr_list_sector >= emummc_img->part_sector[0] &&
1024
curr_list_sector < emummc_img->part_end[0] && emummc_img->part_type[0] != 0x83))
1025
{
1026
s_printf(&emummc_img->part_path[0], "emuMMC/%s", emummc_img->dirlist->name[emummc_idx]);
1027
emummc_img->part_sector[0] = curr_list_sector;
1028
emummc_img->part_end[0] = 0;
1029
}
1030
else if (emummc_img->part_sector[1] && curr_list_sector >= emummc_img->part_sector[1] &&
1031
curr_list_sector < emummc_img->part_end[1] && emummc_img->part_type[1] != 0x83)
1032
{
1033
s_printf(&emummc_img->part_path[1 * 128], "emuMMC/%s", emummc_img->dirlist->name[emummc_idx]);
1034
emummc_img->part_sector[1] = curr_list_sector;
1035
emummc_img->part_end[1] = 0;
1036
}
1037
else if (emummc_img->part_sector[2] && curr_list_sector >= emummc_img->part_sector[2] &&
1038
curr_list_sector < emummc_img->part_end[2] && emummc_img->part_type[2] != 0x83)
1039
{
1040
s_printf(&emummc_img->part_path[2 * 128], "emuMMC/%s", emummc_img->dirlist->name[emummc_idx]);
1041
emummc_img->part_sector[2] = curr_list_sector;
1042
emummc_img->part_end[2] = 0;
1043
}
1044
}
1045
emummc_idx++;
1046
}
1047
1048
emummc_idx = 0;
1049
u32 file_based_idx = 0;
1050
1051
// Sanitize the directory list with sd file based ones.
1052
while (emummc_img->dirlist->name[emummc_idx])
1053
{
1054
s_printf(path, "emuMMC/%s/file_based", emummc_img->dirlist->name[emummc_idx]);
1055
1056
if (!f_stat(path, NULL))
1057
{
1058
char *tmp = emummc_img->dirlist->name[emummc_idx];
1059
memcpy(emummc_img->dirlist->name[file_based_idx], tmp, strlen(tmp) + 1);
1060
file_based_idx++;
1061
}
1062
emummc_idx++;
1063
}
1064
emummc_img->dirlist->name[file_based_idx] = NULL;
1065
1066
out0:;
1067
static lv_style_t h_style;
1068
lv_style_copy(&h_style, &lv_style_transp);
1069
h_style.body.padding.inner = 0;
1070
h_style.body.padding.hor = LV_DPI - (LV_DPI / 4);
1071
h_style.body.padding.ver = LV_DPI / 6;
1072
1073
// Create SD Raw Partitions container.
1074
lv_obj_t *h1 = lv_cont_create(win, NULL);
1075
lv_cont_set_style(h1, &h_style);
1076
lv_cont_set_fit(h1, false, true);
1077
lv_obj_set_width(h1, (LV_HOR_RES / 9) * 4);
1078
lv_obj_set_click(h1, false);
1079
lv_cont_set_layout(h1, LV_LAYOUT_OFF);
1080
1081
lv_obj_t *label_sep = lv_label_create(h1, NULL);
1082
lv_label_set_static_text(label_sep, "");
1083
1084
lv_obj_t *label_txt = lv_label_create(h1, NULL);
1085
lv_label_set_static_text(label_txt, "SD Raw Partitions");
1086
lv_obj_set_style(label_txt, lv_theme_get_current()->label.prim);
1087
lv_obj_align(label_txt, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, -(LV_DPI / 2));
1088
1089
lv_obj_t *line_sep = lv_line_create(h1, NULL);
1090
static const lv_point_t line_pp[] = { {0, 0}, { LV_HOR_RES - (LV_DPI - (LV_DPI / 4)) * 2, 0} };
1091
lv_line_set_points(line_sep, line_pp, 2);
1092
lv_line_set_style(line_sep, lv_theme_get_current()->line.decor);
1093
lv_obj_align(line_sep, label_txt, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 8);
1094
1095
lv_obj_t *btn = NULL;
1096
lv_btn_ext_t *ext;
1097
lv_obj_t *btn_label = NULL;
1098
lv_obj_t *lv_desc = NULL;
1099
char *txt_buf = malloc(SZ_16K);
1100
1101
// Create RAW buttons.
1102
for (u32 raw_btn_idx = 0; raw_btn_idx < 3; raw_btn_idx++)
1103
{
1104
btn = lv_btn_create(h1, btn);
1105
ext = lv_obj_get_ext_attr(btn);
1106
ext->idx = raw_btn_idx;
1107
btn_label = lv_label_create(btn, btn_label);
1108
1109
lv_btn_set_state(btn, LV_BTN_STATE_REL);
1110
lv_obj_set_click(btn, true);
1111
1112
if (emummc_img->part_type[raw_btn_idx] != 0x83)
1113
{
1114
s_printf(txt_buf, "SD RAW %d", raw_btn_idx + 1);
1115
lv_label_set_text(btn_label, txt_buf);
1116
}
1117
1118
if (!emummc_img->part_sector[raw_btn_idx] || emummc_img->part_type[raw_btn_idx] == 0x83 || !emummc_img->part_path[raw_btn_idx * 128])
1119
{
1120
lv_btn_set_state(btn, LV_BTN_STATE_INA);
1121
lv_obj_set_click(btn, false);
1122
1123
if (emummc_img->part_type[raw_btn_idx] == 0x83)
1124
lv_label_set_static_text(btn_label, "Linux");
1125
}
1126
1127
if (!raw_btn_idx)
1128
{
1129
lv_btn_set_fit(btn, false, true);
1130
lv_obj_set_width(btn, LV_DPI * 3);
1131
lv_obj_align(btn, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 2, LV_DPI / 5);
1132
}
1133
else
1134
lv_obj_align(btn, lv_desc, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3);
1135
1136
lv_btn_set_action(btn, LV_BTN_ACTION_CLICK, _save_raw_emummc_cfg_action);
1137
1138
lv_desc = lv_label_create(h1, lv_desc);
1139
lv_label_set_recolor(lv_desc, true);
1140
lv_obj_set_style(lv_desc, &hint_small_style);
1141
1142
s_printf(txt_buf, "Sector start: 0x%08X\nFolder: %s", emummc_img->part_sector[raw_btn_idx], &emummc_img->part_path[raw_btn_idx * 128]);
1143
lv_label_set_text(lv_desc, txt_buf);
1144
lv_obj_align(lv_desc, btn, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 5);
1145
}
1146
free(txt_buf);
1147
1148
// Create SD File Based container.
1149
lv_obj_t *h2 = lv_cont_create(win, NULL);
1150
lv_cont_set_style(h2, &h_style);
1151
lv_cont_set_fit(h2, false, true);
1152
lv_obj_set_width(h2, (LV_HOR_RES / 9) * 4);
1153
lv_obj_set_click(h2, false);
1154
lv_cont_set_layout(h2, LV_LAYOUT_OFF);
1155
lv_obj_align(h2, h1, LV_ALIGN_OUT_RIGHT_TOP, LV_DPI * 17 / 29, 0);
1156
1157
label_sep = lv_label_create(h2, NULL);
1158
lv_label_set_static_text(label_sep, "");
1159
1160
lv_obj_t *label_txt3 = lv_label_create(h2, NULL);
1161
lv_label_set_static_text(label_txt3, "SD File Based");
1162
lv_obj_set_style(label_txt3, lv_theme_get_current()->label.prim);
1163
lv_obj_align(label_txt3, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, -LV_DPI / 7);
1164
1165
line_sep = lv_line_create(h2, line_sep);
1166
lv_obj_align(line_sep, label_txt3, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 2), LV_DPI / 8);
1167
lv_line_set_style(line_sep, lv_theme_get_current()->line.decor);
1168
1169
lv_obj_t *list_sd_based = lv_list_create(h2, NULL);
1170
lv_obj_align(list_sd_based, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 2, LV_DPI / 4);
1171
1172
lv_obj_set_size(list_sd_based, LV_HOR_RES * 4 / 10, LV_VER_RES * 6 / 10);
1173
lv_list_set_single_mode(list_sd_based, true);
1174
1175
if (!emummc_img->dirlist)
1176
goto out1;
1177
1178
emummc_idx = 0;
1179
1180
// Add file based to the list.
1181
while (emummc_img->dirlist->name[emummc_idx])
1182
{
1183
s_printf(path, "emuMMC/%s", emummc_img->dirlist->name[emummc_idx]);
1184
1185
lv_list_add(list_sd_based, NULL, path, _save_file_emummc_cfg_action);
1186
1187
emummc_idx++;
1188
}
1189
1190
out1:
1191
free(path);
1192
sd_unmount();
1193
1194
return LV_RES_OK;
1195
}
1196
1197
lv_res_t create_win_emummc_tools(lv_obj_t *btn)
1198
{
1199
lv_obj_t *win = nyx_create_standard_window(SYMBOL_EDIT" emuMMC Manage");
1200
1201
// Set resources to be managed by other windows.
1202
emummc_manage_window = win;
1203
emummc_tools = (void *)create_win_emummc_tools;
1204
1205
sd_mount();
1206
1207
emummc_cfg_t emu_info;
1208
load_emummc_cfg(&emu_info);
1209
1210
sd_unmount();
1211
1212
static lv_style_t h_style;
1213
lv_style_copy(&h_style, &lv_style_transp);
1214
h_style.body.padding.inner = 0;
1215
h_style.body.padding.hor = LV_DPI - (LV_DPI / 4);
1216
h_style.body.padding.ver = LV_DPI / 9;
1217
1218
// Create emuMMC Info & Selection container.
1219
lv_obj_t *h1 = lv_cont_create(win, NULL);
1220
lv_cont_set_style(h1, &h_style);
1221
lv_cont_set_fit(h1, false, true);
1222
lv_obj_set_width(h1, (LV_HOR_RES / 9) * 4);
1223
lv_obj_set_click(h1, false);
1224
lv_cont_set_layout(h1, LV_LAYOUT_OFF);
1225
1226
lv_obj_t *label_sep = lv_label_create(h1, NULL);
1227
lv_label_set_static_text(label_sep, "");
1228
1229
lv_obj_t *label_txt = lv_label_create(h1, NULL);
1230
lv_label_set_static_text(label_txt, "emuMMC Info & Selection");
1231
lv_obj_set_style(label_txt, lv_theme_get_current()->label.prim);
1232
lv_obj_align(label_txt, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, -LV_DPI / 9);
1233
1234
lv_obj_t *line_sep = lv_line_create(h1, NULL);
1235
static const lv_point_t line_pp[] = { {0, 0}, { LV_HOR_RES - (LV_DPI - (LV_DPI / 4)) * 2, 0} };
1236
lv_line_set_points(line_sep, line_pp, 2);
1237
lv_line_set_style(line_sep, lv_theme_get_current()->line.decor);
1238
lv_obj_align(line_sep, label_txt, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 8);
1239
1240
// Create emuMMC info labels.
1241
lv_obj_t *label_btn = lv_label_create(h1, NULL);
1242
lv_label_set_recolor(label_btn, true);
1243
lv_label_set_static_text(label_btn, emu_info.enabled ? "#96FF00 "SYMBOL_OK" Enabled!#" : "#FF8000 "SYMBOL_CLOSE" Disabled!#");
1244
lv_obj_align(label_btn, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 4);
1245
1246
lv_obj_t *label_txt2 = lv_label_create(h1, NULL);
1247
lv_label_set_recolor(label_txt2, true);
1248
char *txt_buf = (char *)malloc(SZ_16K);
1249
1250
if (emu_info.enabled)
1251
{
1252
if (emu_info.sector)
1253
s_printf(txt_buf, "#00DDFF Type:# SD Raw Partition\n#00DDFF Sector:# 0x%08X\n#00DDFF Nintendo folder:# %s",
1254
emu_info.sector, emu_info.nintendo_path ? emu_info.nintendo_path : "");
1255
else
1256
s_printf(txt_buf, "#00DDFF Type:# SD File\n#00DDFF Base folder:# %s\n#00DDFF Nintendo folder:# %s",
1257
emu_info.path ? emu_info.path : "", emu_info.nintendo_path ? emu_info.nintendo_path : "");
1258
1259
lv_label_set_text(label_txt2, txt_buf);
1260
}
1261
else
1262
{
1263
lv_label_set_static_text(label_txt2, "emuMMC is disabled and eMMC will be used for boot.\n\n");
1264
}
1265
1266
if (emu_info.path)
1267
free(emu_info.path);
1268
if (emu_info.nintendo_path)
1269
free(emu_info.nintendo_path);
1270
free(txt_buf);
1271
1272
lv_obj_set_style(label_txt2, &hint_small_style);
1273
lv_obj_align(label_txt2, label_btn, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3);
1274
1275
// Create Change emuMMC button.
1276
lv_obj_t *btn2 = lv_btn_create(h1, NULL);
1277
lv_btn_set_fit(btn2, true, true);
1278
label_btn = lv_label_create(btn2, NULL);
1279
lv_label_set_static_text(label_btn, SYMBOL_SETTINGS" Change emuMMC");
1280
lv_obj_align(btn2, label_txt2, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI * 6 / 10);
1281
lv_btn_set_action(btn2, LV_BTN_ACTION_CLICK, _create_change_emummc_window);
1282
1283
label_txt2 = lv_label_create(h1, NULL);
1284
lv_label_set_recolor(label_txt2, true);
1285
lv_label_set_static_text(label_txt2,
1286
"Choose between images created in the emuMMC folder\n"
1287
"or in SD card partitions. You can have at most 3 partition\n"
1288
"based and countless file based.");
1289
1290
lv_obj_set_style(label_txt2, &hint_small_style);
1291
lv_obj_align(label_txt2, btn2, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3);
1292
1293
// Create emuMMC Tools container.
1294
lv_obj_t *h2 = lv_cont_create(win, NULL);
1295
lv_cont_set_style(h2, &h_style);
1296
lv_cont_set_fit(h2, false, true);
1297
lv_obj_set_width(h2, (LV_HOR_RES / 9) * 4);
1298
lv_obj_set_click(h2, false);
1299
lv_cont_set_layout(h2, LV_LAYOUT_OFF);
1300
lv_obj_align(h2, h1, LV_ALIGN_OUT_RIGHT_TOP, LV_DPI * 17 / 29, 0);
1301
1302
label_sep = lv_label_create(h2, NULL);
1303
lv_label_set_static_text(label_sep, "");
1304
1305
lv_obj_t *label_txt3 = lv_label_create(h2, NULL);
1306
lv_label_set_static_text(label_txt3, "emuMMC Tools");
1307
lv_obj_set_style(label_txt3, lv_theme_get_current()->label.prim);
1308
lv_obj_align(label_txt3, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, 0);
1309
1310
line_sep = lv_line_create(h2, line_sep);
1311
lv_obj_align(line_sep, label_txt3, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 8);
1312
1313
// Create Create emuMMC button.
1314
lv_obj_t *btn3 = lv_btn_create(h2, btn2);
1315
label_btn = lv_label_create(btn3, NULL);
1316
lv_btn_set_fit(btn3, true, true);
1317
lv_label_set_static_text(label_btn, SYMBOL_DRIVE" Create emuMMC");
1318
lv_obj_align(btn3, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 4);
1319
lv_btn_set_action(btn3, LV_BTN_ACTION_CLICK, _create_mbox_emummc_create);
1320
1321
lv_obj_t *label_txt4 = lv_label_create(h2, NULL);
1322
lv_label_set_recolor(label_txt4, true);
1323
lv_label_set_static_text(label_txt4,
1324
"Allows you to create a new #C7EA46 SD File# or #C7EA46 SD Raw Partition#\n"
1325
"emuMMC. You can create it from eMMC or a eMMC Backup.");
1326
1327
lv_obj_set_style(label_txt4, &hint_small_style);
1328
lv_obj_align(label_txt4, btn3, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3);
1329
1330
// Create Migrate emuMMC button.
1331
lv_obj_t *btn4 = lv_btn_create(h2, btn2);
1332
label_btn = lv_label_create(btn4, NULL);
1333
lv_label_set_static_text(label_btn, SYMBOL_SHUFFLE" Migrate emuMMC");
1334
lv_obj_align(btn4, label_txt4, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 2);
1335
lv_btn_set_action(btn4, LV_BTN_ACTION_CLICK, NULL);
1336
lv_btn_set_action(btn4, LV_BTN_ACTION_CLICK, _create_mbox_emummc_migrate);
1337
1338
label_txt4 = lv_label_create(h2, NULL);
1339
lv_label_set_recolor(label_txt4, true);
1340
lv_label_set_static_text(label_txt4,
1341
"Migrate a backup to a #C7EA46 SD File# or repair existing #C7EA46 SD Raw Partition#.\n"
1342
"Additionally it allows you to migrate from other emunand\nsolutions.");
1343
lv_obj_set_style(label_txt4, &hint_small_style);
1344
lv_obj_align(label_txt4, btn4, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3);
1345
1346
return LV_RES_OK;
1347
}
1348
1349