mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-18 11:16:10 +00:00
thermosphere: make the pending virq list ordering stable
This commit is contained in:
parent
c42aef6ba7
commit
d56185e432
2 changed files with 88 additions and 22 deletions
|
@ -36,9 +36,23 @@
|
||||||
|
|
||||||
#define TEMPORARY __attribute__((section(".tempbss")))
|
#define TEMPORARY __attribute__((section(".tempbss")))
|
||||||
|
|
||||||
|
#define PANIC(...) do { DEBUG(__VA_ARGS__); panic(); } while (false)
|
||||||
|
|
||||||
#define FOREACH_BIT(tmpmsk, var, word) for (u64 tmpmsk = (word), var = __builtin_ctzll(word); tmpmsk != 0; tmpmsk &= ~BITL(var), var = __builtin_ctzll(tmpmsk))
|
#define FOREACH_BIT(tmpmsk, var, word) for (u64 tmpmsk = (word), var = __builtin_ctzll(word); tmpmsk != 0; tmpmsk &= ~BITL(var), var = __builtin_ctzll(tmpmsk))
|
||||||
|
|
||||||
#define PANIC(...) do { DEBUG(__VA_ARGS__); panic(); } while (false)
|
#define _DECLARE_ASM_ARITHMETIC_UNARY_HELPER(sz, regalloc, op)\
|
||||||
|
static inline u##sz __##op##sz(u##sz n)\
|
||||||
|
{\
|
||||||
|
u##sz res;\
|
||||||
|
__asm__ __volatile__ (#op " %" #regalloc "[res], %" #regalloc "[n]" : [res] "=r" (res) : [n] "r" (n));\
|
||||||
|
return res;\
|
||||||
|
}
|
||||||
|
|
||||||
|
#define _DECLARE_ASM_ARITHMETIC_UNARY_HELPER64(op) _DECLARE_ASM_ARITHMETIC_UNARY_HELPER(64, x, op)
|
||||||
|
#define _DECLARE_ASM_ARITHMETIC_UNARY_HELPER32(op) _DECLARE_ASM_ARITHMETIC_UNARY_HELPER(32, w, op)
|
||||||
|
|
||||||
|
_DECLARE_ASM_ARITHMETIC_UNARY_HELPER64(rbit)
|
||||||
|
_DECLARE_ASM_ARITHMETIC_UNARY_HELPER32(rbit)
|
||||||
|
|
||||||
static inline void __dsb_sy(void)
|
static inline void __dsb_sy(void)
|
||||||
{
|
{
|
||||||
|
|
|
@ -143,31 +143,16 @@ void vgicDebugPrintLrList(void)
|
||||||
DEBUG("]\n");
|
DEBUG("]\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Note: ordered by priority
|
static inline void vgicInsertVirqStateBefore(VirqStateList *list, VirqState *pos, VirqState *elem)
|
||||||
static void vgicEnqueueVirqState(VirqStateList *list, VirqState *elem)
|
|
||||||
{
|
{
|
||||||
VirqState *pos;
|
|
||||||
|
|
||||||
if (vgicIsStateQueued(elem)) {
|
|
||||||
PANIC("vgicEnqueueVirqState: unsanitized argument idx=%u previd=%u nextid=%u\n", (u32)vgicGetVirqStateIndex(elem), elem->listPrev, elem->listNext);
|
|
||||||
}
|
|
||||||
|
|
||||||
++list->size;
|
++list->size;
|
||||||
// Empty list
|
// Empty list
|
||||||
if (list->first == vgicGetQueueEnd()) {
|
if (list->first == vgicGetQueueEnd()) {
|
||||||
list->first = list->last = elem;
|
list->first = list->last = elem;
|
||||||
elem->listPrev = elem->listNext = VIRQLIST_END_ID;
|
elem->listPrev = elem->listNext = VIRQLIST_END_ID;
|
||||||
//vgicDebugPrintList(list);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (pos = list->first; pos != vgicGetQueueEnd(); pos = vgicGetNextQueuedVirqState(pos)) {
|
|
||||||
// Lowest priority number is higher
|
|
||||||
if (elem->priority <= pos->priority) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (pos == vgicGetQueueEnd()) {
|
if (pos == vgicGetQueueEnd()) {
|
||||||
// Insert after last
|
// Insert after last
|
||||||
pos = list->last;
|
pos = list->last;
|
||||||
|
@ -194,7 +179,73 @@ static void vgicEnqueueVirqState(VirqStateList *list, VirqState *elem)
|
||||||
prev->listNext = idx;
|
prev->listNext = idx;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//vgicDebugPrintList(list);
|
}
|
||||||
|
|
||||||
|
// Currently unused
|
||||||
|
static inline void vgicInsertVirqStateAfter(VirqStateList *list, VirqState *pos, VirqState *elem)
|
||||||
|
{
|
||||||
|
++list->size;
|
||||||
|
// Empty list
|
||||||
|
if (list->first == vgicGetQueueEnd()) {
|
||||||
|
list->first = list->last = elem;
|
||||||
|
elem->listPrev = elem->listNext = VIRQLIST_END_ID;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pos == vgicGetQueueEnd()) {
|
||||||
|
// Insert before first
|
||||||
|
pos = list->first;
|
||||||
|
|
||||||
|
elem->listPrev = pos->listPrev;
|
||||||
|
elem->listNext = vgicGetVirqStateIndex(pos);
|
||||||
|
pos->listPrev = vgicGetVirqStateIndex(elem);
|
||||||
|
list->first = elem;
|
||||||
|
} else {
|
||||||
|
// Otherwise, insert after
|
||||||
|
u32 idx = vgicGetVirqStateIndex(elem);
|
||||||
|
u32 posidx = vgicGetVirqStateIndex(pos);
|
||||||
|
|
||||||
|
u32 nextidx = pos->listPrev;
|
||||||
|
VirqState *next = vgicGetNextQueuedVirqState(pos);
|
||||||
|
|
||||||
|
elem->listPrev = posidx;
|
||||||
|
elem->listNext = nextidx;
|
||||||
|
|
||||||
|
pos->listNext = idx;
|
||||||
|
|
||||||
|
if (pos == list->last) {
|
||||||
|
list->last = elem;
|
||||||
|
} else {
|
||||||
|
next->listPrev = idx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int vgicCompareVirqState(const VirqState *a, const VirqState *b)
|
||||||
|
{
|
||||||
|
// Lower priority number is higher; we sort by descending priority, ie. ascending priority number
|
||||||
|
// Put the interrupts that were previously in the LR before the one which don't
|
||||||
|
int n1 = (int)(a->priority - b->priority);
|
||||||
|
return n1 == 0 ? (int)b->handled - (int)a->handled : n1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Note: ordered by priority
|
||||||
|
static void vgicEnqueueVirqState(VirqStateList *list, VirqState *elem)
|
||||||
|
{
|
||||||
|
VirqState *pos;
|
||||||
|
|
||||||
|
if (vgicIsStateQueued(elem)) {
|
||||||
|
PANIC("vgicEnqueueVirqState: unsanitized argument idx=%u previd=%u nextid=%u\n", (u32)vgicGetVirqStateIndex(elem), elem->listPrev, elem->listNext);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (pos = list->first; pos != vgicGetQueueEnd(); pos = vgicGetNextQueuedVirqState(pos)) {
|
||||||
|
// Sort predicate should be stable
|
||||||
|
if (vgicCompareVirqState(elem, pos) < 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
vgicInsertVirqStateBefore(list, pos, elem);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void vgicDequeueVirqState(VirqStateList *list, VirqState *elem)
|
static void vgicDequeueVirqState(VirqStateList *list, VirqState *elem)
|
||||||
|
@ -220,7 +271,6 @@ static void vgicDequeueVirqState(VirqStateList *list, VirqState *elem)
|
||||||
}
|
}
|
||||||
|
|
||||||
elem->listPrev = elem->listNext = VIRQLIST_INVALID_ID;
|
elem->listPrev = elem->listNext = VIRQLIST_INVALID_ID;
|
||||||
//vgicDebugPrintList(list);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void vgicNotifyOtherCoreList(u32 coreList)
|
static inline void vgicNotifyOtherCoreList(u32 coreList)
|
||||||
|
@ -855,12 +905,14 @@ void vgicUpdateState(void)
|
||||||
u32 coreId = currentCoreCtx->coreId;
|
u32 coreId = currentCoreCtx->coreId;
|
||||||
|
|
||||||
// First, put back inactive interrupts into the queue, handle some SGI stuff
|
// First, put back inactive interrupts into the queue, handle some SGI stuff
|
||||||
u64 usedMap = g_vgicUsedLrMap[coreId];
|
// Need to handle the LRs in reverse order to keep list stability
|
||||||
|
u64 usedMap = __rbit64(g_vgicUsedLrMap[coreId]);
|
||||||
FOREACH_BIT (tmp, pos, usedMap) {
|
FOREACH_BIT (tmp, pos, usedMap) {
|
||||||
if (!vgicUpdateListRegister(&gich->lr[pos])) {
|
if (!vgicUpdateListRegister(&gich->lr[63 - pos])) {
|
||||||
g_vgicUsedLrMap[coreId] &= ~BITL(pos);
|
usedMap &= ~BITL(pos);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
g_vgicUsedLrMap[coreId] = __rbit64(usedMap);
|
||||||
|
|
||||||
// Then, clean the list up
|
// Then, clean the list up
|
||||||
vgicCleanupPendingList();
|
vgicCleanupPendingList();
|
||||||
|
|
Loading…
Reference in a new issue