2020-07-29 17:02:21 -04:00
/*
* main . c
*
* Copyright ( c ) 2020 , DarkMatterCore < pabloacurielz @ gmail . com > .
*
* This file is part of nxdumptool ( https : //github.com/DarkMatterCore/nxdumptool).
*
2021-03-25 15:26:58 -04:00
* nxdumptool is free software : you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation , either version 3 of the License , or
* ( at your option ) any later version .
2020-07-29 17:02:21 -04:00
*
2021-03-25 15:26:58 -04:00
* nxdumptool is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
2020-07-29 17:02:21 -04:00
*
* You should have received a copy of the GNU General Public License
2021-03-25 15:26:58 -04:00
* along with this program . If not , see < https : //www.gnu.org/licenses/>.
2020-07-29 17:02:21 -04:00
*/
2021-03-26 00:35:14 -04:00
# include "nxdt_utils.h"
2020-07-29 17:02:21 -04:00
# include "bktr.h"
# include "gamecard.h"
# include "usb.h"
# include "title.h"
2020-08-19 13:20:26 -04:00
# define BLOCK_SIZE USB_TRANSFER_BUFFER_SIZE
2020-07-29 17:02:21 -04:00
2021-09-04 00:17:17 -04:00
bool g_borealisInitialized = false ;
2021-04-25 19:10:34 -04:00
static PadState g_padState = { 0 } ;
2020-07-29 17:02:21 -04:00
static Mutex g_fileMutex = 0 ;
static CondVar g_readCondvar = 0 , g_writeCondvar = 0 ;
typedef struct
{
//FILE *fileobj;
2020-08-19 13:20:26 -04:00
RomFileSystemContext * romfs_ctx ;
2020-07-29 17:02:21 -04:00
BktrContext * bktr_ctx ;
void * data ;
size_t data_size ;
size_t data_written ;
size_t total_size ;
bool read_error ;
bool write_error ;
bool transfer_cancelled ;
} ThreadSharedData ;
2021-04-25 19:10:34 -04:00
static void utilsScanPads ( void )
{
padUpdate ( & g_padState ) ;
}
static u64 utilsGetButtonsDown ( void )
{
return padGetButtonsDown ( & g_padState ) ;
}
static u64 utilsGetButtonsHeld ( void )
{
return padGetButtons ( & g_padState ) ;
}
static void utilsWaitForButtonPress ( u64 flag )
{
/* Don't consider stick movement as button inputs. */
if ( ! flag ) flag = ~ ( HidNpadButton_StickLLeft | HidNpadButton_StickLRight | HidNpadButton_StickLUp | HidNpadButton_StickLDown | HidNpadButton_StickRLeft | HidNpadButton_StickRRight | \
HidNpadButton_StickRUp | HidNpadButton_StickRDown ) ;
while ( appletMainLoop ( ) )
{
utilsScanPads ( ) ;
if ( utilsGetButtonsDown ( ) & flag ) break ;
}
}
2020-07-29 17:02:21 -04:00
static void consolePrint ( const char * text , . . . )
{
va_list v ;
va_start ( v , text ) ;
vfprintf ( stdout , text , v ) ;
va_end ( v ) ;
consoleUpdate ( NULL ) ;
}
2020-08-18 01:04:13 -04:00
static void read_thread_func ( void * arg )
2020-07-29 17:02:21 -04:00
{
ThreadSharedData * shared_data = ( ThreadSharedData * ) arg ;
2020-08-19 13:20:26 -04:00
if ( ! shared_data | | ! shared_data - > data | | ! shared_data - > total_size | | ( ! shared_data - > romfs_ctx & & ! shared_data - > bktr_ctx ) )
2020-07-29 17:02:21 -04:00
{
shared_data - > read_error = true ;
2020-08-18 01:04:13 -04:00
goto end ;
2020-07-29 17:02:21 -04:00
}
2020-08-19 13:20:26 -04:00
u8 * buf = malloc ( BLOCK_SIZE ) ;
2020-07-29 17:02:21 -04:00
if ( ! buf )
{
shared_data - > read_error = true ;
2020-08-18 01:04:13 -04:00
goto end ;
2020-07-29 17:02:21 -04:00
}
u64 file_table_offset = 0 ;
2020-08-19 13:20:26 -04:00
u64 file_table_size = ( shared_data - > bktr_ctx ? shared_data - > bktr_ctx - > patch_romfs_ctx . file_table_size : shared_data - > romfs_ctx - > file_table_size ) ;
2020-07-29 17:02:21 -04:00
RomFileSystemFileEntry * file_entry = NULL ;
char path [ FS_MAX_PATH ] = { 0 } ;
2020-08-19 13:20:26 -04:00
while ( file_table_offset < file_table_size )
2020-07-29 17:02:21 -04:00
{
/* Check if the transfer has been cancelled by the user */
if ( shared_data - > transfer_cancelled )
{
condvarWakeAll ( & g_writeCondvar ) ;
break ;
}
/* Retrieve RomFS file entry information */
2020-08-19 13:20:26 -04:00
if ( shared_data - > bktr_ctx )
{
shared_data - > read_error = ( ! ( file_entry = bktrGetFileEntryByOffset ( shared_data - > bktr_ctx , file_table_offset ) ) | | \
! bktrGeneratePathFromFileEntry ( shared_data - > bktr_ctx , file_entry , path , FS_MAX_PATH , RomFileSystemPathIllegalCharReplaceType_IllegalFsChars ) ) ;
} else {
shared_data - > read_error = ( ! ( file_entry = romfsGetFileEntryByOffset ( shared_data - > romfs_ctx , file_table_offset ) ) | | \
! romfsGeneratePathFromFileEntry ( shared_data - > romfs_ctx , file_entry , path , FS_MAX_PATH , RomFileSystemPathIllegalCharReplaceType_IllegalFsChars ) ) ;
}
2020-07-29 17:02:21 -04:00
if ( shared_data - > read_error )
{
condvarWakeAll ( & g_writeCondvar ) ;
break ;
}
/* Wait until the previous file data chunk has been written */
mutexLock ( & g_fileMutex ) ;
if ( shared_data - > data_size & & ! shared_data - > write_error ) condvarWait ( & g_readCondvar , & g_fileMutex ) ;
mutexUnlock ( & g_fileMutex ) ;
if ( shared_data - > write_error ) break ;
/* Send current file properties */
2020-10-26 02:39:33 -04:00
shared_data - > read_error = ! usbSendFilePropertiesCommon ( file_entry - > size , path ) ;
2020-07-29 17:02:21 -04:00
if ( shared_data - > read_error )
{
condvarWakeAll ( & g_writeCondvar ) ;
break ;
}
2020-08-19 13:20:26 -04:00
for ( u64 offset = 0 , blksize = BLOCK_SIZE ; offset < file_entry - > size ; offset + = blksize )
2020-07-29 17:02:21 -04:00
{
if ( blksize > ( file_entry - > size - offset ) ) blksize = ( file_entry - > size - offset ) ;
/* Check if the transfer has been cancelled by the user */
if ( shared_data - > transfer_cancelled )
{
condvarWakeAll ( & g_writeCondvar ) ;
break ;
}
/* Read current file data chunk */
2020-08-19 13:20:26 -04:00
shared_data - > read_error = ( shared_data - > bktr_ctx ? ! bktrReadFileEntryData ( shared_data - > bktr_ctx , file_entry , buf , blksize , offset ) : \
! romfsReadFileEntryData ( shared_data - > romfs_ctx , file_entry , buf , blksize , offset ) ) ;
2020-07-29 17:02:21 -04:00
if ( shared_data - > read_error )
{
condvarWakeAll ( & g_writeCondvar ) ;
break ;
}
/* Wait until the previous file data chunk has been written */
mutexLock ( & g_fileMutex ) ;
if ( shared_data - > data_size & & ! shared_data - > write_error ) condvarWait ( & g_readCondvar , & g_fileMutex ) ;
if ( shared_data - > write_error )
{
mutexUnlock ( & g_fileMutex ) ;
break ;
}
/* Copy current file data chunk to the shared buffer */
memcpy ( shared_data - > data , buf , blksize ) ;
shared_data - > data_size = blksize ;
/* Wake up the write thread to continue writing data */
mutexUnlock ( & g_fileMutex ) ;
condvarWakeAll ( & g_writeCondvar ) ;
}
if ( shared_data - > read_error | | shared_data - > write_error | | shared_data - > transfer_cancelled ) break ;
file_table_offset + = ALIGN_UP ( sizeof ( RomFileSystemFileEntry ) + file_entry - > name_length , 4 ) ;
}
free ( buf ) ;
2020-08-18 01:04:13 -04:00
end :
threadExit ( ) ;
2020-07-29 17:02:21 -04:00
}
2020-08-18 01:04:13 -04:00
static void write_thread_func ( void * arg )
2020-07-29 17:02:21 -04:00
{
ThreadSharedData * shared_data = ( ThreadSharedData * ) arg ;
if ( ! shared_data | | ! shared_data - > data )
{
shared_data - > write_error = true ;
2020-08-18 01:04:13 -04:00
goto end ;
2020-07-29 17:02:21 -04:00
}
while ( shared_data - > data_written < shared_data - > total_size )
{
/* Wait until the current file data chunk has been read */
mutexLock ( & g_fileMutex ) ;
if ( ! shared_data - > data_size & & ! shared_data - > read_error ) condvarWait ( & g_writeCondvar , & g_fileMutex ) ;
if ( shared_data - > read_error | | shared_data - > transfer_cancelled )
{
2021-02-16 08:22:14 -04:00
if ( shared_data - > transfer_cancelled ) usbCancelFileTransfer ( ) ;
2020-07-29 17:02:21 -04:00
mutexUnlock ( & g_fileMutex ) ;
break ;
}
//shared_data->write_error = (fwrite(shared_data->data, 1, shared_data->data_size, shared_data->fileobj) != shared_data->data_size);
/* Write current file data chunk */
shared_data - > write_error = ! usbSendFileData ( shared_data - > data , shared_data - > data_size ) ;
if ( ! shared_data - > write_error )
{
shared_data - > data_written + = shared_data - > data_size ;
shared_data - > data_size = 0 ;
}
/* Wake up the read thread to continue reading data */
mutexUnlock ( & g_fileMutex ) ;
condvarWakeAll ( & g_readCondvar ) ;
if ( shared_data - > write_error ) break ;
}
2020-08-18 01:04:13 -04:00
end :
threadExit ( ) ;
2020-07-29 17:02:21 -04:00
}
2020-09-19 02:59:31 -04:00
u8 get_program_id_offset ( TitleInfo * info , u32 program_count )
{
if ( program_count < = 1 ) return 0 ;
u8 id_offset = 0 ;
u32 selected_idx = 0 , page_size = 30 , scroll = 0 ;
char nca_id_str [ 0x21 ] = { 0 } ;
2022-02-19 06:03:06 +01:00
bool applet_status = true ;
2020-09-19 02:59:31 -04:00
NcmContentInfo * * content_infos = calloc ( program_count , sizeof ( NcmContentInfo * ) ) ;
if ( ! content_infos ) return 0 ;
for ( u32 i = 0 , j = 0 ; i < info - > content_count & & j < program_count ; i + + )
{
if ( info - > content_infos [ i ] . content_type ! = NcmContentType_Program ) continue ;
content_infos [ j + + ] = & ( info - > content_infos [ i ] ) ;
}
2022-02-19 06:03:06 +01:00
while ( ( applet_status = appletMainLoop ( ) ) )
2020-09-19 02:59:31 -04:00
{
consoleClear ( ) ;
printf ( " select a program nca to dump the romfs from. \n \n " ) ;
for ( u32 i = scroll ; i < program_count ; i + + )
{
if ( i > = ( scroll + page_size ) ) break ;
2021-05-11 18:36:15 -04:00
utilsGenerateHexStringFromData ( nca_id_str , sizeof ( nca_id_str ) , content_infos [ i ] - > content_id . c , sizeof ( content_infos [ i ] - > content_id . c ) , false ) ;
2020-09-19 02:59:31 -04:00
printf ( " %s%s.nca (ID offset #%u) \n " , i = = selected_idx ? " -> " : " " , nca_id_str , content_infos [ i ] - > id_offset ) ;
}
printf ( " \n " ) ;
consoleUpdate ( NULL ) ;
u64 btn_down = 0 , btn_held = 0 ;
2022-02-19 06:03:06 +01:00
while ( ( applet_status = appletMainLoop ( ) ) )
2020-09-19 02:59:31 -04:00
{
2020-12-04 02:38:44 -04:00
utilsScanPads ( ) ;
btn_down = utilsGetButtonsDown ( ) ;
btn_held = utilsGetButtonsHeld ( ) ;
2020-09-19 02:59:31 -04:00
if ( btn_down | | btn_held ) break ;
}
2022-02-19 06:03:06 +01:00
if ( ! applet_status ) break ;
2020-12-04 02:38:44 -04:00
if ( btn_down & HidNpadButton_A )
2020-09-19 02:59:31 -04:00
{
id_offset = content_infos [ selected_idx ] - > id_offset ;
break ;
} else
2020-12-04 02:38:44 -04:00
if ( ( btn_down & HidNpadButton_Down ) | | ( btn_held & ( HidNpadButton_StickLDown | HidNpadButton_StickRDown ) ) )
2020-09-19 02:59:31 -04:00
{
selected_idx + + ;
if ( selected_idx > = program_count )
{
2020-12-04 02:38:44 -04:00
if ( btn_down & HidNpadButton_Down )
2020-09-19 02:59:31 -04:00
{
selected_idx = scroll = 0 ;
} else {
selected_idx = ( program_count - 1 ) ;
}
} else
if ( selected_idx > = ( scroll + ( page_size / 2 ) ) & & program_count > ( scroll + page_size ) )
{
scroll + + ;
}
} else
2020-12-04 02:38:44 -04:00
if ( ( btn_down & HidNpadButton_Up ) | | ( btn_held & ( HidNpadButton_StickLUp | HidNpadButton_StickRUp ) ) )
2020-09-19 02:59:31 -04:00
{
selected_idx - - ;
if ( selected_idx = = UINT32_MAX )
{
2020-12-04 02:38:44 -04:00
if ( btn_down & HidNpadButton_Up )
2020-09-19 02:59:31 -04:00
{
selected_idx = ( program_count - 1 ) ;
scroll = ( program_count > = page_size ? ( program_count - page_size ) : 0 ) ;
} else {
selected_idx = 0 ;
}
} else
if ( selected_idx < ( scroll + ( page_size / 2 ) ) & & scroll > 0 )
{
scroll - - ;
}
}
2020-12-04 02:38:44 -04:00
if ( btn_held & ( HidNpadButton_StickLDown | HidNpadButton_StickRDown | HidNpadButton_StickLUp | HidNpadButton_StickRUp ) ) svcSleepThread ( 50000000 ) ; // 50 ms
2020-09-19 02:59:31 -04:00
}
free ( content_infos ) ;
2022-02-19 06:03:06 +01:00
return ( applet_status ? id_offset : ( u8 ) program_count ) ;
2020-09-19 02:59:31 -04:00
}
2022-03-17 20:29:44 +01:00
static TitleInfo * get_latest_patch_info ( TitleInfo * patch_info )
{
if ( ! patch_info | | patch_info - > meta_key . type ! = NcmContentMetaType_Patch ) return NULL ;
TitleInfo * output = patch_info , * tmp = patch_info ;
u32 highest_version = patch_info - > version . value ;
while ( ( tmp = tmp - > next ) ! = NULL )
{
if ( tmp - > version . value > highest_version )
{
output = tmp ;
highest_version = output - > version . value ;
}
}
return output ;
}
2020-07-29 17:02:21 -04:00
int main ( int argc , char * argv [ ] )
{
int ret = 0 ;
2021-05-18 08:32:43 -04:00
if ( ! utilsInitializeResources ( argc , ( const char * * ) argv ) )
2020-07-29 17:02:21 -04:00
{
ret = - 1 ;
goto out ;
}
2021-04-25 19:10:34 -04:00
/* Configure input. */
/* Up to 8 different, full controller inputs. */
/* Individual Joy-Cons not supported. */
padConfigureInput ( 8 , HidNpadStyleSet_NpadFullCtrl ) ;
padInitializeWithMask ( & g_padState , 0x1000000FFUL ) ;
2021-03-08 10:44:11 -04:00
consoleInit ( NULL ) ;
2020-07-30 17:43:50 -04:00
u32 app_count = 0 ;
2020-07-29 17:02:21 -04:00
TitleApplicationMetadata * * app_metadata = NULL ;
TitleUserApplicationData user_app_data = { 0 } ;
2020-08-03 17:41:00 -04:00
u32 selected_idx = 0 , page_size = 30 , scroll = 0 ;
2022-02-19 06:03:06 +01:00
bool applet_status = true , exit_prompt = true ;
2020-07-30 17:43:50 -04:00
2020-07-29 17:02:21 -04:00
u8 * buf = NULL ;
NcaContext * base_nca_ctx = NULL , * update_nca_ctx = NULL ;
2020-08-19 13:20:26 -04:00
RomFileSystemContext romfs_ctx = { 0 } ;
2020-07-29 17:02:21 -04:00
BktrContext bktr_ctx = { 0 } ;
ThreadSharedData shared_data = { 0 } ;
2020-08-18 01:04:13 -04:00
Thread read_thread = { 0 } , write_thread = { 0 } ;
2020-07-29 17:02:21 -04:00
app_metadata = titleGetApplicationMetadataEntries ( false , & app_count ) ;
if ( ! app_metadata | | ! app_count )
{
consolePrint ( " app metadata failed \n " ) ;
goto out2 ;
}
consolePrint ( " app metadata succeeded \n " ) ;
2020-08-19 13:20:26 -04:00
buf = usbAllocatePageAlignedBuffer ( BLOCK_SIZE ) ;
2020-07-29 17:02:21 -04:00
if ( ! buf )
{
consolePrint ( " buf failed \n " ) ;
goto out2 ;
}
consolePrint ( " buf succeeded \n " ) ;
base_nca_ctx = calloc ( 1 , sizeof ( NcaContext ) ) ;
if ( ! base_nca_ctx )
{
consolePrint ( " base nca ctx buf failed \n " ) ;
goto out2 ;
}
consolePrint ( " base nca ctx buf succeeded \n " ) ;
update_nca_ctx = calloc ( 1 , sizeof ( NcaContext ) ) ;
if ( ! update_nca_ctx )
{
consolePrint ( " update nca ctx buf failed \n " ) ;
goto out2 ;
}
consolePrint ( " update nca ctx buf succeeded \n " ) ;
utilsSleep ( 1 ) ;
2022-02-19 06:03:06 +01:00
while ( ( applet_status = appletMainLoop ( ) ) )
2020-07-29 17:02:21 -04:00
{
consoleClear ( ) ;
2020-10-13 20:15:21 -04:00
printf ( " select a user application to dump its romfs. \n if an update is available, patch romfs data will be dumped instead. \n data will be transferred via usb. \n press b to exit. \n \n " ) ;
2020-10-10 06:10:50 -04:00
printf ( " title: %u / %u \n " , selected_idx + 1 , app_count ) ;
printf ( " selected title: %016lX - %s \n \n " , app_metadata [ selected_idx ] - > title_id , app_metadata [ selected_idx ] - > lang_entry . name ) ;
2020-07-29 17:02:21 -04:00
2020-08-03 17:41:00 -04:00
for ( u32 i = scroll ; i < app_count ; i + + )
2020-07-30 17:43:50 -04:00
{
2020-08-03 17:41:00 -04:00
if ( i > = ( scroll + page_size ) ) break ;
printf ( " %s%016lX - %s \n " , i = = selected_idx ? " -> " : " " , app_metadata [ i ] - > title_id , app_metadata [ i ] - > lang_entry . name ) ;
2020-07-30 17:43:50 -04:00
}
printf ( " \n " ) ;
2020-07-29 17:02:21 -04:00
2020-07-30 17:43:50 -04:00
consoleUpdate ( NULL ) ;
2020-07-29 17:02:21 -04:00
2020-07-30 17:43:50 -04:00
u64 btn_down = 0 , btn_held = 0 ;
2022-02-19 06:03:06 +01:00
while ( ( applet_status = appletMainLoop ( ) ) )
2020-07-29 17:02:21 -04:00
{
2020-12-04 02:38:44 -04:00
utilsScanPads ( ) ;
btn_down = utilsGetButtonsDown ( ) ;
btn_held = utilsGetButtonsHeld ( ) ;
2020-07-30 17:43:50 -04:00
if ( btn_down | | btn_held ) break ;
2020-07-29 17:02:21 -04:00
2020-07-30 17:43:50 -04:00
if ( titleIsGameCardInfoUpdated ( ) )
2020-07-29 17:02:21 -04:00
{
free ( app_metadata ) ;
app_metadata = titleGetApplicationMetadataEntries ( false , & app_count ) ;
if ( ! app_metadata )
{
consolePrint ( " \n app metadata failed \n " ) ;
goto out2 ;
}
2020-08-03 17:41:00 -04:00
selected_idx = scroll = 0 ;
2020-07-29 17:02:21 -04:00
break ;
}
}
2022-02-19 06:03:06 +01:00
if ( ! applet_status ) break ;
2020-12-04 02:38:44 -04:00
if ( btn_down & HidNpadButton_A )
2020-07-29 17:02:21 -04:00
{
2020-08-19 13:20:26 -04:00
if ( ! titleGetUserApplicationData ( app_metadata [ selected_idx ] - > title_id , & user_app_data ) | | ! user_app_data . app_info )
2020-07-29 17:02:21 -04:00
{
2020-08-19 13:20:26 -04:00
consolePrint ( " \n the selected title doesn't have available base content. \n " ) ;
2020-07-29 17:02:21 -04:00
utilsSleep ( 3 ) ;
2021-05-31 21:12:15 -04:00
titleFreeUserApplicationData ( & user_app_data ) ;
2020-07-29 17:02:21 -04:00
continue ;
}
break ;
} else
2020-12-04 02:38:44 -04:00
if ( ( btn_down & HidNpadButton_Down ) | | ( btn_held & ( HidNpadButton_StickLDown | HidNpadButton_StickRDown ) ) )
2020-07-29 17:02:21 -04:00
{
2020-07-30 17:43:50 -04:00
selected_idx + + ;
if ( selected_idx > = app_count )
2020-07-29 17:02:21 -04:00
{
2020-12-04 02:38:44 -04:00
if ( btn_down & HidNpadButton_Down )
2020-07-30 17:43:50 -04:00
{
2020-08-03 17:41:00 -04:00
selected_idx = scroll = 0 ;
2020-07-30 17:43:50 -04:00
} else {
selected_idx = ( app_count - 1 ) ;
}
} else
2020-08-03 17:41:00 -04:00
if ( selected_idx > = ( scroll + ( page_size / 2 ) ) & & app_count > ( scroll + page_size ) )
2020-07-30 17:43:50 -04:00
{
2020-08-03 17:41:00 -04:00
scroll + + ;
2020-07-29 17:02:21 -04:00
}
} else
2020-12-04 02:38:44 -04:00
if ( ( btn_down & HidNpadButton_Up ) | | ( btn_held & ( HidNpadButton_StickLUp | HidNpadButton_StickRUp ) ) )
2020-07-29 17:02:21 -04:00
{
2020-07-30 17:43:50 -04:00
selected_idx - - ;
if ( selected_idx = = UINT32_MAX )
2020-07-29 17:02:21 -04:00
{
2020-12-04 02:38:44 -04:00
if ( btn_down & HidNpadButton_Up )
2020-07-30 17:43:50 -04:00
{
selected_idx = ( app_count - 1 ) ;
2020-08-03 17:41:00 -04:00
scroll = ( app_count > = page_size ? ( app_count - page_size ) : 0 ) ;
2020-07-30 17:43:50 -04:00
} else {
selected_idx = 0 ;
}
} else
2020-08-03 17:41:00 -04:00
if ( selected_idx < ( scroll + ( page_size / 2 ) ) & & scroll > 0 )
2020-07-30 17:43:50 -04:00
{
2020-08-03 17:41:00 -04:00
scroll - - ;
2020-07-29 17:02:21 -04:00
}
} else
2020-12-04 02:38:44 -04:00
if ( btn_down & HidNpadButton_B )
2020-07-29 17:02:21 -04:00
{
2020-07-30 17:43:50 -04:00
exit_prompt = false ;
2020-07-29 17:02:21 -04:00
goto out2 ;
}
2020-07-30 17:43:50 -04:00
2020-12-04 02:38:44 -04:00
if ( btn_held & ( HidNpadButton_StickLDown | HidNpadButton_StickRDown | HidNpadButton_StickLUp | HidNpadButton_StickRUp ) ) svcSleepThread ( 50000000 ) ; // 50 ms
2020-07-29 17:02:21 -04:00
}
2022-02-19 06:03:06 +01:00
if ( ! applet_status )
{
exit_prompt = false ;
goto out2 ;
}
2020-09-19 02:59:31 -04:00
u32 program_count = titleGetContentCountByType ( user_app_data . app_info , NcmContentType_Program ) ;
if ( ! program_count )
{
consolePrint ( " base app has no program ncas! \n " ) ;
goto out2 ;
}
u8 program_id_offset = get_program_id_offset ( user_app_data . app_info , program_count ) ;
2022-02-19 06:03:06 +01:00
if ( program_id_offset > = program_count )
{
exit_prompt = false ;
goto out2 ;
}
2020-09-19 02:59:31 -04:00
2020-07-29 17:02:21 -04:00
consoleClear ( ) ;
2020-09-19 02:59:31 -04:00
consolePrint ( " selected title: \n %s (%016lX) \n \n " , app_metadata [ selected_idx ] - > lang_entry . name , app_metadata [ selected_idx ] - > title_id + program_id_offset ) ;
2020-07-29 17:02:21 -04:00
if ( ! ncaInitializeContext ( base_nca_ctx , user_app_data . app_info - > storage_id , ( user_app_data . app_info - > storage_id = = NcmStorageId_GameCard ? GameCardHashFileSystemPartitionType_Secure : 0 ) , \
2020-10-14 09:23:49 -04:00
titleGetContentInfoByTypeAndIdOffset ( user_app_data . app_info , NcmContentType_Program , program_id_offset ) , NULL ) )
2020-07-29 17:02:21 -04:00
{
consolePrint ( " nca initialize base ctx failed \n " ) ;
goto out2 ;
}
2020-08-19 13:20:26 -04:00
if ( user_app_data . patch_info )
2020-07-29 17:02:21 -04:00
{
2022-03-17 20:29:44 +01:00
TitleInfo * latest_patch = get_latest_patch_info ( user_app_data . patch_info ) ;
if ( ! ncaInitializeContext ( update_nca_ctx , latest_patch - > storage_id , ( latest_patch - > storage_id = = NcmStorageId_GameCard ? GameCardHashFileSystemPartitionType_Secure : 0 ) , \
titleGetContentInfoByTypeAndIdOffset ( latest_patch , NcmContentType_Program , program_id_offset ) , NULL ) )
2020-08-19 13:20:26 -04:00
{
consolePrint ( " nca initialize update ctx failed \n " ) ;
goto out2 ;
}
2020-10-13 10:00:03 -04:00
if ( ! bktrInitializeContext ( & bktr_ctx , & ( base_nca_ctx - > fs_ctx [ 1 ] ) , & ( update_nca_ctx - > fs_ctx [ 1 ] ) ) )
2020-08-19 13:20:26 -04:00
{
consolePrint ( " bktr initialize ctx failed \n " ) ;
goto out2 ;
}
shared_data . bktr_ctx = & bktr_ctx ;
bktrGetTotalDataSize ( & bktr_ctx , & ( shared_data . total_size ) ) ;
consolePrint ( " bktr initialize ctx succeeded \n " ) ;
} else {
2020-10-13 10:00:03 -04:00
if ( ! romfsInitializeContext ( & romfs_ctx , & ( base_nca_ctx - > fs_ctx [ 1 ] ) ) )
2020-08-19 13:20:26 -04:00
{
consolePrint ( " romfs initialize ctx failed \n " ) ;
goto out2 ;
}
shared_data . romfs_ctx = & romfs_ctx ;
romfsGetTotalDataSize ( & romfs_ctx , & ( shared_data . total_size ) ) ;
consolePrint ( " romfs initialize ctx succeeded \n " ) ;
2020-07-29 17:02:21 -04:00
}
shared_data . data = buf ;
shared_data . data_size = 0 ;
shared_data . data_written = 0 ;
consolePrint ( " waiting for usb connection... " ) ;
time_t start = time ( NULL ) ;
2021-06-25 17:05:02 -04:00
u8 usb_host_speed = UsbHostSpeed_None ;
2020-07-29 17:02:21 -04:00
2022-02-19 06:03:06 +01:00
while ( ( applet_status = appletMainLoop ( ) ) )
2020-07-29 17:02:21 -04:00
{
time_t now = time ( NULL ) ;
if ( ( now - start ) > = 10 ) break ;
consolePrint ( " %lu " , now - start ) ;
2021-06-25 17:05:02 -04:00
if ( ( usb_host_speed = usbIsReady ( ) ) ) break ;
2020-07-29 17:02:21 -04:00
utilsSleep ( 1 ) ;
}
2022-02-19 06:03:06 +01:00
if ( ! applet_status )
{
exit_prompt = false ;
goto out2 ;
}
2020-07-29 17:02:21 -04:00
consolePrint ( " \n " ) ;
2021-06-25 17:05:02 -04:00
if ( ! usb_host_speed )
2020-07-29 17:02:21 -04:00
{
consolePrint ( " usb connection failed \n " ) ;
goto out2 ;
}
consolePrint ( " creating threads \n " ) ;
2020-08-18 01:04:13 -04:00
utilsCreateThread ( & read_thread , read_thread_func , & shared_data , 2 ) ;
utilsCreateThread ( & write_thread , write_thread_func , & shared_data , 2 ) ;
2020-07-29 17:02:21 -04:00
u8 prev_time = 0 ;
u64 prev_size = 0 ;
u8 percent = 0 ;
time_t btn_cancel_start_tmr = 0 , btn_cancel_end_tmr = 0 ;
bool btn_cancel_cur_state = false , btn_cancel_prev_state = false ;
2021-08-03 02:37:04 -04:00
utilsSetLongRunningProcessState ( true ) ;
2020-08-03 14:13:24 -04:00
2020-07-29 17:02:21 -04:00
consolePrint ( " hold b to cancel \n \n " ) ;
start = time ( NULL ) ;
while ( shared_data . data_written < shared_data . total_size )
{
if ( shared_data . read_error | | shared_data . write_error ) break ;
2022-02-06 03:30:42 +01:00
struct tm ts = { 0 } ;
2020-07-29 17:02:21 -04:00
time_t now = time ( NULL ) ;
2022-02-06 03:30:42 +01:00
localtime_r ( & now , & ts ) ;
2020-07-29 17:02:21 -04:00
size_t size = shared_data . data_written ;
2020-12-04 02:38:44 -04:00
utilsScanPads ( ) ;
btn_cancel_cur_state = ( utilsGetButtonsHeld ( ) & HidNpadButton_B ) ;
2020-07-29 17:02:21 -04:00
if ( btn_cancel_cur_state & & btn_cancel_cur_state ! = btn_cancel_prev_state )
{
btn_cancel_start_tmr = now ;
} else
if ( btn_cancel_cur_state & & btn_cancel_cur_state = = btn_cancel_prev_state )
{
btn_cancel_end_tmr = now ;
if ( ( btn_cancel_end_tmr - btn_cancel_start_tmr ) > = 3 )
{
mutexLock ( & g_fileMutex ) ;
shared_data . transfer_cancelled = true ;
mutexUnlock ( & g_fileMutex ) ;
break ;
}
} else {
btn_cancel_start_tmr = btn_cancel_end_tmr = 0 ;
}
btn_cancel_prev_state = btn_cancel_cur_state ;
2022-02-06 03:30:42 +01:00
if ( prev_time = = ts . tm_sec | | prev_size = = size ) continue ;
2020-07-29 17:02:21 -04:00
percent = ( u8 ) ( ( size * 100 ) / shared_data . total_size ) ;
2022-02-06 03:30:42 +01:00
prev_time = ts . tm_sec ;
2020-07-29 17:02:21 -04:00
prev_size = size ;
printf ( " %lu / %lu (%u%%) | Time elapsed: %lu \n " , size , shared_data . total_size , percent , ( now - start ) ) ;
consoleUpdate ( NULL ) ;
}
start = ( time ( NULL ) - start ) ;
consolePrint ( " \n waiting for threads to join \n " ) ;
2020-08-18 01:04:13 -04:00
utilsJoinThread ( & read_thread ) ;
2020-07-29 17:02:21 -04:00
consolePrint ( " read_thread done: %lu \n " , time ( NULL ) ) ;
2020-08-18 01:04:13 -04:00
utilsJoinThread ( & write_thread ) ;
2020-07-29 17:02:21 -04:00
consolePrint ( " write_thread done: %lu \n " , time ( NULL ) ) ;
2021-08-03 02:37:04 -04:00
utilsSetLongRunningProcessState ( false ) ;
2020-08-03 14:13:24 -04:00
2020-07-29 17:02:21 -04:00
if ( shared_data . read_error | | shared_data . write_error )
{
consolePrint ( " usb transfer error \n " ) ;
goto out2 ;
}
if ( shared_data . transfer_cancelled )
{
consolePrint ( " process cancelled \n " ) ;
goto out2 ;
}
consolePrint ( " process completed in %lu seconds \n " , start ) ;
out2 :
2020-07-30 17:43:50 -04:00
if ( exit_prompt )
{
consolePrint ( " press any button to exit \n " ) ;
2020-12-04 02:38:44 -04:00
utilsWaitForButtonPress ( 0 ) ;
2020-07-30 17:43:50 -04:00
}
2020-07-29 17:02:21 -04:00
2020-08-19 13:20:26 -04:00
romfsFreeContext ( & romfs_ctx ) ;
2020-07-29 17:02:21 -04:00
bktrFreeContext ( & bktr_ctx ) ;
if ( update_nca_ctx ) free ( update_nca_ctx ) ;
if ( base_nca_ctx ) free ( base_nca_ctx ) ;
2021-05-31 21:12:15 -04:00
titleFreeUserApplicationData ( & user_app_data ) ;
2020-07-29 17:02:21 -04:00
if ( buf ) free ( buf ) ;
if ( app_metadata ) free ( app_metadata ) ;
out :
utilsCloseResources ( ) ;
consoleExit ( NULL ) ;
return ret ;
}