nxdumptool/source/dumper.c

269 lines
7.8 KiB
C
Raw Normal View History

2018-05-15 18:00:19 +02:00
#include "dumper.h"
#include "fsext.h"
#include <stdio.h>
#include <malloc.h>
#include <dirent.h>
#include <memory.h>
2018-05-16 23:29:43 +02:00
#include <limits.h>
#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
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)))
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");
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:53:26 +02:00
printf("Handle = %x\n", handle);
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);
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))) {
printf("\nfsStorageRead error\n");
success = false;
break;
2018-05-15 18:00:19 +02:00
}
if (fwrite(buf, 1, n, outFile) != n) {
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");
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();
}
}
if (success) {
printf(C_CLEAR_LINE "\rDone!\n");
syncDisplay();
}
2018-05-15 18:00:19 +02:00
free(buf);
2018-05-15 18:00:19 +02:00
fclose(outFile);
fsStorageClose(&gameCardStorage);
return success;
}
2018-05-16 23:29:43 +02:00
#define NAME_BUF_LEN 4096
bool copyFile(const char* source, const char* dest, bool doSplitting) {
printf("Copying %s...", source);
2018-05-16 23:29:43 +02:00
syncDisplay();
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");
if (!outFile) {
printf("\nFailed to open output file\n");
2018-05-16 23:29:43 +02:00
fclose(inFile);
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;
size_t last_report = 0;
printf(" [00%%]");
syncDisplay();
2018-05-16 23:29:43 +02:00
int lastp = 0;
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);
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;
}
last_report = total;
}
}
if (success)
printf("\b\b\b\b\bDone!\n");
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) {
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)) {
closedir(dir);
return false;
}
} else {
2018-05-16 23:29:43 +02:00
if (!copyFile(sbuf, dbuf, splitting)) {
closedir(dir);
return false;
}
}
}
closedir(dir);
return true;
}
2018-05-16 23:29:43 +02:00
bool copyDirectory(const char* source, const char* dest, bool splitting) {
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
}