2018-05-15 18:00:19 +02:00
# include <stdio.h>
2019-06-05 18:44:18 -04:00
# include <stdlib.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-06-21 02:42:46 -04:00
# include <unistd.h>
2018-06-25 21:11:18 -04:00
# include <math.h>
2019-05-01 16:24:13 -04:00
# include <ctype.h>
2018-05-15 18:00:19 +02:00
2018-06-23 20:48:34 -04:00
# include "crc32_fast.h"
2018-06-21 02:42:46 -04:00
# include "dumper.h"
2019-06-05 18:44:18 -04:00
# include "fs_ext.h"
2018-06-21 02:42:46 -04:00
# include "ui.h"
2019-06-05 18:44:18 -04:00
# include "nca.h"
2019-06-08 21:36:21 -04:00
# include "keys.h"
2018-05-16 23:29:43 +02:00
2019-04-23 01:14:57 -04:00
/* Extern variables */
2019-06-28 16:13:54 -04:00
extern bool runningSxOs ;
2019-06-08 21:36:21 -04:00
extern FsDeviceOperator fsOperatorInstance ;
2019-06-19 23:56:14 -04:00
extern nca_keyset_t nca_keyset ;
2018-06-21 02:42:46 -04:00
extern u64 freeSpace ;
2018-05-16 17:10:30 +02:00
2018-06-21 02:42:46 -04:00
extern int breaks ;
2019-04-21 12:27:33 -04:00
extern int font_height ;
2018-05-15 18:31:46 +02:00
2019-06-28 16:13:54 -04:00
extern u64 gameCardSize , trimmedCardSize ;
2019-05-01 16:24:13 -04:00
extern char trimmedCardSizeStr [ 32 ] ;
2018-05-15 18:53:26 +02:00
2019-06-05 18:44:18 -04:00
extern u8 * hfs0_header ;
2018-06-21 02:42:46 -04:00
extern u64 hfs0_offset , hfs0_size ;
extern u32 hfs0_partition_cnt ;
2018-05-15 18:31:46 +02:00
2019-06-05 18:44:18 -04:00
extern u8 * partitionHfs0Header ;
2019-04-23 01:14:57 -04:00
extern u64 partitionHfs0HeaderOffset , partitionHfs0HeaderSize ;
extern u32 partitionHfs0FileCount , partitionHfs0StrTableSize ;
2018-06-30 02:37:42 -04:00
2019-06-08 21:36:21 -04:00
extern u32 titleAppCount ;
extern FsStorageId * titleAppStorageId ;
2019-04-23 01:14:57 -04:00
2019-06-08 21:36:21 -04:00
extern u32 titlePatchCount ;
extern FsStorageId * titlePatchStorageId ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
extern u32 titleAddOnCount ;
extern FsStorageId * titleAddOnStorageId ;
2019-06-05 18:44:18 -04:00
2019-06-19 23:56:14 -04:00
extern u32 sdCardTitleAppCount ;
extern u32 sdCardTitlePatchCount ;
extern u32 sdCardTitleAddOnCount ;
extern u32 nandUserTitleAppCount ;
extern u32 nandUserTitlePatchCount ;
extern u32 nandUserTitleAddOnCount ;
2019-04-23 01:14:57 -04:00
extern AppletType programAppletType ;
2019-06-08 21:36:21 -04:00
extern exefs_ctx_t exeFsContext ;
2019-06-05 18:44:18 -04:00
extern romfs_ctx_t romFsContext ;
2019-06-28 16:13:54 -04:00
extern bktr_ctx_t bktrContext ;
2019-06-05 18:44:18 -04:00
extern char curRomFsPath [ NAME_BUF_LEN ] ;
2019-06-28 16:13:54 -04:00
extern u32 curRomFsDirOffset ;
2019-06-05 18:44:18 -04:00
2019-05-01 16:24:13 -04:00
extern char strbuf [ NAME_BUF_LEN * 4 ] ;
2019-04-23 01:14:57 -04:00
2019-06-28 16:13:54 -04:00
extern char * filenameBuffer ;
extern char * filenames [ FILENAME_MAX_CNT ] ;
extern int filenamesCount ;
extern u8 * fileNormalIconBuf ;
2019-09-14 22:45:27 -04:00
extern u8 * dumpBuf ;
2019-06-08 21:36:21 -04:00
void workaroundPartitionZeroAccess ( )
2018-06-21 02:42:46 -04:00
{
2019-04-21 12:27:33 -04:00
FsGameCardHandle handle ;
2019-06-08 21:36:21 -04:00
if ( R_FAILED ( fsDeviceOperatorGetGameCardHandle ( & fsOperatorInstance , & handle ) ) ) return ;
2019-04-21 12:27:33 -04:00
FsStorage gameCardStorage ;
if ( R_FAILED ( fsOpenGameCardStorage ( & gameCardStorage , & handle , 0 ) ) ) return ;
fsStorageClose ( & gameCardStorage ) ;
2018-06-21 02:42:46 -04:00
}
2018-05-15 18:00:19 +02:00
2019-09-14 22:45:27 -04:00
bool dumpCartridgeImage ( bool isFat32 , bool setXciArchiveBit , bool keepCert , bool trimDump , bool calcCrc )
2018-06-21 02:42:46 -04:00
{
2019-06-05 18:44:18 -04:00
u64 partitionOffset = 0 , xciDataSize = 0 , n ;
2019-04-21 12:27:33 -04:00
u64 partitionSizes [ ISTORAGE_PARTITION_CNT ] ;
2019-06-05 18:44:18 -04:00
char partitionSizesStr [ ISTORAGE_PARTITION_CNT ] [ 32 ] = { ' \0 ' } , xciDataSizeStr [ 32 ] = { ' \0 ' } , filename [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
2019-04-21 12:27:33 -04:00
u32 partition ;
Result result ;
FsGameCardHandle handle ;
FsStorage gameCardStorage ;
2019-05-01 16:24:13 -04:00
bool proceed = true , success = false , fat32_error = false ;
2019-04-21 12:27:33 -04:00
FILE * outFile = NULL ;
u8 splitIndex = 0 ;
u32 crc1 = 0 , crc2 = 0 ;
2019-09-14 22:45:27 -04:00
memset ( dumpBuf , 0 , DUMP_BUFFER_SIZE ) ;
2019-06-05 18:44:18 -04:00
progress_ctx_t progressCtx ;
memset ( & progressCtx , 0 , sizeof ( progress_ctx_t ) ) ;
char tmp_idx [ 5 ] ;
2019-04-21 12:27:33 -04:00
2019-05-01 16:24:13 -04:00
size_t write_res ;
2019-08-15 01:21:51 -04:00
u64 part_size = ( ! setXciArchiveBit ? SPLIT_FILE_XCI_PART_SIZE : SPLIT_FILE_NSP_PART_SIZE ) ;
2019-06-28 16:13:54 -04:00
char * dumpName = generateFullDumpName ( ) ;
2019-05-01 16:24:13 -04:00
if ( ! dumpName )
{
2019-06-28 16:13:54 -04:00
// We're probably dealing with a forced XCI dump
dumpName = calloc ( 16 , sizeof ( char ) ) ;
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
sprintf ( dumpName , " gamecard " ) ;
2019-05-01 16:24:13 -04:00
}
2019-04-21 12:27:33 -04:00
for ( partition = 0 ; partition < ISTORAGE_PARTITION_CNT ; partition + + )
{
2019-06-05 18:44:18 -04:00
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "Getting partition #%u size...", partition);
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + + ; */
2019-04-21 12:27:33 -04:00
2019-06-28 16:13:54 -04:00
if ( partition = = ( ISTORAGE_PARTITION_CNT - 1 ) & & runningSxOs )
2019-04-21 12:27:33 -04:00
{
2019-06-28 16:13:54 -04:00
// Total size for IStorage instances is maxed out under SX OS, so let's manually reduce the size for the last instance
u64 partitionSizesSum = 0 ;
for ( int i = 0 ; i < ( ISTORAGE_PARTITION_CNT - 1 ) ; i + + ) partitionSizesSum + = partitionSizes [ i ] ;
// Substract the total ECC block size as well as the size for previous IStorage instances
partitionSizes [ partition ] = ( ( gameCardSize - ( ( gameCardSize / GAMECARD_ECC_BLOCK_SIZE ) * GAMECARD_ECC_DATA_SIZE ) ) - partitionSizesSum ) ;
xciDataSize + = partitionSizes [ partition ] ;
convertSize ( partitionSizes [ partition ] , partitionSizesStr [ partition ] , sizeof ( partitionSizesStr [ partition ] ) / sizeof ( partitionSizesStr [ partition ] [ 0 ] ) ) ;
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "Partition #%u size: %s (%lu bytes).", partition, partitionSizesStr[partition], partitionSizes[partition]);
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-28 16:13:54 -04:00
breaks + = 2 ; */
} else {
workaroundPartitionZeroAccess ( ) ;
2019-04-21 12:27:33 -04:00
2019-06-28 16:13:54 -04:00
if ( R_SUCCEEDED ( result = fsDeviceOperatorGetGameCardHandle ( & fsOperatorInstance , & handle ) ) )
2019-04-21 12:27:33 -04:00
{
2019-06-28 16:13:54 -04:00
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "GetGameCardHandle succeeded: 0x%08X", handle.value);
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
breaks + + ; */
2019-06-28 16:13:54 -04:00
if ( R_SUCCEEDED ( result = fsOpenGameCardStorage ( & gameCardStorage , & handle , partition ) ) )
2019-04-21 12:27:33 -04:00
{
2019-06-28 16:13:54 -04:00
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "OpenGameCardStorage succeeded: 0x%08X", handle.value);
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-28 16:13:54 -04:00
breaks + + ; */
if ( R_SUCCEEDED ( result = fsStorageGetSize ( & gameCardStorage , & ( partitionSizes [ partition ] ) ) ) )
{
xciDataSize + = partitionSizes [ partition ] ;
convertSize ( partitionSizes [ partition ] , partitionSizesStr [ partition ] , sizeof ( partitionSizesStr [ partition ] ) / sizeof ( partitionSizesStr [ partition ] [ 0 ] ) ) ;
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "Partition #%u size: %s (%lu bytes).", partition, partitionSizesStr[partition], partitionSizes[partition]);
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ; */
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " StorageGetSize failed! (0x%08X) " , result ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
}
fsStorageClose ( & gameCardStorage ) ;
2019-04-21 12:27:33 -04:00
} else {
2019-06-28 16:13:54 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " OpenGameCardStorage failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
proceed = false ;
}
} else {
2019-06-28 16:13:54 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " GetGameCardHandle failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
proceed = false ;
}
}
uiRefreshDisplay ( ) ;
}
if ( proceed )
{
convertSize ( xciDataSize , xciDataSizeStr , sizeof ( xciDataSizeStr ) / sizeof ( xciDataSizeStr [ 0 ] ) ) ;
2019-06-05 18:44:18 -04:00
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "XCI data size: %s (%lu bytes).", xciDataSizeStr, xciDataSize);
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ; */
2019-04-21 12:27:33 -04:00
if ( trimDump )
{
2019-06-05 18:44:18 -04:00
progressCtx . totalSize = trimmedCardSize ;
snprintf ( progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) , " %s " , trimmedCardSizeStr ) ;
2019-04-21 12:27:33 -04:00
// Change dump size for the last IStorage partition
u64 partitionSizesSum = 0 ;
for ( int i = 0 ; i < ( ISTORAGE_PARTITION_CNT - 1 ) ; i + + ) partitionSizesSum + = partitionSizes [ i ] ;
partitionSizes [ ISTORAGE_PARTITION_CNT - 1 ] = ( trimmedCardSize - partitionSizesSum ) ;
} else {
2019-06-05 18:44:18 -04:00
progressCtx . totalSize = xciDataSize ;
snprintf ( progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) , " %s " , xciDataSizeStr ) ;
2019-04-21 12:27:33 -04:00
}
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output dump size: %s (%lu bytes). " , progressCtx . totalSizeStr , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
breaks + + ;
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize < = freeSpace )
2019-04-21 12:27:33 -04:00
{
breaks + + ;
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 )
2019-04-21 12:27:33 -04:00
{
2019-06-05 18:44:18 -04:00
if ( setXciArchiveBit )
{
2019-06-19 23:56:14 -04:00
// Temporary, we'll use this to check if the dump already exists (it should have the archive bit set if so)
2019-06-08 21:36:21 -04:00
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s.xci " , XCI_DUMP_PATH , dumpName ) ;
2019-06-19 23:56:14 -04:00
} else {
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s.xc%u " , XCI_DUMP_PATH , dumpName , splitIndex ) ;
}
} else {
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s.xci " , XCI_DUMP_PATH , dumpName ) ;
}
// Check if the dump already exists
if ( checkIfFileExists ( filename ) )
{
// Ask the user if they want to proceed anyway
int cur_breaks = breaks ;
breaks + + ;
proceed = yesNoPrompt ( " You have already dumped this content. Do you wish to proceed anyway? " ) ;
if ( ! proceed )
{
uiDrawString ( " Process canceled. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
} else {
// Remove the prompt from the screen
breaks = cur_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
}
}
if ( proceed )
{
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 & & setXciArchiveBit )
{
2019-06-05 18:44:18 -04:00
// Since we may actually be dealing with an existing directory with the archive bit set or unset, let's try both
// Better safe than sorry
unlink ( filename ) ;
2019-08-15 01:21:51 -04:00
fsdevDeleteDirectoryRecursively ( filename ) ;
2019-06-05 18:44:18 -04:00
mkdir ( filename , 0744 ) ;
sprintf ( tmp_idx , " /%02u " , splitIndex ) ;
strcat ( filename , tmp_idx ) ;
}
2019-06-19 23:56:14 -04:00
outFile = fopen ( filename , " wb " ) ;
if ( outFile )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dump procedure started. Hold %s to cancel. " , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
2019-04-23 01:14:57 -04:00
{
2019-09-14 22:45:27 -04:00
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-23 01:14:57 -04:00
breaks + = 2 ;
2019-09-14 22:45:27 -04:00
}
progressCtx . line_offset = ( breaks + 4 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . start ) ) ;
for ( partition = 0 ; partition < ISTORAGE_PARTITION_CNT ; partition + + )
{
n = DUMP_BUFFER_SIZE ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
workaroundPartitionZeroAccess ( ) ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
if ( R_SUCCEEDED ( result = fsDeviceOperatorGetGameCardHandle ( & fsOperatorInstance , & handle ) ) )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
if ( R_SUCCEEDED ( result = fsOpenGameCardStorage ( & gameCardStorage , & handle , partition ) ) )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
for ( partitionOffset = 0 ; partitionOffset < partitionSizes [ partition ] ; partitionOffset + = n , progressCtx . curOffset + = n )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 4 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( filename , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dumping IStorage partition #%u... " , partition ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
if ( DUMP_BUFFER_SIZE > ( partitionSizes [ partition ] - partitionOffset ) ) n = ( partitionSizes [ partition ] - partitionOffset ) ;
if ( R_FAILED ( result = fsStorageRead ( & gameCardStorage , partitionOffset , dumpBuf , n ) ) )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " StorageRead failed (0x%08X) at offset 0x%016lX for partition #%u " , result , partitionOffset , partition ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
// Remove gamecard certificate
if ( progressCtx . curOffset = = 0 & & ! keepCert ) memset ( dumpBuf + CERT_OFFSET , 0xFF , CERT_SIZE ) ;
if ( calcCrc )
{
if ( ! trimDump )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
if ( keepCert )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
if ( progressCtx . curOffset = = 0 )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
// Update CRC32 (with gamecard certificate)
crc32 ( dumpBuf , n , & crc1 ) ;
// Backup gamecard certificate to an array
char tmpCert [ CERT_SIZE ] = { ' \0 ' } ;
memcpy ( tmpCert , dumpBuf + CERT_OFFSET , CERT_SIZE ) ;
// Remove gamecard certificate from buffer
memset ( dumpBuf + CERT_OFFSET , 0xFF , CERT_SIZE ) ;
// Update CRC32 (without gamecard certificate)
crc32 ( dumpBuf , n , & crc2 ) ;
// Restore gamecard certificate to buffer
memcpy ( dumpBuf + CERT_OFFSET , tmpCert , CERT_SIZE ) ;
2019-04-21 12:27:33 -04:00
} else {
2019-09-14 22:45:27 -04:00
// Update CRC32 (with gamecard certificate)
crc32 ( dumpBuf , n , & crc1 ) ;
// Update CRC32 (without gamecard certificate)
crc32 ( dumpBuf , n , & crc2 ) ;
2019-04-21 12:27:33 -04:00
}
} else {
// Update CRC32
2019-09-14 22:45:27 -04:00
crc32 ( dumpBuf , n , & crc2 ) ;
}
} else {
// Update CRC32
crc32 ( dumpBuf , n , & crc1 ) ;
}
}
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 & & ( progressCtx . curOffset + n ) > = ( ( splitIndex + 1 ) * part_size ) )
{
u64 new_file_chunk_size = ( ( progressCtx . curOffset + n ) - ( ( splitIndex + 1 ) * part_size ) ) ;
u64 old_file_chunk_size = ( n - new_file_chunk_size ) ;
if ( old_file_chunk_size > 0 )
{
write_res = fwrite ( dumpBuf , 1 , old_file_chunk_size , outFile ) ;
if ( write_res ! = old_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , old_file_chunk_size , progressCtx . curOffset , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
2019-04-21 12:27:33 -04:00
}
}
2019-09-14 22:45:27 -04:00
fclose ( outFile ) ;
outFile = NULL ;
if ( new_file_chunk_size > 0 | | ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
splitIndex + + ;
2019-06-05 18:44:18 -04:00
2019-09-14 22:45:27 -04:00
if ( setXciArchiveBit )
2019-06-05 18:44:18 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s.xci/%02u " , XCI_DUMP_PATH , dumpName , splitIndex ) ;
} else {
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s.xc%u " , XCI_DUMP_PATH , dumpName , splitIndex ) ;
2019-06-05 18:44:18 -04:00
}
2019-09-14 22:45:27 -04:00
outFile = fopen ( filename , " wb " ) ;
if ( ! outFile )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file for part #%u! " , splitIndex ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
2019-06-05 18:44:18 -04:00
2019-09-14 22:45:27 -04:00
if ( new_file_chunk_size > 0 )
2019-06-05 18:44:18 -04:00
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf + old_file_chunk_size , 1 , new_file_chunk_size , outFile ) ;
if ( write_res ! = new_file_chunk_size )
2019-06-19 23:56:14 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , new_file_chunk_size , progressCtx . curOffset + old_file_chunk_size , splitIndex , write_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
proceed = false ;
break ;
}
}
2019-04-21 12:27:33 -04:00
}
2019-09-14 22:45:27 -04:00
} else {
write_res = fwrite ( dumpBuf , 1 , n , outFile ) ;
if ( write_res ! = n )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX! (wrote %lu bytes) " , n , progressCtx . curOffset , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
if ( ( progressCtx . curOffset + n ) > FAT32_FILESIZE_LIMIT )
2019-05-01 16:24:13 -04:00
{
2019-09-14 22:45:27 -04:00
uiDrawString ( " You're probably using a FAT32 partition. Make sure to enable the \" Split output dump \" option. " , 8 , ( ( progressCtx . line_offset + 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
fat32_error = true ;
2019-05-01 16:24:13 -04:00
}
2019-09-14 22:45:27 -04:00
proceed = false ;
break ;
2019-04-21 12:27:33 -04:00
}
}
2019-09-14 22:45:27 -04:00
printProgressBar ( & progressCtx , true , n ) ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
if ( ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
if ( cancelProcessCheck ( & progressCtx ) )
{
uiDrawString ( " Process canceled. " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
2019-04-21 12:27:33 -04:00
}
2019-09-14 22:45:27 -04:00
}
if ( progressCtx . curOffset > = progressCtx . totalSize ) success = true ;
// Support empty files
if ( ! partitionSizes [ partition ] )
{
uiFill ( 0 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 4 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( filename , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dumping IStorage partition #%u... " , partition ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
printProgressBar ( & progressCtx , false , 0 ) ;
2019-04-21 12:27:33 -04:00
}
2019-09-14 22:45:27 -04:00
fsStorageClose ( & gameCardStorage ) ;
2019-04-21 12:27:33 -04:00
} else {
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " OpenGameCardStorage failed for partition #%u! (0x%08X) " , partition , result ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
proceed = false ;
}
2019-09-14 22:45:27 -04:00
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " GetGameCardHandle failed for partition #%u! (0x%08X) " , partition , result ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
2019-04-21 12:27:33 -04:00
}
2019-09-14 22:45:27 -04:00
if ( ! proceed )
{
setProgressBarError ( & progressCtx ) ;
break ;
}
2019-04-21 12:27:33 -04:00
}
2019-09-14 22:45:27 -04:00
breaks = ( progressCtx . line_offset + 2 ) ;
if ( fat32_error ) breaks + = 2 ;
2019-06-19 23:56:14 -04:00
if ( outFile ) fclose ( outFile ) ;
2019-04-21 12:27:33 -04:00
2019-06-19 23:56:14 -04:00
if ( success )
2019-04-21 12:27:33 -04:00
{
2019-06-19 23:56:14 -04:00
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
2019-04-21 12:27:33 -04:00
2019-06-19 23:56:14 -04:00
if ( calcCrc )
2019-04-21 12:27:33 -04:00
{
2019-06-19 23:56:14 -04:00
breaks + + ;
if ( ! trimDump )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
if ( keepCert )
2019-06-19 23:56:14 -04:00
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " XCI dump CRC32 checksum (with certificate): %08X " , crc1 ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
breaks + + ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " XCI dump CRC32 checksum (without certificate): %08X " , crc2 ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " XCI dump CRC32 checksum: %08X " , crc2 ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
}
breaks + = 2 ;
uiDrawString ( " Starting verification process using XML database from NSWDB.COM... " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
breaks + + ;
2019-06-19 23:56:14 -04:00
uiRefreshDisplay ( ) ;
gameCardDumpNSWDBCheck ( crc2 ) ;
2019-04-21 12:27:33 -04:00
} else {
2019-06-19 23:56:14 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " XCI dump CRC32 checksum: %08X " , crc1 ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
2019-06-19 23:56:14 -04:00
breaks + + ;
uiDrawString ( " Dump verification disabled (not compatible with trimmed dumps). " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
}
}
2019-06-19 23:56:14 -04:00
// Set archive bit (only for FAT32 and if the required option is enabled)
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 & & setXciArchiveBit )
2019-05-01 16:24:13 -04:00
{
2019-06-08 21:36:21 -04:00
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s.xci " , XCI_DUMP_PATH , dumpName ) ;
2019-06-19 23:56:14 -04:00
if ( R_FAILED ( result = fsdevSetArchiveBit ( filename ) ) )
2019-06-05 18:44:18 -04:00
{
2019-06-19 23:56:14 -04:00
breaks + = 2 ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Warning: failed to set archive bit on output directory! (0x%08X) " , result ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
}
2019-05-01 16:24:13 -04:00
}
} else {
2019-06-19 23:56:14 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 )
{
if ( setXciArchiveBit )
{
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s.xci " , XCI_DUMP_PATH , dumpName ) ;
2019-08-15 01:21:51 -04:00
fsdevDeleteDirectoryRecursively ( filename ) ;
2019-06-19 23:56:14 -04:00
} else {
for ( u8 i = 0 ; i < = splitIndex ; i + + )
{
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s.xc%u " , XCI_DUMP_PATH , dumpName , i ) ;
unlink ( filename ) ;
}
}
} else {
unlink ( filename ) ;
}
2019-05-01 16:24:13 -04:00
}
2019-06-19 23:56:14 -04:00
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file \" %s \" ! " , filename ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
}
}
} else {
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
}
}
breaks + = 2 ;
free ( dumpName ) ;
return success ;
}
2019-06-28 16:13:54 -04:00
bool dumpNintendoSubmissionPackage ( nspDumpType selectedNspDumpType , u32 titleIndex , bool isFat32 , bool calcCrc , bool removeConsoleData , bool tiklessDump , bool batch )
2019-05-01 16:24:13 -04:00
{
Result result ;
u32 i = 0 , j = 0 ;
u32 written = 0 ;
u32 total = 0 ;
2019-06-19 23:56:14 -04:00
u32 titleCount = 0 ;
u32 ncmTitleIndex = 0 ;
2019-06-05 18:44:18 -04:00
u32 titleNcaCount = 0 ;
2019-06-08 21:36:21 -04:00
u32 partition = 0 ;
FsStorageId curStorageId ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
FsGameCardHandle handle ;
FsStorage gameCardStorage ;
memset ( & gameCardStorage , 0 , sizeof ( FsStorage ) ) ;
2019-05-01 16:24:13 -04:00
NcmContentMetaDatabase ncmDb ;
2019-06-05 18:44:18 -04:00
memset ( & ncmDb , 0 , sizeof ( NcmContentMetaDatabase ) ) ;
2019-06-08 21:36:21 -04:00
NcmContentMetaRecordsHeader contentRecordsHeader ;
memset ( & contentRecordsHeader , 0 , sizeof ( NcmContentMetaRecordsHeader ) ) ;
u64 contentRecordsHeaderReadSize = 0 ;
2019-05-01 16:24:13 -04:00
NcmContentStorage ncmStorage ;
2019-06-05 18:44:18 -04:00
memset ( & ncmStorage , 0 , sizeof ( NcmContentStorage ) ) ;
NcmApplicationContentMetaKey * titleList = NULL ;
NcmContentRecord * titleContentRecords = NULL ;
size_t titleListSize = sizeof ( NcmApplicationContentMetaKey ) ;
2019-05-01 16:24:13 -04:00
cnmt_xml_program_info xml_program_info ;
cnmt_xml_content_info * xml_content_info = NULL ;
NcmNcaId ncaId ;
2019-06-05 18:44:18 -04:00
u8 ncaHeader [ NCA_FULL_HEADER_LENGTH ] = { 0 } ;
2019-05-01 16:24:13 -04:00
nca_header_t dec_nca_header ;
2019-06-05 18:44:18 -04:00
nca_program_mod_data ncaProgramMod ;
memset ( & ncaProgramMod , 0 , sizeof ( nca_program_mod_data ) ) ;
ncaProgramMod . hash_table = NULL ;
ncaProgramMod . block_data [ 0 ] = NULL ;
ncaProgramMod . block_data [ 1 ] = NULL ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
nca_cnmt_mod_data ncaCnmtMod ;
memset ( & ncaCnmtMod , 0 , sizeof ( nca_cnmt_mod_data ) ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
title_rights_ctx rights_info ;
memset ( & rights_info , 0 , sizeof ( title_rights_ctx ) ) ;
u32 cnmtNcaIndex = 0 ;
u8 * cnmtNcaBuf = NULL ;
bool cnmtFound = false ;
char * cnmtXml = NULL ;
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
u32 programNcaIndex = 0 ;
u64 programInfoXmlSize = 0 ;
char * programInfoXml = NULL ;
2019-06-05 18:44:18 -04:00
u32 nacpNcaIndex = 0 ;
2019-06-28 16:13:54 -04:00
u64 nacpXmlSize = 0 ;
2019-06-05 18:44:18 -04:00
char * nacpXml = NULL ;
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
u8 nacpIconCnt = 0 ;
nacp_icons_ctx * nacpIcons = NULL ;
u32 legalInfoNcaIndex = 0 ;
u64 legalInfoXmlSize = 0 ;
char * legalInfoXml = NULL ;
2019-06-05 18:44:18 -04:00
u32 nspFileCount = 0 ;
2019-05-01 16:24:13 -04:00
pfs0_header nspPfs0Header ;
pfs0_entry_table * nspPfs0EntryTable = NULL ;
char * nspPfs0StrTable = NULL ;
2019-06-05 18:44:18 -04:00
u64 nspPfs0StrTableSize = 0 ;
u64 full_nsp_header_size = 0 ;
u64 hash_table_dump_buffer_start = 0 ;
u64 hash_table_dump_buffer_end = 0 ;
u64 block0_dump_buffer_start = 0 ;
u64 block0_dump_buffer_end = 0 ;
u64 block1_dump_buffer_start = 0 ;
u64 block1_dump_buffer_end = 0 ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
Sha256Context nca_hash_ctx ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
char dumpPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
u64 n , nca_offset ;
2019-05-01 16:24:13 -04:00
FILE * outFile = NULL ;
u8 splitIndex = 0 ;
u32 crc = 0 ;
2019-06-19 23:56:14 -04:00
bool proceed = true , success = false , dumping = false , fat32_error = false , removeFile = true ;
2019-05-01 16:24:13 -04:00
2019-09-14 22:45:27 -04:00
memset ( dumpBuf , 0 , DUMP_BUFFER_SIZE ) ;
2019-06-05 18:44:18 -04:00
progress_ctx_t progressCtx ;
memset ( & progressCtx , 0 , sizeof ( progress_ctx_t ) ) ;
char tmp_idx [ 5 ] ;
2019-05-01 16:24:13 -04:00
size_t write_res ;
2019-06-05 18:44:18 -04:00
int initial_breaks = breaks ;
2019-05-01 16:24:13 -04:00
2019-06-19 23:56:14 -04:00
if ( ( selectedNspDumpType = = DUMP_APP_NSP & & ! titleAppStorageId ) | | ( selectedNspDumpType = = DUMP_PATCH_NSP & & ! titlePatchStorageId ) | | ( selectedNspDumpType = = DUMP_ADDON_NSP & & ! titleAddOnStorageId ) )
2019-06-05 18:44:18 -04:00
{
2019-06-19 23:56:14 -04:00
uiDrawString ( " Error: title storage ID unavailable! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
breaks + = 2 ;
return false ;
}
2019-06-19 23:56:14 -04:00
curStorageId = ( selectedNspDumpType = = DUMP_APP_NSP ? titleAppStorageId [ titleIndex ] : ( selectedNspDumpType = = DUMP_PATCH_NSP ? titlePatchStorageId [ titleIndex ] : titleAddOnStorageId [ titleIndex ] ) ) ;
if ( curStorageId ! = FsStorageId_GameCard & & curStorageId ! = FsStorageId_SdCard & & curStorageId ! = FsStorageId_NandUser )
2019-06-05 18:44:18 -04:00
{
2019-06-19 23:56:14 -04:00
uiDrawString ( " Error: invalid title storage ID! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
breaks + = 2 ;
return false ;
}
2019-06-19 23:56:14 -04:00
switch ( curStorageId )
2019-05-01 16:24:13 -04:00
{
2019-06-19 23:56:14 -04:00
case FsStorageId_GameCard :
titleCount = ( selectedNspDumpType = = DUMP_APP_NSP ? titleAppCount : ( selectedNspDumpType = = DUMP_PATCH_NSP ? titlePatchCount : titleAddOnCount ) ) ;
ncmTitleIndex = titleIndex ;
break ;
case FsStorageId_SdCard :
titleCount = ( selectedNspDumpType = = DUMP_APP_NSP ? sdCardTitleAppCount : ( selectedNspDumpType = = DUMP_PATCH_NSP ? sdCardTitlePatchCount : sdCardTitleAddOnCount ) ) ;
ncmTitleIndex = titleIndex ;
break ;
case FsStorageId_NandUser :
titleCount = ( selectedNspDumpType = = DUMP_APP_NSP ? nandUserTitleAppCount : ( selectedNspDumpType = = DUMP_PATCH_NSP ? nandUserTitlePatchCount : nandUserTitleAddOnCount ) ) ;
ncmTitleIndex = ( titleIndex - ( selectedNspDumpType = = DUMP_APP_NSP ? sdCardTitleAppCount : ( selectedNspDumpType = = DUMP_PATCH_NSP ? sdCardTitlePatchCount : sdCardTitleAddOnCount ) ) ) ;
break ;
default :
break ;
}
if ( ! titleCount )
{
uiDrawString ( " Error: invalid title type count! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
breaks + = 2 ;
return false ;
}
2019-06-19 23:56:14 -04:00
if ( ncmTitleIndex > ( titleCount - 1 ) )
2019-05-01 16:24:13 -04:00
{
2019-06-19 23:56:14 -04:00
uiDrawString ( " Error: invalid title index! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-08 21:36:21 -04:00
breaks + = 2 ;
return false ;
2019-06-05 18:44:18 -04:00
}
2019-06-19 23:56:14 -04:00
titleListSize * = titleCount ;
2019-06-08 21:36:21 -04:00
char * dumpName = generateNSPDumpName ( selectedNspDumpType , titleIndex ) ;
if ( ! dumpName )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
2019-06-05 18:44:18 -04:00
}
2019-06-08 21:36:21 -04:00
// If we're dealing with a gamecard, call workaroundPartitionZeroAccess() and read the secure partition header. Otherwise, ncmContentStorageReadContentIdFile() will fail with error 0x00171002
2019-06-19 23:56:14 -04:00
// Also open an IStorage instance for the HFS0 Secure partition, since we may also need it if we're dealing with a Patch with titlekey crypto, in order to retrieve the tik file
2019-06-08 21:36:21 -04:00
if ( curStorageId = = FsStorageId_GameCard )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
partition = ( hfs0_partition_cnt - 1 ) ; // Select the secure partition
workaroundPartitionZeroAccess ( ) ;
if ( ! getPartitionHfs0Header ( partition ) ) goto out ;
if ( ! partitionHfs0FileCount )
{
uiDrawString ( " The Secure HFS0 partition is empty! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
goto out ;
}
if ( R_FAILED ( result = fsDeviceOperatorGetGameCardHandle ( & fsOperatorInstance , & handle ) ) )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " GetGameCardHandle failed! (0x%08X) " , result ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
goto out ;
}
if ( R_FAILED ( result = fsOpenGameCardStorage ( & gameCardStorage , & handle , HFS0_TO_ISTORAGE_IDX ( hfs0_partition_cnt , partition ) ) ) )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " OpenGameCardStorage failed! (0x%08X) " , result ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
goto out ;
}
2019-05-01 16:24:13 -04:00
}
2019-06-28 16:13:54 -04:00
if ( ! batch )
{
uiDrawString ( " Retrieving information from encrypted NCA content files... " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + + ;
}
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
titleList = calloc ( 1 , titleListSize ) ;
if ( ! titleList )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to allocate memory for the ApplicationContentMetaKey struct! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-08 21:36:21 -04:00
if ( R_FAILED ( result = ncmOpenContentMetaDatabase ( curStorageId , & ncmDb ) ) )
2019-05-01 16:24:13 -04:00
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Error: ncmOpenContentMetaDatabase failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-05 18:44:18 -04:00
u8 filter = ( ( u8 ) selectedNspDumpType + META_DB_REGULAR_APPLICATION ) ;
if ( R_FAILED ( result = ncmContentMetaDatabaseListApplication ( & ncmDb , filter , titleList , titleListSize , & written , & total ) ) )
2019-05-01 16:24:13 -04:00
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Error: ncmContentMetaDatabaseListApplication failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
if ( ! written | | ! total )
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: ncmContentMetaDatabaseListApplication wrote no entries to output buffer! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-19 23:56:14 -04:00
if ( written ! = total )
2019-06-08 21:36:21 -04:00
{
2019-06-19 23:56:14 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Error: title count mismatch in ncmContentMetaDatabaseListApplication (%u != %u) " , written , total ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
goto out ;
}
2019-06-19 23:56:14 -04:00
if ( R_FAILED ( result = ncmContentMetaDatabaseGet ( & ncmDb , & ( titleList [ ncmTitleIndex ] . metaRecord ) , sizeof ( NcmContentMetaRecordsHeader ) , & contentRecordsHeader , & contentRecordsHeaderReadSize ) ) )
2019-05-01 16:24:13 -04:00
{
2019-06-08 21:36:21 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Error: ncmContentMetaDatabaseGet failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-08 21:36:21 -04:00
titleNcaCount = ( u32 ) ( contentRecordsHeader . numContentRecords ) ;
titleContentRecords = calloc ( titleNcaCount , sizeof ( NcmContentRecord ) ) ;
2019-06-05 18:44:18 -04:00
if ( ! titleContentRecords )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to allocate memory for the ContentRecord struct! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-19 23:56:14 -04:00
if ( R_FAILED ( result = ncmContentMetaDatabaseListContentInfo ( & ncmDb , & ( titleList [ ncmTitleIndex ] . metaRecord ) , 0 , titleContentRecords , titleNcaCount * sizeof ( NcmContentRecord ) , & written ) ) )
2019-05-01 16:24:13 -04:00
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Error: ncmContentMetaDatabaseListContentInfo failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-05 18:44:18 -04:00
// Fill information for our CNMT XML
2019-05-01 16:24:13 -04:00
memset ( & xml_program_info , 0 , sizeof ( cnmt_xml_program_info ) ) ;
2019-06-19 23:56:14 -04:00
xml_program_info . type = titleList [ ncmTitleIndex ] . metaRecord . type ;
xml_program_info . title_id = titleList [ ncmTitleIndex ] . metaRecord . titleId ;
xml_program_info . version = titleList [ ncmTitleIndex ] . metaRecord . version ;
2019-06-05 18:44:18 -04:00
xml_program_info . nca_cnt = titleNcaCount ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
xml_content_info = calloc ( titleNcaCount , sizeof ( cnmt_xml_content_info ) ) ;
2019-05-01 16:24:13 -04:00
if ( ! xml_content_info )
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to allocate memory for the CNMT XML content info struct! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-08 21:36:21 -04:00
if ( R_FAILED ( result = ncmOpenContentStorage ( curStorageId , & ncmStorage ) ) )
2019-05-01 16:24:13 -04:00
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Error: ncmOpenContentStorage failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-05 18:44:18 -04:00
// Fill our CNMT XML content records, leaving the CNMT NCA at the end
u32 titleRecordIndex ;
for ( i = 0 , titleRecordIndex = 0 ; titleRecordIndex < titleNcaCount ; i + + , titleRecordIndex + + )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
if ( ! cnmtFound & & titleContentRecords [ titleRecordIndex ] . type = = NcmContentType_CNMT )
{
cnmtFound = true ;
cnmtNcaIndex = titleRecordIndex ;
i - - ;
continue ;
}
2019-06-19 23:56:14 -04:00
// Skip Delta Fragments or any other unknown content types if we're not dealing with Patch titles dumped from installed SD/eMMC (with tiklessDump disabled)
if ( titleContentRecords [ titleRecordIndex ] . type > = NCA_CONTENT_TYPE_DELTA & & ( curStorageId = = FsStorageId_GameCard | | selectedNspDumpType ! = DUMP_PATCH_NSP | | tiklessDump ) )
2019-06-05 18:44:18 -04:00
{
xml_program_info . nca_cnt - - ;
i - - ;
continue ;
}
// Fill information for our CNMT XML
xml_content_info [ i ] . type = titleContentRecords [ titleRecordIndex ] . type ;
memcpy ( xml_content_info [ i ] . nca_id , titleContentRecords [ titleRecordIndex ] . ncaId . c , 16 ) ; // Temporary
convertDataToHexString ( titleContentRecords [ titleRecordIndex ] . ncaId . c , 16 , xml_content_info [ i ] . nca_id_str , 33 ) ; // Temporary
convertNcaSizeToU64 ( titleContentRecords [ titleRecordIndex ] . size , & ( xml_content_info [ i ] . size ) ) ;
convertDataToHexString ( xml_content_info [ i ] . hash , 32 , xml_content_info [ i ] . hash_str , 65 ) ; // Temporary
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
memcpy ( & ncaId , & ( titleContentRecords [ titleRecordIndex ] . ncaId ) , sizeof ( NcmNcaId ) ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
if ( R_FAILED ( result = ncmContentStorageReadContentIdFile ( & ncmStorage , & ncaId , 0 , ncaHeader , NCA_FULL_HEADER_LENGTH ) ) )
2019-05-01 16:24:13 -04:00
{
2019-06-19 23:56:14 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to read header from NCA \" %s \" ! (0x%08X) " , xml_content_info [ i ] . nca_id_str , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
// Decrypt the NCA header
2019-06-08 21:36:21 -04:00
// Don't retrieve the ticket and/or titlekey if we're dealing with a Patch with titlekey crypto bundled with the inserted gamecard
if ( ! decryptNcaHeader ( ncaHeader , NCA_FULL_HEADER_LENGTH , & dec_nca_header , & rights_info , xml_content_info [ i ] . decrypted_nca_keys , ( curStorageId ! = FsStorageId_GameCard ) ) )
2019-06-05 18:44:18 -04:00
{
proceed = false ;
break ;
}
bool has_rights_id = false ;
for ( j = 0 ; j < 0x10 ; j + + )
{
if ( dec_nca_header . rights_id [ j ] ! = 0 )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
has_rights_id = true ;
2019-05-01 16:24:13 -04:00
break ;
}
2019-06-05 18:44:18 -04:00
}
2019-06-08 21:36:21 -04:00
// Fill information for our CNMT XML
xml_content_info [ i ] . keyblob = ( dec_nca_header . crypto_type2 > dec_nca_header . crypto_type ? dec_nca_header . crypto_type2 : dec_nca_header . crypto_type ) ;
if ( curStorageId = = FsStorageId_GameCard )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
if ( selectedNspDumpType = = DUMP_APP_NSP | | selectedNspDumpType = = DUMP_ADDON_NSP )
2019-05-01 16:24:13 -04:00
{
2019-06-08 21:36:21 -04:00
if ( has_rights_id )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Error: Rights ID field in NCA header not empty! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
proceed = false ;
break ;
}
2019-06-08 21:36:21 -04:00
// Modify distribution type
dec_nca_header . distribution = 0 ;
// Patch ACID pubkey and recreate NCA NPDM signature if we're dealing with the Program NCA
if ( xml_content_info [ i ] . type = = NcmContentType_Program )
{
if ( ! processProgramNca ( & ncmStorage , & ncaId , & dec_nca_header , & ( xml_content_info [ i ] ) , & ncaProgramMod ) )
{
proceed = false ;
break ;
}
}
} else
if ( selectedNspDumpType = = DUMP_PATCH_NSP )
{
if ( ! has_rights_id )
{
2019-06-28 16:13:54 -04:00
// We could be dealing with a custom XCI mounted through SX OS, so let's change back the content distribution method
2019-06-08 21:36:21 -04:00
dec_nca_header . distribution = 0 ;
2019-06-19 23:56:14 -04:00
} else {
if ( ! rights_info . retrieved_tik )
{
// Retrieve tik file
if ( ! getPartitionHfs0FileByName ( & gameCardStorage , rights_info . tik_filename , ( u8 * ) ( & ( rights_info . tik_data ) ) , ETICKET_TIK_FILE_SIZE ) )
{
proceed = false ;
break ;
}
memcpy ( rights_info . enc_titlekey , rights_info . tik_data . titlekey_block , 0x10 ) ;
// Load external keys
if ( ! loadExternalKeys ( ) )
{
proceed = false ;
break ;
}
// Decrypt titlekey
u8 crypto_type = xml_content_info [ i ] . keyblob ;
if ( crypto_type ) crypto_type - - ;
Aes128Context titlekey_aes_ctx ;
aes128ContextCreate ( & titlekey_aes_ctx , nca_keyset . titlekeks [ crypto_type ] , false ) ;
aes128DecryptBlock ( & titlekey_aes_ctx , rights_info . dec_titlekey , rights_info . enc_titlekey ) ;
2019-06-28 16:13:54 -04:00
rights_info . retrieved_tik = true ;
}
memset ( xml_content_info [ i ] . decrypted_nca_keys , 0 , NCA_KEY_AREA_SIZE ) ;
memcpy ( xml_content_info [ i ] . decrypted_nca_keys + ( NCA_KEY_AREA_KEY_SIZE * 2 ) , rights_info . dec_titlekey , 0x10 ) ;
// Mess with the NCA header if we're dealing with a content with a populated rights ID field and if tiklessDump is true (removeConsoleData is ignored)
if ( tiklessDump )
{
2019-06-19 23:56:14 -04:00
// Generate new encrypted NCA key area using titlekey
if ( ! generateEncryptedNcaKeyAreaWithTitlekey ( & dec_nca_header , xml_content_info [ i ] . decrypted_nca_keys ) )
{
proceed = false ;
break ;
}
// Remove rights ID from NCA
memset ( dec_nca_header . rights_id , 0 , 0x10 ) ;
2019-06-28 16:13:54 -04:00
// Patch ACID pubkey and recreate NCA NPDM signature if we're dealing with the Program NCA
if ( xml_content_info [ i ] . type = = NcmContentType_Program )
{
if ( ! processProgramNca ( & ncmStorage , & ncaId , & dec_nca_header , & ( xml_content_info [ i ] ) , & ncaProgramMod ) )
{
proceed = false ;
break ;
}
}
2019-06-19 23:56:14 -04:00
}
2019-06-08 21:36:21 -04:00
}
2019-05-01 16:24:13 -04:00
}
2019-06-05 18:44:18 -04:00
} else
2019-06-08 21:36:21 -04:00
if ( curStorageId = = FsStorageId_SdCard | | curStorageId = = FsStorageId_NandUser )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
// Only mess with the NCA header if we're dealing with a content with a populated rights ID field, and if both removeConsoleData and tiklessDump are true
if ( has_rights_id & & removeConsoleData & & tiklessDump )
2019-05-01 16:24:13 -04:00
{
2019-06-08 21:36:21 -04:00
// Generate new encrypted NCA key area using titlekey
if ( ! generateEncryptedNcaKeyAreaWithTitlekey ( & dec_nca_header , xml_content_info [ i ] . decrypted_nca_keys ) )
{
proceed = false ;
break ;
}
// Remove rights ID from NCA
memset ( dec_nca_header . rights_id , 0 , 0x10 ) ;
2019-06-28 16:13:54 -04:00
// Patch ACID pubkey and recreate NCA NPDM signature if we're dealing with the Program NCA
if ( xml_content_info [ i ] . type = = NcmContentType_Program )
{
if ( ! processProgramNca ( & ncmStorage , & ncaId , & dec_nca_header , & ( xml_content_info [ i ] ) , & ncaProgramMod ) )
{
proceed = false ;
break ;
}
}
}
}
// Generate programinfo.xml
if ( ! programInfoXml & & xml_content_info [ i ] . type = = NcmContentType_Program )
{
programNcaIndex = i ;
if ( ! generateProgramInfoXml ( & ncmStorage , & ncaId , & dec_nca_header , xml_content_info [ i ] . decrypted_nca_keys , & ncaProgramMod , & programInfoXml , & programInfoXmlSize ) )
{
proceed = false ;
break ;
2019-05-01 16:24:13 -04:00
}
2019-06-05 18:44:18 -04:00
}
2019-06-28 16:13:54 -04:00
// Retrieve NACP data (XML and icons)
2019-06-05 18:44:18 -04:00
if ( ! nacpXml & & xml_content_info [ i ] . type = = NcmContentType_Icon )
{
nacpNcaIndex = i ;
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
if ( ! retrieveNacpDataFromNca ( & ncmStorage , & ncaId , & dec_nca_header , xml_content_info [ i ] . decrypted_nca_keys , & nacpXml , & nacpXmlSize , & nacpIcons , & nacpIconCnt ) )
{
proceed = false ;
break ;
}
}
// Retrieve legalinfo.xml
if ( ! legalInfoXml & & xml_content_info [ i ] . type = = NcmContentType_Info )
{
legalInfoNcaIndex = i ;
if ( ! retrieveLegalInfoXmlFromNca ( & ncmStorage , & ncaId , & dec_nca_header , xml_content_info [ i ] . decrypted_nca_keys , & legalInfoXml , & legalInfoXmlSize ) )
2019-05-01 16:24:13 -04:00
{
proceed = false ;
break ;
}
}
// Reencrypt header
if ( ! encryptNcaHeader ( & dec_nca_header , xml_content_info [ i ] . encrypted_header_mod , NCA_FULL_HEADER_LENGTH ) )
{
proceed = false ;
break ;
}
}
2019-06-08 21:36:21 -04:00
if ( ! proceed ) goto out ;
2019-05-01 16:24:13 -04:00
if ( proceed & & ! cnmtFound )
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to find CNMT NCA! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-05 18:44:18 -04:00
// Update NCA counter just in case we found any delta fragments
titleNcaCount = xml_program_info . nca_cnt ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
// Fill information for our CNMT XML
xml_content_info [ titleNcaCount - 1 ] . type = titleContentRecords [ cnmtNcaIndex ] . type ;
memcpy ( xml_content_info [ titleNcaCount - 1 ] . nca_id , titleContentRecords [ cnmtNcaIndex ] . ncaId . c , 16 ) ; // Temporary
convertDataToHexString ( titleContentRecords [ cnmtNcaIndex ] . ncaId . c , 16 , xml_content_info [ titleNcaCount - 1 ] . nca_id_str , 33 ) ; // Temporary
convertNcaSizeToU64 ( titleContentRecords [ cnmtNcaIndex ] . size , & ( xml_content_info [ titleNcaCount - 1 ] . size ) ) ;
convertDataToHexString ( xml_content_info [ titleNcaCount - 1 ] . hash , 32 , xml_content_info [ titleNcaCount - 1 ] . hash_str , 65 ) ; // Temporary
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
memcpy ( & ncaId , & ( titleContentRecords [ cnmtNcaIndex ] . ncaId ) , sizeof ( NcmNcaId ) ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
// Update CNMT index
cnmtNcaIndex = ( titleNcaCount - 1 ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
cnmtNcaBuf = malloc ( xml_content_info [ cnmtNcaIndex ] . size ) ;
if ( ! cnmtNcaBuf )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to allocate memory for CNMT NCA data! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-05 18:44:18 -04:00
if ( R_FAILED ( result = ncmContentStorageReadContentIdFile ( & ncmStorage , & ncaId , 0 , cnmtNcaBuf , xml_content_info [ cnmtNcaIndex ] . size ) ) )
2019-05-01 16:24:13 -04:00
{
2019-06-19 23:56:14 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to read CNMT NCA \" %s \" ! (0x%08X) " , xml_content_info [ cnmtNcaIndex ] . nca_id_str , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-05 18:44:18 -04:00
// Retrieve CNMT NCA data
2019-06-08 21:36:21 -04:00
if ( ! retrieveCnmtNcaData ( curStorageId , selectedNspDumpType , cnmtNcaBuf , & xml_program_info , & ( xml_content_info [ cnmtNcaIndex ] ) , & ncaCnmtMod , & rights_info , ( removeConsoleData & & tiklessDump ) ) ) goto out ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
// Generate a placeholder CNMT XML. It's length will be used to calculate the final output dump size
/*breaks++;
uiDrawString ( " Generating placeholder CNMT XML... " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + + ; */
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
// Make sure that the output buffer for our CNMT XML is big enough
2019-09-14 22:45:27 -04:00
cnmtXml = calloc ( NSP_XML_BUFFER_SIZE , sizeof ( char ) ) ;
2019-06-05 18:44:18 -04:00
if ( ! cnmtXml )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to allocate memory for the CNMT XML! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-05 18:44:18 -04:00
generateCnmtXml ( & xml_program_info , xml_content_info , cnmtXml ) ;
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
bool includeTikAndCert = ( rights_info . has_rights_id & & ! tiklessDump ) ;
2019-06-08 21:36:21 -04:00
2019-06-28 16:13:54 -04:00
if ( includeTikAndCert )
2019-05-01 16:24:13 -04:00
{
2019-06-08 21:36:21 -04:00
if ( curStorageId = = FsStorageId_GameCard )
{
// Ticket files from Patch titles bundled with gamecards have a different layout
// Let's convert it to a normal "common" ticket
memset ( rights_info . tik_data . signature , 0xFF , 0x100 ) ;
memset ( ( u8 * ) ( & ( rights_info . tik_data ) ) + 0x190 , 0 , 0x130 ) ;
rights_info . tik_data . unk1 = 0x02 ; // Always 0x02 ?
rights_info . tik_data . master_key_rev = rights_info . rights_id [ 15 ] ;
memcpy ( rights_info . tik_data . rights_id , rights_info . rights_id , 0x10 ) ;
rights_info . tik_data . unk4 [ 4 ] = 0xC0 ; // Always 0xC0 ?
rights_info . tik_data . unk4 [ 5 ] = 0x02 ; // Always 0x02 ?
} else
if ( curStorageId = = FsStorageId_SdCard | | curStorageId = = FsStorageId_NandUser )
{
// Only mess with the ticket data if removeConsoleData is true, if tiklessDump is false and if we're dealing with a personalized ticket
if ( removeConsoleData & & rights_info . tik_data . titlekey_type = = ETICKET_TITLEKEY_PERSONALIZED )
{
memset ( rights_info . tik_data . signature , 0xFF , 0x100 ) ;
memset ( rights_info . tik_data . sig_issuer , 0 , 0x40 ) ;
sprintf ( rights_info . tik_data . sig_issuer , " Root-CA00000003-XS00000020 " ) ;
memset ( rights_info . tik_data . titlekey_block , 0 , 0x100 ) ;
memcpy ( rights_info . tik_data . titlekey_block , rights_info . enc_titlekey , 0x10 ) ;
rights_info . tik_data . titlekey_type = ETICKET_TITLEKEY_COMMON ;
rights_info . tik_data . ticket_id = 0 ;
rights_info . tik_data . device_id = 0 ;
rights_info . tik_data . account_id = 0 ;
}
}
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
// Retrieve cert file
2019-06-08 21:36:21 -04:00
if ( ! retrieveCertData ( rights_info . cert_data , ( rights_info . tik_data . titlekey_type = = ETICKET_TITLEKEY_PERSONALIZED ) ) ) goto out ;
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
// File count = NCA count + CNMT XML + tik + cert
2019-06-05 18:44:18 -04:00
nspFileCount = ( titleNcaCount + 3 ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
// Calculate PFS0 String Table size
2019-06-28 16:13:54 -04:00
nspPfs0StrTableSize = ( ( ( nspFileCount - 4 ) * NSP_NCA_FILENAME_LENGTH ) + ( NSP_CNMT_FILENAME_LENGTH * 2 ) + NSP_TIK_FILENAME_LENGTH + NSP_CERT_FILENAME_LENGTH ) ;
2019-06-05 18:44:18 -04:00
} else {
// File count = NCA count + CNMT XML
nspFileCount = ( titleNcaCount + 1 ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
// Calculate PFS0 String Table size
nspPfs0StrTableSize = ( ( ( nspFileCount - 2 ) * NSP_NCA_FILENAME_LENGTH ) + ( NSP_CNMT_FILENAME_LENGTH * 2 ) ) ;
2019-05-01 16:24:13 -04:00
}
2019-06-28 16:13:54 -04:00
// Add our programinfo.xml if we created it
if ( programInfoXml )
{
nspFileCount + + ;
nspPfs0StrTableSize + = NSP_PROGRAM_XML_FILENAME_LENGTH ;
}
2019-06-05 18:44:18 -04:00
// Add our NACP XML if we created it
if ( nacpXml )
2019-06-28 16:13:54 -04:00
{
// Add icons if we retrieved them
if ( nacpIcons & & nacpIconCnt )
{
for ( i = 0 ; i < nacpIconCnt ; i + + )
{
nspFileCount + + ;
nspPfs0StrTableSize + = ( strlen ( nacpIcons [ i ] . filename ) + 1 ) ;
}
}
nspFileCount + + ;
nspPfs0StrTableSize + = NSP_NACP_XML_FILENAME_LENGTH ;
}
// Add our legalinfo.xml if we retrieved it
if ( legalInfoXml )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
nspFileCount + + ;
2019-06-28 16:13:54 -04:00
nspPfs0StrTableSize + = NSP_LEGAL_XML_FILENAME_LENGTH ;
2019-06-05 18:44:18 -04:00
}
2019-05-01 16:24:13 -04:00
// Start NSP creation
2019-06-05 18:44:18 -04:00
/*breaks++;
uiDrawString ( " Generating placeholder PFS0 header... " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-05-01 16:24:13 -04:00
uiRefreshDisplay ( ) ;
2019-06-05 18:44:18 -04:00
breaks + + ; */
2019-05-01 16:24:13 -04:00
memset ( & nspPfs0Header , 0 , sizeof ( pfs0_header ) ) ;
2019-06-05 18:44:18 -04:00
nspPfs0Header . magic = bswap_32 ( PFS0_MAGIC ) ;
nspPfs0Header . file_cnt = nspFileCount ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
nspPfs0EntryTable = calloc ( nspFileCount , sizeof ( pfs0_entry_table ) ) ;
2019-05-01 16:24:13 -04:00
if ( ! nspPfs0EntryTable )
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Unable to allocate memory for the PFS0 file entries! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-05 18:44:18 -04:00
// Make sure we have enough space
nspPfs0StrTable = calloc ( nspPfs0StrTableSize * 2 , sizeof ( char ) ) ;
2019-05-01 16:24:13 -04:00
if ( ! nspPfs0StrTable )
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Unable to allocate memory for the PFS0 string table! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
// Determine our full NSP header size
2019-06-28 16:13:54 -04:00
full_nsp_header_size = ( sizeof ( pfs0_header ) + ( ( u64 ) nspFileCount * sizeof ( pfs0_entry_table ) ) + nspPfs0StrTableSize ) ;
// Round up our full NSP header size to a 0x10-byte boundary
if ( ! ( full_nsp_header_size % 0x10 ) ) full_nsp_header_size + + ; // If it's already rounded, add more padding
2019-06-05 18:44:18 -04:00
full_nsp_header_size = round_up ( full_nsp_header_size , 0x10 ) ;
2019-05-01 16:24:13 -04:00
// Determine our String Table size
2019-06-28 16:13:54 -04:00
nspPfs0Header . str_table_size = ( full_nsp_header_size - ( sizeof ( pfs0_header ) + ( ( u64 ) nspFileCount * sizeof ( pfs0_entry_table ) ) ) ) ;
2019-05-01 16:24:13 -04:00
// Calculate total dump size
2019-06-05 18:44:18 -04:00
progressCtx . totalSize = full_nsp_header_size ;
2019-06-28 16:13:54 -04:00
2019-06-05 18:44:18 -04:00
for ( i = 0 ; i < titleNcaCount ; i + + ) progressCtx . totalSize + = xml_content_info [ i ] . size ;
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
progressCtx . totalSize + = strlen ( cnmtXml ) ;
if ( programInfoXml ) progressCtx . totalSize + = programInfoXmlSize ;
if ( nacpXml )
{
if ( nacpIcons & & nacpIconCnt )
{
for ( i = 0 ; i < nacpIconCnt ; i + + ) progressCtx . totalSize + = nacpIcons [ i ] . icon_size ;
}
progressCtx . totalSize + = nacpXmlSize ;
}
if ( legalInfoXml ) progressCtx . totalSize + = legalInfoXmlSize ;
if ( includeTikAndCert ) progressCtx . totalSize + = ( ETICKET_TIK_FILE_SIZE + ETICKET_CERT_FILE_SIZE ) ;
2019-06-05 18:44:18 -04:00
convertSize ( progressCtx . totalSize , progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) ) ;
2019-06-28 16:13:54 -04:00
if ( ! batch )
{
breaks + + ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Total NSP dump size: %s (%lu bytes). " , progressCtx . totalSizeStr , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + + ;
}
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize > freeSpace )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-19 23:56:14 -04:00
// Temporary, we'll use this to check if the dump already exists (it should have the archive bit set if so)
2019-06-08 21:36:21 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp " , NSP_DUMP_PATH , dumpName ) ;
2019-06-05 18:44:18 -04:00
2019-06-19 23:56:14 -04:00
// Check if the dump already exists
2019-06-28 16:13:54 -04:00
if ( ! batch & & checkIfFileExists ( dumpPath ) )
2019-06-19 23:56:14 -04:00
{
// Ask the user if they want to proceed anyway
int cur_breaks = breaks ;
breaks + + ;
proceed = yesNoPrompt ( " You have already dumped this content. Do you wish to proceed anyway? " ) ;
if ( ! proceed )
{
uiDrawString ( " Process canceled. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
removeFile = false ;
goto out ;
} else {
// Remove the prompt from the screen
breaks = cur_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
}
}
2019-06-05 18:44:18 -04:00
// Since we may actually be dealing with an existing directory with the archive bit set or unset, let's try both
// Better safe than sorry
unlink ( dumpPath ) ;
2019-08-15 01:21:51 -04:00
fsdevDeleteDirectoryRecursively ( dumpPath ) ;
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 )
2019-05-01 16:24:13 -04:00
{
mkdir ( dumpPath , 0744 ) ;
2019-06-05 18:44:18 -04:00
sprintf ( tmp_idx , " /%02u " , splitIndex ) ;
strcat ( dumpPath , tmp_idx ) ;
2019-05-01 16:24:13 -04:00
}
outFile = fopen ( dumpPath , " wb " ) ;
if ( ! outFile )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file \" %s \" ! " , dumpPath ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-28 16:13:54 -04:00
if ( ! batch )
{
breaks + + ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dump procedure started. Hold %s to cancel. " , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + = 2 ;
}
2019-05-01 16:24:13 -04:00
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
breaks + = 2 ;
}
2019-06-05 18:44:18 -04:00
// Write placeholder zeroes
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , full_nsp_header_size , outFile ) ;
2019-06-28 16:13:54 -04:00
if ( write_res ! = full_nsp_header_size )
2019-05-01 16:24:13 -04:00
{
2019-06-28 16:13:54 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes placeholder data to file offset 0x%016lX! (wrote %lu bytes) " , full_nsp_header_size , ( u64 ) 0 , write_res ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-28 16:13:54 -04:00
progressCtx . curOffset = full_nsp_header_size ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
// Calculate DUMP_BUFFER_SIZE block numbers for the modified Program NCA data blocks
2019-06-28 16:13:54 -04:00
if ( ncaProgramMod . block_mod_cnt > 0 )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
hash_table_dump_buffer_start = ( ( ncaProgramMod . hash_table_offset / DUMP_BUFFER_SIZE ) * DUMP_BUFFER_SIZE ) ;
hash_table_dump_buffer_end = ( ( ( ncaProgramMod . hash_table_offset + ncaProgramMod . hash_table_size ) / DUMP_BUFFER_SIZE ) * DUMP_BUFFER_SIZE ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
block0_dump_buffer_start = ( ( ncaProgramMod . block_offset [ 0 ] / DUMP_BUFFER_SIZE ) * DUMP_BUFFER_SIZE ) ;
block0_dump_buffer_end = ( ( ( ncaProgramMod . block_offset [ 0 ] + ncaProgramMod . block_size [ 0 ] ) / DUMP_BUFFER_SIZE ) * DUMP_BUFFER_SIZE ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
if ( ncaProgramMod . block_mod_cnt = = 2 )
{
block1_dump_buffer_start = ( ( ncaProgramMod . block_offset [ 1 ] / DUMP_BUFFER_SIZE ) * DUMP_BUFFER_SIZE ) ;
block1_dump_buffer_end = ( ( ( ncaProgramMod . block_offset [ 1 ] + ncaProgramMod . block_size [ 1 ] ) / DUMP_BUFFER_SIZE ) * DUMP_BUFFER_SIZE ) ;
}
}
progressCtx . line_offset = ( breaks + 4 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . start ) ) ;
dumping = true ;
// Dump all NCAs excluding the CNMT NCA
for ( i = 0 ; i < ( titleNcaCount - 1 ) ; i + + )
{
2019-05-01 16:24:13 -04:00
n = DUMP_BUFFER_SIZE ;
2019-06-05 18:44:18 -04:00
2019-05-01 16:24:13 -04:00
memcpy ( ncaId . c , xml_content_info [ i ] . nca_id , 16 ) ;
2019-06-05 18:44:18 -04:00
sha256ContextCreate ( & nca_hash_ctx ) ;
for ( nca_offset = 0 ; nca_offset < xml_content_info [ i ] . size ; nca_offset + = n , progressCtx . curOffset + = n )
2019-05-01 16:24:13 -04:00
{
2019-06-08 21:36:21 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 4 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( dumpPath , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dumping NCA content \" %s \" ... " , xml_content_info [ i ] . nca_id_str ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
2019-05-01 16:24:13 -04:00
if ( DUMP_BUFFER_SIZE > ( xml_content_info [ i ] . size - nca_offset ) ) n = ( xml_content_info [ i ] . size - nca_offset ) ;
2019-09-14 22:45:27 -04:00
if ( R_FAILED ( result = ncmContentStorageReadContentIdFile ( & ncmStorage , & ncaId , nca_offset , dumpBuf , n ) ) )
2019-05-01 16:24:13 -04:00
{
2019-06-19 23:56:14 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to read %lu bytes chunk at offset 0x%016lX from NCA \" %s \" ! (0x%08X) " , n , nca_offset , xml_content_info [ i ] . nca_id_str , result ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
proceed = false ;
break ;
}
// Replace NCA header with our modified one
2019-09-14 22:45:27 -04:00
if ( nca_offset = = 0 ) memcpy ( dumpBuf , xml_content_info [ i ] . encrypted_header_mod , NCA_FULL_HEADER_LENGTH ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
// Replace modified Program NCA data blocks
2019-06-28 16:13:54 -04:00
if ( ncaProgramMod . block_mod_cnt > 0 & & xml_content_info [ i ] . type = = NcmContentType_Program )
2019-06-05 18:44:18 -04:00
{
u64 program_nca_prev_write ;
u64 program_nca_next_write ;
if ( nca_offset = = hash_table_dump_buffer_start | | nca_offset = = hash_table_dump_buffer_end )
{
if ( hash_table_dump_buffer_start = = hash_table_dump_buffer_end )
{
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf + ( ncaProgramMod . hash_table_offset - hash_table_dump_buffer_start ) , ncaProgramMod . hash_table , ncaProgramMod . hash_table_size ) ;
2019-06-05 18:44:18 -04:00
} else {
program_nca_prev_write = ( DUMP_BUFFER_SIZE - ( ncaProgramMod . hash_table_offset - hash_table_dump_buffer_start ) ) ;
program_nca_next_write = ( ncaProgramMod . hash_table_size - program_nca_prev_write ) ;
if ( nca_offset = = hash_table_dump_buffer_start )
{
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf + ( ncaProgramMod . hash_table_offset - hash_table_dump_buffer_start ) , ncaProgramMod . hash_table , program_nca_prev_write ) ;
2019-06-05 18:44:18 -04:00
} else {
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , ncaProgramMod . hash_table + program_nca_prev_write , program_nca_next_write ) ;
2019-06-05 18:44:18 -04:00
}
}
}
if ( nca_offset = = block0_dump_buffer_start | | nca_offset = = block0_dump_buffer_end )
{
if ( block0_dump_buffer_start = = block0_dump_buffer_end )
{
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf + ( ncaProgramMod . block_offset [ 0 ] - block0_dump_buffer_start ) , ncaProgramMod . block_data [ 0 ] , ncaProgramMod . block_size [ 0 ] ) ;
2019-06-05 18:44:18 -04:00
} else {
program_nca_prev_write = ( DUMP_BUFFER_SIZE - ( ncaProgramMod . block_offset [ 0 ] - block0_dump_buffer_start ) ) ;
program_nca_next_write = ( ncaProgramMod . block_size [ 0 ] - program_nca_prev_write ) ;
if ( nca_offset = = block0_dump_buffer_start )
{
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf + ( ncaProgramMod . block_offset [ 0 ] - block0_dump_buffer_start ) , ncaProgramMod . block_data [ 0 ] , program_nca_prev_write ) ;
2019-06-05 18:44:18 -04:00
} else {
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , ncaProgramMod . block_data [ 0 ] + program_nca_prev_write , program_nca_next_write ) ;
2019-06-05 18:44:18 -04:00
}
}
}
if ( ncaProgramMod . block_mod_cnt = = 2 & & ( nca_offset = = block1_dump_buffer_start | | nca_offset = = block1_dump_buffer_end ) )
{
if ( block1_dump_buffer_start = = block1_dump_buffer_end )
{
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf + ( ncaProgramMod . block_offset [ 1 ] - block1_dump_buffer_start ) , ncaProgramMod . block_data [ 1 ] , ncaProgramMod . block_size [ 1 ] ) ;
2019-06-05 18:44:18 -04:00
} else {
program_nca_prev_write = ( DUMP_BUFFER_SIZE - ( ncaProgramMod . block_offset [ 1 ] - block1_dump_buffer_start ) ) ;
program_nca_next_write = ( ncaProgramMod . block_size [ 1 ] - program_nca_prev_write ) ;
if ( nca_offset = = block1_dump_buffer_start )
{
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf + ( ncaProgramMod . block_offset [ 1 ] - block1_dump_buffer_start ) , ncaProgramMod . block_data [ 1 ] , program_nca_prev_write ) ;
2019-06-05 18:44:18 -04:00
} else {
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , ncaProgramMod . block_data [ 1 ] + program_nca_prev_write , program_nca_next_write ) ;
2019-06-05 18:44:18 -04:00
}
}
}
}
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
// Update SHA-256 calculation
2019-09-14 22:45:27 -04:00
sha256ContextUpdate ( & nca_hash_ctx , dumpBuf , n ) ;
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 & & ( progressCtx . curOffset + n ) > = ( ( splitIndex + 1 ) * SPLIT_FILE_NSP_PART_SIZE ) )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
u64 new_file_chunk_size = ( ( progressCtx . curOffset + n ) - ( ( splitIndex + 1 ) * SPLIT_FILE_NSP_PART_SIZE ) ) ;
2019-05-01 16:24:13 -04:00
u64 old_file_chunk_size = ( n - new_file_chunk_size ) ;
if ( old_file_chunk_size > 0 )
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , old_file_chunk_size , outFile ) ;
2019-05-01 16:24:13 -04:00
if ( write_res ! = old_file_chunk_size )
{
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , old_file_chunk_size , progressCtx . curOffset , splitIndex , write_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
proceed = false ;
break ;
}
}
fclose ( outFile ) ;
2019-06-05 18:44:18 -04:00
outFile = NULL ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
if ( new_file_chunk_size > 0 | | ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
splitIndex + + ;
2019-06-08 21:36:21 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp/%02u " , NSP_DUMP_PATH , dumpName , splitIndex ) ;
2019-06-05 18:44:18 -04:00
outFile = fopen ( dumpPath , " wb " ) ;
if ( ! outFile )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file for part #%u! " , splitIndex ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
proceed = false ;
break ;
}
2019-06-05 18:44:18 -04:00
if ( new_file_chunk_size > 0 )
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf + old_file_chunk_size , 1 , new_file_chunk_size , outFile ) ;
2019-06-05 18:44:18 -04:00
if ( write_res ! = new_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , new_file_chunk_size , progressCtx . curOffset + old_file_chunk_size , splitIndex , write_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
proceed = false ;
break ;
}
}
2019-05-01 16:24:13 -04:00
}
} else {
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , n , outFile ) ;
2019-05-01 16:24:13 -04:00
if ( write_res ! = n )
{
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX! (wrote %lu bytes) " , n , progressCtx . curOffset , write_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
if ( ( progressCtx . curOffset + n ) > FAT32_FILESIZE_LIMIT )
2019-04-21 12:27:33 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " You're probably using a FAT32 partition. Make sure to enable the \" Split output dump \" option. " , 8 , ( ( progressCtx . line_offset + 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-05-01 16:24:13 -04:00
fat32_error = true ;
2019-04-21 12:27:33 -04:00
}
2019-05-01 16:24:13 -04:00
proceed = false ;
break ;
}
}
2019-06-05 18:44:18 -04:00
printProgressBar ( & progressCtx , true , n ) ;
2019-05-01 16:24:13 -04:00
2019-08-15 01:21:51 -04:00
if ( ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-05-01 16:24:13 -04:00
{
2019-08-15 01:21:51 -04:00
if ( cancelProcessCheck ( & progressCtx ) )
2019-05-01 16:24:13 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Process canceled. " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
proceed = false ;
break ;
2019-04-21 12:27:33 -04:00
}
}
2019-05-01 16:24:13 -04:00
}
if ( ! proceed )
{
2019-06-05 18:44:18 -04:00
setProgressBarError ( & progressCtx ) ;
2019-05-01 16:24:13 -04:00
break ;
}
// Support empty files
if ( ! xml_content_info [ i ] . size )
{
2019-06-08 21:36:21 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 4 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-05-01 16:24:13 -04:00
2019-06-08 21:36:21 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , strrchr ( dumpPath , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dumping NCA content \" %s \" ... " , xml_content_info [ i ] . nca_id_str ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
printProgressBar ( & progressCtx , false , 0 ) ;
2019-05-01 16:24:13 -04:00
}
2019-06-05 18:44:18 -04:00
// Update content info
sha256ContextGetHash ( & nca_hash_ctx , xml_content_info [ i ] . hash ) ;
convertDataToHexString ( xml_content_info [ i ] . hash , 32 , xml_content_info [ i ] . hash_str , 65 ) ;
memcpy ( xml_content_info [ i ] . nca_id , xml_content_info [ i ] . hash , 16 ) ;
convertDataToHexString ( xml_content_info [ i ] . nca_id , 16 , xml_content_info [ i ] . nca_id_str , 33 ) ;
2019-05-01 16:24:13 -04:00
}
if ( ! proceed ) goto out ;
2019-06-05 18:44:18 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 4 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-05-01 16:24:13 -04:00
2019-06-08 21:36:21 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( dumpPath , ' / ' ) + 1 ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-28 16:13:54 -04:00
uiDrawString ( " Writing PFS0 header... " , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
uiRefreshDisplay ( ) ;
// Now we can patch our CNMT NCA and generate our proper CNMT XML
if ( ! patchCnmtNca ( cnmtNcaBuf , xml_content_info [ cnmtNcaIndex ] . size , & xml_program_info , xml_content_info , & ncaCnmtMod ) )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
setProgressBarError ( & progressCtx ) ;
2019-05-01 16:24:13 -04:00
goto out ;
}
2019-06-05 18:44:18 -04:00
generateCnmtXml ( & xml_program_info , xml_content_info , cnmtXml ) ;
// Fill our Entry and String Tables
u64 file_offset = 0 ;
u32 filename_offset = 0 ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
for ( i = 0 ; i < nspFileCount ; i + + )
2019-05-01 16:24:13 -04:00
{
2019-06-28 16:13:54 -04:00
char ncaFileName [ 100 ] = { ' \0 ' } ;
2019-06-05 18:44:18 -04:00
u64 cur_file_size = 0 ;
2019-06-28 16:13:54 -04:00
if ( i < titleNcaCount )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
// Always reserve the first titleNcaCount entries for our NCA contents
sprintf ( ncaFileName , " %s.%s " , xml_content_info [ i ] . nca_id_str , ( i = = cnmtNcaIndex ? " cnmt.nca " : " nca " ) ) ;
cur_file_size = xml_content_info [ i ] . size ;
} else
if ( i = = titleNcaCount )
{
// Reserve the entry right after our NCA contents for the CNMT XML
2019-06-05 18:44:18 -04:00
sprintf ( ncaFileName , " %s.cnmt.xml " , xml_content_info [ cnmtNcaIndex ] . nca_id_str ) ;
cur_file_size = strlen ( cnmtXml ) ;
} else {
2019-06-28 16:13:54 -04:00
// Deal with additional files packed into the PFS0, in the following order:
// programinfo.xml (if available)
// NACP icons (if available)
// NACP XML (if available)
// legalinfo.xml (if available)
// Ticket (if available)
// Certificate chain (if available)
if ( programInfoXml & & i = = ( titleNcaCount + 1 ) )
{
// programinfo.xml entry
sprintf ( ncaFileName , " %s.programinfo.xml " , xml_content_info [ programNcaIndex ] . nca_id_str ) ;
cur_file_size = programInfoXmlSize ;
} else
if ( nacpIcons & & nacpIconCnt & & ( ( ! programInfoXml & & i < = ( titleNcaCount + nacpIconCnt ) ) | | ( programInfoXml & & i < = ( titleNcaCount + 1 + nacpIconCnt ) ) ) )
{
// NACP icon entry
u32 icon_idx = ( ! programInfoXml ? ( i - ( titleNcaCount + 1 ) ) : ( i - ( titleNcaCount + 2 ) ) ) ;
sprintf ( ncaFileName , nacpIcons [ icon_idx ] . filename ) ;
cur_file_size = nacpIcons [ icon_idx ] . icon_size ;
} else
if ( nacpXml & & ( ( ! programInfoXml & & i = = ( titleNcaCount + nacpIconCnt + 1 ) ) | | ( programInfoXml & & i = = ( titleNcaCount + 1 + nacpIconCnt + 1 ) ) ) )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
// NACP XML entry
// If there are no icons, this will effectively make it the next entry after the CNMT XML
2019-06-05 18:44:18 -04:00
sprintf ( ncaFileName , " %s.nacp.xml " , xml_content_info [ nacpNcaIndex ] . nca_id_str ) ;
2019-06-28 16:13:54 -04:00
cur_file_size = nacpXmlSize ;
} else
if ( legalInfoXml & & ( ( ! includeTikAndCert & & i = = ( nspFileCount - 1 ) ) | | ( includeTikAndCert & & i = = ( nspFileCount - 3 ) ) ) )
{
// legalinfo.xml entry
// If none of the previous conditions are met, assume we're dealing with a legalinfo.xml depending on the includeTikAndCert and counter values
sprintf ( ncaFileName , " %s.legalinfo.xml " , xml_content_info [ legalInfoNcaIndex ] . nca_id_str ) ;
cur_file_size = legalInfoXmlSize ;
2019-06-05 18:44:18 -04:00
} else {
2019-06-28 16:13:54 -04:00
// tik/cert entry
sprintf ( ncaFileName , " %s " , ( i = = ( nspFileCount - 2 ) ? rights_info . tik_filename : rights_info . cert_filename ) ) ;
cur_file_size = ( i = = ( nspFileCount - 2 ) ? ETICKET_TIK_FILE_SIZE : ETICKET_CERT_FILE_SIZE ) ;
2019-06-05 18:44:18 -04:00
}
}
nspPfs0EntryTable [ i ] . file_size = cur_file_size ;
nspPfs0EntryTable [ i ] . file_offset = file_offset ;
nspPfs0EntryTable [ i ] . filename_offset = filename_offset ;
strcpy ( nspPfs0StrTable + filename_offset , ncaFileName ) ;
file_offset + = nspPfs0EntryTable [ i ] . file_size ;
filename_offset + = ( strlen ( ncaFileName ) + 1 ) ;
2019-05-01 16:24:13 -04:00
}
2019-06-28 16:13:54 -04:00
// Write our full PFS0 header
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , & nspPfs0Header , sizeof ( pfs0_header ) ) ;
memcpy ( dumpBuf + sizeof ( pfs0_header ) , nspPfs0EntryTable , ( u64 ) nspFileCount * sizeof ( pfs0_entry_table ) ) ;
memcpy ( dumpBuf + sizeof ( pfs0_header ) + ( ( u64 ) nspFileCount * sizeof ( pfs0_entry_table ) ) , nspPfs0StrTable , nspPfs0Header . str_table_size ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 )
{
if ( outFile )
{
fclose ( outFile ) ;
outFile = NULL ;
}
2019-06-08 21:36:21 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp/%02u " , NSP_DUMP_PATH , dumpName , 0 ) ;
2019-06-05 18:44:18 -04:00
outFile = fopen ( dumpPath , " rb+ " ) ;
if ( ! outFile )
{
setProgressBarError ( & progressCtx ) ;
2019-06-28 16:13:54 -04:00
uiDrawString ( " Failed to re-open output file for part #0! " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
goto out ;
}
} else {
rewind ( outFile ) ;
}
2019-05-01 16:24:13 -04:00
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , full_nsp_header_size , outFile ) ;
2019-06-28 16:13:54 -04:00
if ( write_res ! = full_nsp_header_size )
2019-06-05 18:44:18 -04:00
{
setProgressBarError ( & progressCtx ) ;
2019-06-28 16:13:54 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes PFS0 header to file offset 0x%016lX! (wrote %lu bytes) " , full_nsp_header_size , ( u64 ) 0 , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
goto out ;
}
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 )
{
if ( outFile )
{
fclose ( outFile ) ;
outFile = NULL ;
}
2019-06-08 21:36:21 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp/%02u " , NSP_DUMP_PATH , dumpName , splitIndex ) ;
2019-06-05 18:44:18 -04:00
outFile = fopen ( dumpPath , " rb+ " ) ;
if ( ! outFile )
{
setProgressBarError ( & progressCtx ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to re-open output file for part #%u! " , splitIndex ) ;
2019-06-28 16:13:54 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
goto out ;
}
2019-06-28 16:13:54 -04:00
}
fseek ( outFile , 0 , SEEK_END ) ;
// Now let's write the rest of the data, including our modified CNMT NCA
for ( i = ( titleNcaCount - 1 ) ; i < nspFileCount ; i + + )
{
n = DUMP_BUFFER_SIZE ;
2019-06-05 18:44:18 -04:00
2019-06-28 16:13:54 -04:00
char ncaFileName [ 100 ] = { ' \0 ' } ;
u64 cur_file_size = 0 ;
if ( i = = ( titleNcaCount - 1 ) )
{
// CNMT NCA
sprintf ( ncaFileName , " %s.cnmt.nca " , xml_content_info [ i ] . nca_id_str ) ;
cur_file_size = xml_content_info [ cnmtNcaIndex ] . size ;
} else
if ( i = = titleNcaCount )
{
// CNMT XML
sprintf ( ncaFileName , " %s.cnmt.xml " , xml_content_info [ cnmtNcaIndex ] . nca_id_str ) ;
cur_file_size = strlen ( cnmtXml ) ;
} else {
if ( programInfoXml & & i = = ( titleNcaCount + 1 ) )
{
// programinfo.xml entry
sprintf ( ncaFileName , " %s.programinfo.xml " , xml_content_info [ programNcaIndex ] . nca_id_str ) ;
cur_file_size = programInfoXmlSize ;
} else
if ( nacpIcons & & nacpIconCnt & & ( ( ! programInfoXml & & i < = ( titleNcaCount + nacpIconCnt ) ) | | ( programInfoXml & & i < = ( titleNcaCount + 1 + nacpIconCnt ) ) ) )
{
// NACP icon entry
u32 icon_idx = ( ! programInfoXml ? ( i - ( titleNcaCount + 1 ) ) : ( i - ( titleNcaCount + 2 ) ) ) ;
sprintf ( ncaFileName , nacpIcons [ icon_idx ] . filename ) ;
cur_file_size = nacpIcons [ icon_idx ] . icon_size ;
} else
if ( nacpXml & & ( ( ! programInfoXml & & i = = ( titleNcaCount + nacpIconCnt + 1 ) ) | | ( programInfoXml & & i = = ( titleNcaCount + 1 + nacpIconCnt + 1 ) ) ) )
{
// NACP XML entry
sprintf ( ncaFileName , " %s.nacp.xml " , xml_content_info [ nacpNcaIndex ] . nca_id_str ) ;
cur_file_size = nacpXmlSize ;
} else
if ( legalInfoXml & & ( ( ! includeTikAndCert & & i = = ( nspFileCount - 1 ) ) | | ( includeTikAndCert & & i = = ( nspFileCount - 3 ) ) ) )
{
// legalinfo.xml entry
sprintf ( ncaFileName , " %s.legalinfo.xml " , xml_content_info [ legalInfoNcaIndex ] . nca_id_str ) ;
cur_file_size = legalInfoXmlSize ;
} else {
// tik/cert entry
sprintf ( ncaFileName , " %s " , ( i = = ( nspFileCount - 2 ) ? rights_info . tik_filename : rights_info . cert_filename ) ) ;
cur_file_size = ( i = = ( nspFileCount - 2 ) ? ETICKET_TIK_FILE_SIZE : ETICKET_CERT_FILE_SIZE ) ;
}
}
2019-06-05 18:44:18 -04:00
2019-06-28 16:13:54 -04:00
for ( nca_offset = 0 ; nca_offset < cur_file_size ; nca_offset + = n , progressCtx . curOffset + = n )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 4 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-06-05 18:44:18 -04:00
2019-06-28 16:13:54 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( dumpPath , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Writing \" %s \" ... " , ncaFileName ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
if ( DUMP_BUFFER_SIZE > ( cur_file_size - nca_offset ) ) n = ( cur_file_size - nca_offset ) ;
// Retrieve data from its respective source
if ( i = = ( titleNcaCount - 1 ) )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
// CNMT NCA
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , cnmtNcaBuf + nca_offset , n ) ;
2019-06-28 16:13:54 -04:00
} else
if ( i = = titleNcaCount )
{
// CNMT XML
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , cnmtXml + nca_offset , n ) ;
2019-06-28 16:13:54 -04:00
} else {
if ( programInfoXml & & i = = ( titleNcaCount + 1 ) )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
// programinfo.xml entry
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , programInfoXml + nca_offset , n ) ;
2019-06-28 16:13:54 -04:00
} else
if ( nacpIcons & & nacpIconCnt & & ( ( ! programInfoXml & & i < = ( titleNcaCount + nacpIconCnt ) ) | | ( programInfoXml & & i < = ( titleNcaCount + 1 + nacpIconCnt ) ) ) )
{
// NACP icon entry
u32 icon_idx = ( ! programInfoXml ? ( i - ( titleNcaCount + 1 ) ) : ( i - ( titleNcaCount + 2 ) ) ) ;
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , nacpIcons [ icon_idx ] . icon_data + nca_offset , n ) ;
2019-06-28 16:13:54 -04:00
} else
if ( nacpXml & & ( ( ! programInfoXml & & i = = ( titleNcaCount + nacpIconCnt + 1 ) ) | | ( programInfoXml & & i = = ( titleNcaCount + 1 + nacpIconCnt + 1 ) ) ) )
{
// NACP XML entry
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , nacpXml + nca_offset , n ) ;
2019-06-28 16:13:54 -04:00
} else
if ( legalInfoXml & & ( ( ! includeTikAndCert & & i = = ( nspFileCount - 1 ) ) | | ( includeTikAndCert & & i = = ( nspFileCount - 3 ) ) ) )
{
// legalinfo.xml entry
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , legalInfoXml + nca_offset , n ) ;
2019-06-28 16:13:54 -04:00
} else {
// tik/cert entry
if ( i = = ( nspFileCount - 2 ) )
{
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , ( u8 * ) ( & ( rights_info . tik_data ) ) + nca_offset , n ) ;
2019-06-28 16:13:54 -04:00
} else {
2019-09-14 22:45:27 -04:00
memcpy ( dumpBuf , rights_info . cert_data + nca_offset , n ) ;
2019-06-28 16:13:54 -04:00
}
2019-06-05 18:44:18 -04:00
}
}
2019-06-28 16:13:54 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 & & ( progressCtx . curOffset + n ) > = ( ( splitIndex + 1 ) * SPLIT_FILE_NSP_PART_SIZE ) )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
u64 new_file_chunk_size = ( ( progressCtx . curOffset + n ) - ( ( splitIndex + 1 ) * SPLIT_FILE_NSP_PART_SIZE ) ) ;
u64 old_file_chunk_size = ( n - new_file_chunk_size ) ;
2019-06-05 18:44:18 -04:00
2019-06-28 16:13:54 -04:00
if ( old_file_chunk_size > 0 )
2019-06-05 18:44:18 -04:00
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , old_file_chunk_size , outFile ) ;
2019-06-28 16:13:54 -04:00
if ( write_res ! = old_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , old_file_chunk_size , progressCtx . curOffset , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
2019-06-05 18:44:18 -04:00
}
2019-06-28 16:13:54 -04:00
fclose ( outFile ) ;
outFile = NULL ;
2019-06-05 18:44:18 -04:00
2019-06-28 16:13:54 -04:00
if ( new_file_chunk_size > 0 | | ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
splitIndex + + ;
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp/%02u " , NSP_DUMP_PATH , dumpName , splitIndex ) ;
outFile = fopen ( dumpPath , " wb " ) ;
if ( ! outFile )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file for part #%u! " , splitIndex ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
if ( new_file_chunk_size > 0 )
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf + old_file_chunk_size , 1 , new_file_chunk_size , outFile ) ;
2019-06-28 16:13:54 -04:00
if ( write_res ! = new_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , new_file_chunk_size , progressCtx . curOffset + old_file_chunk_size , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
}
}
} else {
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , n , outFile ) ;
2019-06-28 16:13:54 -04:00
if ( write_res ! = n )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX! (wrote %lu bytes) " , n , progressCtx . curOffset , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
if ( ( progressCtx . curOffset + n ) > FAT32_FILESIZE_LIMIT )
{
uiDrawString ( " You're probably using a FAT32 partition. Make sure to enable the \" Split output dump \" option. " , 8 , ( ( progressCtx . line_offset + 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
fat32_error = true ;
}
proceed = false ;
break ;
2019-06-05 18:44:18 -04:00
}
}
2019-06-28 16:13:54 -04:00
printProgressBar ( & progressCtx , true , n ) ;
2019-08-15 01:21:51 -04:00
if ( ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-06-05 18:44:18 -04:00
{
2019-08-15 01:21:51 -04:00
if ( cancelProcessCheck ( & progressCtx ) )
2019-06-28 16:13:54 -04:00
{
uiDrawString ( " Process canceled. " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
2019-06-05 18:44:18 -04:00
}
}
2019-06-28 16:13:54 -04:00
if ( ! proceed )
2019-06-05 18:44:18 -04:00
{
setProgressBarError ( & progressCtx ) ;
2019-06-28 16:13:54 -04:00
break ;
}
// Support empty files
if ( ! cur_file_size )
{
uiFill ( 0 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 4 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-06-05 18:44:18 -04:00
2019-06-28 16:13:54 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , strrchr ( dumpPath , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
2019-06-28 16:13:54 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Writing \" %s \" ... " , ncaFileName ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
printProgressBar ( & progressCtx , false , 0 ) ;
2019-06-05 18:44:18 -04:00
}
}
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
if ( ! proceed ) goto out ;
dumping = false ;
breaks = ( progressCtx . line_offset + 2 ) ;
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
if ( progressCtx . curOffset < progressCtx . totalSize )
{
setProgressBarError ( & progressCtx ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Unexpected underdump error! Wrote %lu bytes, expected %lu bytes. " , progressCtx . curOffset , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
goto out ;
}
2019-05-01 16:24:13 -04:00
2019-06-05 18:44:18 -04:00
success = true ;
2019-05-01 16:24:13 -04:00
// Finalize dump
2019-06-28 16:13:54 -04:00
if ( ! batch )
{
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
progressCtx . progress = 100 ;
progressCtx . remainingTime = 0 ;
printProgressBar ( & progressCtx , false , 0 ) ;
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
uiRefreshDisplay ( ) ;
}
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
if ( ! batch & & calcCrc )
2019-05-01 16:24:13 -04:00
{
2019-06-05 18:44:18 -04:00
breaks + = 2 ;
uiDrawString ( " CRC32 checksum calculation will begin in 5 seconds... " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
delay ( 5 ) ;
breaks = initial_breaks ;
uiFill ( 0 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , FB_HEIGHT - ( breaks * ( font_height + ( font_height / 4 ) ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Calculating CRC32 checksum. Hold %s to cancel. " , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
if ( outFile )
{
fclose ( outFile ) ;
outFile = NULL ;
}
2019-06-08 21:36:21 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp " , NSP_DUMP_PATH , dumpName ) ;
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 )
{
splitIndex = 0 ;
sprintf ( tmp_idx , " /%02u " , splitIndex ) ;
strcat ( dumpPath , tmp_idx ) ;
}
outFile = fopen ( dumpPath , " rb " ) ;
if ( outFile )
{
n = DUMP_BUFFER_SIZE ;
progressCtx . start = progressCtx . now = progressCtx . remainingTime = 0 ;
progressCtx . lastSpeed = progressCtx . averageSpeed = 0.0 ;
size_t read_res ;
progressCtx . line_offset = ( breaks + 2 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . start ) ) ;
for ( progressCtx . curOffset = 0 ; progressCtx . curOffset < progressCtx . totalSize ; progressCtx . curOffset + = n )
{
2019-06-08 21:36:21 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " File: \" %s \" . " , strrchr ( dumpPath , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
if ( DUMP_BUFFER_SIZE > ( progressCtx . totalSize - progressCtx . curOffset ) ) n = ( progressCtx . totalSize - progressCtx . curOffset ) ;
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 & & ( progressCtx . curOffset + n ) > = ( ( splitIndex + 1 ) * SPLIT_FILE_NSP_PART_SIZE ) )
{
u64 new_file_chunk_size = ( ( progressCtx . curOffset + n ) - ( ( splitIndex + 1 ) * SPLIT_FILE_NSP_PART_SIZE ) ) ;
u64 old_file_chunk_size = ( n - new_file_chunk_size ) ;
if ( old_file_chunk_size > 0 )
{
2019-09-14 22:45:27 -04:00
read_res = fread ( dumpBuf , 1 , old_file_chunk_size , outFile ) ;
2019-06-05 18:44:18 -04:00
if ( read_res ! = old_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to read %lu bytes chunk from offset 0x%016lX from part #%02u! (read %lu bytes) " , old_file_chunk_size , progressCtx . curOffset , splitIndex , read_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
proceed = false ;
break ;
}
}
fclose ( outFile ) ;
outFile = NULL ;
if ( new_file_chunk_size > 0 | | ( progressCtx . curOffset + n ) < progressCtx . totalSize )
{
splitIndex + + ;
2019-06-08 21:36:21 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp/%02u " , NSP_DUMP_PATH , dumpName , splitIndex ) ;
2019-06-05 18:44:18 -04:00
outFile = fopen ( dumpPath , " rb " ) ;
if ( ! outFile )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to re-open output file for part #%u! " , splitIndex ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
proceed = false ;
break ;
}
if ( new_file_chunk_size > 0 )
{
2019-09-14 22:45:27 -04:00
read_res = fread ( dumpBuf + old_file_chunk_size , 1 , new_file_chunk_size , outFile ) ;
2019-06-05 18:44:18 -04:00
if ( read_res ! = new_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to read %lu bytes chunk from offset 0x%016lX from part #%02u! (read %lu bytes) " , new_file_chunk_size , progressCtx . curOffset + old_file_chunk_size , splitIndex , read_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
proceed = false ;
break ;
}
}
}
} else {
2019-09-14 22:45:27 -04:00
read_res = fread ( dumpBuf , 1 , n , outFile ) ;
2019-06-05 18:44:18 -04:00
if ( read_res ! = n )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to read %lu bytes chunk from offset 0x%016lX! (read %lu bytes) " , n , progressCtx . curOffset , read_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
proceed = false ;
break ;
}
}
// Update CRC32
2019-09-14 22:45:27 -04:00
crc32 ( dumpBuf , n , & crc ) ;
2019-06-05 18:44:18 -04:00
printProgressBar ( & progressCtx , true , n ) ;
2019-08-15 01:21:51 -04:00
if ( ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-06-05 18:44:18 -04:00
{
2019-08-15 01:21:51 -04:00
if ( cancelProcessCheck ( & progressCtx ) )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Process canceled. " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
proceed = false ;
break ;
}
}
}
2019-06-08 21:36:21 -04:00
breaks = ( progressCtx . line_offset + 2 ) ;
2019-06-05 18:44:18 -04:00
if ( proceed )
{
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
breaks + + ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " NSP dump CRC32 checksum: %08X " , crc ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
} else {
setProgressBarError ( & progressCtx ) ;
}
} else {
uiDrawString ( " Failed to re-open output file in read mode! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
}
2019-05-01 16:24:13 -04:00
}
// Set archive bit (only for FAT32)
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 )
2019-05-01 16:24:13 -04:00
{
2019-06-08 21:36:21 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp " , NSP_DUMP_PATH , dumpName ) ;
2019-05-01 16:24:13 -04:00
if ( R_FAILED ( result = fsdevSetArchiveBit ( dumpPath ) ) )
{
2019-06-05 18:44:18 -04:00
breaks + = 2 ;
2019-05-01 16:24:13 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Warning: failed to set archive bit on output directory! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
}
}
2019-06-28 16:13:54 -04:00
out :
if ( outFile ) fclose ( outFile ) ;
if ( ! success )
{
if ( dumping )
{
breaks + = 6 ;
if ( fat32_error ) breaks + = 2 ;
}
if ( removeFile )
{
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp " , NSP_DUMP_PATH , dumpName ) ;
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & isFat32 )
{
2019-08-15 01:21:51 -04:00
fsdevDeleteDirectoryRecursively ( dumpPath ) ;
2019-06-28 16:13:54 -04:00
} else {
unlink ( dumpPath ) ;
}
}
}
if ( nspPfs0StrTable ) free ( nspPfs0StrTable ) ;
if ( nspPfs0EntryTable ) free ( nspPfs0EntryTable ) ;
if ( cnmtXml ) free ( cnmtXml ) ;
if ( cnmtNcaBuf ) free ( cnmtNcaBuf ) ;
if ( ncaProgramMod . block_data [ 1 ] ) free ( ncaProgramMod . block_data [ 1 ] ) ;
if ( ncaProgramMod . block_data [ 0 ] ) free ( ncaProgramMod . block_data [ 0 ] ) ;
if ( ncaProgramMod . hash_table ) free ( ncaProgramMod . hash_table ) ;
if ( legalInfoXml ) free ( legalInfoXml ) ;
if ( nacpIcons ) free ( nacpIcons ) ;
if ( nacpXml ) free ( nacpXml ) ;
if ( programInfoXml ) free ( programInfoXml ) ;
serviceClose ( & ( ncmStorage . s ) ) ;
if ( xml_content_info ) free ( xml_content_info ) ;
if ( titleContentRecords ) free ( titleContentRecords ) ;
serviceClose ( & ( ncmDb . s ) ) ;
if ( titleList ) free ( titleList ) ;
if ( curStorageId = = FsStorageId_GameCard )
{
fsStorageClose ( & gameCardStorage ) ;
if ( partitionHfs0Header )
{
free ( partitionHfs0Header ) ;
partitionHfs0Header = NULL ;
partitionHfs0HeaderOffset = 0 ;
partitionHfs0HeaderSize = 0 ;
partitionHfs0FileCount = 0 ;
partitionHfs0StrTableSize = 0 ;
}
}
free ( dumpName ) ;
if ( ! batch ) breaks + = 2 ;
return success ;
}
bool dumpNintendoSubmissionPackageBatch ( bool dumpAppTitles , bool dumpPatchTitles , bool dumpAddOnTitles , bool isFat32 , bool removeConsoleData , bool tiklessDump , bool skipDumpedTitles , batchModeSourceStorage batchModeSrc )
{
if ( ( ! dumpAppTitles & & ! dumpPatchTitles & & ! dumpAddOnTitles ) | | ( batchModeSrc = = BATCH_SOURCE_ALL & & ( ( dumpAppTitles & & ! titleAppCount ) | | ( dumpPatchTitles & & ! titlePatchCount ) | | ( dumpAddOnTitles & & ! titleAddOnCount ) ) ) | | ( batchModeSrc = = BATCH_SOURCE_SDCARD & & ( ( dumpAppTitles & & ! sdCardTitleAppCount ) | | ( dumpPatchTitles & & ! sdCardTitlePatchCount ) | | ( dumpAddOnTitles & & ! sdCardTitleAddOnCount ) ) ) | | ( batchModeSrc = = BATCH_SOURCE_EMMC & & ( ( dumpAppTitles & & ! nandUserTitleAppCount ) | | ( dumpPatchTitles & & ! nandUserTitlePatchCount ) | | ( dumpAddOnTitles & & ! nandUserTitleAddOnCount ) ) ) )
{
uiDrawString ( " Error: invalid parameters to perform batch NSP dump! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
u32 i , j ;
u32 totalTitleCount = 0 , totalAppCount = 0 , totalPatchCount = 0 , totalAddOnCount = 0 ;
u32 titleCount , titleIndex ;
char * dumpName = NULL ;
char dumpPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
char curName [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
int initial_breaks = breaks , cur_breaks ;
const u32 maxSummaryFileCount = 6 ;
u32 summaryPage = 0 ;
memset ( filenameBuffer , 0 , FILENAME_BUFFER_SIZE ) ;
filenamesCount = 0 ;
char * nextFilename = filenameBuffer ;
bool proceed = true ;
if ( dumpAppTitles )
{
titleCount = ( batchModeSrc = = BATCH_SOURCE_ALL ? titleAppCount : ( batchModeSrc = = BATCH_SOURCE_SDCARD ? sdCardTitleAppCount : nandUserTitleAppCount ) ) ;
for ( i = 0 ; i < titleCount ; i + + )
{
titleIndex = ( ( batchModeSrc = = BATCH_SOURCE_ALL | | batchModeSrc = = BATCH_SOURCE_SDCARD ) ? i : ( i + sdCardTitleAppCount ) ) ;
dumpName = generateNSPDumpName ( DUMP_APP_NSP , titleIndex ) ;
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp " , NSP_DUMP_PATH , dumpName ) ;
free ( dumpName ) ;
dumpName = NULL ;
// Check if this title has already been dumped
if ( skipDumpedTitles & & checkIfFileExists ( dumpPath ) ) continue ;
snprintf ( curName , sizeof ( curName ) / sizeof ( curName [ 0 ] ) , strrchr ( dumpPath , ' / ' ) + 1 ) ;
// Fix entry name length
u32 strWidth = uiGetStrWidth ( curName ) ;
if ( ( 8 + strWidth ) > = ( FB_WIDTH - ( font_height * 5 ) ) )
{
while ( ( 8 + strWidth ) > = ( FB_WIDTH - ( font_height * 5 ) ) )
{
curName [ strlen ( curName ) - 1 ] = ' \0 ' ;
strWidth = uiGetStrWidth ( curName ) ;
}
strcat ( curName , " ... " ) ;
}
addStringToFilenameBuffer ( curName , & nextFilename ) ;
totalAppCount + + ;
}
totalTitleCount + = totalAppCount ;
}
if ( dumpPatchTitles )
{
titleCount = ( batchModeSrc = = BATCH_SOURCE_ALL ? titlePatchCount : ( batchModeSrc = = BATCH_SOURCE_SDCARD ? sdCardTitlePatchCount : nandUserTitlePatchCount ) ) ;
for ( i = 0 ; i < titleCount ; i + + )
{
titleIndex = ( ( batchModeSrc = = BATCH_SOURCE_ALL | | batchModeSrc = = BATCH_SOURCE_SDCARD ) ? i : ( i + sdCardTitlePatchCount ) ) ;
dumpName = generateNSPDumpName ( DUMP_PATCH_NSP , titleIndex ) ;
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp " , NSP_DUMP_PATH , dumpName ) ;
free ( dumpName ) ;
dumpName = NULL ;
// Check if this title has already been dumped
if ( skipDumpedTitles & & checkIfFileExists ( dumpPath ) ) continue ;
snprintf ( curName , sizeof ( curName ) / sizeof ( curName [ 0 ] ) , strrchr ( dumpPath , ' / ' ) + 1 ) ;
// Fix entry name length
u32 strWidth = uiGetStrWidth ( curName ) ;
if ( ( 8 + strWidth ) > = ( FB_WIDTH - ( font_height * 5 ) ) )
{
while ( ( 8 + strWidth ) > = ( FB_WIDTH - ( font_height * 5 ) ) )
{
curName [ strlen ( curName ) - 1 ] = ' \0 ' ;
strWidth = uiGetStrWidth ( curName ) ;
}
strcat ( curName , " ... " ) ;
}
addStringToFilenameBuffer ( curName , & nextFilename ) ;
totalPatchCount + + ;
}
totalTitleCount + = totalPatchCount ;
}
if ( dumpAddOnTitles )
{
titleCount = ( batchModeSrc = = BATCH_SOURCE_ALL ? titleAddOnCount : ( batchModeSrc = = BATCH_SOURCE_SDCARD ? sdCardTitleAddOnCount : nandUserTitleAddOnCount ) ) ;
for ( i = 0 ; i < titleCount ; i + + )
{
titleIndex = ( ( batchModeSrc = = BATCH_SOURCE_ALL | | batchModeSrc = = BATCH_SOURCE_SDCARD ) ? i : ( i + sdCardTitleAddOnCount ) ) ;
dumpName = generateNSPDumpName ( DUMP_ADDON_NSP , titleIndex ) ;
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp " , NSP_DUMP_PATH , dumpName ) ;
free ( dumpName ) ;
dumpName = NULL ;
// Check if this title has already been dumped
if ( skipDumpedTitles & & checkIfFileExists ( dumpPath ) ) continue ;
snprintf ( curName , sizeof ( curName ) / sizeof ( curName [ 0 ] ) , strrchr ( dumpPath , ' / ' ) + 1 ) ;
// Fix entry name length
u32 strWidth = uiGetStrWidth ( curName ) ;
if ( ( 8 + strWidth ) > = ( FB_WIDTH - ( font_height * 5 ) ) )
{
while ( ( 8 + strWidth ) > = ( FB_WIDTH - ( font_height * 5 ) ) )
{
curName [ strlen ( curName ) - 1 ] = ' \0 ' ;
strWidth = uiGetStrWidth ( curName ) ;
}
strcat ( curName , " ... " ) ;
}
addStringToFilenameBuffer ( curName , & nextFilename ) ;
totalAddOnCount + + ;
}
totalTitleCount + = totalAddOnCount ;
}
if ( ! totalTitleCount )
{
uiDrawString ( " You have already dumped all titles matching the selected settings! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
return false ;
}
// Display summary
uiDrawString ( " Summary: " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
strbuf [ 0 ] = ' \0 ' ;
if ( totalAppCount )
{
snprintf ( curName , sizeof ( curName ) / sizeof ( curName [ 0 ] ) , " BASE: %u " , totalAppCount ) ;
strcat ( strbuf , curName ) ;
}
if ( totalPatchCount )
{
if ( totalAppCount ) strcat ( strbuf , " | " ) ;
snprintf ( curName , sizeof ( curName ) / sizeof ( curName [ 0 ] ) , " UPD: %u " , totalPatchCount ) ;
strcat ( strbuf , curName ) ;
}
if ( totalAddOnCount )
{
if ( totalAppCount | | totalPatchCount ) strcat ( strbuf , " | " ) ;
snprintf ( curName , sizeof ( curName ) / sizeof ( curName [ 0 ] ) , " DLC: %u " , totalAddOnCount ) ;
strcat ( strbuf , curName ) ;
}
strcat ( strbuf , " | " ) ;
snprintf ( curName , sizeof ( curName ) / sizeof ( curName [ 0 ] ) , " Total: %u " , totalTitleCount ) ;
strcat ( strbuf , curName ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + + ;
while ( true )
{
cur_breaks = breaks ;
uiFill ( 0 , 8 + ( cur_breaks * ( font_height + ( font_height / 4 ) ) ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( cur_breaks * ( font_height + ( font_height / 4 ) ) ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
if ( totalTitleCount > maxSummaryFileCount )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Current page: %u " , summaryPage + 1 ) ;
uiDrawString ( strbuf , 8 , ( cur_breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
cur_breaks + + ;
}
cur_breaks + + ;
for ( i = ( summaryPage * maxSummaryFileCount ) ; i < ( ( summaryPage * maxSummaryFileCount ) + maxSummaryFileCount ) ; i + + )
{
if ( i > = totalTitleCount ) break ;
uiDrawIcon ( fileNormalIconBuf , BROWSER_ICON_DIMENSION , BROWSER_ICON_DIMENSION , 8 , 8 + ( cur_breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) ) ;
uiDrawString ( filenames [ i ] , BROWSER_ICON_DIMENSION + 8 , ( cur_breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
cur_breaks + + ;
}
cur_breaks + + ;
if ( totalTitleCount > maxSummaryFileCount )
{
uiDrawString ( " [ " NINTENDO_FONT_L " / " NINTENDO_FONT_R " / " NINTENDO_FONT_ZL " / " NINTENDO_FONT_ZR " ] Change page | [ " NINTENDO_FONT_A " ] Proceed | [ " NINTENDO_FONT_B " ] Cancel " , 8 , ( cur_breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
} else {
uiDrawString ( " [ " NINTENDO_FONT_A " ] Proceed | [ " NINTENDO_FONT_B " ] Cancel " , 8 , ( cur_breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
}
uiRefreshDisplay ( ) ;
hidScanInput ( ) ;
u32 keysDown = hidKeysDown ( CONTROLLER_P1_AUTO ) ;
if ( keysDown & KEY_A )
{
proceed = true ;
break ;
} else
if ( keysDown & KEY_B )
{
proceed = false ;
break ;
} else
if ( ( ( keysDown & KEY_L ) | | ( keysDown & KEY_ZL ) ) & & totalTitleCount > maxSummaryFileCount )
{
if ( summaryPage > 0 ) summaryPage - - ;
} else
if ( ( ( keysDown & KEY_R ) | | ( keysDown & KEY_ZR ) ) & & totalTitleCount > maxSummaryFileCount )
{
if ( ( ( summaryPage * maxSummaryFileCount ) + maxSummaryFileCount ) < totalTitleCount ) summaryPage + + ;
}
}
if ( ! proceed )
{
breaks = ( cur_breaks + 2 ) ;
uiDrawString ( " Process canceled " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
breaks = initial_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dump procedure started. Hold %s to cancel. " , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
initial_breaks = breaks ;
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
j = 0 ;
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
if ( totalAppCount )
2019-05-01 16:24:13 -04:00
{
2019-06-28 16:13:54 -04:00
titleCount = ( batchModeSrc = = BATCH_SOURCE_ALL ? titleAppCount : ( batchModeSrc = = BATCH_SOURCE_SDCARD ? sdCardTitleAppCount : nandUserTitleAppCount ) ) ;
for ( i = 0 ; i < titleCount ; i + + )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
breaks = initial_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
titleIndex = ( ( batchModeSrc = = BATCH_SOURCE_ALL | | batchModeSrc = = BATCH_SOURCE_SDCARD ) ? i : ( i + sdCardTitleAppCount ) ) ;
dumpName = generateNSPDumpName ( DUMP_APP_NSP , titleIndex ) ;
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp " , NSP_DUMP_PATH , dumpName ) ;
free ( dumpName ) ;
dumpName = NULL ;
// Check if this title has already been dumped
if ( skipDumpedTitles & & checkIfFileExists ( dumpPath ) ) continue ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Title: %u / %u. " , j + 1 , totalTitleCount ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + = 2 ;
// Dump title
if ( ! dumpNintendoSubmissionPackage ( DUMP_APP_NSP , titleIndex , isFat32 , false , removeConsoleData , tiklessDump , true ) ) return false ;
j + + ;
2019-06-05 18:44:18 -04:00
}
2019-06-28 16:13:54 -04:00
}
if ( totalPatchCount )
{
titleCount = ( batchModeSrc = = BATCH_SOURCE_ALL ? titlePatchCount : ( batchModeSrc = = BATCH_SOURCE_SDCARD ? sdCardTitlePatchCount : nandUserTitlePatchCount ) ) ;
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
for ( i = 0 ; i < titleCount ; i + + )
2019-05-01 16:24:13 -04:00
{
2019-06-28 16:13:54 -04:00
breaks = initial_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-06-19 23:56:14 -04:00
2019-06-28 16:13:54 -04:00
titleIndex = ( ( batchModeSrc = = BATCH_SOURCE_ALL | | batchModeSrc = = BATCH_SOURCE_SDCARD ) ? i : ( i + sdCardTitlePatchCount ) ) ;
dumpName = generateNSPDumpName ( DUMP_PATCH_NSP , titleIndex ) ;
if ( ! dumpName )
2019-06-19 23:56:14 -04:00
{
2019-06-28 16:13:54 -04:00
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
2019-06-19 23:56:14 -04:00
}
2019-06-28 16:13:54 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp " , NSP_DUMP_PATH , dumpName ) ;
free ( dumpName ) ;
dumpName = NULL ;
// Check if this title has already been dumped
if ( skipDumpedTitles & & checkIfFileExists ( dumpPath ) ) continue ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Title: %u / %u. " , j + 1 , totalTitleCount ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + = 2 ;
// Dump title
if ( ! dumpNintendoSubmissionPackage ( DUMP_PATCH_NSP , titleIndex , isFat32 , false , removeConsoleData , tiklessDump , true ) ) return false ;
j + + ;
2019-04-21 12:27:33 -04:00
}
}
2019-06-28 16:13:54 -04:00
if ( totalAddOnCount )
2019-05-01 16:24:13 -04:00
{
2019-06-28 16:13:54 -04:00
titleCount = ( batchModeSrc = = BATCH_SOURCE_ALL ? titleAddOnCount : ( batchModeSrc = = BATCH_SOURCE_SDCARD ? sdCardTitleAddOnCount : nandUserTitleAddOnCount ) ) ;
2019-06-08 21:36:21 -04:00
2019-06-28 16:13:54 -04:00
for ( i = 0 ; i < titleCount ; i + + )
2019-06-08 21:36:21 -04:00
{
2019-06-28 16:13:54 -04:00
breaks = initial_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
titleIndex = ( ( batchModeSrc = = BATCH_SOURCE_ALL | | batchModeSrc = = BATCH_SOURCE_SDCARD ) ? i : ( i + sdCardTitleAddOnCount ) ) ;
dumpName = generateNSPDumpName ( DUMP_ADDON_NSP , titleIndex ) ;
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s.nsp " , NSP_DUMP_PATH , dumpName ) ;
free ( dumpName ) ;
dumpName = NULL ;
// Check if this title has already been dumped
if ( skipDumpedTitles & & checkIfFileExists ( dumpPath ) ) continue ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Title: %u / %u. " , j + 1 , totalTitleCount ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + = 2 ;
// Dump title
if ( ! dumpNintendoSubmissionPackage ( DUMP_ADDON_NSP , titleIndex , isFat32 , false , removeConsoleData , tiklessDump , true ) ) return false ;
j + + ;
2019-06-08 21:36:21 -04:00
}
2019-05-01 16:24:13 -04:00
}
2019-06-28 16:13:54 -04:00
uiDrawString ( " Process successfully completed! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
2019-06-05 18:44:18 -04:00
2019-04-21 12:27:33 -04:00
breaks + = 2 ;
2019-06-28 16:13:54 -04:00
return true ;
2018-06-21 02:42:46 -04:00
}
2018-05-15 18:00:19 +02:00
2019-06-08 21:36:21 -04:00
bool dumpRawHfs0Partition ( u32 partition , bool doSplitting )
2018-06-21 02:42:46 -04:00
{
2019-04-21 12:27:33 -04:00
Result result ;
2019-06-05 18:44:18 -04:00
u64 partitionOffset ;
2019-06-19 23:56:14 -04:00
bool proceed = true , success = false , fat32_error = false ;
2019-06-05 18:44:18 -04:00
u64 n = DUMP_BUFFER_SIZE ;
2019-04-21 12:27:33 -04:00
FsGameCardHandle handle ;
FsStorage gameCardStorage ;
2019-06-05 18:44:18 -04:00
char filename [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
2019-04-21 12:27:33 -04:00
FILE * outFile = NULL ;
u8 splitIndex = 0 ;
2019-09-14 22:45:27 -04:00
memset ( dumpBuf , 0 , DUMP_BUFFER_SIZE ) ;
2019-06-05 18:44:18 -04:00
progress_ctx_t progressCtx ;
memset ( & progressCtx , 0 , sizeof ( progress_ctx_t ) ) ;
2019-04-21 12:27:33 -04:00
2019-05-01 16:24:13 -04:00
size_t write_res ;
2019-06-28 16:13:54 -04:00
char * dumpName = generateFullDumpName ( ) ;
2019-05-01 16:24:13 -04:00
if ( ! dumpName )
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
breaks + = 2 ;
return false ;
}
2019-06-08 21:36:21 -04:00
workaroundPartitionZeroAccess ( ) ;
2019-04-21 12:27:33 -04:00
2019-06-08 21:36:21 -04:00
if ( R_SUCCEEDED ( result = fsDeviceOperatorGetGameCardHandle ( & fsOperatorInstance , & handle ) ) )
2019-04-21 12:27:33 -04:00
{
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "GetGameCardHandle succeeded: 0x%08X", handle.value);
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
breaks + + ; */
// Ugly hack
// The IStorage instance returned for partition == 0 contains the gamecard header, the gamecard certificate, the root HFS0 header and:
// * The "update" (0) partition and the "normal" (1) partition (for gamecard type 0x01)
// * The "update" (0) partition, the "logo" (1) partition and the "normal" (2) partition (for gamecard type 0x02)
// The IStorage instance returned for partition == 1 contains the "secure" partition (which can either be 2 or 3 depending on the gamecard type)
// This ugly hack makes sure we just dump the *actual* raw HFS0 partition, without preceding data, padding, etc.
2019-06-28 16:13:54 -04:00
// Oddly enough, IFileSystem instances actually point to the specified partition ID filesystem. I don't understand why it doesn't work like that for IStorage, but whatever
2019-04-21 12:27:33 -04:00
// NOTE: Using partition == 2 returns error 0x149002, and using higher values probably do so, too
2019-04-23 01:14:57 -04:00
if ( R_SUCCEEDED ( result = fsOpenGameCardStorage ( & gameCardStorage , & handle , HFS0_TO_ISTORAGE_IDX ( hfs0_partition_cnt , partition ) ) ) )
2019-04-21 12:27:33 -04:00
{
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "OpenGameCardStorage succeeded: 0x%08X", handle.value);
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
breaks + + ; */
2019-06-05 18:44:18 -04:00
if ( getHfs0EntryDetails ( hfs0_header , hfs0_offset , hfs0_size , hfs0_partition_cnt , partition , true , 0 , & partitionOffset , & ( progressCtx . totalSize ) ) )
2019-04-21 12:27:33 -04:00
{
2019-06-05 18:44:18 -04:00
convertSize ( progressCtx . totalSize , progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " HFS0 partition size: %s (%lu bytes). " , progressCtx . totalSizeStr , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
2019-04-21 12:27:33 -04:00
2019-06-05 18:44:18 -04:00
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "HFS0 partition offset (relative to IStorage instance): 0x%016lX", partitionOffset);
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ; */
2019-04-21 12:27:33 -04:00
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize < = freeSpace )
2019-04-21 12:27:33 -04:00
{
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & doSplitting )
2019-04-21 12:27:33 -04:00
{
2019-06-08 21:36:21 -04:00
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s - Partition %u (%s).hfs0.%02u " , HFS0_DUMP_PATH , dumpName , partition , GAMECARD_PARTITION_NAME ( hfs0_partition_cnt , partition ) , splitIndex ) ;
2019-04-21 12:27:33 -04:00
} else {
2019-06-08 21:36:21 -04:00
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s - Partition %u (%s).hfs0 " , HFS0_DUMP_PATH , dumpName , partition , GAMECARD_PARTITION_NAME ( hfs0_partition_cnt , partition ) ) ;
2019-04-21 12:27:33 -04:00
}
2019-06-19 23:56:14 -04:00
// Check if the dump already exists
if ( checkIfFileExists ( filename ) )
2019-04-21 12:27:33 -04:00
{
2019-06-19 23:56:14 -04:00
// Ask the user if they want to proceed anyway
int cur_breaks = breaks ;
proceed = yesNoPrompt ( " You have already dumped this content. Do you wish to proceed anyway? " ) ;
if ( ! proceed )
2019-04-21 12:27:33 -04:00
{
2019-06-19 23:56:14 -04:00
uiDrawString ( " Process canceled. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
} else {
// Remove the prompt from the screen
breaks = cur_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
}
}
if ( proceed )
{
outFile = fopen ( filename , " wb " ) ;
if ( outFile )
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dumping raw HFS0 partition #%u. Hold %s to cancel. " , partition , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
2019-04-23 01:14:57 -04:00
{
2019-09-14 22:45:27 -04:00
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-23 01:14:57 -04:00
breaks + = 2 ;
2019-09-14 22:45:27 -04:00
}
uiRefreshDisplay ( ) ;
progressCtx . line_offset = ( breaks + 2 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . start ) ) ;
for ( progressCtx . curOffset = 0 ; progressCtx . curOffset < progressCtx . totalSize ; progressCtx . curOffset + = n )
{
uiFill ( 0 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-06-05 18:44:18 -04:00
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( filename , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
if ( DUMP_BUFFER_SIZE > ( progressCtx . totalSize - progressCtx . curOffset ) ) n = ( progressCtx . totalSize - progressCtx . curOffset ) ;
2019-06-19 23:56:14 -04:00
2019-09-14 22:45:27 -04:00
if ( R_FAILED ( result = fsStorageRead ( & gameCardStorage , partitionOffset + progressCtx . curOffset , dumpBuf , n ) ) )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " StorageRead failed (0x%08X) at offset 0x%016lX " , result , partitionOffset + progressCtx . curOffset ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
2019-06-19 23:56:14 -04:00
2019-09-14 22:45:27 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & doSplitting & & ( progressCtx . curOffset + n ) > = ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
u64 new_file_chunk_size = ( ( progressCtx . curOffset + n ) - ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) ) ;
u64 old_file_chunk_size = ( n - new_file_chunk_size ) ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
if ( old_file_chunk_size > 0 )
2019-06-19 23:56:14 -04:00
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , old_file_chunk_size , outFile ) ;
if ( write_res ! = old_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , old_file_chunk_size , progressCtx . curOffset , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
2019-06-19 23:56:14 -04:00
}
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
fclose ( outFile ) ;
outFile = NULL ;
if ( new_file_chunk_size > 0 | | ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
splitIndex + + ;
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s - Partition %u (%s).hfs0.%02u " , HFS0_DUMP_PATH , dumpName , partition , GAMECARD_PARTITION_NAME ( hfs0_partition_cnt , partition ) , splitIndex ) ;
2019-06-05 18:44:18 -04:00
2019-09-14 22:45:27 -04:00
outFile = fopen ( filename , " wb " ) ;
if ( ! outFile )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file for part #%u! " , splitIndex ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
2019-04-21 12:27:33 -04:00
}
2019-06-05 18:44:18 -04:00
2019-09-14 22:45:27 -04:00
if ( new_file_chunk_size > 0 )
2019-06-05 18:44:18 -04:00
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf + old_file_chunk_size , 1 , new_file_chunk_size , outFile ) ;
if ( write_res ! = new_file_chunk_size )
2019-06-05 18:44:18 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , new_file_chunk_size , progressCtx . curOffset + old_file_chunk_size , splitIndex , write_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
break ;
}
}
2019-04-21 12:27:33 -04:00
}
2019-09-14 22:45:27 -04:00
} else {
write_res = fwrite ( dumpBuf , 1 , n , outFile ) ;
if ( write_res ! = n )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX! (wrote %lu bytes) " , n , progressCtx . curOffset , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
if ( ( progressCtx . curOffset + n ) > FAT32_FILESIZE_LIMIT )
2019-05-01 16:24:13 -04:00
{
2019-09-14 22:45:27 -04:00
uiDrawString ( " You're probably using a FAT32 partition. Make sure to enable file splitting. " , 8 , ( ( progressCtx . line_offset + 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
fat32_error = true ;
2019-05-01 16:24:13 -04:00
}
2019-09-14 22:45:27 -04:00
break ;
2019-04-21 12:27:33 -04:00
}
}
2019-09-14 22:45:27 -04:00
printProgressBar ( & progressCtx , true , n ) ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
if ( ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
if ( cancelProcessCheck ( & progressCtx ) )
{
uiDrawString ( " Process canceled. " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
2019-04-21 12:27:33 -04:00
}
2019-09-14 22:45:27 -04:00
}
if ( progressCtx . curOffset > = progressCtx . totalSize ) success = true ;
// Support empty files
if ( ! progressCtx . totalSize )
{
uiFill ( 0 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( filename , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
progressCtx . progress = 100 ;
printProgressBar ( & progressCtx , false , 0 ) ;
}
breaks = ( progressCtx . line_offset + 2 ) ;
if ( success )
{
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
2019-06-05 18:44:18 -04:00
2019-09-14 22:45:27 -04:00
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
2019-04-21 12:27:33 -04:00
} else {
2019-09-14 22:45:27 -04:00
setProgressBarError ( & progressCtx ) ;
if ( fat32_error ) breaks + = 2 ;
2019-04-21 12:27:33 -04:00
}
2019-06-19 23:56:14 -04:00
if ( outFile ) fclose ( outFile ) ;
if ( ! success )
2019-04-21 12:27:33 -04:00
{
2019-06-19 23:56:14 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & doSplitting )
2019-04-21 12:27:33 -04:00
{
2019-06-19 23:56:14 -04:00
for ( u8 i = 0 ; i < = splitIndex ; i + + )
{
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s - Partition %u (%s).hfs0.%02u " , HFS0_DUMP_PATH , dumpName , partition , GAMECARD_PARTITION_NAME ( hfs0_partition_cnt , partition ) , i ) ;
unlink ( filename ) ;
}
} else {
2019-06-05 18:44:18 -04:00
unlink ( filename ) ;
2019-04-21 12:27:33 -04:00
}
}
2019-06-19 23:56:14 -04:00
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file \" %s \" ! " , filename ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
}
} else {
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
} else {
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to get partition details from the root HFS0 header! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
fsStorageClose ( & gameCardStorage ) ;
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " OpenGameCardStorage failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " GetGameCardHandle failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
breaks + = 2 ;
2019-05-01 16:24:13 -04:00
free ( dumpName ) ;
2019-04-23 01:14:57 -04:00
2019-04-21 12:27:33 -04:00
return success ;
2018-06-21 02:42:46 -04:00
}
2018-05-16 23:29:43 +02:00
2019-06-08 21:36:21 -04:00
bool copyFileFromHfs0 ( u32 partition , const char * source , const char * dest , const u64 file_offset , const u64 size , progress_ctx_t * progressCtx , bool doSplitting )
2018-06-21 02:42:46 -04:00
{
2019-04-23 01:14:57 -04:00
Result result ;
2019-05-01 16:24:13 -04:00
bool success = false , fat32_error = false ;
2019-04-21 12:27:33 -04:00
char splitFilename [ NAME_BUF_LEN ] = { ' \0 ' } ;
size_t destLen = strlen ( dest ) ;
2019-04-23 01:14:57 -04:00
FILE * outFile = NULL ;
u64 off , n = DUMP_BUFFER_SIZE ;
2019-04-21 12:27:33 -04:00
u8 splitIndex = 0 ;
2019-04-23 01:14:57 -04:00
FsGameCardHandle handle ;
FsStorage gameCardStorage ;
2019-05-01 16:24:13 -04:00
size_t write_res ;
2019-09-14 22:45:27 -04:00
memset ( dumpBuf , 0 , DUMP_BUFFER_SIZE ) ;
2019-06-08 21:36:21 -04:00
uiFill ( 0 , ( ( progressCtx - > line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 4 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-04-21 12:27:33 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Copying \" %s \" ... " , source ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
if ( ( destLen + 1 ) < NAME_BUF_LEN )
{
2019-06-08 21:36:21 -04:00
if ( R_SUCCEEDED ( result = fsDeviceOperatorGetGameCardHandle ( & fsOperatorInstance , & handle ) ) )
2019-04-21 12:27:33 -04:00
{
2019-04-23 01:14:57 -04:00
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "GetGameCardHandle succeeded: 0x%08X", handle.value);
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-23 01:14:57 -04:00
breaks + + ; */
2019-04-21 12:27:33 -04:00
2019-06-05 18:44:18 -04:00
// Same ugly hack from dumpRawHfs0Partition()
2019-04-23 01:14:57 -04:00
if ( R_SUCCEEDED ( result = fsOpenGameCardStorage ( & gameCardStorage , & handle , HFS0_TO_ISTORAGE_IDX ( hfs0_partition_cnt , partition ) ) ) )
2019-04-21 12:27:33 -04:00
{
2019-04-23 01:14:57 -04:00
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "OpenGameCardStorage succeeded: 0x%08X", handle.value);
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-23 01:14:57 -04:00
breaks + + ; */
2019-05-01 16:24:13 -04:00
if ( size > FAT32_FILESIZE_LIMIT & & doSplitting ) snprintf ( splitFilename , sizeof ( splitFilename ) / sizeof ( splitFilename [ 0 ] ) , " %s.%02u " , dest , splitIndex ) ;
2019-04-23 01:14:57 -04:00
2019-05-01 16:24:13 -04:00
outFile = fopen ( ( ( size > FAT32_FILESIZE_LIMIT & & doSplitting ) ? splitFilename : dest ) , " wb " ) ;
2019-04-23 01:14:57 -04:00
if ( outFile )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
for ( off = 0 ; off < size ; off + = n , progressCtx - > curOffset + = n )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
uiFill ( 0 , ( ( progressCtx - > line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , ( ( size > FAT32_FILESIZE_LIMIT & & doSplitting ) ? ( strrchr ( splitFilename , ' / ' ) + 1 ) : ( strrchr ( dest , ' / ' ) + 1 ) ) ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
if ( DUMP_BUFFER_SIZE > ( size - off ) ) n = ( size - off ) ;
if ( R_FAILED ( result = fsStorageRead ( & gameCardStorage , file_offset + off , dumpBuf , n ) ) )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " StorageRead failed (0x%08X) at offset 0x%016lX " , result , file_offset + off ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
if ( size > FAT32_FILESIZE_LIMIT & & doSplitting & & ( off + n ) > = ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) )
{
u64 new_file_chunk_size = ( ( off + n ) - ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) ) ;
u64 old_file_chunk_size = ( n - new_file_chunk_size ) ;
2019-04-23 01:14:57 -04:00
2019-09-14 22:45:27 -04:00
if ( old_file_chunk_size > 0 )
2019-04-23 01:14:57 -04:00
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , old_file_chunk_size , outFile ) ;
if ( write_res ! = old_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , old_file_chunk_size , off , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
2019-04-23 01:14:57 -04:00
}
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
fclose ( outFile ) ;
outFile = NULL ;
if ( new_file_chunk_size > 0 | | ( off + n ) < size )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
splitIndex + + ;
snprintf ( splitFilename , sizeof ( splitFilename ) / sizeof ( splitFilename [ 0 ] ) , " %s.%02u " , dest , splitIndex ) ;
2019-04-23 01:14:57 -04:00
2019-09-14 22:45:27 -04:00
outFile = fopen ( splitFilename , " wb " ) ;
if ( ! outFile )
2019-04-23 01:14:57 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file for part #%u! " , splitIndex ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
2019-04-23 01:14:57 -04:00
}
2019-09-14 22:45:27 -04:00
if ( new_file_chunk_size > 0 )
2019-04-23 01:14:57 -04:00
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf + old_file_chunk_size , 1 , new_file_chunk_size , outFile ) ;
if ( write_res ! = new_file_chunk_size )
2019-04-23 01:14:57 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , new_file_chunk_size , off + old_file_chunk_size , splitIndex , write_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-23 01:14:57 -04:00
break ;
}
2019-04-21 12:27:33 -04:00
}
}
2019-09-14 22:45:27 -04:00
} else {
write_res = fwrite ( dumpBuf , 1 , n , outFile ) ;
if ( write_res ! = n )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX! (wrote %lu bytes) " , n , off , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
if ( ( off + n ) > FAT32_FILESIZE_LIMIT )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
uiDrawString ( " You're probably using a FAT32 partition. Make sure to enable file splitting. " , 8 , ( ( progressCtx - > line_offset + 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
fat32_error = true ;
2019-04-21 12:27:33 -04:00
}
2019-09-14 22:45:27 -04:00
break ;
2019-04-21 12:27:33 -04:00
}
}
2019-09-14 22:45:27 -04:00
printProgressBar ( progressCtx , true , n ) ;
2019-04-23 01:14:57 -04:00
2019-09-14 22:45:27 -04:00
if ( ( off + n ) < size | | ( progressCtx - > curOffset + n ) < progressCtx - > totalSize )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
if ( cancelProcessCheck ( progressCtx ) )
{
uiDrawString ( " Process canceled. " , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
2019-04-21 12:27:33 -04:00
}
2019-09-14 22:45:27 -04:00
}
if ( off > = size ) success = true ;
// Support empty files
if ( ! size )
{
uiFill ( 0 , ( ( progressCtx - > line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , ( ( size > FAT32_FILESIZE_LIMIT & & doSplitting ) ? ( strrchr ( splitFilename , ' / ' ) + 1 ) : ( strrchr ( dest , ' / ' ) + 1 ) ) ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
2019-09-14 22:45:27 -04:00
if ( progressCtx - > totalSize = = size ) progressCtx - > progress = 100 ;
printProgressBar ( progressCtx , false , 0 ) ;
}
if ( ! success )
{
setProgressBarError ( progressCtx ) ;
breaks = ( progressCtx - > line_offset + 2 ) ;
if ( fat32_error ) breaks + = 2 ;
2019-04-21 12:27:33 -04:00
}
2019-04-23 01:14:57 -04:00
if ( outFile ) fclose ( outFile ) ;
2019-04-21 12:27:33 -04:00
2019-06-05 18:44:18 -04:00
if ( ! success )
2019-04-21 12:27:33 -04:00
{
2019-05-01 16:24:13 -04:00
if ( size > FAT32_FILESIZE_LIMIT & & doSplitting )
2019-04-23 01:14:57 -04:00
{
for ( u8 i = 0 ; i < = splitIndex ; i + + )
{
snprintf ( splitFilename , sizeof ( splitFilename ) / sizeof ( splitFilename [ 0 ] ) , " %s.%02u " , dest , i ) ;
2019-06-05 18:44:18 -04:00
unlink ( splitFilename ) ;
2019-04-23 01:14:57 -04:00
}
} else {
2019-06-05 18:44:18 -04:00
unlink ( dest ) ;
2019-04-23 01:14:57 -04:00
}
2019-04-21 12:27:33 -04:00
}
2019-04-23 01:14:57 -04:00
} else {
2019-06-08 21:36:21 -04:00
uiDrawString ( " Failed to open output file! " , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
}
2019-04-23 01:14:57 -04:00
fsStorageClose ( & gameCardStorage ) ;
2019-04-21 12:27:33 -04:00
} else {
2019-04-23 01:14:57 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " OpenGameCardStorage failed! (0x%08X) " , result ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
} else {
2019-04-23 01:14:57 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " GetGameCardHandle failed! (0x%08X) " , result ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
} else {
2019-04-23 01:14:57 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Destination path is too long! (%lu bytes) " , destLen ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
return success ;
2018-06-21 02:42:46 -04:00
}
2018-05-16 23:29:43 +02:00
2019-06-08 21:36:21 -04:00
bool copyHfs0Contents ( u32 partition , hfs0_entry_table * partitionEntryTable , progress_ctx_t * progressCtx , const char * dest , bool splitting )
2018-06-21 02:42:46 -04:00
{
2019-04-23 01:14:57 -04:00
if ( ! dest | | ! * dest )
2019-04-21 12:27:33 -04:00
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: destination directory is empty. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-23 01:14:57 -04:00
return false ;
2019-04-21 12:27:33 -04:00
}
2019-04-23 01:14:57 -04:00
if ( ! partitionHfs0Header | | ! partitionEntryTable )
2019-04-21 12:27:33 -04:00
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " HFS0 partition header information unavailable! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
return false ;
}
if ( ! progressCtx )
{
uiDrawString ( " Error: invalid progress context. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
return false ;
}
2019-04-23 01:14:57 -04:00
char dbuf [ NAME_BUF_LEN ] = { ' \0 ' } ;
size_t dest_len = strlen ( dest ) ;
if ( ( dest_len + 1 ) > = NAME_BUF_LEN )
2019-04-21 12:27:33 -04:00
{
2019-04-23 01:14:57 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Destination directory name is too long! (%lu bytes) " , dest_len ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
return false ;
}
strcpy ( dbuf , dest ) ;
mkdir ( dbuf , 0744 ) ;
2019-04-23 01:14:57 -04:00
dbuf [ dest_len ] = ' / ' ;
dest_len + + ;
2019-04-21 12:27:33 -04:00
2019-04-23 01:14:57 -04:00
u32 i ;
bool success ;
2019-06-05 18:44:18 -04:00
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx - > start ) ) ;
2019-04-23 01:14:57 -04:00
for ( i = 0 ; i < partitionHfs0FileCount ; i + + )
2019-04-21 12:27:33 -04:00
{
2019-04-23 01:14:57 -04:00
u32 filename_offset = ( HFS0_ENTRY_TABLE_ADDR + ( sizeof ( hfs0_entry_table ) * partitionHfs0FileCount ) + partitionEntryTable [ i ] . filename_offset ) ;
2019-06-05 18:44:18 -04:00
char * filename = ( ( char * ) partitionHfs0Header + filename_offset ) ;
2019-04-23 01:14:57 -04:00
strcpy ( dbuf + dest_len , filename ) ;
2019-04-21 12:27:33 -04:00
2019-06-05 18:44:18 -04:00
removeIllegalCharacters ( dbuf + dest_len ) ;
2019-04-23 01:14:57 -04:00
u64 file_offset = ( partitionHfs0HeaderSize + partitionEntryTable [ i ] . file_offset ) ;
if ( HFS0_TO_ISTORAGE_IDX ( hfs0_partition_cnt , partition ) = = 0 ) file_offset + = partitionHfs0HeaderOffset ;
2019-04-21 12:27:33 -04:00
2019-06-08 21:36:21 -04:00
success = copyFileFromHfs0 ( partition , filename , dbuf , file_offset , partitionEntryTable [ i ] . file_size , progressCtx , splitting ) ;
2019-04-23 01:14:57 -04:00
if ( ! success ) break ;
2019-04-21 12:27:33 -04:00
}
return success ;
2018-06-21 02:42:46 -04:00
}
2018-05-16 19:08:20 +02:00
2019-06-28 16:13:54 -04:00
bool dumpHfs0PartitionData ( u32 partition , bool doSplitting )
2018-06-21 02:42:46 -04:00
{
2019-04-21 12:27:33 -04:00
bool success = false ;
2019-04-23 01:14:57 -04:00
u32 i ;
hfs0_entry_table * entryTable = NULL ;
2019-06-05 18:44:18 -04:00
char dumpPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
progress_ctx_t progressCtx ;
memset ( & progressCtx , 0 , sizeof ( progress_ctx_t ) ) ;
2019-05-01 16:24:13 -04:00
2019-06-28 16:13:54 -04:00
char * dumpName = generateFullDumpName ( ) ;
2019-05-01 16:24:13 -04:00
if ( ! dumpName )
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-05-01 16:24:13 -04:00
breaks + = 2 ;
return false ;
}
2019-04-21 12:27:33 -04:00
2019-06-08 21:36:21 -04:00
workaroundPartitionZeroAccess ( ) ;
2019-04-21 12:27:33 -04:00
2019-05-01 16:24:13 -04:00
if ( getPartitionHfs0Header ( partition ) )
2019-04-21 12:27:33 -04:00
{
2019-04-23 01:14:57 -04:00
if ( partitionHfs0FileCount )
2019-04-21 12:27:33 -04:00
{
2019-06-05 18:44:18 -04:00
entryTable = calloc ( partitionHfs0FileCount , sizeof ( hfs0_entry_table ) ) ;
2019-04-23 01:14:57 -04:00
if ( entryTable )
2019-04-21 12:27:33 -04:00
{
2019-04-23 01:14:57 -04:00
memcpy ( entryTable , partitionHfs0Header + HFS0_ENTRY_TABLE_ADDR , sizeof ( hfs0_entry_table ) * partitionHfs0FileCount ) ;
// Calculate total size
2019-06-05 18:44:18 -04:00
for ( i = 0 ; i < partitionHfs0FileCount ; i + + ) progressCtx . totalSize + = entryTable [ i ] . file_size ;
2019-04-23 01:14:57 -04:00
2019-06-05 18:44:18 -04:00
convertSize ( progressCtx . totalSize , progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Total partition data size: %s (%lu bytes). " , progressCtx . totalSizeStr , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
2019-04-21 12:27:33 -04:00
2019-06-05 18:44:18 -04:00
if ( progressCtx . totalSize < = freeSpace )
2019-04-21 12:27:33 -04:00
{
2019-06-08 21:36:21 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s - Partition %u (%s) " , HFS0_DUMP_PATH , dumpName , partition , GAMECARD_PARTITION_NAME ( hfs0_partition_cnt , partition ) ) ;
2019-04-21 12:27:33 -04:00
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Copying partition #%u data to \" %s/ \" . Hold %s to cancel. " , partition , dumpPath , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-04-21 12:27:33 -04:00
breaks + = 2 ;
2019-04-23 01:14:57 -04:00
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-23 01:14:57 -04:00
breaks + = 2 ;
}
2019-04-21 12:27:33 -04:00
2019-04-23 01:14:57 -04:00
uiRefreshDisplay ( ) ;
2019-06-08 21:36:21 -04:00
progressCtx . line_offset = ( breaks + 4 ) ;
2019-06-28 16:13:54 -04:00
success = copyHfs0Contents ( partition , entryTable , & progressCtx , dumpPath , doSplitting ) ;
2019-06-05 18:44:18 -04:00
2019-04-23 01:14:57 -04:00
if ( success )
2019-04-21 12:27:33 -04:00
{
2019-06-28 16:13:54 -04:00
breaks = ( progressCtx . line_offset + 2 ) ;
2019-06-05 18:44:18 -04:00
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
2019-04-21 12:27:33 -04:00
} else {
2019-08-15 01:21:51 -04:00
removeDirectoryWithVerbose ( dumpPath , " Deleting output directory. Please wait... " ) ;
2019-04-21 12:27:33 -04:00
}
} else {
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
2019-04-23 01:14:57 -04:00
free ( entryTable ) ;
2019-04-21 12:27:33 -04:00
} else {
2019-06-05 18:44:18 -04:00
uiDrawString ( " Unable to allocate memory for the HFS0 file entries! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
} else {
2019-06-05 18:44:18 -04:00
uiDrawString ( " The selected partition is empty! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
2019-04-23 01:14:57 -04:00
free ( partitionHfs0Header ) ;
partitionHfs0Header = NULL ;
2019-06-05 18:44:18 -04:00
partitionHfs0HeaderOffset = 0 ;
2019-04-23 01:14:57 -04:00
partitionHfs0HeaderSize = 0 ;
partitionHfs0FileCount = 0 ;
partitionHfs0StrTableSize = 0 ;
2019-04-21 12:27:33 -04:00
}
breaks + = 2 ;
2019-05-01 16:24:13 -04:00
free ( dumpName ) ;
2019-04-21 12:27:33 -04:00
return success ;
2018-05-16 19:08:20 +02:00
}
2019-06-28 16:13:54 -04:00
bool dumpFileFromHfs0Partition ( u32 partition , u32 file , char * filename , bool doSplitting )
2018-06-21 02:42:46 -04:00
{
2019-04-23 01:14:57 -04:00
if ( ! partitionHfs0Header )
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " HFS0 partition header information unavailable! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
2019-04-23 01:14:57 -04:00
return false ;
}
2019-05-01 16:24:13 -04:00
if ( ! filename | | ! * filename )
2019-04-23 01:14:57 -04:00
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Filename unavailable! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
2019-04-23 01:14:57 -04:00
return false ;
}
2019-06-05 18:44:18 -04:00
progress_ctx_t progressCtx ;
memset ( & progressCtx , 0 , sizeof ( progress_ctx_t ) ) ;
2019-06-28 16:13:54 -04:00
char * dumpName = generateFullDumpName ( ) ;
2019-05-01 16:24:13 -04:00
if ( ! dumpName )
2019-04-21 12:27:33 -04:00
{
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
2019-04-23 01:14:57 -04:00
return false ;
}
u64 file_offset = 0 ;
u64 file_size = 0 ;
2019-06-19 23:56:14 -04:00
bool proceed = true , success = false ;
2019-04-23 01:14:57 -04:00
if ( getHfs0EntryDetails ( partitionHfs0Header , partitionHfs0HeaderOffset , partitionHfs0HeaderSize , partitionHfs0FileCount , file , false , partition , & file_offset , & file_size ) )
{
2019-06-05 18:44:18 -04:00
progressCtx . totalSize = file_size ;
convertSize ( progressCtx . totalSize , progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " File size: %s (%lu bytes). " , progressCtx . totalSizeStr , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + + ;
2019-04-23 01:14:57 -04:00
if ( file_size < = freeSpace )
2019-04-21 12:27:33 -04:00
{
2019-05-01 16:24:13 -04:00
char destCopyPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
2019-06-05 18:44:18 -04:00
char fixedFilename [ NAME_BUF_LEN ] = { ' \0 ' } ;
sprintf ( fixedFilename , filename ) ;
removeIllegalCharacters ( fixedFilename ) ;
2019-06-08 21:36:21 -04:00
snprintf ( destCopyPath , sizeof ( destCopyPath ) / sizeof ( destCopyPath [ 0 ] ) , " %s%s - Partition %u (%s) " , HFS0_DUMP_PATH , dumpName , partition , GAMECARD_PARTITION_NAME ( hfs0_partition_cnt , partition ) ) ;
2019-04-21 12:27:33 -04:00
2019-04-23 01:14:57 -04:00
if ( ( strlen ( destCopyPath ) + 1 + strlen ( filename ) ) < NAME_BUF_LEN )
{
mkdir ( destCopyPath , 0744 ) ;
2019-06-19 23:56:14 -04:00
strcat ( destCopyPath , " / " ) ;
strcat ( destCopyPath , fixedFilename ) ;
2019-06-05 18:44:18 -04:00
breaks + + ;
2019-04-23 01:14:57 -04:00
2019-06-19 23:56:14 -04:00
// Check if the dump already exists
if ( checkIfFileExists ( destCopyPath ) )
2019-04-23 01:14:57 -04:00
{
2019-06-19 23:56:14 -04:00
// Ask the user if they want to proceed anyway
int cur_breaks = breaks ;
proceed = yesNoPrompt ( " You have already dumped this content. Do you wish to proceed anyway? " ) ;
if ( ! proceed )
{
uiDrawString ( " Process canceled. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
} else {
// Remove the prompt from the screen
breaks = cur_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
}
2019-04-23 01:14:57 -04:00
}
2019-06-19 23:56:14 -04:00
if ( proceed )
2019-06-05 18:44:18 -04:00
{
2019-06-19 23:56:14 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Hold %s to cancel. " , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
{
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
}
uiRefreshDisplay ( ) ;
progressCtx . line_offset = ( breaks + 4 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . start ) ) ;
2019-06-28 16:13:54 -04:00
success = copyFileFromHfs0 ( partition , filename , destCopyPath , file_offset , file_size , & progressCtx , doSplitting ) ;
2019-06-05 18:44:18 -04:00
2019-06-19 23:56:14 -04:00
if ( success )
{
2019-06-28 16:13:54 -04:00
breaks = ( progressCtx . line_offset + 2 ) ;
2019-06-19 23:56:14 -04:00
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
}
2019-06-05 18:44:18 -04:00
}
2019-04-23 01:14:57 -04:00
} else {
2019-06-05 18:44:18 -04:00
breaks + + ;
2019-04-23 01:14:57 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Destination path is too long! (%lu bytes) " , strlen ( destCopyPath ) + 1 + strlen ( filename ) ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-23 01:14:57 -04:00
}
2019-04-21 12:27:33 -04:00
} else {
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
} else {
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: unable to get file details from the partition HFS0 header! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
2019-05-01 16:24:13 -04:00
free ( dumpName ) ;
2019-06-05 18:44:18 -04:00
breaks + = 2 ;
2019-04-21 12:27:33 -04:00
return success ;
2018-05-16 19:08:20 +02:00
}
2019-06-28 16:13:54 -04:00
bool dumpExeFsSectionData ( u32 titleIndex , bool usePatch , bool doSplitting )
2018-06-21 02:42:46 -04:00
{
2019-06-08 21:36:21 -04:00
u64 n ;
FILE * outFile ;
u8 splitIndex ;
2019-06-05 18:44:18 -04:00
bool proceed = true , success = false , fat32_error = false ;
2019-06-08 21:36:21 -04:00
char dumpPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } , curDumpPath [ NAME_BUF_LEN * 4 ] = { ' \0 ' } ;
char tmp_idx [ 5 ] ;
2019-09-14 22:45:27 -04:00
memset ( dumpBuf , 0 , DUMP_BUFFER_SIZE ) ;
2019-06-08 21:36:21 -04:00
progress_ctx_t progressCtx ;
memset ( & progressCtx , 0 , sizeof ( progress_ctx_t ) ) ;
2019-06-05 18:44:18 -04:00
2019-05-01 16:24:13 -04:00
size_t write_res ;
2019-06-08 21:36:21 -04:00
u32 i ;
u64 offset ;
2019-06-05 18:44:18 -04:00
2019-06-28 16:13:54 -04:00
if ( ( ! usePatch & & ! titleAppCount ) | | ( usePatch & & ! titlePatchCount ) )
2019-06-08 21:36:21 -04:00
{
2019-06-28 16:13:54 -04:00
uiDrawString ( " Error: invalid title count! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-08 21:36:21 -04:00
breaks + = 2 ;
return false ;
}
2019-06-05 18:44:18 -04:00
2019-06-28 16:13:54 -04:00
if ( ( ! usePatch & & titleIndex > ( titleAppCount - 1 ) ) | | ( usePatch & & titleIndex > ( titlePatchCount - 1 ) ) )
2019-05-01 16:24:13 -04:00
{
2019-06-28 16:13:54 -04:00
uiDrawString ( " Error: invalid title index! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-08 21:36:21 -04:00
breaks + = 2 ;
2019-05-01 16:24:13 -04:00
return false ;
}
2019-04-21 12:27:33 -04:00
2019-06-28 16:13:54 -04:00
char * dumpName = generateNSPDumpName ( ( ! usePatch ? DUMP_APP_NSP : DUMP_PATCH_NSP ) , titleIndex ) ;
2019-06-08 21:36:21 -04:00
if ( ! dumpName )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
2019-06-05 18:44:18 -04:00
return false ;
}
2019-04-21 12:27:33 -04:00
2019-06-08 21:36:21 -04:00
// Retrieve ExeFS from Program NCA
2019-09-14 22:45:27 -04:00
if ( ! readNcaExeFsSection ( titleIndex , usePatch ) )
2019-06-08 21:36:21 -04:00
{
free ( dumpName ) ;
breaks + = 2 ;
return false ;
}
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
// Calculate total dump size
if ( ! calculateExeFsExtractedDataSize ( & ( progressCtx . totalSize ) ) ) goto out ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
convertSize ( progressCtx . totalSize , progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Extracted ExeFS dump size: %s (%lu bytes). " , progressCtx . totalSizeStr , progressCtx . totalSize ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-08 21:36:21 -04:00
uiRefreshDisplay ( ) ;
breaks + + ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
if ( progressCtx . totalSize > freeSpace )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
goto out ;
}
2019-06-08 21:36:21 -04:00
// Prepare output dump path
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s " , EXEFS_DUMP_PATH , dumpName ) ;
mkdir ( dumpPath , 0744 ) ;
// Start dump process
breaks + + ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dump procedure started. Hold %s to cancel. " , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + = 2 ;
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
{
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
}
progressCtx . line_offset = ( breaks + 4 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . start ) ) ;
for ( i = 0 ; i < exeFsContext . exefs_header . file_cnt ; i + + )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
n = DUMP_BUFFER_SIZE ;
outFile = NULL ;
splitIndex = 0 ;
2019-04-21 12:27:33 -04:00
2019-06-08 21:36:21 -04:00
char * exeFsFilename = ( exeFsContext . exefs_str_table + exeFsContext . exefs_entries [ i ] . filename_offset ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
// Check if we're dealing with a nameless file
if ( ! strlen ( exeFsFilename ) )
{
uiDrawString ( " Error: file entry without name in ExeFS section! " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
snprintf ( curDumpPath , sizeof ( curDumpPath ) / sizeof ( curDumpPath [ 0 ] ) , " %s/%s " , dumpPath , exeFsFilename ) ;
removeIllegalCharacters ( curDumpPath + strlen ( dumpPath ) + 1 ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
if ( exeFsContext . exefs_entries [ i ] . file_size > FAT32_FILESIZE_LIMIT & & doSplitting )
{
sprintf ( tmp_idx , " .%02u " , splitIndex ) ;
strcat ( curDumpPath , tmp_idx ) ;
}
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
outFile = fopen ( curDumpPath , " wb " ) ;
if ( ! outFile )
{
uiDrawString ( " Failed to open output file! " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Copying \" %s \" ... " , exeFsFilename ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
for ( offset = 0 ; offset < exeFsContext . exefs_entries [ i ] . file_size ; offset + = n , progressCtx . curOffset + = n )
2019-04-21 12:27:33 -04:00
{
2019-06-08 21:36:21 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-04-21 12:27:33 -04:00
2019-06-08 21:36:21 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( curDumpPath , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
uiRefreshDisplay ( ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
if ( DUMP_BUFFER_SIZE > ( exeFsContext . exefs_entries [ i ] . file_size - offset ) ) n = ( exeFsContext . exefs_entries [ i ] . file_size - offset ) ;
breaks = ( progressCtx . line_offset + 2 ) ;
2019-09-14 22:45:27 -04:00
proceed = processNcaCtrSectionBlock ( & ( exeFsContext . ncmStorage ) , & ( exeFsContext . ncaId ) , & ( exeFsContext . aes_ctx ) , exeFsContext . exefs_data_offset + exeFsContext . exefs_entries [ i ] . file_offset + offset , dumpBuf , n , false ) ;
2019-06-08 21:36:21 -04:00
breaks = ( progressCtx . line_offset - 4 ) ;
if ( ! proceed ) break ;
if ( exeFsContext . exefs_entries [ i ] . file_size > FAT32_FILESIZE_LIMIT & & doSplitting & & ( offset + n ) > = ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
u64 new_file_chunk_size = ( ( offset + n ) - ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) ) ;
u64 old_file_chunk_size = ( n - new_file_chunk_size ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
if ( old_file_chunk_size > 0 )
2019-06-05 18:44:18 -04:00
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , old_file_chunk_size , outFile ) ;
2019-06-08 21:36:21 -04:00
if ( write_res ! = old_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , old_file_chunk_size , offset , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
}
fclose ( outFile ) ;
outFile = NULL ;
if ( new_file_chunk_size > 0 | | ( offset + n ) < exeFsContext . exefs_entries [ i ] . file_size )
{
char * tmp = strrchr ( curDumpPath , ' . ' ) ;
if ( tmp ! = NULL ) * tmp = ' \0 ' ;
splitIndex + + ;
sprintf ( tmp_idx , " .%02u " , splitIndex ) ;
strcat ( curDumpPath , tmp_idx ) ;
outFile = fopen ( curDumpPath , " wb " ) ;
if ( ! outFile )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file for part #%u! " , splitIndex ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
if ( new_file_chunk_size > 0 )
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf + old_file_chunk_size , 1 , new_file_chunk_size , outFile ) ;
2019-06-08 21:36:21 -04:00
if ( write_res ! = new_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , new_file_chunk_size , offset + old_file_chunk_size , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
}
}
} else {
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , n , outFile ) ;
2019-06-08 21:36:21 -04:00
if ( write_res ! = n )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX! (wrote %lu bytes) " , n , offset , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
if ( ( offset + n ) > FAT32_FILESIZE_LIMIT )
{
uiDrawString ( " You're probably using a FAT32 partition. Make sure to enable file splitting. " , 8 , ( ( progressCtx . line_offset + 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
fat32_error = true ;
}
proceed = false ;
break ;
}
}
printProgressBar ( & progressCtx , true , n ) ;
2019-08-15 01:21:51 -04:00
if ( ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-06-08 21:36:21 -04:00
{
2019-08-15 01:21:51 -04:00
if ( cancelProcessCheck ( & progressCtx ) )
2019-06-08 21:36:21 -04:00
{
uiDrawString ( " Process canceled. " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
}
}
if ( outFile ) fclose ( outFile ) ;
if ( ! proceed ) break ;
// Support empty files
if ( ! exeFsContext . exefs_entries [ i ] . file_size )
{
uiFill ( 0 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( curDumpPath , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
if ( progressCtx . totalSize = = exeFsContext . exefs_entries [ i ] . file_size ) progressCtx . progress = 100 ;
printProgressBar ( & progressCtx , true , 0 ) ;
}
}
if ( proceed )
{
if ( progressCtx . curOffset > = progressCtx . totalSize )
{
success = true ;
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Unexpected underdump error! Wrote %lu bytes, expected %lu bytes. " , progressCtx . curOffset , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
}
}
breaks = ( progressCtx . line_offset + 2 ) ;
if ( success )
{
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
} else {
setProgressBarError ( & progressCtx ) ;
if ( fat32_error ) breaks + = 2 ;
2019-08-15 01:21:51 -04:00
removeDirectoryWithVerbose ( dumpPath , " Deleting output directory. Please wait... " ) ;
2019-06-08 21:36:21 -04:00
}
out :
freeExeFsContext ( ) ;
free ( dumpName ) ;
breaks + = 2 ;
return success ;
}
2019-06-28 16:13:54 -04:00
bool dumpFileFromExeFsSection ( u32 titleIndex , u32 fileIndex , bool usePatch , bool doSplitting )
2019-06-08 21:36:21 -04:00
{
2019-06-28 16:13:54 -04:00
if ( ! exeFsContext . exefs_header . file_cnt | | fileIndex > ( exeFsContext . exefs_header . file_cnt - 1 ) | | ! exeFsContext . exefs_entries | | ! exeFsContext . exefs_str_table | | exeFsContext . exefs_data_offset < = exeFsContext . exefs_offset | | ( ! usePatch & & titleIndex > ( titleAppCount - 1 ) ) | | ( usePatch & & titleIndex > ( titlePatchCount - 1 ) ) )
2019-06-08 21:36:21 -04:00
{
uiDrawString ( " Error: invalid parameters to parse file entry from ExeFS section! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
u64 n = DUMP_BUFFER_SIZE ;
FILE * outFile = NULL ;
u8 splitIndex = 0 ;
2019-06-19 23:56:14 -04:00
bool proceed = true , success = false , fat32_error = false , removeFile = true ;
2019-06-08 21:36:21 -04:00
char dumpPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
char tmp_idx [ 5 ] ;
2019-09-14 22:45:27 -04:00
memset ( dumpBuf , 0 , DUMP_BUFFER_SIZE ) ;
2019-06-08 21:36:21 -04:00
progress_ctx_t progressCtx ;
memset ( & progressCtx , 0 , sizeof ( progress_ctx_t ) ) ;
size_t write_res ;
char * exeFsFilename = ( exeFsContext . exefs_str_table + exeFsContext . exefs_entries [ fileIndex ] . filename_offset ) ;
// Check if we're dealing with a nameless file
if ( ! strlen ( exeFsFilename ) )
{
uiDrawString ( " Error: file entry without name in ExeFS section! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
2019-06-28 16:13:54 -04:00
char * dumpName = generateNSPDumpName ( ( ! usePatch ? DUMP_APP_NSP : DUMP_PATCH_NSP ) , titleIndex ) ;
2019-06-08 21:36:21 -04:00
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
// Generate output path
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s " , EXEFS_DUMP_PATH , dumpName ) ;
mkdir ( dumpPath , 0744 ) ;
strcat ( dumpPath , " / " ) ;
size_t cur_len = strlen ( dumpPath ) ;
strcat ( dumpPath , exeFsFilename ) ;
removeIllegalCharacters ( dumpPath + cur_len ) ;
2019-06-19 23:56:14 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & doSplitting )
{
sprintf ( tmp_idx , " .%02u " , splitIndex ) ;
strcat ( dumpPath , tmp_idx ) ;
}
2019-06-08 21:36:21 -04:00
progressCtx . totalSize = exeFsContext . exefs_entries [ fileIndex ] . file_size ;
convertSize ( progressCtx . totalSize , progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " File size: %s (%lu bytes). " , progressCtx . totalSizeStr , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-19 23:56:14 -04:00
breaks + + ;
if ( progressCtx . totalSize > freeSpace )
{
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
goto out ;
}
breaks + + ;
// Check if the dump already exists
if ( checkIfFileExists ( dumpPath ) )
{
// Ask the user if they want to proceed anyway
int cur_breaks = breaks ;
proceed = yesNoPrompt ( " You have already dumped this content. Do you wish to proceed anyway? " ) ;
if ( ! proceed )
{
uiDrawString ( " Process canceled. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
removeFile = false ;
goto out ;
} else {
// Remove the prompt from the screen
breaks = cur_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
}
}
2019-06-08 21:36:21 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Hold %s to cancel. " , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
{
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
}
// Start dump process
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Copying \" %s \" ... " , exeFsFilename ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
outFile = fopen ( dumpPath , " wb " ) ;
if ( ! outFile )
{
uiDrawString ( " Failed to open output file! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
goto out ;
}
progressCtx . line_offset = ( breaks + 2 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . start ) ) ;
for ( progressCtx . curOffset = 0 ; progressCtx . curOffset < progressCtx . totalSize ; progressCtx . curOffset + = n )
{
uiFill ( 0 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( dumpPath , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
if ( DUMP_BUFFER_SIZE > ( progressCtx . totalSize - progressCtx . curOffset ) ) n = ( progressCtx . totalSize - progressCtx . curOffset ) ;
breaks = ( progressCtx . line_offset + 2 ) ;
2019-09-14 22:45:27 -04:00
proceed = processNcaCtrSectionBlock ( & ( exeFsContext . ncmStorage ) , & ( exeFsContext . ncaId ) , & ( exeFsContext . aes_ctx ) , exeFsContext . exefs_data_offset + exeFsContext . exefs_entries [ fileIndex ] . file_offset + progressCtx . curOffset , dumpBuf , n , false ) ;
2019-06-08 21:36:21 -04:00
breaks = ( progressCtx . line_offset - 2 ) ;
if ( ! proceed ) break ;
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & doSplitting & & ( progressCtx . curOffset + n ) > = ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) )
{
u64 new_file_chunk_size = ( ( progressCtx . curOffset + n ) - ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) ) ;
u64 old_file_chunk_size = ( n - new_file_chunk_size ) ;
if ( old_file_chunk_size > 0 )
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , old_file_chunk_size , outFile ) ;
2019-06-08 21:36:21 -04:00
if ( write_res ! = old_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , old_file_chunk_size , progressCtx . curOffset , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
}
fclose ( outFile ) ;
outFile = NULL ;
if ( new_file_chunk_size > 0 | | ( progressCtx . curOffset + n ) < progressCtx . totalSize )
{
char * tmp = strrchr ( dumpPath , ' . ' ) ;
if ( tmp ! = NULL ) * tmp = ' \0 ' ;
splitIndex + + ;
sprintf ( tmp_idx , " .%02u " , splitIndex ) ;
strcat ( dumpPath , tmp_idx ) ;
outFile = fopen ( dumpPath , " wb " ) ;
if ( ! outFile )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file for part #%u! " , splitIndex ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
if ( new_file_chunk_size > 0 )
2019-04-21 12:27:33 -04:00
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf + old_file_chunk_size , 1 , new_file_chunk_size , outFile ) ;
2019-06-05 18:44:18 -04:00
if ( write_res ! = new_file_chunk_size )
2019-04-21 12:27:33 -04:00
{
2019-06-08 21:36:21 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , new_file_chunk_size , progressCtx . curOffset + old_file_chunk_size , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
break ;
}
}
}
} else {
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , n , outFile ) ;
2019-06-05 18:44:18 -04:00
if ( write_res ! = n )
{
2019-06-08 21:36:21 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX! (wrote %lu bytes) " , n , progressCtx . curOffset , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
if ( ( progressCtx . curOffset + n ) > FAT32_FILESIZE_LIMIT )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " You're probably using a FAT32 partition. Make sure to enable file splitting. " , 8 , ( ( progressCtx . line_offset + 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
fat32_error = true ;
}
break ;
}
}
2019-06-08 21:36:21 -04:00
printProgressBar ( & progressCtx , true , n ) ;
2019-06-05 18:44:18 -04:00
2019-08-15 01:21:51 -04:00
if ( ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-06-05 18:44:18 -04:00
{
2019-08-15 01:21:51 -04:00
if ( cancelProcessCheck ( & progressCtx ) )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Process canceled. " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
break ;
}
}
}
2019-06-08 21:36:21 -04:00
if ( progressCtx . curOffset > = progressCtx . totalSize ) success = true ;
2019-06-05 18:44:18 -04:00
// Support empty files
2019-06-08 21:36:21 -04:00
if ( ! progressCtx . totalSize )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( dumpPath , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
progressCtx . progress = 100 ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
printProgressBar ( & progressCtx , false , 0 ) ;
}
breaks = ( progressCtx . line_offset + 2 ) ;
if ( success )
{
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
} else {
setProgressBarError ( & progressCtx ) ;
2019-06-05 18:44:18 -04:00
}
out :
if ( outFile ) fclose ( outFile ) ;
if ( ! success )
{
if ( fat32_error ) breaks + = 2 ;
2019-06-19 23:56:14 -04:00
if ( removeFile )
2019-06-05 18:44:18 -04:00
{
2019-06-19 23:56:14 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & doSplitting )
2019-06-08 21:36:21 -04:00
{
2019-06-19 23:56:14 -04:00
for ( u8 i = 0 ; i < = splitIndex ; i + + )
{
char * tmp = strrchr ( dumpPath , ' . ' ) ;
if ( tmp ! = NULL ) * tmp = ' \0 ' ;
sprintf ( tmp_idx , " .%02u " , splitIndex ) ;
strcat ( dumpPath , tmp_idx ) ;
unlink ( dumpPath ) ;
}
} else {
2019-06-08 21:36:21 -04:00
unlink ( dumpPath ) ;
}
}
}
free ( dumpName ) ;
breaks + = 2 ;
return success ;
}
2019-06-28 16:13:54 -04:00
bool recursiveDumpRomFsFile ( u32 file_offset , char * romfs_path , char * output_path , progress_ctx_t * progressCtx , bool usePatch , bool doSplitting )
2019-06-08 21:36:21 -04:00
{
2019-06-28 16:13:54 -04:00
if ( ( ! usePatch & & ( ! romFsContext . romfs_filetable_size | | file_offset > romFsContext . romfs_filetable_size | | ! romFsContext . romfs_file_entries ) ) | | ( usePatch & & ( ! bktrContext . romfs_filetable_size | | file_offset > bktrContext . romfs_filetable_size | | ! bktrContext . romfs_file_entries ) ) | | ! romfs_path | | ! output_path | | ! progressCtx )
2019-06-08 21:36:21 -04:00
{
uiDrawString ( " Error: invalid parameters to parse file entry from RomFS section! " , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
return false ;
}
size_t orig_romfs_path_len = strlen ( romfs_path ) ;
size_t orig_output_path_len = strlen ( output_path ) ;
u64 n = DUMP_BUFFER_SIZE ;
FILE * outFile = NULL ;
u8 splitIndex = 0 ;
bool proceed = true , success = false , fat32_error = false ;
2019-08-15 01:21:51 -04:00
u32 romfs_file_offset = file_offset ;
romfs_file * entry = NULL ;
2019-06-08 21:36:21 -04:00
u64 off = 0 ;
size_t write_res ;
char tmp_idx [ 5 ] ;
2019-09-14 22:45:27 -04:00
memset ( dumpBuf , 0 , DUMP_BUFFER_SIZE ) ;
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
while ( romfs_file_offset ! = ROMFS_ENTRY_EMPTY )
2019-06-08 21:36:21 -04:00
{
2019-08-15 01:21:51 -04:00
romfs_path [ orig_romfs_path_len ] = ' \0 ' ;
output_path [ orig_output_path_len ] = ' \0 ' ;
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
entry = ( ! usePatch ? ( romfs_file * ) ( ( u8 * ) romFsContext . romfs_file_entries + romfs_file_offset ) : ( romfs_file * ) ( ( u8 * ) bktrContext . romfs_file_entries + romfs_file_offset ) ) ;
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
// Check if we're dealing with a nameless file
if ( ! entry - > nameLen )
{
uiDrawString ( " Error: file entry without name in RomFS section! " , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
if ( ( orig_romfs_path_len + 1 + entry - > nameLen ) > = ( NAME_BUF_LEN * 2 ) | | ( orig_output_path_len + 1 + entry - > nameLen ) > = ( NAME_BUF_LEN * 2 ) )
{
uiDrawString ( " Error: RomFS section file path is too long! " , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
break ;
}
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
// Generate current path
strcat ( romfs_path , " / " ) ;
strncat ( romfs_path , ( char * ) entry - > name , entry - > nameLen ) ;
2019-06-28 16:13:54 -04:00
2019-08-15 01:21:51 -04:00
strcat ( output_path , " / " ) ;
strncat ( output_path , ( char * ) entry - > name , entry - > nameLen ) ;
removeIllegalCharacters ( output_path + orig_output_path_len + 1 ) ;
// Start dump process
uiFill ( 0 , ( ( progressCtx - > line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 4 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Copying \" romfs:%s \" ... " , romfs_path ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset - 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
if ( entry - > dataSize > FAT32_FILESIZE_LIMIT & & doSplitting )
2019-06-28 16:13:54 -04:00
{
2019-08-15 01:21:51 -04:00
sprintf ( tmp_idx , " .%02u " , splitIndex ) ;
strcat ( output_path , tmp_idx ) ;
2019-06-28 16:13:54 -04:00
}
2019-08-15 01:21:51 -04:00
outFile = fopen ( output_path , " wb " ) ;
if ( ! outFile )
{
uiDrawString ( " Failed to open output file! " , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
break ;
}
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
n = DUMP_BUFFER_SIZE ;
splitIndex = 0 ;
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
for ( off = 0 ; off < entry - > dataSize ; off + = n , progressCtx - > curOffset + = n )
2019-06-08 21:36:21 -04:00
{
2019-08-15 01:21:51 -04:00
uiFill ( 0 , ( ( progressCtx - > line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( output_path , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
if ( DUMP_BUFFER_SIZE > ( entry - > dataSize - off ) ) n = ( entry - > dataSize - off ) ;
breaks = ( progressCtx - > line_offset + 2 ) ;
if ( ! usePatch )
2019-06-08 21:36:21 -04:00
{
2019-09-14 22:45:27 -04:00
proceed = processNcaCtrSectionBlock ( & ( romFsContext . ncmStorage ) , & ( romFsContext . ncaId ) , & ( romFsContext . aes_ctx ) , romFsContext . romfs_filedata_offset + entry - > dataOff + off , dumpBuf , n , false ) ;
2019-08-15 01:21:51 -04:00
} else {
2019-09-14 22:45:27 -04:00
proceed = readBktrSectionBlock ( bktrContext . romfs_filedata_offset + entry - > dataOff + off , dumpBuf , n ) ;
2019-06-08 21:36:21 -04:00
}
2019-08-15 01:21:51 -04:00
breaks = ( progressCtx - > line_offset - 4 ) ;
if ( ! proceed ) break ;
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
if ( entry - > dataSize > FAT32_FILESIZE_LIMIT & & doSplitting & & ( off + n ) > = ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) )
2019-06-05 18:44:18 -04:00
{
2019-08-15 01:21:51 -04:00
u64 new_file_chunk_size = ( ( off + n ) - ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) ) ;
u64 old_file_chunk_size = ( n - new_file_chunk_size ) ;
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
if ( old_file_chunk_size > 0 )
2019-06-08 21:36:21 -04:00
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , old_file_chunk_size , outFile ) ;
2019-08-15 01:21:51 -04:00
if ( write_res ! = old_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , old_file_chunk_size , off , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
2019-06-08 21:36:21 -04:00
}
2019-08-15 01:21:51 -04:00
fclose ( outFile ) ;
outFile = NULL ;
if ( new_file_chunk_size > 0 | | ( off + n ) < entry - > dataSize )
2019-06-08 21:36:21 -04:00
{
2019-08-15 01:21:51 -04:00
char * tmp = strrchr ( output_path , ' . ' ) ;
if ( tmp ! = NULL ) * tmp = ' \0 ' ;
splitIndex + + ;
sprintf ( tmp_idx , " .%02u " , splitIndex ) ;
strcat ( output_path , tmp_idx ) ;
outFile = fopen ( output_path , " wb " ) ;
if ( ! outFile )
2019-06-08 21:36:21 -04:00
{
2019-08-15 01:21:51 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file for part #%u! " , splitIndex ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-08-15 01:21:51 -04:00
proceed = false ;
2019-06-08 21:36:21 -04:00
break ;
}
2019-08-15 01:21:51 -04:00
if ( new_file_chunk_size > 0 )
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf + old_file_chunk_size , 1 , new_file_chunk_size , outFile ) ;
2019-08-15 01:21:51 -04:00
if ( write_res ! = new_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , new_file_chunk_size , off + old_file_chunk_size , splitIndex , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
}
}
}
} else {
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , n , outFile ) ;
2019-08-15 01:21:51 -04:00
if ( write_res ! = n )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX! (wrote %lu bytes) " , n , off , write_res ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
if ( ( off + n ) > FAT32_FILESIZE_LIMIT )
{
uiDrawString ( " You're probably using a FAT32 partition. Make sure to enable file splitting. " , 8 , ( ( progressCtx - > line_offset + 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
fat32_error = true ;
}
proceed = false ;
break ;
2019-06-08 21:36:21 -04:00
}
2019-06-05 18:44:18 -04:00
}
2019-08-15 01:21:51 -04:00
printProgressBar ( progressCtx , true , n ) ;
if ( ( off + n ) < entry - > dataSize | | ( progressCtx - > curOffset + n ) < progressCtx - > totalSize )
2019-06-08 21:36:21 -04:00
{
2019-08-15 01:21:51 -04:00
if ( cancelProcessCheck ( progressCtx ) )
2019-06-08 21:36:21 -04:00
{
2019-08-15 01:21:51 -04:00
uiDrawString ( " Process canceled. " , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
proceed = false ;
break ;
2019-06-08 21:36:21 -04:00
}
}
2019-06-05 18:44:18 -04:00
}
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
if ( outFile )
2019-06-08 21:36:21 -04:00
{
2019-08-15 01:21:51 -04:00
fclose ( outFile ) ;
outFile = NULL ;
2019-06-08 21:36:21 -04:00
}
2019-08-15 01:21:51 -04:00
if ( ! proceed | | off < entry - > dataSize ) break ;
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
// Support empty files
if ( ! entry - > dataSize )
{
uiFill ( 0 , ( ( progressCtx - > line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( output_path , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( ( progressCtx - > line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
if ( progressCtx - > totalSize = = entry - > dataSize ) progressCtx - > progress = 100 ;
printProgressBar ( progressCtx , false , 0 ) ;
}
2019-06-08 21:36:21 -04:00
2019-08-15 01:21:51 -04:00
romfs_file_offset = entry - > sibling ;
if ( romfs_file_offset = = ROMFS_ENTRY_EMPTY ) success = true ;
2019-06-05 18:44:18 -04:00
}
2019-06-28 16:13:54 -04:00
if ( ! success )
{
breaks = ( progressCtx - > line_offset + 2 ) ;
if ( fat32_error ) breaks + = 2 ;
}
2019-06-08 21:36:21 -04:00
2019-06-05 18:44:18 -04:00
romfs_path [ orig_romfs_path_len ] = ' \0 ' ;
output_path [ orig_output_path_len ] = ' \0 ' ;
return success ;
}
2019-06-28 16:13:54 -04:00
bool recursiveDumpRomFsDir ( u32 dir_offset , char * romfs_path , char * output_path , progress_ctx_t * progressCtx , bool usePatch , bool dumpSiblingDir , bool doSplitting )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
if ( ( ! usePatch & & ( ! romFsContext . romfs_dirtable_size | | dir_offset > romFsContext . romfs_dirtable_size | | ! romFsContext . romfs_dir_entries | | ! romFsContext . romfs_filetable_size | | ! romFsContext . romfs_file_entries ) ) | | ( usePatch & & ( ! bktrContext . romfs_dirtable_size | | dir_offset > bktrContext . romfs_dirtable_size | | ! bktrContext . romfs_dir_entries | | ! bktrContext . romfs_filetable_size | | ! bktrContext . romfs_file_entries ) ) | | ! romfs_path | | ! output_path | | ! progressCtx )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Error: invalid parameters to parse directory entry from RomFS section! " , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
return false ;
}
size_t orig_romfs_path_len = strlen ( romfs_path ) ;
size_t orig_output_path_len = strlen ( output_path ) ;
2019-06-28 16:13:54 -04:00
romfs_dir * entry = ( ! usePatch ? ( romfs_dir * ) ( ( u8 * ) romFsContext . romfs_dir_entries + dir_offset ) : ( romfs_dir * ) ( ( u8 * ) bktrContext . romfs_dir_entries + dir_offset ) ) ;
2019-06-05 18:44:18 -04:00
// Check if we're dealing with a nameless directory that's not the root directory
if ( ! entry - > nameLen & & dir_offset > 0 )
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Error: directory entry without name in RomFS section! " , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
return false ;
}
if ( ( orig_romfs_path_len + 1 + entry - > nameLen ) > = ( NAME_BUF_LEN * 2 ) | | ( orig_output_path_len + 1 + entry - > nameLen ) > = ( NAME_BUF_LEN * 2 ) )
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Error: RomFS section directory path is too long! " , 8 , ( ( progressCtx - > line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
return false ;
}
// Generate current path
if ( entry - > nameLen )
{
strcat ( romfs_path , " / " ) ;
strncat ( romfs_path , ( char * ) entry - > name , entry - > nameLen ) ;
strcat ( output_path , " / " ) ;
strncat ( output_path , ( char * ) entry - > name , entry - > nameLen ) ;
removeIllegalCharacters ( output_path + orig_output_path_len + 1 ) ;
mkdir ( output_path , 0744 ) ;
}
if ( entry - > childFile ! = ROMFS_ENTRY_EMPTY )
{
2019-06-28 16:13:54 -04:00
if ( ! recursiveDumpRomFsFile ( entry - > childFile , romfs_path , output_path , progressCtx , usePatch , doSplitting ) )
2019-06-05 18:44:18 -04:00
{
romfs_path [ orig_romfs_path_len ] = ' \0 ' ;
output_path [ orig_output_path_len ] = ' \0 ' ;
return false ;
}
}
if ( entry - > childDir ! = ROMFS_ENTRY_EMPTY )
{
2019-06-28 16:13:54 -04:00
if ( ! recursiveDumpRomFsDir ( entry - > childDir , romfs_path , output_path , progressCtx , usePatch , true , doSplitting ) )
2019-06-05 18:44:18 -04:00
{
romfs_path [ orig_romfs_path_len ] = ' \0 ' ;
output_path [ orig_output_path_len ] = ' \0 ' ;
return false ;
}
}
romfs_path [ orig_romfs_path_len ] = ' \0 ' ;
output_path [ orig_output_path_len ] = ' \0 ' ;
2019-06-28 16:13:54 -04:00
if ( dumpSiblingDir & & entry - > sibling ! = ROMFS_ENTRY_EMPTY )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
if ( ! recursiveDumpRomFsDir ( entry - > sibling , romfs_path , output_path , progressCtx , usePatch , true , doSplitting ) ) return false ;
2019-06-05 18:44:18 -04:00
}
return true ;
}
2019-09-14 22:45:27 -04:00
bool dumpRomFsSectionData ( u32 titleIndex , selectedRomFsType curRomFsType , bool doSplitting )
2019-06-05 18:44:18 -04:00
{
progress_ctx_t progressCtx ;
memset ( & progressCtx , 0 , sizeof ( progress_ctx_t ) ) ;
char romFsPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } , dumpPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
bool success = false ;
2019-09-14 22:45:27 -04:00
if ( ( curRomFsType = = ROMFS_TYPE_APP & & ! titleAppCount ) | | ( curRomFsType = = ROMFS_TYPE_PATCH & & ! titlePatchCount ) | | ( curRomFsType = = ROMFS_TYPE_ADDON & & ! titleAddOnCount ) )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
uiDrawString ( " Error: invalid title count! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
breaks + = 2 ;
return false ;
}
2019-09-14 22:45:27 -04:00
if ( ( curRomFsType = = ROMFS_TYPE_APP & & titleIndex > ( titleAppCount - 1 ) ) | | ( curRomFsType = = ROMFS_TYPE_PATCH & & titleIndex > ( titlePatchCount - 1 ) ) | | ( curRomFsType = = ROMFS_TYPE_ADDON & & titleIndex > ( titleAddOnCount - 1 ) ) )
2019-06-05 18:44:18 -04:00
{
2019-06-28 16:13:54 -04:00
uiDrawString ( " Error: invalid title index! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
breaks + = 2 ;
return false ;
}
2019-09-14 22:45:27 -04:00
char * dumpName = generateNSPDumpName ( ( curRomFsType = = ROMFS_TYPE_APP ? DUMP_APP_NSP : ( curRomFsType = = ROMFS_TYPE_PATCH ? DUMP_PATCH_NSP : DUMP_ADDON_NSP ) ) , titleIndex ) ;
2019-06-05 18:44:18 -04:00
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
// Retrieve RomFS from Program NCA
2019-09-14 22:45:27 -04:00
if ( ! readNcaRomFsSection ( titleIndex , curRomFsType ) )
2019-06-08 21:36:21 -04:00
{
free ( dumpName ) ;
breaks + = 2 ;
return false ;
}
2019-06-05 18:44:18 -04:00
// Calculate total dump size
2019-09-14 22:45:27 -04:00
if ( ! calculateRomFsFullExtractedSize ( ( curRomFsType = = ROMFS_TYPE_PATCH ) , & ( progressCtx . totalSize ) ) ) goto out ;
2019-06-05 18:44:18 -04:00
convertSize ( progressCtx . totalSize , progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Extracted RomFS dump size: %s (%lu bytes). " , progressCtx . totalSizeStr , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + + ;
if ( progressCtx . totalSize > freeSpace )
{
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
goto out ;
}
// Prepare output dump path
2019-06-08 21:36:21 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s " , ROMFS_DUMP_PATH , dumpName ) ;
2019-06-05 18:44:18 -04:00
mkdir ( dumpPath , 0744 ) ;
// Start dump process
breaks + + ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dump procedure started. Hold %s to cancel. " , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + = 2 ;
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
{
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
}
progressCtx . line_offset = ( breaks + 4 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . start ) ) ;
2019-09-14 22:45:27 -04:00
success = recursiveDumpRomFsDir ( 0 , romFsPath , dumpPath , & progressCtx , ( curRomFsType = = ROMFS_TYPE_PATCH ) , true , doSplitting ) ;
2019-06-05 18:44:18 -04:00
if ( success )
{
2019-06-28 16:13:54 -04:00
breaks = ( progressCtx . line_offset + 2 ) ;
2019-06-05 18:44:18 -04:00
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
} else {
setProgressBarError ( & progressCtx ) ;
2019-08-15 01:21:51 -04:00
removeDirectoryWithVerbose ( dumpPath , " Deleting output directory. Please wait... " ) ;
2019-06-05 18:44:18 -04:00
}
out :
2019-09-14 22:45:27 -04:00
if ( curRomFsType = = ROMFS_TYPE_PATCH ) freeBktrContext ( ) ;
2019-06-28 16:13:54 -04:00
2019-06-05 18:44:18 -04:00
freeRomFsContext ( ) ;
free ( dumpName ) ;
breaks + = 2 ;
return success ;
}
2019-09-14 22:45:27 -04:00
bool dumpFileFromRomFsSection ( u32 titleIndex , u32 file_offset , selectedRomFsType curRomFsType , bool doSplitting )
2019-06-05 18:44:18 -04:00
{
2019-09-14 22:45:27 -04:00
if ( ! romFsContext . romfs_filetable_size | | file_offset > romFsContext . romfs_filetable_size | | ! romFsContext . romfs_file_entries | | ( curRomFsType = = ROMFS_TYPE_APP & & titleIndex > ( titleAppCount - 1 ) ) | | ( curRomFsType = = ROMFS_TYPE_PATCH & & titleIndex > ( titlePatchCount - 1 ) ) | | ( curRomFsType = = ROMFS_TYPE_ADDON & & titleIndex > ( titleAddOnCount - 1 ) ) )
2019-06-05 18:44:18 -04:00
{
uiDrawString ( " Error: invalid parameters to parse file entry from RomFS section! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
u64 n = DUMP_BUFFER_SIZE ;
FILE * outFile = NULL ;
u8 splitIndex = 0 ;
2019-06-19 23:56:14 -04:00
bool proceed = true , success = false , fat32_error = false , removeFile = true ;
2019-06-05 18:44:18 -04:00
char dumpPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
char tmp_idx [ 5 ] ;
2019-09-14 22:45:27 -04:00
memset ( dumpBuf , 0 , DUMP_BUFFER_SIZE ) ;
2019-06-05 18:44:18 -04:00
progress_ctx_t progressCtx ;
memset ( & progressCtx , 0 , sizeof ( progress_ctx_t ) ) ;
size_t write_res ;
2019-09-14 22:45:27 -04:00
romfs_file * entry = ( curRomFsType ! = ROMFS_TYPE_PATCH ? ( romfs_file * ) ( ( u8 * ) romFsContext . romfs_file_entries + file_offset ) : ( romfs_file * ) ( ( u8 * ) bktrContext . romfs_file_entries + file_offset ) ) ;
2019-06-05 18:44:18 -04:00
// Check if we're dealing with a nameless file
if ( ! entry - > nameLen )
{
uiDrawString ( " Error: file entry without name in RomFS section! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
2019-09-14 22:45:27 -04:00
char * dumpName = generateNSPDumpName ( ( curRomFsType = = ROMFS_TYPE_APP ? DUMP_APP_NSP : ( curRomFsType = = ROMFS_TYPE_PATCH ? DUMP_PATCH_NSP : DUMP_ADDON_NSP ) ) , titleIndex ) ;
2019-06-05 18:44:18 -04:00
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
// Generate output path
2019-06-08 21:36:21 -04:00
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s " , ROMFS_DUMP_PATH , dumpName ) ;
2019-06-05 18:44:18 -04:00
mkdir ( dumpPath , 0744 ) ;
// Create subdirectories
char * tmp1 = NULL ;
char * tmp2 = NULL ;
2019-06-08 21:36:21 -04:00
size_t cur_len ;
2019-06-05 18:44:18 -04:00
tmp1 = strchr ( curRomFsPath , ' / ' ) ;
while ( tmp1 ! = NULL )
{
tmp1 + + ;
if ( ! strlen ( tmp1 ) ) break ;
strcat ( dumpPath , " / " ) ;
2019-06-08 21:36:21 -04:00
cur_len = strlen ( dumpPath ) ;
2019-06-05 18:44:18 -04:00
tmp2 = strchr ( tmp1 , ' / ' ) ;
if ( tmp2 ! = NULL )
{
strncat ( dumpPath , tmp1 , tmp2 - tmp1 ) ;
tmp1 = tmp2 ;
} else {
strcat ( dumpPath , tmp1 ) ;
tmp1 = NULL ;
}
removeIllegalCharacters ( dumpPath + cur_len ) ;
mkdir ( dumpPath , 0744 ) ;
}
strcat ( dumpPath , " / " ) ;
2019-06-08 21:36:21 -04:00
cur_len = strlen ( dumpPath ) ;
2019-06-05 18:44:18 -04:00
strncat ( dumpPath , ( char * ) entry - > name , entry - > nameLen ) ;
2019-06-08 21:36:21 -04:00
removeIllegalCharacters ( dumpPath + cur_len ) ;
2019-06-05 18:44:18 -04:00
progressCtx . totalSize = entry - > dataSize ;
convertSize ( progressCtx . totalSize , progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " File size: %s (%lu bytes). " , progressCtx . totalSizeStr , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-19 23:56:14 -04:00
breaks + + ;
if ( progressCtx . totalSize > freeSpace )
{
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
goto out ;
}
breaks + + ;
2019-08-17 06:33:50 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & doSplitting )
{
sprintf ( tmp_idx , " .%02u " , splitIndex ) ;
strcat ( dumpPath , tmp_idx ) ;
}
2019-06-19 23:56:14 -04:00
// Check if the dump already exists
if ( checkIfFileExists ( dumpPath ) )
{
// Ask the user if they want to proceed anyway
int cur_breaks = breaks ;
proceed = yesNoPrompt ( " You have already dumped this content. Do you wish to proceed anyway? " ) ;
if ( ! proceed )
{
uiDrawString ( " Process canceled. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
removeFile = false ;
goto out ;
} else {
// Remove the prompt from the screen
breaks = cur_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
}
}
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Hold %s to cancel. " , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
{
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
}
// Start dump process
2019-08-17 06:33:50 -04:00
if ( strlen ( curRomFsPath ) > 1 )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Copying \" romfs:%s/%.*s \" ... " , curRomFsPath , entry - > nameLen , entry - > name ) ;
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Copying \" romfs:/%.*s \" ... " , entry - > nameLen , entry - > name ) ;
}
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
outFile = fopen ( dumpPath , " wb " ) ;
if ( ! outFile )
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Failed to open output file! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
goto out ;
}
progressCtx . line_offset = ( breaks + 2 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . start ) ) ;
for ( progressCtx . curOffset = 0 ; progressCtx . curOffset < progressCtx . totalSize ; progressCtx . curOffset + = n )
{
2019-06-08 21:36:21 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( dumpPath , ' / ' ) + 1 ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
uiRefreshDisplay ( ) ;
if ( DUMP_BUFFER_SIZE > ( progressCtx . totalSize - progressCtx . curOffset ) ) n = ( progressCtx . totalSize - progressCtx . curOffset ) ;
2019-06-08 21:36:21 -04:00
breaks = ( progressCtx . line_offset + 2 ) ;
2019-06-28 16:13:54 -04:00
2019-09-14 22:45:27 -04:00
if ( curRomFsType ! = ROMFS_TYPE_PATCH )
2019-06-28 16:13:54 -04:00
{
2019-09-14 22:45:27 -04:00
proceed = processNcaCtrSectionBlock ( & ( romFsContext . ncmStorage ) , & ( romFsContext . ncaId ) , & ( romFsContext . aes_ctx ) , romFsContext . romfs_filedata_offset + entry - > dataOff + progressCtx . curOffset , dumpBuf , n , false ) ;
2019-06-28 16:13:54 -04:00
} else {
2019-09-14 22:45:27 -04:00
proceed = readBktrSectionBlock ( bktrContext . romfs_filedata_offset + entry - > dataOff + progressCtx . curOffset , dumpBuf , n ) ;
2019-06-28 16:13:54 -04:00
}
2019-06-08 21:36:21 -04:00
breaks = ( progressCtx . line_offset - 2 ) ;
2019-06-05 18:44:18 -04:00
if ( ! proceed ) break ;
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & doSplitting & & ( progressCtx . curOffset + n ) > = ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) )
{
u64 new_file_chunk_size = ( ( progressCtx . curOffset + n ) - ( ( splitIndex + 1 ) * SPLIT_FILE_GENERIC_PART_SIZE ) ) ;
u64 old_file_chunk_size = ( n - new_file_chunk_size ) ;
if ( old_file_chunk_size > 0 )
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , old_file_chunk_size , outFile ) ;
2019-06-05 18:44:18 -04:00
if ( write_res ! = old_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , old_file_chunk_size , progressCtx . curOffset , splitIndex , write_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
break ;
}
}
fclose ( outFile ) ;
outFile = NULL ;
if ( new_file_chunk_size > 0 | | ( progressCtx . curOffset + n ) < progressCtx . totalSize )
{
char * tmp = strrchr ( dumpPath , ' . ' ) ;
if ( tmp ! = NULL ) * tmp = ' \0 ' ;
splitIndex + + ;
sprintf ( tmp_idx , " .%02u " , splitIndex ) ;
strcat ( dumpPath , tmp_idx ) ;
outFile = fopen ( dumpPath , " wb " ) ;
if ( ! outFile )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file for part #%u! " , splitIndex ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
break ;
}
if ( new_file_chunk_size > 0 )
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf + old_file_chunk_size , 1 , new_file_chunk_size , outFile ) ;
2019-06-05 18:44:18 -04:00
if ( write_res ! = new_file_chunk_size )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX to part #%02u! (wrote %lu bytes) " , new_file_chunk_size , progressCtx . curOffset + old_file_chunk_size , splitIndex , write_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
break ;
}
}
}
} else {
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , n , outFile ) ;
2019-06-05 18:44:18 -04:00
if ( write_res ! = n )
{
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %lu bytes chunk from offset 0x%016lX! (wrote %lu bytes) " , n , progressCtx . curOffset , write_res ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
if ( ( progressCtx . curOffset + n ) > FAT32_FILESIZE_LIMIT )
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " You're probably using a FAT32 partition. Make sure to enable file splitting. " , 8 , ( ( progressCtx . line_offset + 4 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
fat32_error = true ;
}
break ;
}
}
printProgressBar ( & progressCtx , true , n ) ;
2019-08-15 01:21:51 -04:00
if ( ( progressCtx . curOffset + n ) < progressCtx . totalSize )
2019-06-05 18:44:18 -04:00
{
2019-08-15 01:21:51 -04:00
if ( cancelProcessCheck ( & progressCtx ) )
2019-06-05 18:44:18 -04:00
{
2019-06-08 21:36:21 -04:00
uiDrawString ( " Process canceled. " , 8 , ( ( progressCtx . line_offset + 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-06-05 18:44:18 -04:00
break ;
}
}
}
if ( progressCtx . curOffset > = progressCtx . totalSize ) success = true ;
// Support empty files
if ( ! progressCtx . totalSize )
{
2019-06-08 21:36:21 -04:00
uiFill ( 0 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + 8 , FB_WIDTH , ( font_height + ( font_height / 4 ) ) * 2 , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Output file: \" %s \" . " , strrchr ( dumpPath , ' / ' ) + 1 ) ;
2019-06-08 21:36:21 -04:00
uiDrawString ( strbuf , 8 , ( ( progressCtx . line_offset - 2 ) * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
2019-06-05 18:44:18 -04:00
progressCtx . progress = 100 ;
printProgressBar ( & progressCtx , false , 0 ) ;
}
2019-06-08 21:36:21 -04:00
breaks = ( progressCtx . line_offset + 2 ) ;
2019-06-05 18:44:18 -04:00
if ( success )
{
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
} else {
setProgressBarError ( & progressCtx ) ;
}
out :
if ( outFile ) fclose ( outFile ) ;
if ( ! success )
{
if ( fat32_error ) breaks + = 2 ;
2019-06-19 23:56:14 -04:00
if ( removeFile )
2019-06-05 18:44:18 -04:00
{
2019-06-19 23:56:14 -04:00
if ( progressCtx . totalSize > FAT32_FILESIZE_LIMIT & & doSplitting )
2019-06-05 18:44:18 -04:00
{
2019-06-19 23:56:14 -04:00
for ( u8 i = 0 ; i < = splitIndex ; i + + )
{
char * tmp = strrchr ( dumpPath , ' . ' ) ;
if ( tmp ! = NULL ) * tmp = ' \0 ' ;
sprintf ( tmp_idx , " .%02u " , splitIndex ) ;
strcat ( dumpPath , tmp_idx ) ;
unlink ( dumpPath ) ;
}
} else {
2019-06-05 18:44:18 -04:00
unlink ( dumpPath ) ;
}
}
}
free ( dumpName ) ;
breaks + = 2 ;
return success ;
}
2019-09-14 22:45:27 -04:00
bool dumpCurrentDirFromRomFsSection ( u32 titleIndex , selectedRomFsType curRomFsType , bool doSplitting )
2019-06-28 16:13:54 -04:00
{
progress_ctx_t progressCtx ;
memset ( & progressCtx , 0 , sizeof ( progress_ctx_t ) ) ;
char romFsPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } , dumpPath [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
bool success = false ;
2019-09-14 22:45:27 -04:00
if ( ( curRomFsType = = ROMFS_TYPE_APP & & ! titleAppCount ) | | ( curRomFsType = = ROMFS_TYPE_PATCH & & ! titlePatchCount ) | | ( curRomFsType = = ROMFS_TYPE_ADDON & & ! titleAddOnCount ) )
2019-06-28 16:13:54 -04:00
{
uiDrawString ( " Error: invalid title count! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
2019-09-14 22:45:27 -04:00
if ( ( curRomFsType = = ROMFS_TYPE_APP & & titleIndex > ( titleAppCount - 1 ) ) | | ( curRomFsType = = ROMFS_TYPE_PATCH & & titleIndex > ( titlePatchCount - 1 ) ) | | ( curRomFsType = = ROMFS_TYPE_ADDON & & titleIndex > ( titleAddOnCount - 1 ) ) )
2019-06-28 16:13:54 -04:00
{
uiDrawString ( " Error: invalid title index! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
2019-09-14 22:45:27 -04:00
char * dumpName = generateNSPDumpName ( ( curRomFsType = = ROMFS_TYPE_APP ? DUMP_APP_NSP : ( curRomFsType = = ROMFS_TYPE_PATCH ? DUMP_PATCH_NSP : DUMP_ADDON_NSP ) ) , titleIndex ) ;
2019-06-28 16:13:54 -04:00
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
// Calculate total dump size
2019-09-14 22:45:27 -04:00
if ( ! calculateRomFsExtractedDirSize ( curRomFsDirOffset , ( curRomFsType = = ROMFS_TYPE_PATCH ) , & ( progressCtx . totalSize ) ) ) goto out ;
2019-06-28 16:13:54 -04:00
convertSize ( progressCtx . totalSize , progressCtx . totalSizeStr , sizeof ( progressCtx . totalSizeStr ) / sizeof ( progressCtx . totalSizeStr [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Extracted RomFS directory size: %s (%lu bytes). " , progressCtx . totalSizeStr , progressCtx . totalSize ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + + ;
if ( progressCtx . totalSize > freeSpace )
{
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
goto out ;
}
2019-08-15 01:21:51 -04:00
if ( strlen ( curRomFsPath ) > 1 )
{
// Copy the whole current path and remove the last element (current directory) from it
// It will be re-added later
snprintf ( romFsPath , sizeof ( romFsPath ) / sizeof ( romFsPath [ 0 ] ) , curRomFsPath ) ;
char * slash = strrchr ( romFsPath , ' / ' ) ;
if ( slash ) * slash = ' \0 ' ;
}
2019-06-28 16:13:54 -04:00
// Prepare output dump path
snprintf ( dumpPath , sizeof ( dumpPath ) / sizeof ( dumpPath [ 0 ] ) , " %s%s " , ROMFS_DUMP_PATH , dumpName ) ;
mkdir ( dumpPath , 0744 ) ;
// Create subdirectories
char * tmp1 = NULL ;
char * tmp2 = NULL ;
size_t cur_len ;
tmp1 = strchr ( curRomFsPath , ' / ' ) ;
while ( tmp1 ! = NULL )
{
tmp1 + + ;
if ( ! strlen ( tmp1 ) ) break ;
tmp2 = strchr ( tmp1 , ' / ' ) ;
if ( tmp2 ! = NULL )
{
strcat ( dumpPath , " / " ) ;
cur_len = strlen ( dumpPath ) ;
strncat ( dumpPath , tmp1 , tmp2 - tmp1 ) ;
removeIllegalCharacters ( dumpPath + cur_len ) ;
mkdir ( dumpPath , 0744 ) ;
tmp1 = tmp2 ;
} else {
// Skip last entry
tmp1 = NULL ;
}
}
// Start dump process
breaks + + ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dump procedure started. Hold %s to cancel. " , NINTENDO_FONT_B ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
uiRefreshDisplay ( ) ;
breaks + = 2 ;
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
{
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
}
progressCtx . line_offset = ( breaks + 4 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . start ) ) ;
2019-09-14 22:45:27 -04:00
success = recursiveDumpRomFsDir ( curRomFsDirOffset , romFsPath , dumpPath , & progressCtx , ( curRomFsType = = ROMFS_TYPE_PATCH ) , false , doSplitting ) ;
2019-06-28 16:13:54 -04:00
if ( success )
{
breaks = ( progressCtx . line_offset + 2 ) ;
timeGetCurrentTime ( TimeType_LocalSystemClock , & ( progressCtx . now ) ) ;
progressCtx . now - = progressCtx . start ;
formatETAString ( progressCtx . now , progressCtx . etaInfo , sizeof ( progressCtx . etaInfo ) / sizeof ( progressCtx . etaInfo [ 0 ] ) ) ;
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Process successfully completed after %s! " , progressCtx . etaInfo ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
} else {
setProgressBarError ( & progressCtx ) ;
2019-08-15 01:21:51 -04:00
removeDirectoryWithVerbose ( dumpPath , " Deleting output directory. Please wait... " ) ;
2019-06-28 16:13:54 -04:00
}
out :
free ( dumpName ) ;
breaks + = 2 ;
return success ;
}
2019-06-08 21:36:21 -04:00
bool dumpGameCardCertificate ( )
2019-06-05 18:44:18 -04:00
{
u32 crc = 0 ;
Result result ;
FsGameCardHandle handle ;
FsStorage gameCardStorage ;
2019-06-19 23:56:14 -04:00
bool proceed = true , success = false ;
2019-06-05 18:44:18 -04:00
FILE * outFile = NULL ;
char filename [ NAME_BUF_LEN * 2 ] = { ' \0 ' } ;
size_t write_res ;
2019-09-14 22:45:27 -04:00
memset ( dumpBuf , 0 , DUMP_BUFFER_SIZE ) ;
2019-06-28 16:13:54 -04:00
char * dumpName = generateFullDumpName ( ) ;
2019-06-05 18:44:18 -04:00
if ( ! dumpName )
{
uiDrawString ( " Error: unable to generate output dump name! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
return false ;
}
2019-06-08 21:36:21 -04:00
workaroundPartitionZeroAccess ( ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
if ( R_SUCCEEDED ( result = fsDeviceOperatorGetGameCardHandle ( & fsOperatorInstance , & handle ) ) )
2019-06-05 18:44:18 -04:00
{
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "GetGameCardHandle succeeded: 0x%08X", handle.value);
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + + ; */
if ( R_SUCCEEDED ( result = fsOpenGameCardStorage ( & gameCardStorage , & handle , 0 ) ) )
{
/*snprintf(strbuf, sizeof(strbuf) / sizeof(strbuf[0]), "OpenGameCardStorage succeeded: 0x%08X", handle.value);
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + + ; */
if ( CERT_SIZE < = freeSpace )
{
2019-09-14 22:45:27 -04:00
if ( R_SUCCEEDED ( result = fsStorageRead ( & gameCardStorage , CERT_OFFSET , dumpBuf , CERT_SIZE ) ) )
2019-06-05 18:44:18 -04:00
{
// Calculate CRC32
2019-09-14 22:45:27 -04:00
crc32 ( dumpBuf , CERT_SIZE , & crc ) ;
2019-06-05 18:44:18 -04:00
2019-06-08 21:36:21 -04:00
snprintf ( filename , sizeof ( filename ) / sizeof ( filename [ 0 ] ) , " %s%s - Certificate (%08X).bin " , CERT_DUMP_PATH , dumpName , crc ) ;
2019-06-05 18:44:18 -04:00
2019-06-19 23:56:14 -04:00
// Check if the dump already exists
if ( checkIfFileExists ( filename ) )
2019-06-05 18:44:18 -04:00
{
2019-06-19 23:56:14 -04:00
// Ask the user if they want to proceed anyway
int cur_breaks = breaks ;
proceed = yesNoPrompt ( " You have already dumped this content. Do you wish to proceed anyway? " ) ;
if ( ! proceed )
{
uiDrawString ( " Process canceled. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
} else {
// Remove the prompt from the screen
breaks = cur_breaks ;
uiFill ( 0 , 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , FB_WIDTH , FB_HEIGHT - ( 8 + ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) ) , BG_COLOR_RGB , BG_COLOR_RGB , BG_COLOR_RGB ) ;
}
2019-06-05 18:44:18 -04:00
}
2019-06-19 23:56:14 -04:00
if ( proceed )
2019-06-05 18:44:18 -04:00
{
2019-06-19 23:56:14 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Dumping gamecard certificate to \" %s \" ... " , strrchr ( filename , ' / ' ) + 1 ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 255 , 255 ) ;
breaks + = 2 ;
if ( programAppletType ! = AppletType_Application & & programAppletType ! = AppletType_SystemApplication )
{
uiDrawString ( " Do not press the " NINTENDO_FONT_HOME " button. Doing so could corrupt the SD card filesystem. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
breaks + = 2 ;
}
uiRefreshDisplay ( ) ;
outFile = fopen ( filename , " wb " ) ;
if ( outFile )
2019-06-05 18:44:18 -04:00
{
2019-09-14 22:45:27 -04:00
write_res = fwrite ( dumpBuf , 1 , CERT_SIZE , outFile ) ;
2019-06-19 23:56:14 -04:00
if ( write_res = = CERT_SIZE )
{
success = true ;
uiDrawString ( " Process successfully completed! " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 0 , 255 , 0 ) ;
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to write %u bytes certificate data! (wrote %lu bytes) " , CERT_SIZE , write_res ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
}
fclose ( outFile ) ;
if ( ! success ) unlink ( filename ) ;
2019-06-05 18:44:18 -04:00
} else {
2019-06-19 23:56:14 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " Failed to open output file \" %s \" ! " , filename ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
}
2019-04-21 12:27:33 -04:00
}
} else {
2019-06-05 18:44:18 -04:00
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " StorageRead failed (0x%08X) at offset 0x%08X " , result , CERT_OFFSET ) ;
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
} else {
2019-06-05 18:44:18 -04:00
uiDrawString ( " Error: not enough free space available in the SD card. " , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
fsStorageClose ( & gameCardStorage ) ;
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " OpenGameCardStorage failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
} else {
snprintf ( strbuf , sizeof ( strbuf ) / sizeof ( strbuf [ 0 ] ) , " GetGameCardHandle failed! (0x%08X) " , result ) ;
2019-06-05 18:44:18 -04:00
uiDrawString ( strbuf , 8 , ( breaks * ( font_height + ( font_height / 4 ) ) ) + ( font_height / 8 ) , 255 , 0 , 0 ) ;
2019-04-21 12:27:33 -04:00
}
breaks + = 2 ;
2019-05-01 16:24:13 -04:00
free ( dumpName ) ;
2019-04-21 12:27:33 -04:00
return success ;
2018-06-21 02:42:46 -04:00
}