mirror of
https://gitlab.com/sortix/sortix.git
synced 2023-02-13 20:55:38 -05:00
147 lines
4.2 KiB
C++
147 lines
4.2 KiB
C++
/******************************************************************************
|
|
|
|
COPYRIGHT(C) JONAS 'SORTIE' TERMANSEN 2011.
|
|
|
|
This file is part of Sortix.
|
|
|
|
Sortix is free software: you can redistribute it and/or modify it under the
|
|
terms of the GNU General Public License as published by the Free Software
|
|
Foundation, either version 3 of the License, or (at your option) any later
|
|
version.
|
|
|
|
Sortix is distributed in the hope that 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 Sortix. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
interrupt.cpp
|
|
High level interrupt service routines and interrupt request handlers.
|
|
|
|
******************************************************************************/
|
|
|
|
#include "platform.h"
|
|
#include "log.h"
|
|
#include "interrupt.h"
|
|
#include "panic.h"
|
|
|
|
#include "process.h" // Hack for SIGSEGV
|
|
#include "sound.h" // Hack for SIGSEGV
|
|
#include "thread.h" // HACK FOR SIGSEGV
|
|
#include "syscall.h" // HACK FOR SIGSEGV
|
|
#include "scheduler.h" // HACK FOR SIGSEGV
|
|
|
|
namespace Sortix
|
|
{
|
|
void SysExit(int status); // HACK
|
|
|
|
namespace Interrupt
|
|
{
|
|
const bool DEBUG_EXCEPTION = false;
|
|
const bool DEBUG_IRQ = false;
|
|
|
|
size_t numknownexceptions = 20;
|
|
const char* exceptions[] =
|
|
{ "Divide by zero", "Debug", "Non maskable interrupt", "Breakpoint",
|
|
"Into detected overflow", "Out of bounds", "Invalid opcode",
|
|
"No coprocessor", "Double fault", "Coprocessor segment overrun",
|
|
"Bad TSS", "Segment not present", "Stack fault",
|
|
"General protection fault", "Page fault", "Unknown interrupt",
|
|
"Coprocessor fault", "Alignment check", "Machine check",
|
|
"SIMD Floating-Point" };
|
|
|
|
Handler interrupthandlers[256];
|
|
|
|
void Init()
|
|
{
|
|
for ( size_t i = 0; i < 256; i++ )
|
|
{
|
|
interrupthandlers[i] = NULL;
|
|
}
|
|
}
|
|
|
|
void RegisterHandler(uint8_t n, Interrupt::Handler handler)
|
|
{
|
|
interrupthandlers[n] = handler;
|
|
}
|
|
|
|
// This gets called from our ASM interrupt handler stub.
|
|
extern "C" void ISRHandler(Sortix::CPU::InterruptRegisters* regs)
|
|
{
|
|
if ( regs->int_no < 32 )
|
|
{
|
|
const char* message = ( regs->int_no < numknownexceptions )
|
|
? exceptions[regs->int_no] : "Unknown";
|
|
|
|
if ( DEBUG_EXCEPTION ) { regs->LogRegisters(); Log::Print("\n"); }
|
|
|
|
#ifdef PLATFORM_X64
|
|
addr_t ip = regs->rip;
|
|
#else
|
|
addr_t ip = regs->eip;
|
|
#endif
|
|
|
|
// Halt and catch fire if we are the kernel.
|
|
if ( (regs->cs & (0x4-1)) == 0 )
|
|
{
|
|
PanicF("Unhandled CPU Exception id %zu '%s' at ip=0x%zx "
|
|
"(cr2=0x%p, err_code=0x%p)", regs->int_no, message,
|
|
ip, regs->cr2, regs->err_code);
|
|
}
|
|
|
|
Log::Print("The current program has crashed and was terminated:\n");
|
|
Log::PrintF("%s exception at ip=0x%zx (cr2=0x%p, err_code=0x%p)\n",
|
|
message, ip, regs->cr2, regs->err_code);
|
|
|
|
Sound::Mute();
|
|
|
|
CurrentProcess()->Exit(139);
|
|
Scheduler::ProcessTerminated(regs);
|
|
|
|
return;
|
|
}
|
|
|
|
if ( interrupthandlers[regs->int_no] != NULL )
|
|
{
|
|
interrupthandlers[regs->int_no](regs);
|
|
}
|
|
}
|
|
|
|
// This gets called from our ASM interrupt handler stub.
|
|
extern "C" void IRQHandler(Sortix::CPU::InterruptRegisters* regs)
|
|
{
|
|
// TODO! IRQ 7 and 15 might be spurious and might need to be ignored.
|
|
// See http://wiki.osdev.org/PIC for details (section Spurious IRQs).
|
|
if ( regs->int_no == 32 + 7 || regs->int_no == 32 + 15 ) { return; }
|
|
|
|
if ( DEBUG_IRQ )
|
|
{
|
|
Log::PrintF("IRQ%u ", regs->int_no-32);
|
|
regs->LogRegisters();
|
|
Log::Print("\n");
|
|
}
|
|
|
|
// Send an EOI (end of interrupt) signal to the PICs.
|
|
|
|
// Send reset signal to slave if this interrupt involved the slave.
|
|
if (regs->int_no >= 40) { CPU::OutPortB(0xA0, 0x20); }
|
|
|
|
// Send reset signal to master.
|
|
CPU::OutPortB(0x20, 0x20);
|
|
|
|
if ( interrupthandlers[regs->int_no] )
|
|
{
|
|
interrupthandlers[regs->int_no](regs);
|
|
}
|
|
}
|
|
|
|
extern "C" void interrupt_handler(Sortix::CPU::InterruptRegisters* regs)
|
|
{
|
|
size_t int_no = regs->int_no;
|
|
if ( 32 <= int_no && int_no < 48 ) { IRQHandler(regs); }
|
|
else { ISRHandler(regs); }
|
|
}
|
|
}
|
|
}
|