#include #include #include #include #include <3ds.h> #include "section.h" #include "task/uitask.h" #include "../error.h" #include "../info.h" #include "../prompt.h" #include "../resources.h" #include "../ui.h" #include "../../core/error.h" #include "../../core/screen.h" #include "../../core/util.h" static Result dumpnand_is_src_directory(void* data, u32 index, bool* isDirectory) { *isDirectory = false; return 0; } static Result dumpnand_make_dst_directory(void* data, u32 index) { return 0; } static Result dumpnand_open_src(void* data, u32 index, u32* handle) { return FSUSER_OpenFileDirectly(handle, ARCHIVE_NAND_W_FS, fsMakePath(PATH_EMPTY, ""), fsMakePath(PATH_UTF16, u"/"), FS_OPEN_READ, 0); } static Result dumpnand_close_src(void* data, u32 index, bool succeeded, u32 handle) { return FSFILE_Close(handle); } static Result dumpnand_get_src_size(void* data, u32 handle, u64* size) { return FSFILE_GetSize(handle, size); } static Result dumpnand_read_src(void* data, u32 handle, u32* bytesRead, void* buffer, u64 offset, u32 size) { return FSFILE_Read(handle, bytesRead, offset, buffer, size); } static Result dumpnand_open_dst(void* data, u32 index, void* initialReadBlock, u64 size, u32* handle) { Result res = 0; FS_Archive sdmcArchive = 0; if(R_SUCCEEDED(res = FSUSER_OpenArchive(&sdmcArchive, ARCHIVE_SDMC, fsMakePath(PATH_EMPTY, "")))) { if(R_SUCCEEDED(res = util_ensure_dir(sdmcArchive, "/fbi/")) && R_SUCCEEDED(res = util_ensure_dir(sdmcArchive, "/fbi/nand/"))) { time_t t = time(NULL); struct tm* timeInfo = localtime(&t); char path[FILE_PATH_MAX]; strftime(path, sizeof(path), "/fbi/nand/NAND_%m-%d-%y_%H-%M-%S.bin", timeInfo); FS_Path* fsPath = util_make_path_utf8(path); if(fsPath != NULL) { res = FSUSER_OpenFileDirectly(handle, ARCHIVE_SDMC, fsMakePath(PATH_EMPTY, ""), *fsPath, FS_OPEN_WRITE | FS_OPEN_CREATE, 0); util_free_path_utf8(fsPath); } else { res = R_APP_OUT_OF_MEMORY; } } FSUSER_CloseArchive(sdmcArchive); } return res; } static Result dumpnand_close_dst(void* data, u32 index, bool succeeded, u32 handle) { return FSFILE_Close(handle); } static Result dumpnand_write_dst(void* data, u32 handle, u32* bytesWritten, void* buffer, u64 offset, u32 size) { return FSFILE_Write(handle, bytesWritten, offset, buffer, size, 0); } static Result dumpnand_suspend_transfer(void* data, u32 index, u32* srcHandle, u32* dstHandle) { return 0; } static Result dumpnand_restore_transfer(void* data, u32 index, u32* srcHandle, u32* dstHandle) { return 0; } static Result dumpnand_suspend(void* data, u32 index) { return 0; } static Result dumpnand_restore(void* data, u32 index) { return 0; } static bool dumpnand_error(void* data, u32 index, Result res, ui_view** errorView) { *errorView = error_display_res(NULL, NULL, res, "Failed to dump NAND."); return true; } static void dumpnand_update(ui_view* view, void* data, float* progress, char* text) { data_op_data* dumpData = (data_op_data*) data; if(dumpData->finished) { ui_pop(); info_destroy(view); if(R_SUCCEEDED(dumpData->result)) { prompt_display_notify("Success", "NAND dumped.", COLOR_TEXT, NULL, NULL, NULL); } free(dumpData); return; } if(hidKeysDown() & KEY_B) { svcSignalEvent(dumpData->cancelEvent); } *progress = dumpData->currTotal != 0 ? (float) ((double) dumpData->currProcessed / (double) dumpData->currTotal) : 0; snprintf(text, PROGRESS_TEXT_MAX, "%.2f %s / %.2f %s\n%.2f %s/s, ETA %s", ui_get_display_size(dumpData->currProcessed), ui_get_display_size_units(dumpData->currProcessed), ui_get_display_size(dumpData->currTotal), ui_get_display_size_units(dumpData->currTotal), ui_get_display_size(dumpData->bytesPerSecond), ui_get_display_size_units(dumpData->bytesPerSecond), ui_get_display_eta(dumpData->estimatedRemainingSeconds)); } static void dumpnand_onresponse(ui_view* view, void* data, u32 response) { if(response == PROMPT_YES) { data_op_data* dumpData = (data_op_data*) data; Result res = task_data_op(dumpData); if(R_SUCCEEDED(res)) { info_display("Dumping NAND", "Press B to cancel.", true, data, dumpnand_update, NULL); } else { error_display_res(NULL, NULL, res, "Failed to initiate NAND dump."); free(data); } } else { free(data); } } void dumpnand_open() { data_op_data* data = (data_op_data*) calloc(1, sizeof(data_op_data)); if(data == NULL) { error_display(NULL, NULL, "Failed to allocate dump NAND data."); return; } data->data = data; data->op = DATAOP_COPY; data->bufferSize = 256 * 1024; data->copyEmpty = true; data->total = 1; data->isSrcDirectory = dumpnand_is_src_directory; data->makeDstDirectory = dumpnand_make_dst_directory; data->openSrc = dumpnand_open_src; data->closeSrc = dumpnand_close_src; data->getSrcSize = dumpnand_get_src_size; data->readSrc = dumpnand_read_src; data->openDst = dumpnand_open_dst; data->closeDst = dumpnand_close_dst; data->writeDst = dumpnand_write_dst; data->suspendTransfer = dumpnand_suspend_transfer; data->restoreTransfer = dumpnand_restore_transfer; data->suspend = dumpnand_suspend; data->restore = dumpnand_restore; data->error = dumpnand_error; data->finished = true; prompt_display_yes_no("Confirmation", "Dump raw NAND image to the SD card?", COLOR_TEXT, data, NULL, dumpnand_onresponse); }