1
0
Fork 0
mirror of https://github.com/Atmosphere-NX/Atmosphere.git synced 2024-11-15 00:16:48 +00:00

thermosphere: fix guest access to irq 25, etc; we don't need to raise VI manually

See Armv8a TRM "Virtual IRQ exception"
This commit is contained in:
TuxSH 2019-12-26 00:05:36 +00:00
parent cdf3bc6942
commit 676a895cca
3 changed files with 47 additions and 35 deletions

View file

@ -15,7 +15,6 @@
*/ */
#include "irq.h" #include "irq.h"
#include "platform/interrupt_config.h"
#include "core_ctx.h" #include "core_ctx.h"
#include "debug_log.h" #include "debug_log.h"
#include "vgic.h" #include "vgic.h"
@ -134,11 +133,6 @@ void initIrq(void)
recursiveSpinlockUnlockRestoreIrq(&g_irqManager.lock, flags); recursiveSpinlockUnlockRestoreIrq(&g_irqManager.lock, flags);
} }
bool isGuestIrq(u16 id)
{
return true;
}
void handleIrqException(ExceptionStackFrame *frame, bool isLowerEl, bool isA32) void handleIrqException(ExceptionStackFrame *frame, bool isLowerEl, bool isA32)
{ {
(void)isLowerEl; (void)isLowerEl;
@ -178,6 +172,8 @@ void handleIrqException(ExceptionStackFrame *frame, bool isLowerEl, bool isA32)
// Priority drop // Priority drop
gicc->eoir = iar; gicc->eoir = iar;
isGuestInterrupt = isGuestInterrupt && irqIsGuest(irqId);
recursiveSpinlockLock(&g_irqManager.lock); recursiveSpinlockLock(&g_irqManager.lock);
if (!isGuestInterrupt) { if (!isGuestInterrupt) {

View file

@ -20,6 +20,7 @@
#include "spinlock.h" #include "spinlock.h"
#include "exceptions.h" #include "exceptions.h"
#include "utils.h" #include "utils.h"
#include "platform/interrupt_config.h"
#define IRQ_PRIORITY_HOST 0 #define IRQ_PRIORITY_HOST 0
#define IRQ_PRIORITY_GUEST 1 #define IRQ_PRIORITY_GUEST 1
@ -46,7 +47,6 @@ extern IrqManager g_irqManager;
void initIrq(void); void initIrq(void);
void handleIrqException(ExceptionStackFrame *frame, bool isLowerEl, bool isA32); void handleIrqException(ExceptionStackFrame *frame, bool isLowerEl, bool isA32);
bool isGuestIrq(u16 id);
static inline void generateSgiForAllOthers(ThermosphereSgi id) static inline void generateSgiForAllOthers(ThermosphereSgi id)
{ {
@ -67,3 +67,13 @@ static inline void generateSgiForAll(ThermosphereSgi id)
{ {
generateSgiForList(id, MASK(g_irqManager.numCpuInterfaces)); generateSgiForList(id, MASK(g_irqManager.numCpuInterfaces));
} }
static inline bool irqIsEnabled(u16 id)
{
return (g_irqManager.gic.gicd->isenabler[id / 32] & BIT(id % 32)) != 0;
}
static inline bool irqIsGuest(u16 id)
{
return id != GIC_IRQID_MAINTENANCE && id != GIC_IRQID_HYP_TIMER;
}

View file

@ -191,11 +191,6 @@ static inline bool vgicIsVirqEdgeTriggered(u16 id)
} }
} }
static inline bool vgicIsVirqEnabled(u16 id)
{
return (g_irqManager.gic.gicd->isenabler[id / 32] & BIT(id % 32)) != 0;
}
static inline bool vgicIsVirqPending(VirqState *state) static inline bool vgicIsVirqPending(VirqState *state)
{ {
// In case we emulate ispendr in the future... // In case we emulate ispendr in the future...
@ -254,7 +249,7 @@ static inline u32 vgicGetDistributorImplementerIdentificationRegister(void)
static void vgicSetInterruptEnabledState(u16 id) static void vgicSetInterruptEnabledState(u16 id)
{ {
if (id < 16 || !vgicIsVirqEnabled(id)) { if (id < 16 || !irqIsGuest(id) || irqIsEnabled(id)) {
// Nothing to do... // Nothing to do...
// Also, ignore for SGIs // Also, ignore for SGIs
return; return;
@ -271,7 +266,7 @@ static void vgicSetInterruptEnabledState(u16 id)
static void vgicClearInterruptEnabledState(u16 id) static void vgicClearInterruptEnabledState(u16 id)
{ {
if (id < 16 || !vgicIsVirqEnabled(id)) { if (id < 16 || !irqIsGuest(id) || !irqIsEnabled(id)) {
// Nothing to do... // Nothing to do...
// Also, ignore for SGIs // Also, ignore for SGIs
return; return;
@ -290,11 +285,15 @@ static void vgicClearInterruptEnabledState(u16 id)
static inline bool vgicGetInterruptEnabledState(u16 id) static inline bool vgicGetInterruptEnabledState(u16 id)
{ {
// SGIs are always enabled // SGIs are always enabled
return id < 16 || vgicIsVirqEnabled(id); return id < 16 || (irqIsGuest(id) && irqIsEnabled(id));
} }
static void vgicSetInterruptPriorityByte(u16 id, u8 priority) static void vgicSetInterruptPriorityByte(u16 id, u8 priority)
{ {
if (!irqIsGuest(id)) {
return;
}
// 32 priority levels max, bits [7:3] // 32 priority levels max, bits [7:3]
priority >>= 3; priority >>= 3;
priority &= 0x1F; priority &= 0x1F;
@ -319,13 +318,13 @@ static void vgicSetInterruptPriorityByte(u16 id, u8 priority)
static inline u8 vgicGetInterruptPriorityByte(u16 id) static inline u8 vgicGetInterruptPriorityByte(u16 id)
{ {
return vgicGetVirqState(currentCoreCtx->coreId, id)->priority << 3; return irqIsGuest(id) ? vgicGetVirqState(currentCoreCtx->coreId, id)->priority << 3 : 0;
} }
static void vgicSetInterruptTargets(u16 id, u8 coreList) static void vgicSetInterruptTargets(u16 id, u8 coreList)
{ {
// Ignored for SGIs and PPIs // Ignored for SGIs and PPIs, and non-guest interrupts
if (id < 32) { if (id < 32 || !irqIsGuest(id)) {
return; return;
} }
@ -350,13 +349,13 @@ static void vgicSetInterruptTargets(u16 id, u8 coreList)
static inline u8 vgicGetInterruptTargets(u16 id) static inline u8 vgicGetInterruptTargets(u16 id)
{ {
// For SGIs & PPIs, itargetsr is banked and contains the CPU ID // For SGIs & PPIs, itargetsr is banked and contains the CPU ID
return g_irqManager.gic.gicd->itargetsr[id]; return (id < 32 || irqIsGuest(id)) ? g_irqManager.gic.gicd->itargetsr[id] : 0;
} }
static inline void vgicSetInterruptConfigByte(u16 id, u32 config) static inline void vgicSetInterruptConfigByte(u16 id, u32 config)
{ {
// Ignored for SGIs, implementation defined for PPIs // Ignored for SGIs, implementation defined for PPIs
if (id < 32) { if (id < 32 || !irqIsGuest(id)) {
return; return;
} }
@ -368,7 +367,7 @@ static inline void vgicSetInterruptConfigByte(u16 id, u32 config)
static inline u32 vgicGetInterruptConfigByte(u16 id, u32 config) static inline u32 vgicGetInterruptConfigByte(u16 id, u32 config)
{ {
return vgicIsVirqEdgeTriggered(id) ? 2 : 0; return (irqIsGuest(id) && vgicIsVirqEdgeTriggered(id)) ? 2 : 0;
} }
static void vgicSetSgiPendingState(u16 id, u32 coreId, u32 srcCoreId) static void vgicSetSgiPendingState(u16 id, u32 coreId, u32 srcCoreId)
@ -826,26 +825,23 @@ void vgicUpdateState(void)
for (size_t i = 0; i < numChosen; i++) { for (size_t i = 0; i < numChosen; i++) {
vgicPushListRegisters(chosen, numChosen); vgicPushListRegisters(chosen, numChosen);
} }
// Apparently, the following is not needed because the GIC generates it for us
// Keep this comment, it's not intuitive
/* /*
// Raise vIRQ when applicable. We only need to check for the highest priority // Raise vIRQ when applicable. We only need to check for the highest priority
// TRM: "The GIC always masks an interrupt that has the largest supported priority field value. /*if (newHiPrio < 0x1F && vgicIsInterruptRaisable(newHiPrio)) {
// This provides an additional means of preventing an interrupt being signaled to any processor"
if (false && newHiPrio < 0x1F && vgicIsInterruptRaisable(newHiPrio)) {
DEBUG("enablegrp0 %d\n", (int)gich->vmcr.enableGrp0);
DEBUG("enablegrp1 %d\n", (int)gich->vmcr.enableGrp1);
gich->hcr.npie = true; gich->hcr.npie = true;
u32 hcr = GET_SYSREG(hcr_el2); u32 hcr = GET_SYSREG(hcr_el2);
SET_SYSREG(hcr_el2, hcr | HCR_VI); SET_SYSREG(hcr_el2, hcr | HCR_VI);
} else { } else {
//DEBUG("unraising\n");
gich->hcr.npie = false; gich->hcr.npie = false;
u32 hcr = GET_SYSREG(hcr_el2); u32 hcr = GET_SYSREG(hcr_el2);
SET_SYSREG(hcr_el2, hcr & ~HCR_VI); SET_SYSREG(hcr_el2, hcr & ~HCR_VI);
} }*/
*/
// Enable underflow interrupt when appropriate to do so // Enable underflow interrupt when appropriate to do so
if (vgicGetNumberOfFreeListRegisters() != g_irqManager.numListRegisters) { if (g_irqManager.numListRegisters - vgicGetNumberOfFreeListRegisters() > 1) {
gich->hcr.uie = true; gich->hcr.uie = true;
} else { } else {
gich->hcr.uie = false; gich->hcr.uie = false;
@ -857,7 +853,6 @@ void vgicMaintenanceInterruptHandler(void)
volatile ArmGicV2VirtualInterfaceController *gich = g_irqManager.gic.gich; volatile ArmGicV2VirtualInterfaceController *gich = g_irqManager.gic.gich;
ArmGicV2MaintenanceIntStatRegister misr = g_irqManager.gic.gich->misr; ArmGicV2MaintenanceIntStatRegister misr = g_irqManager.gic.gich->misr;
DEBUG("maintenance\n");
// Force GICV_CTRL to behave like ns-GICC_CTLR, with group 1 being replaced by group 0 // Force GICV_CTRL to behave like ns-GICC_CTLR, with group 1 being replaced by group 0
// Ensure we aren't spammed by maintenance interrupts, either. // Ensure we aren't spammed by maintenance interrupts, either.
if (misr.vgrp0e || misr.vgrp0d || misr.vgrp1e || misr.vgrp1d) { if (misr.vgrp0e || misr.vgrp0d || misr.vgrp1e || misr.vgrp1d) {
@ -865,22 +860,33 @@ void vgicMaintenanceInterruptHandler(void)
} }
if (misr.vgrp0e) { if (misr.vgrp0e) {
DEBUG("maintenance grp0 enabled\n");
gich->hcr.vgrp0eie = false; gich->hcr.vgrp0eie = false;
gich->hcr.vgrp0die = true; gich->hcr.vgrp0die = true;
} else if (misr.vgrp0d) { } else if (misr.vgrp0d) {
DEBUG("maintenance grp0 disabled\n");
gich->hcr.vgrp0eie = true; gich->hcr.vgrp0eie = true;
gich->hcr.vgrp0die = false; gich->hcr.vgrp0die = false;
} }
if (misr.vgrp1e) { if (misr.vgrp1e) {
// Nothing to do since we unset the bit asap // Nothing to do since we cleared the bits above...
} }
if (misr.lrenp) { if (misr.lrenp) {
DEBUG("VGIC: List Register Entry Not Present maintenance interrupt!"); DEBUG("VGIC: List Register Entry Not Present maintenance interrupt!\n");
panic(); panic();
} }
if (misr.eoi) {
DEBUG("SGI EOI maintenance interrupt\n");
}
if (misr.np) {
DEBUG("No Pending maintenance interrupt\n");
}
if (misr.u) {
DEBUG("Underflow maintenance interrupt\n");
}
// The rest should be handled by the main loop... // The rest should be handled by the main loop...
} }