kext: add hid/ir thread locking
This commit is contained in:
parent
e69f89a0d4
commit
991f51831d
@ -36,11 +36,12 @@ void executeFunctionOnCores(SGI0Handler_t func, u8 targetList, u8 targetListFilt
|
|||||||
|
|
||||||
void KScheduler__TriggerCrossCoreInterrupt(KScheduler *this);
|
void KScheduler__TriggerCrossCoreInterrupt(KScheduler *this);
|
||||||
void KThread__DebugReschedule(KThread *this, bool lock);
|
void KThread__DebugReschedule(KThread *this, bool lock);
|
||||||
bool rosalinaThreadLockPredicate(KThread *thread);
|
|
||||||
|
bool rosalinaThreadLockPredicate(KThread *thread, u32 mask);
|
||||||
void rosalinaRescheduleThread(KThread *thread, bool lock);
|
void rosalinaRescheduleThread(KThread *thread, bool lock);
|
||||||
void rosalinaLockThread(KThread *thread);
|
|
||||||
void rosalinaLockAllThreads(void);
|
void rosalinaLockThreads(u32 mask);
|
||||||
void rosalinaUnlockAllThreads(void);
|
void rosalinaUnlockThreads(u32 mask);
|
||||||
|
|
||||||
// Taken from ctrulib:
|
// Taken from ctrulib:
|
||||||
|
|
||||||
@ -49,6 +50,11 @@ static inline void __dsb(void)
|
|||||||
__asm__ __volatile__("mcr p15, 0, %[val], c7, c10, 4" :: [val] "r" (0) : "memory");
|
__asm__ __volatile__("mcr p15, 0, %[val], c7, c10, 4" :: [val] "r" (0) : "memory");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void __dmb(void)
|
||||||
|
{
|
||||||
|
__asm__ __volatile__("mcr p15, 0, %[val], c7, c10, 5" :: [val] "r" (0) : "memory");
|
||||||
|
}
|
||||||
|
|
||||||
static inline void __clrex(void)
|
static inline void __clrex(void)
|
||||||
{
|
{
|
||||||
__asm__ __volatile__("clrex" ::: "memory");
|
__asm__ __volatile__("clrex" ::: "memory");
|
||||||
|
@ -78,7 +78,7 @@ void signalSvcReturn(u8 *pageEnd)
|
|||||||
void postprocessSvc(void)
|
void postprocessSvc(void)
|
||||||
{
|
{
|
||||||
KThread *currentThread = currentCoreContext->objectContext.currentThread;
|
KThread *currentThread = currentCoreContext->objectContext.currentThread;
|
||||||
if(!currentThread->shallTerminate && rosalinaThreadLockPredicate(currentThread))
|
if(!currentThread->shallTerminate && rosalinaThreadLockPredicate(currentThread, rosalinaState & 5))
|
||||||
rosalinaRescheduleThread(currentThread, true);
|
rosalinaRescheduleThread(currentThread, true);
|
||||||
|
|
||||||
officialPostProcessSvc();
|
officialPostProcessSvc();
|
||||||
|
@ -104,14 +104,26 @@ Result KernelSetStateHook(u32 type, u32 varg1, u32 varg2, u32 varg3)
|
|||||||
__ldrex((s32 *)&rosalinaState);
|
__ldrex((s32 *)&rosalinaState);
|
||||||
}
|
}
|
||||||
while(__strex((s32 *)&rosalinaState, (s32)(rosalinaState ^ varg1)));
|
while(__strex((s32 *)&rosalinaState, (s32)(rosalinaState ^ varg1)));
|
||||||
|
__dmb();
|
||||||
|
|
||||||
if(rosalinaState & 2)
|
if(rosalinaState & 2)
|
||||||
hasStartedRosalinaNetworkFuncsOnce = true;
|
hasStartedRosalinaNetworkFuncsOnce = true;
|
||||||
|
|
||||||
if(rosalinaState & 1)
|
// 1: all applet/app/gsp/dsp... threads 4: hid/ir
|
||||||
rosalinaLockAllThreads();
|
if(varg1 & 1)
|
||||||
else if(varg1 & 1)
|
{
|
||||||
rosalinaUnlockAllThreads();
|
if (rosalinaState & 1)
|
||||||
|
rosalinaLockThreads(1);
|
||||||
|
else
|
||||||
|
rosalinaUnlockThreads(1);
|
||||||
|
}
|
||||||
|
if(varg1 & 4)
|
||||||
|
{
|
||||||
|
if (rosalinaState & 4)
|
||||||
|
rosalinaLockThreads(4);
|
||||||
|
else
|
||||||
|
rosalinaUnlockThreads(4);
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -66,17 +66,12 @@ void KThread__DebugReschedule(KThread *this, bool lock)
|
|||||||
KRecursiveLock__Unlock(criticalSectionLock);
|
KRecursiveLock__Unlock(criticalSectionLock);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool rosalinaThreadLockPredicate(KThread *thread)
|
static void rosalinaLockThread(KThread *thread)
|
||||||
{
|
{
|
||||||
KProcess *process = thread->ownerProcess;
|
KThread *syncThread = synchronizationMutex->owner;
|
||||||
if(process == NULL)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
u64 titleId = codeSetOfProcess(process)->titleId;
|
if(syncThread == NULL || syncThread != thread)
|
||||||
u32 highTitleId = (u32)(titleId >> 32), lowTitleId = (u32)(titleId & ~0xF0000001); // clear N3DS and SAFE_FIRM bits
|
rosalinaRescheduleThread(thread, true);
|
||||||
return
|
|
||||||
((rosalinaState & 1) && idOfProcess(process) >= nbSection0Modules &&
|
|
||||||
(highTitleId != 0x00040130 || (highTitleId == 0x00040130 && (lowTitleId == 0x1A02 || lowTitleId == 0x1C02))));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void rosalinaRescheduleThread(KThread *thread, bool lock)
|
void rosalinaRescheduleThread(KThread *thread, bool lock)
|
||||||
@ -89,20 +84,37 @@ void rosalinaRescheduleThread(KThread *thread, bool lock)
|
|||||||
else
|
else
|
||||||
thread->schedulingMask &= ~0x40;
|
thread->schedulingMask &= ~0x40;
|
||||||
|
|
||||||
KScheduler__AdjustThread(currentCoreContext->objectContext.currentScheduler, thread, oldSchedulingMask);
|
if (oldSchedulingMask != thread->schedulingMask)
|
||||||
|
KScheduler__AdjustThread(currentCoreContext->objectContext.currentScheduler, thread, oldSchedulingMask);
|
||||||
|
|
||||||
KRecursiveLock__Unlock(criticalSectionLock);
|
KRecursiveLock__Unlock(criticalSectionLock);
|
||||||
}
|
}
|
||||||
|
|
||||||
void rosalinaLockThread(KThread *thread)
|
bool rosalinaThreadLockPredicate(KThread *thread, u32 mask)
|
||||||
{
|
{
|
||||||
KThread *syncThread = synchronizationMutex->owner;
|
KProcess *process = thread->ownerProcess;
|
||||||
|
if(process == NULL || idOfProcess(process) < nbSection0Modules)
|
||||||
|
return false;
|
||||||
|
|
||||||
if(syncThread == NULL || syncThread != thread)
|
u64 titleId = codeSetOfProcess(process)->titleId;
|
||||||
rosalinaRescheduleThread(thread, true);
|
u32 highTitleId = (u32)(titleId >> 32), lowTitleId = (u32)(titleId & ~0xF0000001); // clear N3DS and SAFE_FIRM bits
|
||||||
|
|
||||||
|
if (mask & 1)
|
||||||
|
{
|
||||||
|
if (highTitleId != 0x00040130) // non-sysmodules
|
||||||
|
return true;
|
||||||
|
else
|
||||||
|
return lowTitleId == 0x1A02 || lowTitleId == 0x1C02 || lowTitleId == 0x2702; // dsp, gsp, csnd
|
||||||
|
}
|
||||||
|
if (mask & 4)
|
||||||
|
{
|
||||||
|
return lowTitleId == 0x1D02 || lowTitleId == 0x3302;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void rosalinaLockAllThreads(void)
|
void rosalinaLockThreads(u32 mask)
|
||||||
{
|
{
|
||||||
bool currentThreadsFound = false;
|
bool currentThreadsFound = false;
|
||||||
|
|
||||||
@ -110,7 +122,7 @@ void rosalinaLockAllThreads(void)
|
|||||||
for(KLinkedListNode *node = threadList->list.nodes.first; node != (KLinkedListNode *)&threadList->list.nodes; node = node->next)
|
for(KLinkedListNode *node = threadList->list.nodes.first; node != (KLinkedListNode *)&threadList->list.nodes; node = node->next)
|
||||||
{
|
{
|
||||||
KThread *thread = (KThread *)node->key;
|
KThread *thread = (KThread *)node->key;
|
||||||
if(!rosalinaThreadLockPredicate(thread))
|
if(!rosalinaThreadLockPredicate(thread, mask))
|
||||||
continue;
|
continue;
|
||||||
if(thread == coreCtxs[thread->coreId].objectContext.currentThread)
|
if(thread == coreCtxs[thread->coreId].objectContext.currentThread)
|
||||||
currentThreadsFound = true;
|
currentThreadsFound = true;
|
||||||
@ -123,7 +135,7 @@ void rosalinaLockAllThreads(void)
|
|||||||
for(KLinkedListNode *node = threadList->list.nodes.first; node != (KLinkedListNode *)&threadList->list.nodes; node = node->next)
|
for(KLinkedListNode *node = threadList->list.nodes.first; node != (KLinkedListNode *)&threadList->list.nodes; node = node->next)
|
||||||
{
|
{
|
||||||
KThread *thread = (KThread *)node->key;
|
KThread *thread = (KThread *)node->key;
|
||||||
if(!rosalinaThreadLockPredicate(thread))
|
if(!rosalinaThreadLockPredicate(thread, mask))
|
||||||
continue;
|
continue;
|
||||||
if(!(thread->schedulingMask & 0x40))
|
if(!(thread->schedulingMask & 0x40))
|
||||||
{
|
{
|
||||||
@ -145,7 +157,7 @@ void rosalinaLockAllThreads(void)
|
|||||||
KRecursiveLock__Unlock(criticalSectionLock);
|
KRecursiveLock__Unlock(criticalSectionLock);
|
||||||
}
|
}
|
||||||
|
|
||||||
void rosalinaUnlockAllThreads(void)
|
void rosalinaUnlockThreads(u32 mask)
|
||||||
{
|
{
|
||||||
for(KLinkedListNode *node = threadList->list.nodes.first; node != (KLinkedListNode *)&threadList->list.nodes; node = node->next)
|
for(KLinkedListNode *node = threadList->list.nodes.first; node != (KLinkedListNode *)&threadList->list.nodes; node = node->next)
|
||||||
{
|
{
|
||||||
@ -154,7 +166,7 @@ void rosalinaUnlockAllThreads(void)
|
|||||||
if((thread->schedulingMask & 0xF) == 2) // thread is terminating
|
if((thread->schedulingMask & 0xF) == 2) // thread is terminating
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if(thread->schedulingMask & 0x40)
|
if((thread->schedulingMask & 0x40) && rosalinaThreadLockPredicate(thread, mask))
|
||||||
rosalinaRescheduleThread(thread, false);
|
rosalinaRescheduleThread(thread, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user