mirror of
https://github.com/DarkMatterCore/nxdumptool.git
synced 2025-01-11 18:26:13 +00:00
b4bf505d4f
* Temporarily changed tracked libusbhsfs branch from main to dev. * Minor improvements in ums.c/h. * Refactored nsp_dumper_sd into nsp_dumper_stor - now capable of dumping NSPs to a mounted partition from a UMS device. * Fixed utilsWaitForButtonPress() when no button mask is provided. * Refactored utilsGetFreeSpaceFromFileSystemByPath() into utilsGetFileSystemStatsByPath().
257 lines
7 KiB
C
257 lines
7 KiB
C
/*
|
|
* ums.c
|
|
*
|
|
* Copyright (c) 2020-2021, DarkMatterCore <pabloacurielz@gmail.com>.
|
|
*
|
|
* This file is part of nxdumptool (https://github.com/DarkMatterCore/nxdumptool).
|
|
*
|
|
* nxdumptool is free software; you can redistribute it and/or modify it
|
|
* under the terms and conditions of the GNU General Public License,
|
|
* version 2, as published by the Free Software Foundation.
|
|
*
|
|
* nxdumptool is distributed in the hope 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.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "utils.h"
|
|
|
|
/* Global variables. */
|
|
|
|
static Mutex g_umsMutex = 0;
|
|
static bool g_umsInterfaceInit = false;
|
|
|
|
static Thread g_umsDetectionThread = {0};
|
|
static UEvent *g_umsStatusChangeEvent = NULL, g_umsDetectionThreadExitEvent = {0};
|
|
static bool g_umsDetectionThreadCreated = false, g_umsDeviceInfoUpdated = false;
|
|
|
|
static u32 g_umsDeviceCount = 0;
|
|
static UsbHsFsDevice *g_umsDevices = NULL;
|
|
|
|
/* Function prototypes. */
|
|
|
|
static bool umsCreateDetectionThread(void);
|
|
static void umsDestroyDetectionThread(void);
|
|
static void umsDetectionThreadFunc(void *arg);
|
|
|
|
static void umsFreeDeviceData(void);
|
|
|
|
bool umsInitialize(void)
|
|
{
|
|
mutexLock(&g_umsMutex);
|
|
|
|
Result rc = 0;
|
|
|
|
bool ret = g_umsInterfaceInit;
|
|
if (ret) goto end;
|
|
|
|
/* Initialize USB Mass Storage Host interface. */
|
|
rc = usbHsFsInitialize(0);
|
|
if (R_FAILED(rc))
|
|
{
|
|
LOGFILE("usbHsFsInitialize failed! (0x%08X).", rc);
|
|
goto end;
|
|
}
|
|
|
|
/* Get USB Mass Storage status change event. */
|
|
g_umsStatusChangeEvent = usbHsFsGetStatusChangeUserEvent();
|
|
|
|
/* Create user-mode exit event. */
|
|
ueventCreate(&g_umsDetectionThreadExitEvent, true);
|
|
|
|
/* Create USB Mass Storage detection thread. */
|
|
if (!(g_umsDetectionThreadCreated = umsCreateDetectionThread())) goto end;
|
|
|
|
ret = g_umsInterfaceInit = true;
|
|
|
|
end:
|
|
mutexUnlock(&g_umsMutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void umsExit(void)
|
|
{
|
|
mutexLock(&g_umsMutex);
|
|
|
|
if (!g_umsInterfaceInit) goto end;
|
|
|
|
/* Destroy USB Mass Storage detection thread. */
|
|
if (g_umsDetectionThreadCreated)
|
|
{
|
|
umsDestroyDetectionThread();
|
|
g_umsDetectionThreadCreated = false;
|
|
}
|
|
|
|
/* Close USB Mass Storage Host interface. */
|
|
usbHsFsExit();
|
|
|
|
g_umsInterfaceInit = false;
|
|
|
|
end:
|
|
mutexUnlock(&g_umsMutex);
|
|
}
|
|
|
|
bool umsIsDeviceInfoUpdated(void)
|
|
{
|
|
mutexLock(&g_umsMutex);
|
|
|
|
bool ret = false;
|
|
|
|
if (g_umsInterfaceInit && g_umsDeviceInfoUpdated)
|
|
{
|
|
ret = true;
|
|
g_umsDeviceInfoUpdated = false;
|
|
}
|
|
|
|
mutexUnlock(&g_umsMutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
UsbHsFsDevice *umsGetDevices(u32 *out_count)
|
|
{
|
|
mutexLock(&g_umsMutex);
|
|
|
|
UsbHsFsDevice *devices = NULL;
|
|
|
|
if (!g_umsInterfaceInit || !out_count)
|
|
{
|
|
LOGFILE("Invalid parameters!");
|
|
goto end;
|
|
}
|
|
|
|
if (g_umsDeviceCount && g_umsDevices)
|
|
{
|
|
/* Allocate memory for the output devices. */
|
|
devices = calloc(g_umsDeviceCount, sizeof(UsbHsFsDevice));
|
|
if (!devices)
|
|
{
|
|
LOGFILE("Failed to allocate memory for %u devices!", g_umsDeviceCount);
|
|
goto end;
|
|
}
|
|
|
|
/* Copy device data. */
|
|
memcpy(devices, g_umsDevices, g_umsDeviceCount * sizeof(UsbHsFsDevice));
|
|
}
|
|
|
|
/* Update output device count. */
|
|
*out_count = ((g_umsDeviceCount && g_umsDevices) ? g_umsDeviceCount : 0);
|
|
|
|
end:
|
|
mutexUnlock(&g_umsMutex);
|
|
|
|
return devices;
|
|
}
|
|
|
|
static bool umsCreateDetectionThread(void)
|
|
{
|
|
if (!utilsCreateThread(&g_umsDetectionThread, umsDetectionThreadFunc, NULL, 1))
|
|
{
|
|
LOGFILE("Failed to create USB Mass Storage detection thread!");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static void umsDestroyDetectionThread(void)
|
|
{
|
|
/* Signal the exit event to terminate the USB Mass Storage detection thread. */
|
|
ueventSignal(&g_umsDetectionThreadExitEvent);
|
|
|
|
/* Wait for the USB Mass Storage detection thread to exit. */
|
|
utilsJoinThread(&g_umsDetectionThread);
|
|
}
|
|
|
|
static void umsDetectionThreadFunc(void *arg)
|
|
{
|
|
(void)arg;
|
|
|
|
Result rc = 0;
|
|
int idx = 0;
|
|
u32 listed_device_count = 0;
|
|
|
|
Waiter status_change_event_waiter = waiterForUEvent(g_umsStatusChangeEvent);
|
|
Waiter exit_event_waiter = waiterForUEvent(&g_umsDetectionThreadExitEvent);
|
|
|
|
while(true)
|
|
{
|
|
/* Wait until an event is triggered. */
|
|
rc = waitMulti(&idx, -1, status_change_event_waiter, exit_event_waiter);
|
|
if (R_FAILED(rc)) continue;
|
|
|
|
/* Exit event triggered. */
|
|
if (idx == 1) break;
|
|
|
|
mutexLock(&g_umsMutex);
|
|
|
|
/* Free USB Mass Storage device data. */
|
|
umsFreeDeviceData();
|
|
|
|
/* Get mounted device count. */
|
|
g_umsDeviceCount = usbHsFsGetMountedDeviceCount();
|
|
LOGFILE("USB Mass Storage status change event triggered! Mounted USB Mass Storage device count: %u.", g_umsDeviceCount);
|
|
|
|
if (g_umsDeviceCount)
|
|
{
|
|
bool fail = false;
|
|
|
|
/* Allocate mounted devices buffer. */
|
|
g_umsDevices = calloc(g_umsDeviceCount, sizeof(UsbHsFsDevice));
|
|
if (g_umsDevices)
|
|
{
|
|
/* List mounted devices. */
|
|
listed_device_count = usbHsFsListMountedDevices(g_umsDevices, g_umsDeviceCount);
|
|
if (listed_device_count)
|
|
{
|
|
/* Check if we got as many devices as we expected. */
|
|
if (listed_device_count == g_umsDeviceCount)
|
|
{
|
|
/* Update USB Mass Storage device info updated flag. */
|
|
g_umsDeviceInfoUpdated = true;
|
|
} else {
|
|
LOGFILE("USB Mass Storage device count mismatch! (%u != %u).", listed_device_count, g_umsDeviceCount);
|
|
fail = true;
|
|
}
|
|
} else {
|
|
LOGFILE("Failed to list mounted USB Mass Storage devices!");
|
|
fail = true;
|
|
}
|
|
} else {
|
|
LOGFILE("Failed to allocate memory for mounted USB Mass Storage devices buffer!");
|
|
fail = true;
|
|
}
|
|
|
|
/* Free USB Mass Storage device data if something went wrong. */
|
|
if (fail) umsFreeDeviceData();
|
|
} else {
|
|
/* Update USB Mass Storage device info updated flag. */
|
|
g_umsDeviceInfoUpdated = true;
|
|
}
|
|
|
|
mutexUnlock(&g_umsMutex);
|
|
}
|
|
|
|
/* Free USB Mass Storage device data. */
|
|
umsFreeDeviceData();
|
|
|
|
threadExit();
|
|
}
|
|
|
|
static void umsFreeDeviceData(void)
|
|
{
|
|
/* Free devices buffer. */
|
|
if (g_umsDevices)
|
|
{
|
|
free(g_umsDevices);
|
|
g_umsDevices = NULL;
|
|
}
|
|
|
|
/* Reset device count. */
|
|
g_umsDeviceCount = 0;
|
|
}
|