2022-05-11 16:54:03 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2022, Timon Kruiper <timonkruiper@gmail.com>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
2023-01-30 11:03:26 +01:00
|
|
|
#include <Kernel/Arch/CPU.h>
|
2022-05-11 16:54:03 +02:00
|
|
|
#include <Kernel/Arch/Interrupts.h>
|
2023-01-08 17:42:07 +01:00
|
|
|
#include <Kernel/Arch/PageFault.h>
|
2022-12-22 11:56:40 +01:00
|
|
|
#include <Kernel/Arch/TrapFrame.h>
|
2022-05-30 10:14:46 +02:00
|
|
|
#include <Kernel/Arch/aarch64/InterruptManagement.h>
|
|
|
|
#include <Kernel/Interrupts/GenericInterruptHandler.h>
|
|
|
|
#include <Kernel/Interrupts/SharedIRQHandler.h>
|
|
|
|
#include <Kernel/Interrupts/UnhandledInterruptHandler.h>
|
2023-01-08 13:23:55 +01:00
|
|
|
#include <Kernel/KSyms.h>
|
2023-01-08 17:42:07 +01:00
|
|
|
#include <Kernel/Panic.h>
|
2023-01-08 13:23:55 +01:00
|
|
|
#include <Kernel/StdLib.h>
|
2022-05-30 10:14:46 +02:00
|
|
|
|
2022-05-11 16:54:03 +02:00
|
|
|
namespace Kernel {
|
|
|
|
|
2023-01-30 15:42:39 +01:00
|
|
|
extern "C" void syscall_handler(TrapFrame const*);
|
|
|
|
|
2023-01-30 12:18:52 +01:00
|
|
|
static void dump_exception_syndrome_register(Aarch64::ESR_EL1 const& esr_el1)
|
|
|
|
{
|
2023-01-30 14:36:27 +01:00
|
|
|
dbgln("Exception Syndrome: EC({:#b}) IL({:#b}) ISS({:#b}) ISS2({:#b})", esr_el1.EC, esr_el1.IL, esr_el1.ISS, esr_el1.ISS2);
|
|
|
|
dbgln(" Class: {}", Aarch64::exception_class_to_string(esr_el1.EC));
|
2023-01-30 12:18:52 +01:00
|
|
|
|
|
|
|
if (Aarch64::exception_class_is_data_abort(esr_el1.EC))
|
2023-01-30 14:36:27 +01:00
|
|
|
dbgln(" Data Fault Status Code: {}", Aarch64::data_fault_status_code_to_string(esr_el1.ISS));
|
|
|
|
if (Aarch64::exception_class_has_set_far(esr_el1.EC))
|
|
|
|
dbgln(" Faulting Virtual Address: 0x{:x}", Aarch64::FAR_EL1::read().virtual_address);
|
2023-01-30 12:18:52 +01:00
|
|
|
}
|
|
|
|
|
2023-01-30 11:03:26 +01:00
|
|
|
void dump_registers(RegisterState const& regs)
|
2023-01-08 13:23:55 +01:00
|
|
|
{
|
2023-01-30 14:36:27 +01:00
|
|
|
auto esr_el1 = Kernel::Aarch64::ESR_EL1::read();
|
|
|
|
dump_exception_syndrome_register(esr_el1);
|
|
|
|
|
|
|
|
// Special registers
|
|
|
|
Aarch64::SPSR_EL1 spsr_el1 = {};
|
|
|
|
memcpy(&spsr_el1, (u8 const*)®s.spsr_el1, sizeof(u64));
|
|
|
|
|
|
|
|
dbgln("Saved Program Status: (NZCV({:#b}) DAIF({:#b}) M({:#b})) / 0x{:x}", ((regs.spsr_el1 >> 28) & 0b1111), ((regs.spsr_el1 >> 6) & 0b1111), regs.spsr_el1 & 0b1111, regs.spsr_el1);
|
|
|
|
dbgln("Exception Link Register: 0x{:x}", regs.elr_el1);
|
|
|
|
dbgln("Stack Pointer (EL0): 0x{:x}", regs.sp_el0);
|
|
|
|
|
2023-01-08 13:23:55 +01:00
|
|
|
dbgln(" x0={:p} x1={:p} x2={:p} x3={:p} x4={:p}", regs.x[0], regs.x[1], regs.x[2], regs.x[3], regs.x[4]);
|
|
|
|
dbgln(" x5={:p} x6={:p} x7={:p} x8={:p} x9={:p}", regs.x[5], regs.x[6], regs.x[7], regs.x[8], regs.x[9]);
|
|
|
|
dbgln("x10={:p} x11={:p} x12={:p} x13={:p} x14={:p}", regs.x[10], regs.x[11], regs.x[12], regs.x[13], regs.x[14]);
|
|
|
|
dbgln("x15={:p} x16={:p} x17={:p} x18={:p} x19={:p}", regs.x[15], regs.x[16], regs.x[17], regs.x[18], regs.x[19]);
|
|
|
|
dbgln("x20={:p} x21={:p} x22={:p} x23={:p} x24={:p}", regs.x[20], regs.x[21], regs.x[22], regs.x[23], regs.x[24]);
|
|
|
|
dbgln("x25={:p} x26={:p} x27={:p} x28={:p} x29={:p}", regs.x[25], regs.x[26], regs.x[27], regs.x[28], regs.x[29]);
|
|
|
|
dbgln("x30={:p}", regs.x[30]);
|
|
|
|
}
|
|
|
|
|
2023-02-18 10:26:30 +01:00
|
|
|
static ErrorOr<PageFault> page_fault_from_exception_syndrome_register(VirtualAddress fault_address, Aarch64::ESR_EL1 esr_el1)
|
2023-01-08 13:23:55 +01:00
|
|
|
{
|
2023-01-08 17:42:07 +01:00
|
|
|
PageFault fault { fault_address };
|
|
|
|
|
|
|
|
u8 data_fault_status_code = esr_el1.ISS & 0x3f;
|
|
|
|
if (data_fault_status_code >= 0b001100 && data_fault_status_code <= 0b001111) {
|
|
|
|
fault.set_type(PageFault::Type::ProtectionViolation);
|
|
|
|
} else if (data_fault_status_code >= 0b000100 && data_fault_status_code <= 0b000111) {
|
|
|
|
fault.set_type(PageFault::Type::PageNotPresent);
|
|
|
|
} else {
|
2023-02-18 10:26:30 +01:00
|
|
|
dbgln("Unknown DFSC: {}", Aarch64::data_fault_status_code_to_string(esr_el1.ISS));
|
|
|
|
return Error::from_errno(EFAULT);
|
2023-01-08 17:42:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
fault.set_access((esr_el1.ISS & (1 << 6)) == (1 << 6) ? PageFault::Access::Write : PageFault::Access::Read);
|
|
|
|
|
2023-01-30 10:28:17 +01:00
|
|
|
fault.set_mode(Aarch64::exception_class_is_data_or_instruction_abort_from_lower_exception_level(esr_el1.EC) ? ExecutionMode::User : ExecutionMode::Kernel);
|
2023-01-08 17:42:07 +01:00
|
|
|
|
2023-01-29 15:10:41 +01:00
|
|
|
if (Aarch64::exception_class_is_instruction_abort(esr_el1.EC))
|
|
|
|
fault.set_instruction_fetch(true);
|
|
|
|
|
2023-01-08 17:42:07 +01:00
|
|
|
return fault;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" void exception_common(Kernel::TrapFrame* trap_frame);
|
|
|
|
extern "C" void exception_common(Kernel::TrapFrame* trap_frame)
|
|
|
|
{
|
|
|
|
Processor::current().enter_trap(*trap_frame, false);
|
|
|
|
|
2023-01-08 13:23:55 +01:00
|
|
|
auto esr_el1 = Kernel::Aarch64::ESR_EL1::read();
|
2023-01-08 17:42:07 +01:00
|
|
|
auto fault_address = Aarch64::FAR_EL1::read().virtual_address;
|
|
|
|
Processor::enable_interrupts();
|
2023-01-08 13:23:55 +01:00
|
|
|
|
2023-01-29 15:10:41 +01:00
|
|
|
if (Aarch64::exception_class_is_data_abort(esr_el1.EC) || Aarch64::exception_class_is_instruction_abort(esr_el1.EC)) {
|
2023-02-18 10:26:30 +01:00
|
|
|
auto page_fault_or_error = page_fault_from_exception_syndrome_register(VirtualAddress(fault_address), esr_el1);
|
|
|
|
if (page_fault_or_error.is_error()) {
|
|
|
|
handle_crash(*trap_frame->regs, "Unknown page fault", SIGSEGV, false);
|
|
|
|
} else {
|
|
|
|
auto page_fault = page_fault_or_error.release_value();
|
|
|
|
page_fault.handle(*trap_frame->regs);
|
|
|
|
}
|
2023-01-30 15:42:39 +01:00
|
|
|
} else if (Aarch64::exception_class_is_svc_instruction_execution(esr_el1.EC)) {
|
|
|
|
syscall_handler(trap_frame);
|
2023-01-08 17:42:07 +01:00
|
|
|
} else {
|
2023-02-18 10:26:30 +01:00
|
|
|
handle_crash(*trap_frame->regs, "Unexpected exception", SIGSEGV, false);
|
2023-01-08 17:42:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Processor::disable_interrupts();
|
|
|
|
Processor::current().exit_trap(*trap_frame);
|
2023-01-08 13:23:55 +01:00
|
|
|
}
|
|
|
|
|
2022-05-30 10:14:46 +02:00
|
|
|
static Array<GenericInterruptHandler*, 64> s_interrupt_handlers;
|
|
|
|
|
2022-12-22 11:56:40 +01:00
|
|
|
extern "C" void handle_interrupt(TrapFrame&);
|
|
|
|
extern "C" void handle_interrupt(TrapFrame& trap_frame)
|
2022-05-30 10:14:46 +02:00
|
|
|
{
|
2022-12-22 11:56:40 +01:00
|
|
|
Processor::current().enter_trap(trap_frame, true);
|
|
|
|
|
2022-05-30 10:14:46 +02:00
|
|
|
for (auto& interrupt_controller : InterruptManagement::the().controllers()) {
|
|
|
|
auto pending_interrupts = interrupt_controller->pending_interrupts();
|
|
|
|
|
|
|
|
// TODO: Add these interrupts as a source of entropy for randomness.
|
|
|
|
u8 irq = 0;
|
|
|
|
while (pending_interrupts) {
|
|
|
|
if ((pending_interrupts & 0b1) != 0b1) {
|
|
|
|
irq += 1;
|
|
|
|
pending_interrupts >>= 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto* handler = s_interrupt_handlers[irq];
|
|
|
|
VERIFY(handler);
|
2022-11-18 21:39:49 +01:00
|
|
|
handler->increment_call_count();
|
2022-12-22 11:56:40 +01:00
|
|
|
handler->handle_interrupt(*trap_frame.regs);
|
2022-05-30 10:14:46 +02:00
|
|
|
handler->eoi();
|
|
|
|
|
|
|
|
irq += 1;
|
|
|
|
pending_interrupts >>= 1;
|
|
|
|
}
|
|
|
|
}
|
2022-12-22 11:56:40 +01:00
|
|
|
|
|
|
|
Processor::current().exit_trap(trap_frame);
|
2022-05-30 10:14:46 +02:00
|
|
|
}
|
|
|
|
|
2022-10-04 13:46:11 +03:00
|
|
|
// FIXME: Share the code below with Arch/x86_64/Interrupts.cpp
|
2022-05-30 10:14:46 +02:00
|
|
|
// While refactoring, the interrupt handlers can also be moved into the InterruptManagement class.
|
|
|
|
GenericInterruptHandler& get_interrupt_handler(u8 interrupt_number)
|
|
|
|
{
|
|
|
|
auto*& handler_slot = s_interrupt_handlers[interrupt_number];
|
|
|
|
VERIFY(handler_slot != nullptr);
|
|
|
|
return *handler_slot;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void revert_to_unused_handler(u8 interrupt_number)
|
2022-05-11 16:54:03 +02:00
|
|
|
{
|
2022-05-30 10:14:46 +02:00
|
|
|
auto handler = new UnhandledInterruptHandler(interrupt_number);
|
|
|
|
handler->register_interrupt_handler();
|
2022-05-11 16:54:03 +02:00
|
|
|
}
|
|
|
|
|
2022-05-30 10:14:46 +02:00
|
|
|
void register_generic_interrupt_handler(u8 interrupt_number, GenericInterruptHandler& handler)
|
2022-05-11 16:54:03 +02:00
|
|
|
{
|
2022-05-30 10:14:46 +02:00
|
|
|
auto*& handler_slot = s_interrupt_handlers[interrupt_number];
|
2023-01-17 20:47:51 +01:00
|
|
|
if (handler_slot == nullptr) {
|
2022-05-30 10:14:46 +02:00
|
|
|
handler_slot = &handler;
|
2023-01-17 20:47:51 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (handler_slot->type() == HandlerType::UnhandledInterruptHandler) {
|
|
|
|
auto* unhandled_handler = static_cast<UnhandledInterruptHandler*>(handler_slot);
|
|
|
|
unhandled_handler->unregister_interrupt_handler();
|
|
|
|
delete unhandled_handler;
|
|
|
|
handler_slot = &handler;
|
|
|
|
return;
|
2022-05-30 10:14:46 +02:00
|
|
|
}
|
2023-04-28 11:03:46 +02:00
|
|
|
if (handler_slot->is_shared_handler()) {
|
2023-01-17 20:47:51 +01:00
|
|
|
VERIFY(handler_slot->type() == HandlerType::SharedIRQHandler);
|
|
|
|
static_cast<SharedIRQHandler*>(handler_slot)->register_handler(handler);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!handler_slot->is_shared_handler()) {
|
|
|
|
if (handler_slot->type() == HandlerType::SpuriousInterruptHandler) {
|
|
|
|
// FIXME: Add support for spurious interrupts on aarch64
|
|
|
|
TODO_AARCH64();
|
|
|
|
}
|
|
|
|
VERIFY(handler_slot->type() == HandlerType::IRQHandler);
|
|
|
|
auto& previous_handler = *handler_slot;
|
|
|
|
handler_slot = nullptr;
|
|
|
|
SharedIRQHandler::initialize(interrupt_number);
|
|
|
|
VERIFY(handler_slot);
|
|
|
|
static_cast<SharedIRQHandler*>(handler_slot)->register_handler(previous_handler);
|
|
|
|
static_cast<SharedIRQHandler*>(handler_slot)->register_handler(handler);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
VERIFY_NOT_REACHED();
|
2022-05-11 16:54:03 +02:00
|
|
|
}
|
|
|
|
|
2022-05-30 10:14:46 +02:00
|
|
|
void unregister_generic_interrupt_handler(u8 interrupt_number, GenericInterruptHandler& handler)
|
2022-05-11 16:54:03 +02:00
|
|
|
{
|
2022-05-30 10:14:46 +02:00
|
|
|
auto*& handler_slot = s_interrupt_handlers[interrupt_number];
|
|
|
|
VERIFY(handler_slot != nullptr);
|
2023-01-17 20:47:51 +01:00
|
|
|
if (handler_slot->type() == HandlerType::UnhandledInterruptHandler)
|
2022-05-30 10:14:46 +02:00
|
|
|
return;
|
2023-04-28 11:03:46 +02:00
|
|
|
if (handler_slot->is_shared_handler()) {
|
2022-05-30 10:14:46 +02:00
|
|
|
VERIFY(handler_slot->type() == HandlerType::SharedIRQHandler);
|
|
|
|
auto* shared_handler = static_cast<SharedIRQHandler*>(handler_slot);
|
|
|
|
shared_handler->unregister_handler(handler);
|
|
|
|
if (!shared_handler->sharing_devices_count()) {
|
|
|
|
handler_slot = nullptr;
|
|
|
|
revert_to_unused_handler(interrupt_number);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!handler_slot->is_shared_handler()) {
|
|
|
|
VERIFY(handler_slot->type() == HandlerType::IRQHandler);
|
|
|
|
handler_slot = nullptr;
|
|
|
|
revert_to_unused_handler(interrupt_number);
|
|
|
|
return;
|
|
|
|
}
|
2022-05-11 16:54:03 +02:00
|
|
|
}
|
|
|
|
|
2022-05-17 11:07:02 +02:00
|
|
|
void initialize_interrupts()
|
|
|
|
{
|
2022-05-30 10:14:46 +02:00
|
|
|
for (u8 i = 0; i < s_interrupt_handlers.size(); ++i) {
|
|
|
|
auto* handler = new UnhandledInterruptHandler(i);
|
|
|
|
handler->register_interrupt_handler();
|
|
|
|
}
|
2022-05-17 11:07:02 +02:00
|
|
|
}
|
|
|
|
|
2022-05-11 16:54:03 +02:00
|
|
|
}
|