mirror of
https://github.com/suchmememanyskill/TegraExplorer.git
synced 2025-01-10 03:37:33 -03:00
Add fw dumping
- also fatfs is stupid - also close files properly on a failed copy - also check for errors during folder readouts - also make sure holding vol- doesn't dump the keys anyway
This commit is contained in:
parent
859ad2cc4c
commit
513bd804b1
18 changed files with 222 additions and 29 deletions
|
@ -4048,7 +4048,8 @@ FRESULT f_write (
|
|||
}
|
||||
if (clst == 0) {
|
||||
EFSPRINTF("DSKFULL");
|
||||
break; /* Could not allocate a new cluster (disk full) */
|
||||
fp->flag |= FA_MODIFIED;
|
||||
ABORT(fs, FR_DISK_ERR); /* Could not allocate a new cluster (disk full) */
|
||||
}
|
||||
if (clst == 1) {
|
||||
EFSPRINTF("CCHK");
|
||||
|
|
|
@ -13,6 +13,7 @@ ErrCode_t FileCopy(const char *locin, const char *locout, u8 options){
|
|||
u64 sizeRemaining, toCopy;
|
||||
u8 *buff;
|
||||
u32 x, y;
|
||||
ErrCode_t err = newErrCode(0);
|
||||
int res = 0;
|
||||
|
||||
gfx_con_getpos(&x, &y);
|
||||
|
@ -47,11 +48,13 @@ ErrCode_t FileCopy(const char *locin, const char *locout, u8 options){
|
|||
toCopy = MIN(sizeRemaining, TConf.FSBuffSize);
|
||||
|
||||
if ((res = f_read(&in, buff, toCopy, NULL))){
|
||||
return newErrCode(res);
|
||||
err = newErrCode(res);
|
||||
break;
|
||||
}
|
||||
|
||||
if ((res = f_write(&out, buff, toCopy, NULL))){
|
||||
return newErrCode(res);
|
||||
err = newErrCode(res);
|
||||
break;
|
||||
}
|
||||
|
||||
sizeRemaining -= toCopy;
|
||||
|
@ -74,5 +77,5 @@ ErrCode_t FileCopy(const char *locin, const char *locout, u8 options){
|
|||
f_chmod(locout, in_info.fattrib, 0x3A);
|
||||
|
||||
//f_stat(locin, &in_info); //somehow stops fatfs from being weird
|
||||
return newErrCode(0);
|
||||
return err;
|
||||
}
|
|
@ -43,3 +43,12 @@ char *GetFileAttribs(FSEntry_t entry){
|
|||
MaskIn(ret, entry.optionUnion, '-');
|
||||
return ret;
|
||||
}
|
||||
|
||||
bool FileExists(char* path){
|
||||
FRESULT fr;
|
||||
FILINFO fno;
|
||||
|
||||
fr = f_stat(path, &fno);
|
||||
|
||||
return !(fr & FR_NO_FILE);
|
||||
}
|
||||
|
|
|
@ -6,3 +6,4 @@ u64 GetFileSize(char *path);
|
|||
char *EscapeFolder(const char *current);
|
||||
char *CombinePaths(const char *current, const char *add);
|
||||
char *GetFileAttribs(FSEntry_t entry);
|
||||
bool FileExists(char* path);
|
|
@ -30,13 +30,7 @@ MenuEntry_t MakeMenuOutFSEntry(FSEntry_t entry){
|
|||
return out;
|
||||
}
|
||||
|
||||
void clearFileVector(Vector_t *v){
|
||||
vecPDefArray(FSEntry_t*, entries, v);
|
||||
for (int i = 0; i < v->count; i++)
|
||||
free(entries[i].name);
|
||||
|
||||
free(v->data);
|
||||
}
|
||||
|
||||
void FileExplorer(char *path){
|
||||
char *storedPath = CpyStr(path);
|
||||
|
@ -48,8 +42,15 @@ void FileExplorer(char *path){
|
|||
|
||||
gfx_clearscreen();
|
||||
gfx_printf("Loading...\r");
|
||||
//gfx_printf(" ");
|
||||
Vector_t fileVec = ReadFolder(storedPath);
|
||||
|
||||
int readRes = 0;
|
||||
Vector_t fileVec = ReadFolder(storedPath, &readRes);
|
||||
if (readRes){
|
||||
clearFileVector(&fileVec);
|
||||
DrawError(newErrCode(readRes));
|
||||
return;
|
||||
}
|
||||
|
||||
vecDefArray(FSEntry_t*, fsEntries, fileVec);
|
||||
|
||||
topEntries[0].name = storedPath;
|
||||
|
|
|
@ -10,6 +10,7 @@
|
|||
#include "../../hid/hid.h"
|
||||
#include <libs/fatfs/ff.h>
|
||||
#include "../../utils/utils.h"
|
||||
#include "../../keys/nca.h"
|
||||
|
||||
MenuEntry_t FileMenuEntries[] = {
|
||||
// Still have to think up the options
|
||||
|
@ -50,17 +51,12 @@ void MoveClipboard(char *path, FSEntry_t entry){
|
|||
free(thing);
|
||||
}
|
||||
|
||||
MenuEntry_t DeleteEntries[] = {
|
||||
{.optionUnion = COLORTORGB(COLOR_YELLOW), .name = "No"},
|
||||
{.R = 255, .name = "Yes"}
|
||||
};
|
||||
|
||||
void DeleteFile(char *path, FSEntry_t entry){
|
||||
gfx_con_setpos(384 + 16, 200 + 16 + 10 * 16);
|
||||
SETCOLOR(COLOR_RED, COLOR_DARKGREY);
|
||||
gfx_printf("Are you sure? ");
|
||||
|
||||
if (!MakeHorizontalMenu(DeleteEntries, 2, 3, COLOR_DARKGREY))
|
||||
if (!MakeYesNoHorzMenu(3, COLOR_DARKGREY))
|
||||
return;
|
||||
|
||||
char *thing = CombinePaths(path, entry.name);
|
||||
|
|
|
@ -1,19 +1,27 @@
|
|||
#include "folderReader.h"
|
||||
#include <libs/fatfs/ff.h>
|
||||
#include "../../utils/utils.h"
|
||||
#include <mem/heap.h>
|
||||
|
||||
Vector_t /* of type FSEntry_t */ ReadFolder(char *path){
|
||||
void clearFileVector(Vector_t *v){
|
||||
vecPDefArray(FSEntry_t*, entries, v);
|
||||
for (int i = 0; i < v->count; i++)
|
||||
free(entries[i].name);
|
||||
|
||||
free(v->data);
|
||||
}
|
||||
|
||||
Vector_t /* of type FSEntry_t */ ReadFolder(char *path, int *res){
|
||||
Vector_t out = newVec(sizeof(FSEntry_t), 16); // we may want to prealloc with the same size as the folder
|
||||
DIR dir;
|
||||
FILINFO fno;
|
||||
int res;
|
||||
|
||||
if ((res = f_opendir(&dir, path))){
|
||||
if ((*res = f_opendir(&dir, path))){
|
||||
// Err!
|
||||
return out;
|
||||
}
|
||||
|
||||
while (!f_readdir(&dir, &fno) && fno.fname[0]) {
|
||||
while (!(*res = f_readdir(&dir, &fno)) && fno.fname[0]) {
|
||||
FSEntry_t newEntry = {.optionUnion = fno.fattrib, .name = CpyStr(fno.fname)};
|
||||
|
||||
if (!newEntry.isDir){
|
||||
|
|
|
@ -3,4 +3,5 @@
|
|||
#include "../../utils/vector.h"
|
||||
#include "../fstypes.h"
|
||||
|
||||
Vector_t /* of type FSEntry_t */ ReadFolder(char *path);
|
||||
void clearFileVector(Vector_t *v);
|
||||
Vector_t /* of type FSEntry_t */ ReadFolder(char *path, int *res);
|
|
@ -19,6 +19,15 @@ void gfx_clearscreen(){
|
|||
RESETCOLOR;
|
||||
}
|
||||
|
||||
MenuEntry_t YesNoEntries[] = {
|
||||
{.optionUnion = COLORTORGB(COLOR_YELLOW), .name = "No"},
|
||||
{.R = 255, .name = "Yes"}
|
||||
};
|
||||
|
||||
int MakeYesNoHorzMenu(int spacesBetween, u32 bg){
|
||||
return MakeHorizontalMenu(YesNoEntries, ARR_LEN(YesNoEntries), spacesBetween, bg);
|
||||
}
|
||||
|
||||
int MakeHorizontalMenu(MenuEntry_t *entries, int len, int spacesBetween, u32 bg){
|
||||
u32 initialX = 0, initialY = 0;
|
||||
u32 highlight = 0;
|
||||
|
|
|
@ -15,3 +15,4 @@
|
|||
|
||||
void gfx_clearscreen();
|
||||
int MakeHorizontalMenu(MenuEntry_t *entries, int len, int spacesBetween, u32 bg);
|
||||
int MakeYesNoHorzMenu(int spacesBetween, u32 bg);
|
31
source/keys/nca.c
Normal file
31
source/keys/nca.c
Normal file
|
@ -0,0 +1,31 @@
|
|||
#include "nca.h"
|
||||
#include "keys.h"
|
||||
#include <libs/fatfs/ff.h>
|
||||
#include <sec/se.h>
|
||||
#include <mem/heap.h>
|
||||
|
||||
// Thanks switchbrew https://switchbrew.org/wiki/NCA_Format
|
||||
// This function is hacky, should change it but am lazy
|
||||
int GetNcaType(char *path){
|
||||
FIL fp;
|
||||
u32 read_bytes = 0;
|
||||
|
||||
if (f_open(&fp, path, FA_READ | FA_OPEN_EXISTING))
|
||||
return -1;
|
||||
|
||||
u8 *dec_header = (u8*)malloc(0x400);
|
||||
|
||||
if (f_lseek(&fp, 0x200) || f_read(&fp, dec_header, 32, &read_bytes) || read_bytes != 32){
|
||||
f_close(&fp);
|
||||
free(dec_header);
|
||||
return -1;
|
||||
}
|
||||
|
||||
se_aes_xts_crypt(7,6,0,1, dec_header + 0x200, dec_header, 32, 1);
|
||||
|
||||
u8 ContentType = dec_header[0x205];
|
||||
|
||||
f_close(&fp);
|
||||
free(dec_header);
|
||||
return ContentType;
|
||||
}
|
12
source/keys/nca.h
Normal file
12
source/keys/nca.h
Normal file
|
@ -0,0 +1,12 @@
|
|||
#pragma once
|
||||
|
||||
enum NcaTypes {
|
||||
Porgram = 0,
|
||||
Meta,
|
||||
Control,
|
||||
Manual,
|
||||
Data,
|
||||
PublicData
|
||||
};
|
||||
|
||||
int GetNcaType(char *path);
|
|
@ -271,6 +271,7 @@ void ipl_main()
|
|||
emu_cfg.enabled = !h_cfg.emummc_force_disable;
|
||||
h_cfg.emummc_force_disable = 1;
|
||||
|
||||
TConf.pkg1ID = "Unk";
|
||||
|
||||
hidInit();
|
||||
|
||||
|
@ -290,7 +291,7 @@ void ipl_main()
|
|||
int res = -1;
|
||||
|
||||
|
||||
if (DumpKeys() || btn_read() & BTN_VOL_DOWN)
|
||||
if (btn_read() & BTN_VOL_DOWN || DumpKeys())
|
||||
res = GetKeysFromFile("sd:/switch/prod.keys");
|
||||
|
||||
TConf.keysDumped = (res > 0) ? 0 : 1;
|
||||
|
|
|
@ -82,6 +82,7 @@ void GptMenu(u8 MMCType){
|
|||
else {
|
||||
if (TConf.curExplorerLoc > LOC_SD)
|
||||
ResetCopyParams();
|
||||
|
||||
TConf.curExplorerLoc = LOC_EMMC;
|
||||
FileExplorer("bis:/");
|
||||
}
|
||||
|
|
|
@ -18,6 +18,10 @@ void SetKeySlots(){
|
|||
se_aes_key_set(3, dumpedKeys.bis_key[1] + AES_128_KEY_SIZE, AES_128_KEY_SIZE);
|
||||
se_aes_key_set(4, dumpedKeys.bis_key[2], AES_128_KEY_SIZE);
|
||||
se_aes_key_set(5, dumpedKeys.bis_key[2] + AES_128_KEY_SIZE, AES_128_KEY_SIZE);
|
||||
|
||||
// Not for bis but whatever
|
||||
se_aes_key_set(6, dumpedKeys.header_key + 0x00, 0x10);
|
||||
se_aes_key_set(7, dumpedKeys.header_key + 0x10, 0x10);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -20,6 +20,7 @@ MenuEntry_t mainMenuEntries[] = {
|
|||
{.optionUnion = COLORTORGB(COLOR_YELLOW), .name = "Emummc"},
|
||||
{.B = 255, .G = 255, .name = "Test Controllers"},
|
||||
{.R = 255, .name = "Cause an exception"},
|
||||
{.optionUnion = COLORTORGB(COLOR_BLUE), .name = "Dump Firmware"},
|
||||
{.optionUnion = COLORTORGB(COLOR_ORANGE), .name = "View dumped keys"},
|
||||
{.R = 255, .name = "Reboot to payload"}
|
||||
};
|
||||
|
@ -73,6 +74,7 @@ menuPaths mainMenuPaths[] = {
|
|||
HandleEMUMMC,
|
||||
TestControllers,
|
||||
CrashTE,
|
||||
DumpSysFw,
|
||||
ViewKeys,
|
||||
RebootToPayload
|
||||
};
|
||||
|
|
|
@ -3,6 +3,20 @@
|
|||
#include "../gfx/gfxutils.h"
|
||||
#include "../gfx/menu.h"
|
||||
#include "../hid/hid.h"
|
||||
#include <libs/fatfs/ff.h>
|
||||
#include "../keys/keys.h"
|
||||
#include "../keys/nca.h"
|
||||
#include <storage/nx_sd.h>
|
||||
#include "../fs/fsutils.h"
|
||||
#include <utils/util.h>
|
||||
#include "../storage/mountmanager.h"
|
||||
#include "../err.h"
|
||||
#include <utils/sprintf.h>
|
||||
#include <mem/heap.h>
|
||||
#include "../tegraexplorer/tconf.h"
|
||||
#include "../fs/readers/folderReader.h"
|
||||
#include <string.h>
|
||||
#include "../fs/fscopy.h"
|
||||
|
||||
void TestControllers(){
|
||||
gfx_clearscreen();
|
||||
|
@ -27,3 +41,100 @@ extern int launch_payload(char *path);
|
|||
void RebootToPayload(){
|
||||
launch_payload("atmosphere/reboot_payload.bin");
|
||||
}
|
||||
|
||||
void DumpSysFw(){
|
||||
char sysPath[25 + 36 + 3 + 1]; // 24 for "bis:/Contents/registered", 36 for ncaName.nca, 3 for /00, and 1 to make sure :)
|
||||
char *baseSdPath;
|
||||
|
||||
u32 timer = get_tmr_s();
|
||||
|
||||
if (!sd_mount())
|
||||
return;
|
||||
|
||||
if (connectMMC(MMC_CONN_EMMC))
|
||||
return;
|
||||
|
||||
if (!TConf.keysDumped)
|
||||
return;
|
||||
|
||||
ErrCode_t err = mountMMCPart("SYSTEM");
|
||||
if (err.err){
|
||||
DrawError(err);
|
||||
return;
|
||||
}
|
||||
|
||||
baseSdPath = malloc(36 + 16);
|
||||
sprintf(baseSdPath, "sd:/tegraexplorer/Firmware/%d (%s)", TConf.pkg1ver, TConf.pkg1ID);
|
||||
int baseSdPathLen = strlen(baseSdPath);
|
||||
|
||||
f_mkdir("sd:/tegraexplorer");
|
||||
f_mkdir("sd:/tegraexplorer/Firmware");
|
||||
|
||||
gfx_clearscreen();
|
||||
|
||||
gfx_printf("Pkg1 id: '%s', kb %d\n", TConf.pkg1ID, TConf.pkg1ver);
|
||||
if (FileExists(baseSdPath)){
|
||||
SETCOLOR(COLOR_ORANGE, COLOR_DEFAULT);
|
||||
gfx_printf("Destination already exists. Replace? ");
|
||||
if (!MakeYesNoHorzMenu(3, COLOR_DEFAULT)){
|
||||
free(baseSdPath);
|
||||
return;
|
||||
}
|
||||
RESETCOLOR;
|
||||
gfx_printf("\nReminder! delete the folder. i can't delete recursively yet");
|
||||
gfx_putc('\n');
|
||||
}
|
||||
|
||||
f_mkdir(baseSdPath);
|
||||
|
||||
gfx_printf("Out: %s\nReading entries...\n", baseSdPath);
|
||||
int readRes = 0;
|
||||
Vector_t fileVec = ReadFolder("bis:/Contents/registered", &readRes);
|
||||
if (readRes){
|
||||
DrawError(newErrCode(readRes));
|
||||
free(baseSdPath);
|
||||
return;
|
||||
}
|
||||
|
||||
gfx_printf("Starting dump...\n");
|
||||
SETCOLOR(COLOR_GREEN, COLOR_DEFAULT);
|
||||
|
||||
int res = 0;
|
||||
int total = 1;
|
||||
vecDefArray(FSEntry_t*, fsEntries, fileVec);
|
||||
for (int i = 0; i < fileVec.count; i++){
|
||||
sprintf(sysPath, (fsEntries[i].isDir) ? "%s/%s/00" : "%s/%s", "bis:/Contents/registered", fsEntries[i].name);
|
||||
int contentType = GetNcaType(sysPath);
|
||||
|
||||
if (contentType < 0){
|
||||
res = 1;
|
||||
break;
|
||||
}
|
||||
|
||||
char *sdPath = malloc(baseSdPathLen + 45);
|
||||
sprintf(sdPath, "%s/%s", baseSdPath, fsEntries[i].name);
|
||||
if (contentType == Meta)
|
||||
memcpy(sdPath + strlen(sdPath) - 4, ".cnmt.nca", 10);
|
||||
|
||||
gfx_printf("[%3d / %3d] %s\r", total, fileVec.count, fsEntries[i].name);
|
||||
total++;
|
||||
err = FileCopy(sysPath, sdPath, 0);
|
||||
free(sdPath);
|
||||
if (err.err){
|
||||
DrawError(err);
|
||||
res = 1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
clearFileVector(&fileVec);
|
||||
|
||||
RESETCOLOR;
|
||||
|
||||
if (res){
|
||||
gfx_printf("\nDump failed...\n");
|
||||
}
|
||||
|
||||
gfx_printf("\n\nDone! Time taken: %ds\nPress any key to exit", get_tmr_s() - timer);
|
||||
free(baseSdPath);
|
||||
hidWait();
|
||||
}
|
|
@ -2,3 +2,4 @@
|
|||
|
||||
void RebootToPayload();
|
||||
void TestControllers();
|
||||
void DumpSysFw();
|
Loading…
Reference in a new issue