From 278afbabfacb090481a7a54e6d587290bb70a984 Mon Sep 17 00:00:00 2001 From: Pablo Curiel Date: Tue, 27 Oct 2020 17:23:19 -0400 Subject: [PATCH] Title handler bugfixes. * Fixed possible invalid memory accesses using TitleInfo pointers from gamecard titles if the gamecard they belong to is taken out. * Fixed invalid for loop condition in _titleGetInfoFromStorageByTitleId() if no gamecard titles have been retrieved. --- code_templates/nsp_dumper_sd.c | 20 --------- code_templates/nsp_dumper_usb.c | 20 --------- source/title.c | 74 ++++++++++++++++++++++++++------- source/title.h | 5 ++- 4 files changed, 63 insertions(+), 56 deletions(-) diff --git a/code_templates/nsp_dumper_sd.c b/code_templates/nsp_dumper_sd.c index 4a4365b..9a65ace 100644 --- a/code_templates/nsp_dumper_sd.c +++ b/code_templates/nsp_dumper_sd.c @@ -751,26 +751,6 @@ int main(int argc, char *argv[]) { consoleClear(); - if (menu == 2 && titleIsGameCardInfoUpdated()) - { - free(app_metadata); - - app_metadata = titleGetApplicationMetadataEntries(false, &app_count); - if (!app_metadata) - { - consolePrint("\napp metadata failed\n"); - goto out2; - } - - menu = selected_idx = scroll = 0; - - title_idx = title_scroll = 0; - type_idx = type_scroll = 0; - list_count = list_idx = 0; - - continue; - } - printf("press b to %s.\n", menu == 0 ? "exit" : "go back"); printf("______________________________\n\n"); diff --git a/code_templates/nsp_dumper_usb.c b/code_templates/nsp_dumper_usb.c index 35e9699..50278e2 100644 --- a/code_templates/nsp_dumper_usb.c +++ b/code_templates/nsp_dumper_usb.c @@ -796,26 +796,6 @@ int main(int argc, char *argv[]) { consoleClear(); - if (menu == 2 && titleIsGameCardInfoUpdated()) - { - free(app_metadata); - - app_metadata = titleGetApplicationMetadataEntries(false, &app_count); - if (!app_metadata) - { - consolePrint("\napp metadata failed\n"); - goto out2; - } - - menu = selected_idx = scroll = 0; - - title_idx = title_scroll = 0; - type_idx = type_scroll = 0; - list_count = list_idx = 0; - - continue; - } - printf("press b to %s.\n", menu == 0 ? "exit" : "go back"); printf("______________________________\n\n"); diff --git a/source/title.c b/source/title.c index af856f8..731efe9 100644 --- a/source/title.c +++ b/source/title.c @@ -35,7 +35,8 @@ typedef struct { static Mutex g_titleMutex = 0; static Thread g_titleGameCardInfoThread = {0}; -static UEvent g_titleGameCardInfoThreadExitEvent = {0}, *g_titleGameCardStatusChangeUserEvent = NULL; +static UEvent g_titleGameCardInfoThreadExitEvent = {0}, *g_titleGameCardStatusChangeUserEvent = NULL, g_titleGameCardUpdateInfoUserEvent = {0}; +static CondVar g_gameCardCondVar = 0; static bool g_titleInterfaceInit = false, g_titleGameCardInfoThreadCreated = false, g_titleGameCardAvailable = false, g_titleGameCardInfoUpdated = false; @@ -477,6 +478,9 @@ bool titleInitialize(void) goto end; } + /* Create usermode gamecard update info event. */ + ueventCreate(&g_titleGameCardUpdateInfoUserEvent, true); + /* Create gamecard title info thread. */ if (!(g_titleGameCardInfoThreadCreated = titleCreateGameCardInfoThread())) goto end; @@ -728,8 +732,22 @@ end: bool titleIsGameCardInfoUpdated(void) { mutexLock(&g_titleMutex); + + /* Check if the gamecard thread detected a gamecard status change. */ bool ret = g_titleGameCardInfoUpdated; - if (ret) g_titleGameCardInfoUpdated = false; /* Update flag to avoid updating application metadata entries in the caller function if it's not needed. */ + if (!ret) goto end; + + /* Signal the gamecard update info user event. */ + ueventSignal(&g_titleGameCardUpdateInfoUserEvent); + + /* Wait for the gamecard thread to wakes us up. */ + condvarWait(&g_gameCardCondVar, &g_titleMutex); + + /* Update output value and gamecard info updated flag (if needed). */ + ret = g_titleGameCardInfoUpdated; + if (ret) g_titleGameCardInfoUpdated = false; + +end: mutexUnlock(&g_titleMutex); return ret; } @@ -1258,8 +1276,6 @@ static bool titleLoadPersistentStorageTitleInfo(void) /* Return right away if title info has already been retrieved. */ if (g_titleInfo || g_titleInfoCount) return true; - g_titleInfoCount = 0; - for(u8 i = NcmStorageId_BuiltInSystem; i <= NcmStorageId_SdCard; i++) { /* Retrieve content meta keys from the current storage. */ @@ -1397,6 +1413,11 @@ static bool titleRetrieveContentMetaKeysFromDatabase(u8 storage_id) /* Update title info count. */ g_titleInfoCount += total; + /* Update success flag. Nothing past this point will generate errors. */ + /* Also jump to the end of this function if we're dealing with eMMC System titles. */ + success = true; + if (storage_id == NcmStorageId_BuiltInSystem) goto end; + /* Update linked lists pointers for user applications, patches and add-on contents. */ g_titleInfoOrphanCount = 0; for(u32 i = 0; i < g_titleInfoCount; i++) @@ -1449,8 +1470,6 @@ static bool titleRetrieveContentMetaKeysFromDatabase(u8 storage_id) } } - success = true; - end: if (meta_keys) free(meta_keys); @@ -1559,14 +1578,11 @@ static void titleGameCardInfoThreadFunc(void *arg) Result rc = 0; int idx = 0; + bool first_run = true; Waiter gamecard_status_event_waiter = waiterForUEvent(g_titleGameCardStatusChangeUserEvent); Waiter exit_event_waiter = waiterForUEvent(&g_titleGameCardInfoThreadExitEvent); - - /* Initial gamecard title info retrieval. */ - mutexLock(&g_titleMutex); - titleRefreshGameCardTitleInfo(); - mutexUnlock(&g_titleMutex); + Waiter update_info_waiter = waiterForUEvent(&g_titleGameCardUpdateInfoUserEvent); while(true) { @@ -1577,10 +1593,40 @@ static void titleGameCardInfoThreadFunc(void *arg) /* Exit event triggered. */ if (idx == 1) break; + if (!first_run) + { + /* Update gamecard info updated flag. */ + mutexLock(&g_titleMutex); + g_titleGameCardInfoUpdated = true; + mutexUnlock(&g_titleMutex); + + /* Wait until another function signals us (titleIsGameCardInfoUpdated() or titleExit()). */ + rc = waitMulti(&idx, -1, update_info_waiter, exit_event_waiter); + if (R_FAILED(rc)) + { + mutexLock(&g_titleMutex); + g_titleGameCardInfoUpdated = false; + mutexUnlock(&g_titleMutex); + continue; + } + + /* Exit event triggered. */ + if (idx == 1) break; + } + /* Update gamecard title info. */ mutexLock(&g_titleMutex); - g_titleGameCardInfoUpdated = titleRefreshGameCardTitleInfo(); + g_titleGameCardInfoUpdated = (titleRefreshGameCardTitleInfo() && !first_run); mutexUnlock(&g_titleMutex); + + if (first_run) + { + /* Disable first run flag. */ + first_run = false; + } else { + /* Wake up titleIsGameCardInfoUpdated(). */ + condvarWakeAll(&g_gameCardCondVar); + } } /* Update gamecard flags. */ @@ -1791,7 +1837,7 @@ static TitleInfo *_titleGetInfoFromStorageByTitleId(u8 storage_id, u64 title_id, /* Speed up gamecard lookups. */ u32 start_idx = (storage_id == NcmStorageId_GameCard ? g_titleInfoGameCardStartIndex : 0); - u32 max_val = ((storage_id == NcmStorageId_GameCard || storage_id == NcmStorageId_Any) ? g_titleInfoCount : g_titleInfoGameCardStartIndex); + u32 max_val = ((storage_id == NcmStorageId_GameCard || storage_id == NcmStorageId_Any) ? g_titleInfoCount : (g_titleInfoGameCardCount ? g_titleInfoGameCardStartIndex : g_titleInfoCount)); for(u32 i = start_idx; i < max_val; i++) { @@ -1804,7 +1850,7 @@ static TitleInfo *_titleGetInfoFromStorageByTitleId(u8 storage_id, u64 title_id, } } - //if (!info) LOGFILE("Unable to find TitleInfo entry with ID \"%016lX\"! (storage ID %u).", title_id, storage_id); + if (!info && lock) LOGFILE("Unable to find TitleInfo entry with ID \"%016lX\"! (storage ID %u).", title_id, storage_id); end: if (lock) mutexUnlock(&g_titleMutex); diff --git a/source/title.h b/source/title.h index 61169ee..89010dc 100644 --- a/source/title.h +++ b/source/title.h @@ -108,8 +108,9 @@ bool titleAreOrphanTitlesAvailable(void); /// Returns NULL if an error occurs. TitleInfo **titleGetInfoFromOrphanTitles(u32 *out_count); -/// Returns true if the gamecard title info entries have been updated (e.g. after a new gamecard has been inserted, of after the current one has been taken out). -/// If titleGetApplicationMetadataEntries() has been previously called, its returned buffer should be freed and a new titleGetApplicationMetadataEntries() call should be issued. +/// Checks if a gamecard status update has been detected by the background gamecard title info thread (e.g. after a new gamecard has been inserted, of after the current one has been taken out). +/// If so, gamecard title info entries will be updated or freed during this call, depending on the current gamecard status. +/// If this function returns true and titleGetApplicationMetadataEntries() has been previously called, its returned buffer should be freed and titleGetApplicationMetadataEntries() should be called again. bool titleIsGameCardInfoUpdated(void); /// Returns a pointer to a dynamically allocated buffer that holds a filename string suitable for output title dumps.