mirror of
https://github.com/switchbrew/libnx.git
synced 2025-06-21 12:32:40 +02:00
tmem: Add tmemCloseHandle(), tmemWaitForPermission(), use those functions in nv.c to fix a race condition (#606)
This commit is contained in:
parent
db592d8aeb
commit
d219884bfa
@ -72,6 +72,21 @@ static inline void* tmemGetAddr(TransferMemory* t){
|
|||||||
return t->map_addr;
|
return t->map_addr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Closes handle of a transfer memory object.
|
||||||
|
* @param t Transfer memory information structure.
|
||||||
|
* @return Result code.
|
||||||
|
*/
|
||||||
|
Result tmemCloseHandle(TransferMemory* t);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Waits until source backing memory permissions match perm.
|
||||||
|
* @param t Transfer memory information structure.
|
||||||
|
* @param perm Permissions which the source backing memory is expected to have before return.
|
||||||
|
* @return Result code.
|
||||||
|
*/
|
||||||
|
Result tmemWaitForPermission(TransferMemory* t, Permission perm);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Frees up resources used by a transfer memory object, unmapping and closing handles, etc.
|
* @brief Frees up resources used by a transfer memory object, unmapping and closing handles, etc.
|
||||||
* @param t Transfer memory information structure.
|
* @param t Transfer memory information structure.
|
||||||
|
@ -100,6 +100,45 @@ Result tmemUnmap(TransferMemory* t)
|
|||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Result tmemCloseHandle(TransferMemory* t)
|
||||||
|
{
|
||||||
|
Result rc = 0;
|
||||||
|
|
||||||
|
if (t->handle != INVALID_HANDLE) {
|
||||||
|
rc = svcCloseHandle(t->handle);
|
||||||
|
t->handle = INVALID_HANDLE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
Result tmemWaitForPermission(TransferMemory* t, Permission perm)
|
||||||
|
{
|
||||||
|
Result rc = 0;
|
||||||
|
|
||||||
|
if ((t->perm & perm) != perm) {
|
||||||
|
MemoryInfo m = {0};
|
||||||
|
u32 p = 0;
|
||||||
|
rc = svcQueryMemory(&m, &p, (u64)(t->src_addr));
|
||||||
|
|
||||||
|
if (R_FAILED(rc)) {
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
while ((m.perm & perm) != perm) {
|
||||||
|
rc = svcQueryMemory(&m, &p, (u64)(t->src_addr));
|
||||||
|
|
||||||
|
if (R_FAILED(rc)) {
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
svcSleepThread(100000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
Result tmemClose(TransferMemory* t)
|
Result tmemClose(TransferMemory* t)
|
||||||
{
|
{
|
||||||
Result rc = 0;
|
Result rc = 0;
|
||||||
@ -109,16 +148,13 @@ Result tmemClose(TransferMemory* t)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (R_SUCCEEDED(rc)) {
|
if (R_SUCCEEDED(rc)) {
|
||||||
if (t->handle != INVALID_HANDLE) {
|
rc = tmemCloseHandle(t);
|
||||||
rc = svcCloseHandle(t->handle);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (t->src_addr != NULL) {
|
if (t->src_addr != NULL) {
|
||||||
__libnx_free(t->src_addr);
|
__libnx_free(t->src_addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
t->src_addr = NULL;
|
t->src_addr = NULL;
|
||||||
t->handle = INVALID_HANDLE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return rc;
|
return rc;
|
||||||
|
@ -55,6 +55,11 @@ Result _nvInitialize(void) {
|
|||||||
if (R_SUCCEEDED(rc))
|
if (R_SUCCEEDED(rc))
|
||||||
rc = _nvCmdInitialize(CUR_PROCESS_HANDLE, g_nvTransfermem.handle, tmem_size);
|
rc = _nvCmdInitialize(CUR_PROCESS_HANDLE, g_nvTransfermem.handle, tmem_size);
|
||||||
|
|
||||||
|
Result rc2 = tmemCloseHandle(&g_nvTransfermem);
|
||||||
|
|
||||||
|
if (R_SUCCEEDED(rc))
|
||||||
|
rc = rc2;
|
||||||
|
|
||||||
// Clone the session handle - the cloned session is used to execute certain commands in parallel
|
// Clone the session handle - the cloned session is used to execute certain commands in parallel
|
||||||
if (R_SUCCEEDED(rc))
|
if (R_SUCCEEDED(rc))
|
||||||
rc = serviceCloneEx(&g_nvSrv, 1, &g_nvSrvClone);
|
rc = serviceCloneEx(&g_nvSrv, 1, &g_nvSrvClone);
|
||||||
@ -73,6 +78,7 @@ Result _nvInitialize(void) {
|
|||||||
void _nvCleanup(void) {
|
void _nvCleanup(void) {
|
||||||
serviceClose(&g_nvSrvClone);
|
serviceClose(&g_nvSrvClone);
|
||||||
serviceClose(&g_nvSrv);
|
serviceClose(&g_nvSrv);
|
||||||
|
tmemWaitForPermission(&g_nvTransfermem, Perm_Rw);
|
||||||
tmemClose(&g_nvTransfermem);
|
tmemClose(&g_nvTransfermem);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user