2018-09-07 15:00:13 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018 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/>.
|
|
|
|
*/
|
|
|
|
|
2018-02-24 13:33:16 +00:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
|
|
|
|
#include "utils.h"
|
|
|
|
#include "interrupt.h"
|
|
|
|
|
|
|
|
/* Global of registered handlers. */
|
2018-02-26 08:42:21 +00:00
|
|
|
static struct {
|
2018-02-24 13:33:16 +00:00
|
|
|
unsigned int id;
|
|
|
|
void (*handler)(void);
|
2018-03-01 18:11:09 +00:00
|
|
|
} g_registered_interrupts[MAX_REGISTERED_INTERRUPTS] = { {0} };
|
2018-02-24 13:33:16 +00:00
|
|
|
|
2018-02-25 19:00:50 +00:00
|
|
|
static unsigned int get_interrupt_id(void) {
|
2018-02-28 01:07:30 +00:00
|
|
|
return GICC_IAR;
|
2018-02-25 19:00:50 +00:00
|
|
|
}
|
2018-02-24 13:33:16 +00:00
|
|
|
|
2018-03-02 23:04:16 +00:00
|
|
|
/* Initializes the GIC. This must be called during wakeup. */
|
2018-02-24 13:33:16 +00:00
|
|
|
void intr_initialize_gic(void) {
|
|
|
|
/* Setup interrupts 0-0x1F as nonsecure with highest non-secure priority. */
|
|
|
|
GICD_IGROUPR[0] = 0xFFFFFFFF;
|
|
|
|
for (unsigned int i = 0; i < 0x20; i++) {
|
2018-02-25 19:00:50 +00:00
|
|
|
GICD_IPRIORITYR[i] = GIC_PRI_HIGHEST_NONSECURE;
|
2018-02-24 13:33:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Setup the GICC. */
|
|
|
|
GICC_CTLR = 0x1D9;
|
2018-02-25 19:00:50 +00:00
|
|
|
GICC_PMR = GIC_PRI_HIGHEST_NONSECURE;
|
2018-02-24 13:33:16 +00:00
|
|
|
GICC_BPR = 7;
|
|
|
|
}
|
|
|
|
|
2018-02-28 03:58:56 +00:00
|
|
|
/* Initializes Interrupt Groups 1-7 in the GIC. Called by pk2ldr. */
|
|
|
|
void intr_initialize_gic_nonsecure(void) {
|
|
|
|
for (unsigned int i = 1; i < 8; i++) {
|
2018-02-28 04:28:34 +00:00
|
|
|
GICD_IGROUPR[i] = 0xFFFFFFFF;
|
2018-02-28 03:58:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned int i = 0x20; i < 0xE0; i++) {
|
|
|
|
GICD_IPRIORITYR[i] = GIC_PRI_HIGHEST_NONSECURE;
|
|
|
|
}
|
|
|
|
GICD_CTLR = 1;
|
|
|
|
}
|
|
|
|
|
2018-02-28 00:35:35 +00:00
|
|
|
/* Sets GICC_CTLR to appropriate pre-sleep value. */
|
|
|
|
void intr_prepare_gicc_for_sleep(void) {
|
|
|
|
GICC_CTLR = 0x1E0;
|
|
|
|
}
|
|
|
|
|
2018-02-24 13:33:16 +00:00
|
|
|
/* Sets an interrupt's group in the GICD. */
|
|
|
|
void intr_set_group(unsigned int id, int group) {
|
|
|
|
GICD_IGROUPR[id >> 5] = (GICD_IGROUPR[id >> 5] & (~(1 << (id & 0x1F)))) | ((group & 1) << (id & 0x1F));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sets an interrupt id as pending in the GICD. */
|
|
|
|
void intr_set_pending(unsigned int id) {
|
|
|
|
GICD_ISPENDR[id >> 5] = 1 << (id & 0x1F);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sets an interrupt's priority in the GICD. */
|
|
|
|
void intr_set_priority(unsigned int id, uint8_t priority) {
|
|
|
|
GICD_IPRIORITYR[id] = priority;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sets an interrupt's target CPU mask in the GICD. */
|
|
|
|
void intr_set_cpu_mask(unsigned int id, uint8_t mask) {
|
|
|
|
GICD_ITARGETSR[id] = mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sets an interrupt's edge/level bits in the GICD. */
|
|
|
|
void intr_set_edge_level(unsigned int id, int edge_level) {
|
2018-02-25 19:00:50 +00:00
|
|
|
GICD_ICFGR[id >> 4] = GICD_ICFGR[id >> 4] & ((~(3 << ((id & 0xF) << 1))) | (((edge_level & 1) << 1) << ((id & 0xF) << 1)));
|
2018-02-24 13:33:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Sets an interrupt's enabled status in the GICD. */
|
|
|
|
void intr_set_enabled(unsigned int id, int enabled) {
|
|
|
|
GICD_ISENABLER[id >> 5] = (enabled & 1) << (id & 0x1F);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* To be called by FIQ handler. */
|
|
|
|
void handle_registered_interrupt(void) {
|
|
|
|
unsigned int interrupt_id = get_interrupt_id();
|
|
|
|
if (interrupt_id <= 0xDF) {
|
|
|
|
bool found_handler = false;
|
|
|
|
for (unsigned int i = 0; i < MAX_REGISTERED_INTERRUPTS; i++) {
|
|
|
|
if (g_registered_interrupts[i].id == interrupt_id) {
|
|
|
|
found_handler = true;
|
|
|
|
g_registered_interrupts[i].handler();
|
|
|
|
/* Mark that interrupt is done. */
|
|
|
|
GICC_EOIR = interrupt_id;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* We must have found a handler, or something went wrong. */
|
|
|
|
if (!found_handler) {
|
2018-02-24 14:20:45 +00:00
|
|
|
generic_panic();
|
2018-02-24 13:33:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Registers an interrupt into the global. */
|
|
|
|
void intr_register_handler(unsigned int id, void (*handler)(void)) {
|
|
|
|
bool registered_handler = false;
|
|
|
|
for (unsigned int i = 0; i < MAX_REGISTERED_INTERRUPTS; i++) {
|
|
|
|
if (g_registered_interrupts[i].id == 0) {
|
|
|
|
g_registered_interrupts[i].handler = handler;
|
|
|
|
g_registered_interrupts[i].id = id;
|
|
|
|
registered_handler = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Failure to register is an error condition. */
|
|
|
|
if (!registered_handler) {
|
2018-02-24 14:20:45 +00:00
|
|
|
generic_panic();
|
2018-02-24 13:33:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|