mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-10 23:04:44 +00:00
thermosphere: add hypervisor timer code
This commit is contained in:
parent
4d8a07943c
commit
0dd5f1f6d4
10 changed files with 169 additions and 17 deletions
|
@ -144,8 +144,8 @@ ifeq ($(PLATFORM), qemu)
|
||||||
|
|
||||||
export QEMU := qemu-system-aarch64
|
export QEMU := qemu-system-aarch64
|
||||||
|
|
||||||
QEMUFLAGS := -nographic -machine virt,virtualization=on,gic-version=2 -cpu cortex-a57 -smp 4 -m 1024\
|
QEMUFLAGS := -nographic -machine virt,virtualization=on,accel=tcg,gic-version=2 -cpu cortex-a57 -smp 4 -m 1024\
|
||||||
-kernel thermosphere.elf -d unimp,guest_errors -semihosting-config enable,target=native -serial mon:stdio
|
-kernel thermosphere.elf -d unimp,guest_errors -semihosting-config enable,target=native -serial mon:stdio
|
||||||
|
|
||||||
qemu: all
|
qemu: all
|
||||||
@$(QEMU) $(QEMUFLAGS)
|
@$(QEMU) $(QEMUFLAGS)
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
#include "core_ctx.h"
|
#include "core_ctx.h"
|
||||||
#include "debug_log.h"
|
#include "debug_log.h"
|
||||||
#include "vgic.h"
|
#include "vgic.h"
|
||||||
|
#include "timer.h"
|
||||||
|
|
||||||
IrqManager g_irqManager = {0};
|
IrqManager g_irqManager = {0};
|
||||||
|
|
||||||
|
@ -183,6 +184,9 @@ void handleIrqException(ExceptionStackFrame *frame, bool isLowerEl, bool isA32)
|
||||||
case GIC_IRQID_MAINTENANCE:
|
case GIC_IRQID_MAINTENANCE:
|
||||||
isMaintenanceInterrupt = true;
|
isMaintenanceInterrupt = true;
|
||||||
break;
|
break;
|
||||||
|
case TIMER_IRQID(CURRENT_TIMER):
|
||||||
|
timerInterruptHandler();
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
isGuestInterrupt = irqId >= 16;
|
isGuestInterrupt = irqId >= 16;
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -77,5 +77,15 @@ static inline bool irqIsGuest(u16 id)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return id != GIC_IRQID_MAINTENANCE && id != GIC_IRQID_HYP_TIMER;
|
bool ret = id != GIC_IRQID_MAINTENANCE && id != GIC_IRQID_NS_PHYS_HYP_TIMER;
|
||||||
|
#if GIC_IRQID_NS_VIRT_HYP_TIMER != GIC_IRQID_SPURIOUS
|
||||||
|
ret = ret && id != GIC_IRQID_NS_VIRT_HYP_TIMER;
|
||||||
|
#endif
|
||||||
|
#if GIC_IRQID_SEC_PHYS_HYP_TIMER != GIC_IRQID_SPURIOUS
|
||||||
|
ret = ret && id != GIC_IRQID_SEC_PHYS_HYP_TIMER;
|
||||||
|
#endif
|
||||||
|
#if GIC_IRQID_SEC_VIRT_HYP_TIMER != GIC_IRQID_SPURIOUS
|
||||||
|
ret = ret && id != GIC_IRQID_SEC_VIRT_HYP_TIMER;
|
||||||
|
#endif
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,7 +11,7 @@
|
||||||
#include "single_step.h"
|
#include "single_step.h"
|
||||||
#include "breakpoints.h"
|
#include "breakpoints.h"
|
||||||
#include "watchpoints.h"
|
#include "watchpoints.h"
|
||||||
|
#include "timer.h"
|
||||||
#include "irq.h"
|
#include "irq.h"
|
||||||
|
|
||||||
extern const u8 __start__[];
|
extern const u8 __start__[];
|
||||||
|
@ -41,6 +41,7 @@ void thermosphereMain(ExceptionStackFrame *frame)
|
||||||
{
|
{
|
||||||
enableTraps();
|
enableTraps();
|
||||||
enableBreakpointsAndWatchpoints();
|
enableBreakpointsAndWatchpoints();
|
||||||
|
timerInit();
|
||||||
initIrq();
|
initIrq();
|
||||||
|
|
||||||
if (currentCoreCtx->isBootCore) {
|
if (currentCoreCtx->isBootCore) {
|
||||||
|
|
|
@ -23,13 +23,18 @@
|
||||||
|
|
||||||
#define GIC_IRQID_PMU 23
|
#define GIC_IRQID_PMU 23
|
||||||
#define GIC_IRQID_MAINTENANCE 25
|
#define GIC_IRQID_MAINTENANCE 25
|
||||||
#define GIC_IRQID_HYP_TIMER 26
|
#define GIC_IRQID_NS_PHYS_HYP_TIMER 26
|
||||||
#define GIC_IRQID_VIRT_TIMER 27
|
#define GIC_IRQID_NS_VIRT_TIMER 27
|
||||||
//#define GIC_IRQID_LEGACY_NFIQ 28 not defined?
|
//#define GIC_IRQID_LEGACY_NFIQ 28 not defined?
|
||||||
#define GIC_IRQID_SEC_PHYS_TIMER 29
|
#define GIC_IRQID_SEC_PHYS_TIMER 29
|
||||||
#define GIC_IRQID_NS_PHYS_TIMER 30
|
#define GIC_IRQID_NS_PHYS_TIMER 30
|
||||||
//#define GIC_IRQID_LEGACY_NIRQ 31 not defined?
|
//#define GIC_IRQID_LEGACY_NIRQ 31 not defined?
|
||||||
|
|
||||||
|
|
||||||
|
#define GIC_IRQID_NS_VIRT_HYP_TIMER GIC_IRQID_SPURIOUS // SBSA: 28. Unimplemented
|
||||||
|
#define GIC_IRQID_SEC_PHYS_HYP_TIMER GIC_IRQID_SPURIOUS // SBSA: 20. Unimplemented
|
||||||
|
#define GIC_IRQID_SEC_VIRT_HYP_TIMER GIC_IRQID_SPURIOUS // SBSA: 19. Unimplemented
|
||||||
|
|
||||||
static inline void initGicV2Pointers(ArmGicV2 *gic)
|
static inline void initGicV2Pointers(ArmGicV2 *gic)
|
||||||
{
|
{
|
||||||
gic->gicd = (volatile ArmGicV2Distributor *)0x08000000ull;
|
gic->gicd = (volatile ArmGicV2Distributor *)0x08000000ull;
|
||||||
|
|
|
@ -21,14 +21,17 @@
|
||||||
// For both guest and host
|
// For both guest and host
|
||||||
#define MAX_NUM_REGISTERED_INTERRUPTS 512
|
#define MAX_NUM_REGISTERED_INTERRUPTS 512
|
||||||
|
|
||||||
#define GIC_IRQID_MAINTENANCE 25
|
#define GIC_IRQID_MAINTENANCE 25
|
||||||
#define GIC_IRQID_HYP_TIMER 26
|
#define GIC_IRQID_NS_PHYS_HYP_TIMER 26
|
||||||
#define GIC_IRQID_VIRT_TIMER 27
|
#define GIC_IRQID_NS_VIRT_TIMER 27
|
||||||
#define GIC_IRQID_LEGACY_NFIQ 28
|
#define GIC_IRQID_LEGACY_NFIQ 28
|
||||||
#define GIC_IRQID_SEC_PHYS_TIMER 29
|
#define GIC_IRQID_SEC_PHYS_TIMER 29
|
||||||
#define GIC_IRQID_NS_PHYS_TIMER 30
|
#define GIC_IRQID_NS_PHYS_TIMER 30
|
||||||
#define GIC_IRQID_LEGACY_NIRQ 31
|
#define GIC_IRQID_LEGACY_NIRQ 31
|
||||||
|
|
||||||
|
#define GIC_IRQID_NS_VIRT_HYP_TIMER GIC_IRQID_SPURIOUS // SBSA: 28. Unimplemented
|
||||||
|
#define GIC_IRQID_SEC_PHYS_HYP_TIMER GIC_IRQID_SPURIOUS // SBSA: 20. Unimplemented
|
||||||
|
#define GIC_IRQID_SEC_VIRT_HYP_TIMER GIC_IRQID_SPURIOUS // SBSA: 19. Unimplemented
|
||||||
|
|
||||||
static inline void initGicV2Pointers(ArmGicV2 *gic)
|
static inline void initGicV2Pointers(ArmGicV2 *gic)
|
||||||
{
|
{
|
||||||
|
|
|
@ -428,12 +428,12 @@
|
||||||
|
|
||||||
#define GET_SYSREG(r) ({\
|
#define GET_SYSREG(r) ({\
|
||||||
u64 __val; \
|
u64 __val; \
|
||||||
asm volatile("mrs %0, " STRINGIZE(r) : "=r" (__val)); \
|
asm volatile("mrs %0, " STRINGIZE(r) : "=r" (__val)); \
|
||||||
__val; \
|
__val; \
|
||||||
})
|
})
|
||||||
|
|
||||||
#define SET_SYSREG(reg, val) do { u64 temp_reg = (val); __asm__ __volatile__ ("msr " #reg ", %0" :: "r"(temp_reg) : "memory"); } while(false)
|
#define SET_SYSREG(reg, val) do { u64 temp_reg = (val); __asm__ __volatile__ ("msr " STRINGIZE(reg) ", %0" :: "r"(temp_reg) : "memory"); } while(false)
|
||||||
#define SET_SYSREG_IMM(reg, imm) do { __asm__ __volatile__ ("msr " #reg ", %0" :: "I"(imm) : "memory"); } while(false)
|
#define SET_SYSREG_IMM(reg, imm) do { __asm__ __volatile__ ("msr " STRINGIZE(reg) ", %0" :: "I"(imm) : "memory"); } while(false)
|
||||||
|
|
||||||
#define SYSREG_OP1_AARCH32_AUTO 0
|
#define SYSREG_OP1_AARCH32_AUTO 0
|
||||||
#define SYSREG_OP1_AARCH64_EL1 0
|
#define SYSREG_OP1_AARCH64_EL1 0
|
||||||
|
|
38
thermosphere/src/timer.c
Normal file
38
thermosphere/src/timer.c
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
/*
|
||||||
|
* 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 "timer.h"
|
||||||
|
#include "irq.h"
|
||||||
|
|
||||||
|
u64 g_timerFreq = 0;
|
||||||
|
|
||||||
|
void timerInit(void)
|
||||||
|
{
|
||||||
|
timerSetInterruptStatus(false, false);
|
||||||
|
if (currentCoreCtx->isBootCore) {
|
||||||
|
g_timerFreq = GET_SYSREG(cntfrq_el0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void timerInterruptHandler(void)
|
||||||
|
{
|
||||||
|
// Disable timer interrupts until reprogrammed
|
||||||
|
timerSetInterruptStatus(false, false);
|
||||||
|
|
||||||
|
// For fun
|
||||||
|
DEBUG("EL2 [core %d]: Timer interrupt at %lums\n", (int)currentCoreCtx->coreId, timerGetSystemTimeMs());
|
||||||
|
timerSetTimeoutMs(1000);
|
||||||
|
}
|
91
thermosphere/src/timer.h
Normal file
91
thermosphere/src/timer.h
Normal file
|
@ -0,0 +1,91 @@
|
||||||
|
/*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "utils.h"
|
||||||
|
#include "sysreg.h"
|
||||||
|
#include "platform/interrupt_config.h"
|
||||||
|
|
||||||
|
#define SECTONSECS 1000000000ull
|
||||||
|
#define SECTOMSECS 1000ull
|
||||||
|
|
||||||
|
// All generic timers possibly defined in the Arm architecture:
|
||||||
|
// (suffix, time source suffix, level, irqid)
|
||||||
|
#define NS_PHYS_TIMER (p, p, 0, GIC_IRQID_NS_PHYS_TIMER)
|
||||||
|
#define NS_VIRT_TIMER (v, v, 0, GIC_IRQID_NS_VIRT_TIMER)
|
||||||
|
#define NS_PHYS_HYP_TIMER (hp, p, 2, GIC_IRQID_NS_PHYS_HYP_TIMER)
|
||||||
|
#define NS_VIRT_HYP_TIMER (hv, v, 2, GIC_IRQID_NS_VIRT_HYP_TIMER)
|
||||||
|
#define SEC_PHYS_TIMER (ps, p, 1, GIC_IRQID_SEC_PHYS_TIMER)
|
||||||
|
#define SEC_PHYS_HYP_TIMER (hps, p, 2, GIC_IRQID_SEC_PHYS_HYP_TIMER)
|
||||||
|
#define SEC_VIRT_HYP_TIMER (hvs, v, 2, GIC_IRQID_SEC_VIRT_HYP_TIMER)
|
||||||
|
|
||||||
|
#define TIMER_IRQID_FIELDS(ign, ign2, ign3, id) id
|
||||||
|
#define TIMER_COUNTER_REG_FIELDS(ign, ts, ign2, ign3) cnt##ts##ct_el0
|
||||||
|
#define TIMER_CTL_REG_FIELDS(t, ign, el, ign2) cnt##t##_ctl_el##el
|
||||||
|
#define TIMER_CVAL_REG_FIELDS(t, ign, el, ign2) cnt##t##_cval_el##el
|
||||||
|
#define TIMER_TVAL_REG_FIELDS(t, ign, el, ign2) cnt##t##_tval_el##el
|
||||||
|
|
||||||
|
#define TIMER_IRQID(name) EVAL(TIMER_IRQID_FIELDS name)
|
||||||
|
#define TIMER_COUNTER_REG(name) EVAL(TIMER_COUNTER_REG_FIELDS name)
|
||||||
|
#define TIMER_CTL_REG(name) EVAL(TIMER_CTL_REG_FIELDS name)
|
||||||
|
#define TIMER_CVAL_REG(name) EVAL(TIMER_CVAL_REG_FIELDS name)
|
||||||
|
#define TIMER_TVAL_REG(name) EVAL(TIMER_TVAL_REG_FIELDS name)
|
||||||
|
|
||||||
|
#define CURRENT_TIMER NS_PHYS_HYP_TIMER
|
||||||
|
|
||||||
|
extern u64 g_timerFreq;
|
||||||
|
|
||||||
|
void timerInit(void);
|
||||||
|
void timerInterruptHandler(void);
|
||||||
|
|
||||||
|
static inline u64 timerGetSystemTick(void)
|
||||||
|
{
|
||||||
|
return GET_SYSREG(TIMER_COUNTER_REG(CURRENT_TIMER));
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline u64 timerGetSystemTimeNs(void)
|
||||||
|
{
|
||||||
|
return timerGetSystemTick() * SECTONSECS / g_timerFreq;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline u64 timerGetSystemTimeMs(void)
|
||||||
|
{
|
||||||
|
return timerGetSystemTick() * SECTOMSECS / g_timerFreq;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void timerSetInterruptStatus(bool enabled, bool masked)
|
||||||
|
{
|
||||||
|
u32 ebit = enabled ? BIT(0) : 0;
|
||||||
|
u32 mbit = masked ? BIT(1) : 0;
|
||||||
|
SET_SYSREG(TIMER_CTL_REG(CURRENT_TIMER), mbit | ebit);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void timerSetTimeoutTicks(u64 ticks)
|
||||||
|
{
|
||||||
|
SET_SYSREG(TIMER_CVAL_REG(CURRENT_TIMER), timerGetSystemTick() + ticks);
|
||||||
|
timerSetInterruptStatus(true, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void timerSetTimeoutNs(u64 ns)
|
||||||
|
{
|
||||||
|
timerSetTimeoutTicks(ns * g_timerFreq / SECTONSECS);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void timerSetTimeoutMs(u64 ms)
|
||||||
|
{
|
||||||
|
timerSetTimeoutTicks(ms * g_timerFreq / SECTOMSECS);
|
||||||
|
}
|
|
@ -46,7 +46,7 @@
|
||||||
static inline u##sz __##op##sz(u##sz n)\
|
static inline u##sz __##op##sz(u##sz n)\
|
||||||
{\
|
{\
|
||||||
u##sz res;\
|
u##sz res;\
|
||||||
__asm__ __volatile__ (#op " %" #regalloc "[res], %" #regalloc "[n]" : [res] "=r" (res) : [n] "r" (n));\
|
__asm__ __volatile__ (STRINGIZE(op) " %" STRINGIZE(regalloc) "[res], %" STRINGIZE(regalloc) "[n]" : [res] "=r" (res) : [n] "r" (n));\
|
||||||
return res;\
|
return res;\
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue