aboutsummaryrefslogtreecommitdiff
path: root/src/kernel/interrupts
diff options
context:
space:
mode:
Diffstat (limited to 'src/kernel/interrupts')
-rw-r--r--src/kernel/interrupts/idt.c44
-rw-r--r--src/kernel/interrupts/interrupts.h8
-rw-r--r--src/kernel/interrupts/irq.c87
-rw-r--r--src/kernel/interrupts/isr.c207
4 files changed, 170 insertions, 176 deletions
diff --git a/src/kernel/interrupts/idt.c b/src/kernel/interrupts/idt.c
index 026b23c..7c762b3 100644
--- a/src/kernel/interrupts/idt.c
+++ b/src/kernel/interrupts/idt.c
@@ -2,16 +2,16 @@
#include <kernel/system.h>
struct idt_entry {
- uint16_t base_low;
- uint16_t sel; // Kernel segment
- uint8_t always0; // Always 0
- uint8_t flags;
- uint16_t base_high;
+ uint16_t base_low;
+ uint16_t sel; // Kernel segment
+ uint8_t always0; // Always 0
+ uint8_t flags;
+ uint16_t base_high;
} __attribute__((packed));
struct idt_ptr {
- unsigned short limit;
- void *base;
+ unsigned short limit;
+ void *base;
} __attribute__((packed));
// Initialize IDT with 256 entries
@@ -23,26 +23,26 @@ extern void idt_load();
void idt_set_gate(unsigned char num, unsigned long base, unsigned short sel, unsigned char flags)
{
- // Specify the interrupt routine's base address
- idt[num].base_low = (uint16_t) (base & 0xFFFF);
- idt[num].base_high = (uint16_t) ((base >> 16) & 0xFFFF);
-
- // Set selector/segment of IDT entry
- idt[num].sel = sel;
- idt[num].always0 = 0;
- idt[num].flags = (uint8_t) (flags | 0x60);
+ // Specify the interrupt routine's base address
+ idt[num].base_low = (uint16_t)(base & 0xFFFF);
+ idt[num].base_high = (uint16_t)((base >> 16) & 0xFFFF);
+
+ // Set selector/segment of IDT entry
+ idt[num].sel = sel;
+ idt[num].always0 = 0;
+ idt[num].flags = (uint8_t)(flags | 0x60);
}
// Install IDT
void idt_install()
{
- // Set IDT pointer and limit
- idtp.limit = (sizeof(struct idt_entry) * 256) - 1;
- idtp.base = &idt;
+ // Set IDT pointer and limit
+ idtp.limit = (sizeof(struct idt_entry) * 256) - 1;
+ idtp.base = &idt;
- // Clear IDT by setting memory cells to 0
- memset(&idt, 0, sizeof(struct idt_entry) * 256);
+ // Clear IDT by setting memory cells to 0
+ memset(&idt, 0, sizeof(struct idt_entry) * 256);
- idt_load();
- vga_log("Installed Interrupt Descriptor Table");
+ idt_load();
+ vga_log("Installed Interrupt Descriptor Table");
} \ No newline at end of file
diff --git a/src/kernel/interrupts/interrupts.h b/src/kernel/interrupts/interrupts.h
index 10612c0..9047104 100644
--- a/src/kernel/interrupts/interrupts.h
+++ b/src/kernel/interrupts/interrupts.h
@@ -22,10 +22,10 @@ void idt_set_gate(unsigned char num, unsigned long base, unsigned short sel, uns
* Registers that get passed into an IRQ handler
*/
struct regs {
- unsigned int gs, fs, es, ds;
- unsigned int edi, esi, ebp, esp, ebx, edx, ecx, eax;
- unsigned int int_no, err_code;
- unsigned int eip, cs, eflags, useresp, ss;
+ unsigned int gs, fs, es, ds;
+ unsigned int edi, esi, ebp, esp, ebx, edx, ecx, eax;
+ unsigned int int_no, err_code;
+ unsigned int eip, cs, eflags, useresp, ss;
};
/**
diff --git a/src/kernel/interrupts/irq.c b/src/kernel/interrupts/irq.c
index dd8ce90..f23e14a 100644
--- a/src/kernel/interrupts/irq.c
+++ b/src/kernel/interrupts/irq.c
@@ -35,80 +35,77 @@ extern void irq14();
extern void irq15();
// Array to handle custom IRQ handlers
-void *irq_routines[16] = {
- 0, 0, 0, 0, 0, 0, 0, 0,
- 0, 0, 0, 0, 0, 0, 0, 0
-};
+void *irq_routines[16] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
// Install custom IRQ handler
void irq_install_handler(int irq, void (*handler)(struct regs *r))
{
- irq_routines[irq] = handler;
+ irq_routines[irq] = handler;
}
// Removes the custom IRQ handler
void irq_uninstall_handler(int irq)
{
- irq_routines[irq] = 0;
+ irq_routines[irq] = 0;
}
int irq_is_installed(int irq)
{
- return irq_routines[irq] != 0;
+ return irq_routines[irq] != 0;
}
// Remap IRQs for protected mode compatibility via the PIC
void irq_remap(void)
{
- outb(0x20, 0x11);
- outb(0xA0, 0x11);
- outb(0x21, 0x20);
- outb(0xA1, 0x28);
- outb(0x21, 0x04);
- outb(0xA1, 0x02);
- outb(0x21, 0x01);
- outb(0xA1, 0x01);
- outb(0x21, 0x0);
- outb(0xA1, 0x0);
+ outb(0x20, 0x11);
+ outb(0xA0, 0x11);
+ outb(0x21, 0x20);
+ outb(0xA1, 0x28);
+ outb(0x21, 0x04);
+ outb(0xA1, 0x02);
+ outb(0x21, 0x01);
+ outb(0xA1, 0x01);
+ outb(0x21, 0x0);
+ outb(0xA1, 0x0);
}
// Map ISRs to the correct entries in the IDT
void irq_install()
{
- irq_remap();
- idt_set_gate(32, (unsigned) irq0, 0x08, 0x8E);
- idt_set_gate(33, (unsigned) irq1, 0x08, 0x8E);
- idt_set_gate(34, (unsigned) irq2, 0x08, 0x8E);
- idt_set_gate(35, (unsigned) irq3, 0x08, 0x8E);
- idt_set_gate(36, (unsigned) irq4, 0x08, 0x8E);
- idt_set_gate(37, (unsigned) irq5, 0x08, 0x8E);
- idt_set_gate(38, (unsigned) irq6, 0x08, 0x8E);
- idt_set_gate(39, (unsigned) irq7, 0x08, 0x8E);
- idt_set_gate(40, (unsigned) irq8, 0x08, 0x8E);
- idt_set_gate(41, (unsigned) irq9, 0x08, 0x8E);
- idt_set_gate(42, (unsigned) irq10, 0x08, 0x8E);
- idt_set_gate(43, (unsigned) irq11, 0x08, 0x8E);
- idt_set_gate(44, (unsigned) irq12, 0x08, 0x8E);
- idt_set_gate(45, (unsigned) irq13, 0x08, 0x8E);
- idt_set_gate(46, (unsigned) irq14, 0x08, 0x8E);
- idt_set_gate(47, (unsigned) irq15, 0x08, 0x8E);
- vga_log("Installed Interrupt Requests");
+ irq_remap();
+ idt_set_gate(32, (unsigned)irq0, 0x08, 0x8E);
+ idt_set_gate(33, (unsigned)irq1, 0x08, 0x8E);
+ idt_set_gate(34, (unsigned)irq2, 0x08, 0x8E);
+ idt_set_gate(35, (unsigned)irq3, 0x08, 0x8E);
+ idt_set_gate(36, (unsigned)irq4, 0x08, 0x8E);
+ idt_set_gate(37, (unsigned)irq5, 0x08, 0x8E);
+ idt_set_gate(38, (unsigned)irq6, 0x08, 0x8E);
+ idt_set_gate(39, (unsigned)irq7, 0x08, 0x8E);
+ idt_set_gate(40, (unsigned)irq8, 0x08, 0x8E);
+ idt_set_gate(41, (unsigned)irq9, 0x08, 0x8E);
+ idt_set_gate(42, (unsigned)irq10, 0x08, 0x8E);
+ idt_set_gate(43, (unsigned)irq11, 0x08, 0x8E);
+ idt_set_gate(44, (unsigned)irq12, 0x08, 0x8E);
+ idt_set_gate(45, (unsigned)irq13, 0x08, 0x8E);
+ idt_set_gate(46, (unsigned)irq14, 0x08, 0x8E);
+ idt_set_gate(47, (unsigned)irq15, 0x08, 0x8E);
+ vga_log("Installed Interrupt Requests");
}
// Handle IRQ ISRs
void irq_handler(struct regs *r)
{
- void (*handler)(struct regs *r);
+ void (*handler)(struct regs * r);
- // Execute custom handler if exists
- handler = irq_routines[r->int_no - 32];
- if (handler)
- handler(r);
+ // Execute custom handler if exists
+ handler = irq_routines[r->int_no - 32];
+ if (handler)
+ handler(r);
- // Send end of interrupt to second (slave) IRQ controller
- if (r->int_no >= 40)
- outb(0xA0, 0x20);
+ // Send end of interrupt to second (slave) IRQ controller
+ if (r->int_no >= 40)
+ outb(0xA0, 0x20);
- // Send end of interrupt to master interrupt controller
- outb(0x20, 0x20);
+ // Send end of interrupt to master interrupt controller
+ outb(0x20, 0x20);
} \ No newline at end of file
diff --git a/src/kernel/interrupts/isr.c b/src/kernel/interrupts/isr.c
index 42dbaa6..61d0fcc 100644
--- a/src/kernel/interrupts/isr.c
+++ b/src/kernel/interrupts/isr.c
@@ -8,127 +8,124 @@
// Install ISRs in IDT
void isrs_install()
{
- idt_set_gate(0, (unsigned) isr0, 0x08, 0x8E);
- idt_set_gate(1, (unsigned) isr1, 0x08, 0x8E);
- idt_set_gate(2, (unsigned) isr2, 0x08, 0x8E);
- idt_set_gate(3, (unsigned) isr3, 0x08, 0x8E);
- idt_set_gate(4, (unsigned) isr4, 0x08, 0x8E);
- idt_set_gate(5, (unsigned) isr5, 0x08, 0x8E);
- idt_set_gate(6, (unsigned) isr6, 0x08, 0x8E);
- idt_set_gate(7, (unsigned) isr7, 0x08, 0x8E);
-
- idt_set_gate(8, (unsigned) isr8, 0x08, 0x8E);
- idt_set_gate(9, (unsigned) isr9, 0x08, 0x8E);
- idt_set_gate(10, (unsigned) isr10, 0x08, 0x8E);
- idt_set_gate(11, (unsigned) isr11, 0x08, 0x8E);
- idt_set_gate(12, (unsigned) isr12, 0x08, 0x8E);
- idt_set_gate(13, (unsigned) isr13, 0x08, 0x8E);
- idt_set_gate(14, (unsigned) isr14, 0x08, 0x8E);
- idt_set_gate(15, (unsigned) isr15, 0x08, 0x8E);
-
- idt_set_gate(16, (unsigned) isr16, 0x08, 0x8E);
- idt_set_gate(17, (unsigned) isr17, 0x08, 0x8E);
- idt_set_gate(18, (unsigned) isr18, 0x08, 0x8E);
- idt_set_gate(19, (unsigned) isr19, 0x08, 0x8E);
- idt_set_gate(20, (unsigned) isr20, 0x08, 0x8E);
- idt_set_gate(21, (unsigned) isr21, 0x08, 0x8E);
- idt_set_gate(22, (unsigned) isr22, 0x08, 0x8E);
- idt_set_gate(23, (unsigned) isr23, 0x08, 0x8E);
-
- idt_set_gate(24, (unsigned) isr24, 0x08, 0x8E);
- idt_set_gate(25, (unsigned) isr25, 0x08, 0x8E);
- idt_set_gate(26, (unsigned) isr26, 0x08, 0x8E);
- idt_set_gate(27, (unsigned) isr27, 0x08, 0x8E);
- idt_set_gate(28, (unsigned) isr28, 0x08, 0x8E);
- idt_set_gate(29, (unsigned) isr29, 0x08, 0x8E);
- idt_set_gate(30, (unsigned) isr30, 0x08, 0x8E);
- idt_set_gate(31, (unsigned) isr31, 0x08, 0x8E);
-
- idt_set_gate(0x80, (unsigned) isr128, 0x08, 0xEE);
-
- vga_log("Installed Interrupt Service Routines");
+ idt_set_gate(0, (unsigned)isr0, 0x08, 0x8E);
+ idt_set_gate(1, (unsigned)isr1, 0x08, 0x8E);
+ idt_set_gate(2, (unsigned)isr2, 0x08, 0x8E);
+ idt_set_gate(3, (unsigned)isr3, 0x08, 0x8E);
+ idt_set_gate(4, (unsigned)isr4, 0x08, 0x8E);
+ idt_set_gate(5, (unsigned)isr5, 0x08, 0x8E);
+ idt_set_gate(6, (unsigned)isr6, 0x08, 0x8E);
+ idt_set_gate(7, (unsigned)isr7, 0x08, 0x8E);
+
+ idt_set_gate(8, (unsigned)isr8, 0x08, 0x8E);
+ idt_set_gate(9, (unsigned)isr9, 0x08, 0x8E);
+ idt_set_gate(10, (unsigned)isr10, 0x08, 0x8E);
+ idt_set_gate(11, (unsigned)isr11, 0x08, 0x8E);
+ idt_set_gate(12, (unsigned)isr12, 0x08, 0x8E);
+ idt_set_gate(13, (unsigned)isr13, 0x08, 0x8E);
+ idt_set_gate(14, (unsigned)isr14, 0x08, 0x8E);
+ idt_set_gate(15, (unsigned)isr15, 0x08, 0x8E);
+
+ idt_set_gate(16, (unsigned)isr16, 0x08, 0x8E);
+ idt_set_gate(17, (unsigned)isr17, 0x08, 0x8E);
+ idt_set_gate(18, (unsigned)isr18, 0x08, 0x8E);
+ idt_set_gate(19, (unsigned)isr19, 0x08, 0x8E);
+ idt_set_gate(20, (unsigned)isr20, 0x08, 0x8E);
+ idt_set_gate(21, (unsigned)isr21, 0x08, 0x8E);
+ idt_set_gate(22, (unsigned)isr22, 0x08, 0x8E);
+ idt_set_gate(23, (unsigned)isr23, 0x08, 0x8E);
+
+ idt_set_gate(24, (unsigned)isr24, 0x08, 0x8E);
+ idt_set_gate(25, (unsigned)isr25, 0x08, 0x8E);
+ idt_set_gate(26, (unsigned)isr26, 0x08, 0x8E);
+ idt_set_gate(27, (unsigned)isr27, 0x08, 0x8E);
+ idt_set_gate(28, (unsigned)isr28, 0x08, 0x8E);
+ idt_set_gate(29, (unsigned)isr29, 0x08, 0x8E);
+ idt_set_gate(30, (unsigned)isr30, 0x08, 0x8E);
+ idt_set_gate(31, (unsigned)isr31, 0x08, 0x8E);
+
+ idt_set_gate(0x80, (unsigned)isr128, 0x08, 0xEE);
+
+ vga_log("Installed Interrupt Service Routines");
}
-irq_handler_t isr_routines[256] = {0};
+irq_handler_t isr_routines[256] = { 0 };
// Install custom IRQ handler
void isr_install_handler(size_t isr, irq_handler_t handler)
{
- isr_routines[isr] = handler;
+ isr_routines[isr] = handler;
}
// Removes the custom IRQ handler
void isr_uninstall_handler(size_t isr)
{
- isr_routines[isr] = 0;
+ isr_routines[isr] = 0;
}
// Error exception messages
-const char *exception_messages[32] = {
- "Division 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",
- "Reserved",
- "Reserved",
- "Reserved",
- "Reserved",
- "Reserved",
-
- "Reserved",
- "Reserved",
- "Reserved",
- "Reserved",
- "Reserved",
- "Reserved",
- "Reserved",
- "Reserved"
-};
+const char *exception_messages[32] = { "Division 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",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved" };
// Master exception/interrupt/fault handler - halt via panic
void fault_handler(struct regs *r)
{
- irq_handler_t handler = isr_routines[r->int_no];
- if (handler) {
- handler(r);
- } else {
- uint32_t faulting_address;
- asm ("mov %%cr2, %0" : "=r" (faulting_address));
-
- serial_printf(
- "\n[DEBUG]\nEIP: 0x%x\nEAX: 0x%x\nEBX: 0x%x\nECX: 0x%x\nEDX: 0x%x\nESP: 0x%x\nFault addr: 0x%x\nErr flag: 0x%x\nErr code: 0x%x\nINT code: 0x%x\nINT msg: %s",
- r->eip, r->eax, r->ebx, r->ecx, r->edx, r->esp, faulting_address, r->eflags, r->err_code, r->int_no,
- exception_messages[r->int_no]
- );
-
- if (r->int_no <= 32) {
- char *message = (char *) exception_messages[r->int_no];
- strcat(message, " Exception");
-
- // Show message if there wasn't an error in video memory
- if (faulting_address != (uint32_t) fb)
- panic(message);
- else
- halt_loop();
- } else {
- panic("Unknown Exception");
- }
- }
+ irq_handler_t handler = isr_routines[r->int_no];
+ if (handler) {
+ handler(r);
+ } else {
+ uint32_t faulting_address;
+ asm("mov %%cr2, %0" : "=r"(faulting_address));
+
+ serial_printf(
+ "\n[DEBUG]\nEIP: 0x%x\nEAX: 0x%x\nEBX: 0x%x\nECX: 0x%x\nEDX: 0x%x\nESP: 0x%x\nFault addr: 0x%x\nErr flag: 0x%x\nErr code: 0x%x\nINT code: 0x%x\nINT msg: %s",
+ r->eip, r->eax, r->ebx, r->ecx, r->edx, r->esp, faulting_address, r->eflags,
+ r->err_code, r->int_no, exception_messages[r->int_no]);
+
+ if (r->int_no <= 32) {
+ char *message = (char *)exception_messages[r->int_no];
+ strcat(message, " Exception");
+
+ // Show message if there wasn't an error in video memory
+ if (faulting_address != (uint32_t)fb)
+ panic(message);
+ else
+ halt_loop();
+ } else {
+ panic("Unknown Exception");
+ }
+ }
} \ No newline at end of file