2019-08-10 23:56:49 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2019 Atmosphère-NX
|
|
|
|
*
|
|
|
|
* This program 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.
|
|
|
|
*
|
|
|
|
* This program 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 "irq.h"
|
|
|
|
#include "core_ctx.h"
|
|
|
|
#include "debug_log.h"
|
2019-08-17 23:40:47 +01:00
|
|
|
#include "vgic.h"
|
2020-01-09 19:24:05 +00:00
|
|
|
#include "timer.h"
|
2019-08-10 23:56:49 +01:00
|
|
|
|
|
|
|
IrqManager g_irqManager = {0};
|
|
|
|
|
|
|
|
static void initGic(void)
|
|
|
|
{
|
|
|
|
// Reinits the GICD and GICC (for non-secure mode, obviously)
|
|
|
|
if (currentCoreCtx->isBootCore && !currentCoreCtx->warmboot) {
|
|
|
|
initGicV2Pointers(&g_irqManager.gic);
|
|
|
|
|
|
|
|
// Disable interrupt handling & global interrupt distribution
|
|
|
|
g_irqManager.gic.gicd->ctlr = 0;
|
|
|
|
|
|
|
|
// Get some info
|
|
|
|
g_irqManager.numSharedInterrupts = 32 * (g_irqManager.gic.gicd->typer & 0x1F); // number of interrupt lines / 32
|
|
|
|
|
|
|
|
// unimplemented priority bits (lowest significant) are RAZ/WI
|
|
|
|
g_irqManager.gic.gicd->ipriorityr[0] = 0xFF;
|
2019-12-23 20:12:02 +00:00
|
|
|
g_irqManager.priorityShift = 8 - __builtin_popcount(g_irqManager.gic.gicd->ipriorityr[0]);
|
2019-08-10 23:56:49 +01:00
|
|
|
g_irqManager.numPriorityLevels = (u8)BIT(__builtin_popcount(g_irqManager.gic.gicd->ipriorityr[0]));
|
|
|
|
|
2019-08-12 22:24:30 +01:00
|
|
|
g_irqManager.numCpuInterfaces = (u8)(1 + ((g_irqManager.gic.gicd->typer >> 5) & 7));
|
2019-08-17 23:40:47 +01:00
|
|
|
g_irqManager.numListRegisters = (u8)(1 + (g_irqManager.gic.gich->vtr & 0x3F));
|
2019-08-10 23:56:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
volatile ArmGicV2Controller *gicc = g_irqManager.gic.gicc;
|
|
|
|
volatile ArmGicV2Distributor *gicd = g_irqManager.gic.gicd;
|
|
|
|
|
|
|
|
// Only one core will reset the GIC state for the shared peripheral interrupts
|
|
|
|
|
|
|
|
u32 numInterrupts = 32;
|
|
|
|
if (currentCoreCtx->isBootCore) {
|
|
|
|
numInterrupts += g_irqManager.numSharedInterrupts;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Filter all interrupts
|
|
|
|
gicc->pmr = 0;
|
|
|
|
|
|
|
|
// Disable interrupt preemption
|
|
|
|
gicc->bpr = 7;
|
|
|
|
|
|
|
|
// Note: the GICD I...n regs are banked for private interrupts
|
|
|
|
|
2019-08-12 22:24:30 +01:00
|
|
|
// Disable all interrupts, clear active status, clear pending status
|
2019-08-10 23:56:49 +01:00
|
|
|
for (u32 i = 0; i < numInterrupts / 32; i++) {
|
|
|
|
gicd->icenabler[i] = 0xFFFFFFFF;
|
|
|
|
gicd->icactiver[i] = 0xFFFFFFFF;
|
|
|
|
gicd->icpendr[i] = 0xFFFFFFFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set priorities to lowest
|
|
|
|
for (u32 i = 0; i < numInterrupts; i++) {
|
|
|
|
gicd->ipriorityr[i] = 0xFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reset icfgr, itargetsr for shared peripheral interrupts
|
|
|
|
for (u32 i = 32 / 16; i < numInterrupts / 16; i++) {
|
2020-01-05 00:33:35 +00:00
|
|
|
gicd->icfgr[i] = 0x55555555;
|
2019-08-10 23:56:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
for (u32 i = 32; i < numInterrupts; i++) {
|
|
|
|
gicd->itargetsr[i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now, reenable interrupts
|
|
|
|
|
|
|
|
// Enable the distributor
|
|
|
|
if (currentCoreCtx->isBootCore) {
|
|
|
|
gicd->ctlr = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Enable the CPU interface. Set EOIModeNS=1 (split prio drop & deactivate priority)
|
|
|
|
gicc->ctlr = BIT(9) | 1;
|
|
|
|
|
|
|
|
// Disable interrupt filtering
|
|
|
|
gicc->pmr = 0xFF;
|
|
|
|
|
2020-01-05 00:33:35 +00:00
|
|
|
currentCoreCtx->gicInterfaceMask = gicd->itargetsr[0];
|
2019-08-10 23:56:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void configureInterrupt(u16 id, u8 prio, bool isLevelSensitive)
|
|
|
|
{
|
|
|
|
volatile ArmGicV2Distributor *gicd = g_irqManager.gic.gicd;
|
2020-01-04 20:18:43 +00:00
|
|
|
gicd->icenabler[id / 32] = BIT(id % 32);
|
2019-08-10 23:56:49 +01:00
|
|
|
|
|
|
|
if (id >= 32) {
|
2020-01-05 00:33:35 +00:00
|
|
|
u32 cfgr = gicd->icfgr[id / 16];
|
2020-01-05 16:04:53 +00:00
|
|
|
cfgr &= ~(3 << IRQ_CFGR_SHIFT(id));
|
|
|
|
cfgr |= (!isLevelSensitive ? 3 : 1) << IRQ_CFGR_SHIFT(id);
|
2020-01-05 00:33:35 +00:00
|
|
|
gicd->icfgr[id / 16] = cfgr;
|
|
|
|
gicd->itargetsr[id] |= currentCoreCtx->gicInterfaceMask;
|
2019-08-10 23:56:49 +01:00
|
|
|
}
|
2020-01-04 20:18:43 +00:00
|
|
|
gicd->icpendr[id / 32] = BIT(id % 32);
|
|
|
|
gicd->ipriorityr[id] = (prio << g_irqManager.priorityShift) & 0xFF;
|
|
|
|
gicd->isenabler[id / 32] = BIT(id % 32);
|
2019-08-10 23:56:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void initIrq(void)
|
|
|
|
{
|
|
|
|
u64 flags = recursiveSpinlockLockMaskIrq(&g_irqManager.lock);
|
|
|
|
|
|
|
|
initGic();
|
2019-08-17 23:40:47 +01:00
|
|
|
vgicInit();
|
2019-08-10 23:56:49 +01:00
|
|
|
|
|
|
|
// Configure the interrupts we use here
|
2019-08-12 22:47:14 +01:00
|
|
|
for (u32 i = 0; i < ThermosphereSgi_Max; i++) {
|
2019-12-23 20:12:02 +00:00
|
|
|
configureInterrupt(i, IRQ_PRIORITY_HOST, false);
|
2019-08-12 22:47:14 +01:00
|
|
|
}
|
|
|
|
|
2019-12-23 20:12:02 +00:00
|
|
|
configureInterrupt(GIC_IRQID_MAINTENANCE, IRQ_PRIORITY_HOST, true);
|
2019-08-10 23:56:49 +01:00
|
|
|
|
|
|
|
recursiveSpinlockUnlockRestoreIrq(&g_irqManager.lock, flags);
|
|
|
|
}
|
|
|
|
|
2020-01-08 22:18:56 +00:00
|
|
|
static inline bool checkRescheduleEmulatedPtimer(ExceptionStackFrame *frame)
|
|
|
|
{
|
|
|
|
// Evaluate if the timer really has expired in the PoV of the guest kernel. If not, reschedule (add missed time delta) it & exit early
|
|
|
|
u64 cval = GET_SYSREG(cntp_cval_el0);
|
|
|
|
if (cval > frame->cntvct_el0) {
|
|
|
|
// It has not: reschedule the timer
|
|
|
|
u64 offsetNow = GET_SYSREG(cntvoff_el2);
|
|
|
|
SET_SYSREG(cntp_cval_el0, cval + (offsetNow - currentCoreCtx->emulPtimerOffsetThen));
|
|
|
|
currentCoreCtx->emulPtimerOffsetThen = offsetNow;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-08-10 23:56:49 +01:00
|
|
|
void handleIrqException(ExceptionStackFrame *frame, bool isLowerEl, bool isA32)
|
|
|
|
{
|
|
|
|
(void)isLowerEl;
|
|
|
|
(void)isA32;
|
|
|
|
volatile ArmGicV2Controller *gicc = g_irqManager.gic.gicc;
|
|
|
|
|
|
|
|
// Acknowledge the interrupt. Interrupt goes from pending to active.
|
|
|
|
u32 iar = gicc->iar;
|
|
|
|
u32 irqId = iar & 0x3FF;
|
2020-01-02 01:40:30 +00:00
|
|
|
u32 srcCore = (iar >> 10) & 7;
|
2019-08-10 23:56:49 +01:00
|
|
|
|
2020-01-02 01:40:30 +00:00
|
|
|
//DEBUG("EL2 [core %d]: Received irq %x\n", (int)currentCoreCtx->coreId, irqId);
|
2019-08-10 23:56:49 +01:00
|
|
|
|
|
|
|
if (irqId == GIC_IRQID_SPURIOUS) {
|
|
|
|
// Spurious interrupt received
|
|
|
|
return;
|
2020-01-08 22:18:56 +00:00
|
|
|
} else if (irqId == GIC_IRQID_NS_PHYS_TIMER && !checkRescheduleEmulatedPtimer(frame)) {
|
|
|
|
// Deactivate the ptimer interrupt, return early
|
|
|
|
gicc->eoir = iar;
|
|
|
|
gicc->dir = iar;
|
|
|
|
return;
|
2019-08-10 23:56:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool isGuestInterrupt = false;
|
2019-08-17 23:40:47 +01:00
|
|
|
bool isMaintenanceInterrupt = false;
|
2019-08-10 23:56:49 +01:00
|
|
|
|
2019-08-12 22:47:14 +01:00
|
|
|
switch (irqId) {
|
|
|
|
case ThermosphereSgi_ExecuteFunction:
|
|
|
|
executeFunctionInterruptHandler(srcCore);
|
|
|
|
break;
|
2019-08-17 23:40:47 +01:00
|
|
|
case ThermosphereSgi_VgicUpdate:
|
|
|
|
// Nothing in particular to do here
|
|
|
|
break;
|
2019-08-12 22:47:14 +01:00
|
|
|
case GIC_IRQID_MAINTENANCE:
|
2019-08-17 23:40:47 +01:00
|
|
|
isMaintenanceInterrupt = true;
|
2019-08-12 22:47:14 +01:00
|
|
|
break;
|
2020-01-09 19:24:05 +00:00
|
|
|
case TIMER_IRQID(CURRENT_TIMER):
|
|
|
|
timerInterruptHandler();
|
|
|
|
break;
|
2019-08-12 22:47:14 +01:00
|
|
|
default:
|
2019-08-17 23:40:47 +01:00
|
|
|
isGuestInterrupt = irqId >= 16;
|
2019-08-12 22:47:14 +01:00
|
|
|
break;
|
|
|
|
}
|
2019-08-10 23:56:49 +01:00
|
|
|
|
|
|
|
// Priority drop
|
|
|
|
gicc->eoir = iar;
|
|
|
|
|
2019-12-26 00:05:36 +00:00
|
|
|
isGuestInterrupt = isGuestInterrupt && irqIsGuest(irqId);
|
|
|
|
|
2019-08-17 23:40:47 +01:00
|
|
|
recursiveSpinlockLock(&g_irqManager.lock);
|
|
|
|
|
2019-08-10 23:56:49 +01:00
|
|
|
if (!isGuestInterrupt) {
|
2019-08-17 23:40:47 +01:00
|
|
|
if (isMaintenanceInterrupt) {
|
|
|
|
vgicMaintenanceInterruptHandler();
|
|
|
|
}
|
2019-08-10 23:56:49 +01:00
|
|
|
// Deactivate the interrupt
|
|
|
|
gicc->dir = iar;
|
|
|
|
} else {
|
2019-08-17 23:40:47 +01:00
|
|
|
vgicEnqueuePhysicalIrq(irqId);
|
2019-08-10 23:56:49 +01:00
|
|
|
}
|
2019-08-17 23:40:47 +01:00
|
|
|
|
|
|
|
// Update vgic state
|
|
|
|
vgicUpdateState();
|
|
|
|
|
|
|
|
recursiveSpinlockUnlock(&g_irqManager.lock);
|
2019-08-10 23:56:49 +01:00
|
|
|
}
|