2018-05-15 18:00:19 +02:00
|
|
|
#include "dumper.h"
|
|
|
|
#include "fsext.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <malloc.h>
|
2018-05-16 19:08:20 +02:00
|
|
|
#include <dirent.h>
|
|
|
|
#include <memory.h>
|
2018-05-16 23:29:43 +02:00
|
|
|
#include <limits.h>
|
2018-05-16 19:08:20 +02:00
|
|
|
#include <sys/stat.h>
|
2018-05-16 23:29:43 +02:00
|
|
|
#include <alloca.h>
|
2018-05-15 18:00:19 +02:00
|
|
|
#include "ccolor.h"
|
|
|
|
#include "util.h"
|
|
|
|
|
2018-05-16 23:29:43 +02:00
|
|
|
#define FILE_MAX INT_MAX
|
|
|
|
#define SPLIT_FILE_MIN 4000000000u
|
|
|
|
|
2018-05-16 17:10:30 +02:00
|
|
|
void workaroundPartitionZeroAccess(FsDeviceOperator* fsOperator) {
|
|
|
|
u32 handle;
|
|
|
|
if (R_FAILED(fsDeviceOperatorGetGameCardHandle(fsOperator, &handle)))
|
|
|
|
return;
|
|
|
|
FsStorage gameCardStorage;
|
2018-05-16 18:21:13 +02:00
|
|
|
if (R_FAILED(fsOpenGameCardStorage(&gameCardStorage, handle, 0)))
|
2018-05-16 17:10:30 +02:00
|
|
|
return;
|
|
|
|
fsStorageClose(&gameCardStorage);
|
|
|
|
}
|
|
|
|
|
2018-05-16 18:21:13 +02:00
|
|
|
bool openPartitionFs(FsFileSystem* ret, FsDeviceOperator* fsOperator, u32 partition) {
|
|
|
|
u32 handle;
|
|
|
|
if (R_FAILED(fsDeviceOperatorGetGameCardHandle(fsOperator, &handle))) {
|
|
|
|
printf("GetGameCardHandle failed\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Result result;
|
|
|
|
if (R_FAILED(result = fsMountGameCard(ret, handle, partition))) {
|
|
|
|
printf("MountGameCard failed %x\n", result);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
printf("Opened card\n");
|
2018-05-16 19:08:20 +02:00
|
|
|
return true;
|
2018-05-16 18:21:13 +02:00
|
|
|
}
|
|
|
|
|
2018-05-15 18:00:19 +02:00
|
|
|
bool dumpPartitionRaw(FsDeviceOperator* fsOperator, u32 partition) {
|
|
|
|
u32 handle;
|
|
|
|
if (R_FAILED(fsDeviceOperatorGetGameCardHandle(fsOperator, &handle))) {
|
|
|
|
printf("GetGameCardHandle failed\n");
|
|
|
|
return false;
|
|
|
|
}
|
2018-05-15 18:31:46 +02:00
|
|
|
|
2018-05-15 18:53:26 +02:00
|
|
|
printf("Handle = %x\n", handle);
|
|
|
|
|
2018-05-15 18:31:46 +02:00
|
|
|
if (partition == 0) {
|
|
|
|
u32 title_ver;
|
|
|
|
fsDeviceOperatorUpdatePartitionInfo(fsOperator, handle, &title_ver, NULL);
|
|
|
|
printf("System title-version = %i\n", title_ver);
|
|
|
|
}
|
|
|
|
|
2018-05-15 18:00:19 +02:00
|
|
|
FsStorage gameCardStorage;
|
|
|
|
Result result;
|
2018-05-16 18:21:13 +02:00
|
|
|
if (R_FAILED(result = fsOpenGameCardStorage(&gameCardStorage, handle, partition))) {
|
2018-05-15 18:00:19 +02:00
|
|
|
printf("MountGameCard failed %x\n", result);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
printf("Opened card\n");
|
|
|
|
|
|
|
|
u64 size;
|
|
|
|
fsStorageGetSize(&gameCardStorage, &size);
|
|
|
|
printf("Total size = %li\n", size);
|
|
|
|
FILE* outFile = fopen("out.bin", "wb");
|
|
|
|
if (!outFile) {
|
|
|
|
printf("Failed to open output file!\n");
|
|
|
|
fsStorageClose(&gameCardStorage);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("Starting...");
|
|
|
|
syncDisplay();
|
|
|
|
|
|
|
|
const size_t bufs = 1024 * 1024;
|
|
|
|
char* buf = (char*) malloc(bufs);
|
2018-05-15 19:05:13 +02:00
|
|
|
bool success = true;
|
2018-05-15 18:00:19 +02:00
|
|
|
for (u64 off = 0; off < size; off += bufs) {
|
|
|
|
u64 n = bufs;
|
|
|
|
if (size - off < n)
|
|
|
|
n = size - off;
|
|
|
|
if (R_FAILED(result = fsStorageRead(&gameCardStorage, off, buf, n))) {
|
2018-05-15 19:05:13 +02:00
|
|
|
printf("\nfsStorageRead error\n");
|
|
|
|
success = false;
|
|
|
|
break;
|
2018-05-15 18:00:19 +02:00
|
|
|
}
|
|
|
|
if (fwrite(buf, 1, n, outFile) != n) {
|
2018-05-15 19:05:13 +02:00
|
|
|
printf("\nfwrite error\n");
|
|
|
|
success = false;
|
|
|
|
break;
|
2018-05-15 18:00:19 +02:00
|
|
|
}
|
|
|
|
if (((off / bufs) % 10) == 0) {
|
2018-05-15 18:53:26 +02:00
|
|
|
hidScanInput();
|
|
|
|
u64 kDown = hidKeysDown(CONTROLLER_P1_AUTO);
|
|
|
|
if (kDown & KEY_B) {
|
|
|
|
printf("\nCancelled\n");
|
2018-05-15 19:05:13 +02:00
|
|
|
success = false;
|
|
|
|
break;
|
2018-05-15 18:53:26 +02:00
|
|
|
}
|
|
|
|
|
2018-05-15 18:00:19 +02:00
|
|
|
printf(C_CLEAR_LINE "\rDumping %i%% [%li / %li bytes]", (int) (off * 100 / size), off, size);
|
|
|
|
syncDisplay();
|
|
|
|
}
|
|
|
|
}
|
2018-05-15 19:05:13 +02:00
|
|
|
if (success) {
|
|
|
|
printf(C_CLEAR_LINE "\rDone!\n");
|
|
|
|
syncDisplay();
|
|
|
|
}
|
2018-05-15 18:00:19 +02:00
|
|
|
|
2018-05-15 19:05:13 +02:00
|
|
|
free(buf);
|
2018-05-15 18:00:19 +02:00
|
|
|
fclose(outFile);
|
|
|
|
fsStorageClose(&gameCardStorage);
|
|
|
|
|
2018-05-15 19:05:13 +02:00
|
|
|
return success;
|
2018-05-16 19:08:20 +02:00
|
|
|
}
|
|
|
|
|
2018-05-16 23:29:43 +02:00
|
|
|
#define NAME_BUF_LEN 4096
|
|
|
|
|
|
|
|
bool copyFile(const char* source, const char* dest, bool doSplitting) {
|
2018-05-16 19:08:20 +02:00
|
|
|
printf("Copying %s...", source);
|
2018-05-16 23:29:43 +02:00
|
|
|
syncDisplay();
|
2018-05-16 19:08:20 +02:00
|
|
|
FILE* inFile = fopen(source, "rb");
|
|
|
|
if (!inFile) {
|
|
|
|
printf("\nFailed to open input file\n");
|
|
|
|
return false;
|
|
|
|
}
|
2018-05-16 23:29:43 +02:00
|
|
|
fseek(inFile, 0L, SEEK_END);
|
|
|
|
long int size = ftell(inFile);
|
|
|
|
fseek(inFile, 0L, SEEK_SET);
|
|
|
|
|
|
|
|
char* splitFilename = NULL;
|
|
|
|
size_t destLen = strlen(dest);
|
|
|
|
if (size > SPLIT_FILE_MIN && doSplitting) {
|
|
|
|
if (destLen + 1 > NAME_BUF_LEN) {
|
|
|
|
printf("\nFilename is too long\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
splitFilename = alloca(NAME_BUF_LEN);
|
|
|
|
strcpy(splitFilename, dest);
|
|
|
|
sprintf(&splitFilename[destLen], ".%02i", 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
FILE* outFile = fopen(splitFilename != NULL ? splitFilename : dest, "wb");
|
2018-05-16 19:08:20 +02:00
|
|
|
if (!outFile) {
|
|
|
|
printf("\nFailed to open output file\n");
|
2018-05-16 23:29:43 +02:00
|
|
|
fclose(inFile);
|
2018-05-16 19:08:20 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const size_t bufs = 1024 * 1024;
|
|
|
|
char* buf = (char*) malloc(bufs);
|
|
|
|
bool success = true;
|
|
|
|
ssize_t n;
|
|
|
|
size_t total = 0;
|
2018-05-16 23:29:43 +02:00
|
|
|
size_t file_off = 0;
|
2018-05-16 19:08:20 +02:00
|
|
|
size_t last_report = 0;
|
|
|
|
printf(" [00%%]");
|
|
|
|
syncDisplay();
|
2018-05-16 23:29:43 +02:00
|
|
|
int lastp = 0;
|
2018-05-16 19:08:20 +02:00
|
|
|
while (true) {
|
2018-05-16 23:29:43 +02:00
|
|
|
if (total - file_off >= FILE_MAX && splitFilename != NULL) {
|
|
|
|
fclose(outFile);
|
|
|
|
file_off += FILE_MAX;
|
|
|
|
sprintf(&splitFilename[destLen], ".%02i", (int) (file_off / FILE_MAX));
|
|
|
|
outFile = fopen(splitFilename, "wb");
|
|
|
|
if (!outFile) {
|
|
|
|
printf("\nFailed to open output file\n");
|
|
|
|
free(buf);
|
|
|
|
fclose(inFile);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t readCount = bufs;
|
|
|
|
if (FILE_MAX - (total - file_off) < readCount)
|
|
|
|
readCount = FILE_MAX - (total - file_off);
|
2018-05-16 19:08:20 +02:00
|
|
|
n = fread(buf, 1, bufs, inFile);
|
|
|
|
if (n <= 0) {
|
|
|
|
if (feof(inFile))
|
|
|
|
break;
|
|
|
|
printf("\nRead error; retrying\n");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (fwrite(buf, 1, n, outFile) != n) {
|
|
|
|
printf("\nWrite error\n");
|
|
|
|
success = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
total += n;
|
|
|
|
if ((total - last_report) > 1024 * 1024) {
|
|
|
|
hidScanInput();
|
|
|
|
u64 kDown = hidKeysDown(CONTROLLER_P1_AUTO);
|
|
|
|
if (kDown & KEY_B) {
|
|
|
|
printf("\nCancelled\n");
|
|
|
|
success = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
int p = (int) (total * 100 / size);
|
|
|
|
if (p >= 100)
|
|
|
|
p = 99;
|
2018-05-16 23:29:43 +02:00
|
|
|
if (lastp != p) {
|
|
|
|
printf("\b\b\b\b%02i%%]", p);
|
|
|
|
syncDisplay();
|
|
|
|
lastp = p;
|
|
|
|
}
|
2018-05-16 19:08:20 +02:00
|
|
|
last_report = total;
|
|
|
|
}
|
|
|
|
}
|
2018-05-16 19:46:55 +02:00
|
|
|
if (success)
|
|
|
|
printf("\b\b\b\b\bDone!\n");
|
2018-05-16 19:08:20 +02:00
|
|
|
free(buf);
|
|
|
|
fclose(inFile);
|
|
|
|
fclose(outFile);
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
2018-05-16 23:29:43 +02:00
|
|
|
bool _copyDirectory(char* sbuf, size_t source_len, char* dbuf, size_t dest_len, bool splitting) {
|
2018-05-16 19:08:20 +02:00
|
|
|
DIR* dir = opendir(sbuf);
|
|
|
|
struct dirent* ent;
|
|
|
|
sbuf[source_len] = '/';
|
|
|
|
dbuf[dest_len] = '/';
|
|
|
|
while ((ent = readdir(dir)) != NULL) {
|
|
|
|
size_t d_name_len = strlen(ent->d_name);
|
|
|
|
if (source_len + 1 + d_name_len + 1 >= NAME_BUF_LEN ||
|
|
|
|
dest_len + 1 + d_name_len + 1 >= NAME_BUF_LEN) {
|
|
|
|
printf("Too long file name!\n");
|
|
|
|
closedir(dir);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
strcpy(sbuf + source_len + 1, ent->d_name);
|
|
|
|
strcpy(dbuf + dest_len + 1, ent->d_name);
|
|
|
|
if (ent->d_type == DT_DIR) {
|
|
|
|
mkdir(dbuf, 0744);
|
2018-05-16 23:29:43 +02:00
|
|
|
if (!_copyDirectory(sbuf, source_len + 1 + d_name_len, dbuf, dest_len + 1 + d_name_len, splitting)) {
|
2018-05-16 19:08:20 +02:00
|
|
|
closedir(dir);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
2018-05-16 23:29:43 +02:00
|
|
|
if (!copyFile(sbuf, dbuf, splitting)) {
|
2018-05-16 19:08:20 +02:00
|
|
|
closedir(dir);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-05-16 19:46:55 +02:00
|
|
|
closedir(dir);
|
2018-05-16 19:08:20 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-05-16 23:29:43 +02:00
|
|
|
bool copyDirectory(const char* source, const char* dest, bool splitting) {
|
2018-05-16 19:08:20 +02:00
|
|
|
char sbuf[NAME_BUF_LEN];
|
|
|
|
char dbuf[NAME_BUF_LEN];
|
|
|
|
size_t source_len = strlen(source);
|
|
|
|
size_t dest_len = strlen(dest);
|
|
|
|
if (source_len + 1 >= NAME_BUF_LEN) {
|
|
|
|
printf("Directory name too long %li: %s\n", source_len + 1, source);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (dest_len + 1 >= NAME_BUF_LEN) {
|
|
|
|
printf("Directory name too long %li: %s\n", dest_len + + 1, dest);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
strcpy(sbuf, source);
|
|
|
|
strcpy(dbuf, dest);
|
|
|
|
mkdir(dbuf, 0744);
|
2018-05-16 23:29:43 +02:00
|
|
|
return _copyDirectory(sbuf, source_len, dbuf, dest_len, splitting);
|
2018-05-15 18:00:19 +02:00
|
|
|
}
|