diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/boot.asm | 8 | ||||
-rw-r--r-- | src/idt/idt.h | 8 | ||||
-rw-r--r-- | src/interrupts/idt.asm | 6 | ||||
-rw-r--r-- | src/interrupts/idt.c (renamed from src/idt/idt.c) | 1 | ||||
-rw-r--r-- | src/interrupts/interrupts.h | 28 | ||||
-rw-r--r-- | src/interrupts/irq.c | 104 | ||||
-rw-r--r-- | src/interrupts/isr.asm | 436 | ||||
-rw-r--r-- | src/interrupts/isr.c | 155 | ||||
-rw-r--r-- | src/kernel.c | 4 |
9 files changed, 734 insertions, 16 deletions
diff --git a/src/boot.asm b/src/boot.asm index b45d64b..5e6c812 100644 --- a/src/boot.asm +++ b/src/boot.asm @@ -49,12 +49,8 @@ gdt_flush: flush2: ret ; Returns to C code -; IDT loader -global idt_load -extern idtp -idt_load: - lidt [idtp] - ret +%include "src/interrupts/idt.asm" +%include "src/interrupts/isr.asm" ; Store the stack SECTION .bss diff --git a/src/idt/idt.h b/src/idt/idt.h deleted file mode 100644 index 5771ff0..0000000 --- a/src/idt/idt.h +++ /dev/null @@ -1,8 +0,0 @@ -#ifndef MELVIX_IDT_H -#define MELVIX_IDT_H - -void idt_install(); - -void idt_set_gate(unsigned char num, unsigned long base, unsigned short sel, unsigned char flags); - -#endif diff --git a/src/interrupts/idt.asm b/src/interrupts/idt.asm new file mode 100644 index 0000000..90eab47 --- /dev/null +++ b/src/interrupts/idt.asm @@ -0,0 +1,6 @@ +; IDT loader +global idt_load +extern idtp +idt_load: + lidt [idtp] + ret diff --git a/src/idt/idt.c b/src/interrupts/idt.c index c442044..641de3a 100644 --- a/src/idt/idt.c +++ b/src/interrupts/idt.c @@ -1,6 +1,5 @@ #include "../memory/memory.h" -/* Defines an IDT entry */ struct idt_entry { unsigned short base_lo; unsigned short sel; // Kernel segment diff --git a/src/interrupts/interrupts.h b/src/interrupts/interrupts.h new file mode 100644 index 0000000..b91e91c --- /dev/null +++ b/src/interrupts/interrupts.h @@ -0,0 +1,28 @@ +#ifndef MELVIX_INTERRUPTS_H +#define MELVIX_INTERRUPTS_H + +// IDT +void idt_install(); + +void idt_set_gate(unsigned char num, unsigned long base, unsigned short sel, unsigned char flags); + +// ISRS +void isrs_install(); + +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; +}; + +// IRQ +void irq_install(); + +void irq_install_handler(int irq, void (*handler)(struct regs *r)); + +void irq_uninstall_handler(int irq); + +void irq_handler(struct regs *r); + +#endif diff --git a/src/interrupts/irq.c b/src/interrupts/irq.c new file mode 100644 index 0000000..686d543 --- /dev/null +++ b/src/interrupts/irq.c @@ -0,0 +1,104 @@ +#include "../io/io.h" +#include "interrupts.h" + +extern void irq0(); + +extern void irq1(); + +extern void irq2(); + +extern void irq3(); + +extern void irq4(); + +extern void irq5(); + +extern void irq6(); + +extern void irq7(); + +extern void irq8(); + +extern void irq9(); + +extern void irq10(); + +extern void irq11(); + +extern void irq12(); + +extern void irq13(); + +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 +}; + +// Install custom IRQ handler +void irq_install_handler(int irq, void (*handler)(struct regs *r)) { + irq_routines[irq] = handler; +} + +// Removes the custom IRQ handler +void irq_uninstall_handler(int irq) { + irq_routines[irq] = 0; +} + +// Remap IRQs for protected mode compatibility via the PIC +void irq_remap(void) { + send(0x20, 0x11); + send(0xA0, 0x11); + send(0x21, 0x20); + send(0xA1, 0x28); + send(0x21, 0x04); + send(0xA1, 0x02); + send(0x21, 0x01); + send(0xA1, 0x01); + send(0x21, 0x0); + send(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); +} + +// Handle IRQ ISRs +void irq_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); + } + + // Send end of interrupt to second (slave) IRQ controller + if (r->int_no >= 40) { + send(0xA0, 0x20); + } + + // Send end of interrupt to master interrupt controller + send(0x20, 0x20); +} diff --git a/src/interrupts/isr.asm b/src/interrupts/isr.asm new file mode 100644 index 0000000..713fecf --- /dev/null +++ b/src/interrupts/isr.asm @@ -0,0 +1,436 @@ +global isr0 +global isr1 +global isr2 +global isr3 +global isr4 +global isr5 +global isr6 +global isr7 +global isr8 +global isr9 +global isr10 +global isr11 +global isr12 +global isr13 +global isr14 +global isr15 +global isr16 +global isr17 +global isr18 +global isr19 +global isr20 +global isr21 +global isr22 +global isr23 +global isr24 +global isr25 +global isr26 +global isr27 +global isr28 +global isr29 +global isr30 +global isr31 + +; 0: Divide By Zero Exception +isr0: + cli + push byte 0 + push byte 0 + jmp isr_common_stub + +; 1: Debug Exception +isr1: + cli + push byte 0 + push byte 1 + jmp isr_common_stub + +; 2: Non Maskable Interrupt Exception +isr2: + cli + push byte 0 + push byte 2 + jmp isr_common_stub + +; 3: Int 3 Exception +isr3: + cli + push byte 0 + push byte 3 + jmp isr_common_stub + +; 4: INTO Exception +isr4: + cli + push byte 0 + push byte 4 + jmp isr_common_stub + +; 5: Out of Bounds Exception +isr5: + cli + push byte 0 + push byte 5 + jmp isr_common_stub + +; 6: Invalid Opcode Exception +isr6: + cli + push byte 0 + push byte 6 + jmp isr_common_stub + +; 7: Coprocessor Not Available Exception +isr7: + cli + push byte 0 + push byte 7 + jmp isr_common_stub + +; 8: Double Fault Exception (With Error Code!) +isr8: + cli + push byte 8 + jmp isr_common_stub + +; 9: Coprocessor Segment Overrun Exception +isr9: + cli + push byte 0 + push byte 9 + jmp isr_common_stub + +; 10: Bad TSS Exception (With Error Code!) +isr10: + cli + push byte 10 + jmp isr_common_stub + +; 11: Segment Not Present Exception (With Error Code!) +isr11: + cli + push byte 11 + jmp isr_common_stub + +; 12: Stack Fault Exception (With Error Code!) +isr12: + cli + push byte 12 + jmp isr_common_stub + +; 13: General Protection Fault Exception (With Error Code!) +isr13: + cli + push byte 13 + jmp isr_common_stub + +; 14: Page Fault Exception (With Error Code!) +isr14: + cli + push byte 14 + jmp isr_common_stub + +; 15: Reserved Exception +isr15: + cli + push byte 0 + push byte 15 + jmp isr_common_stub + +; 16: Floating Point Exception +isr16: + cli + push byte 0 + push byte 16 + jmp isr_common_stub + +; 17: Alignment Check Exception +isr17: + cli + push byte 0 + push byte 17 + jmp isr_common_stub + +; 18: Machine Check Exception +isr18: + cli + push byte 0 + push byte 18 + jmp isr_common_stub + +; 19: Reserved +isr19: + cli + push byte 0 + push byte 19 + jmp isr_common_stub + +; 20: Reserved +isr20: + cli + push byte 0 + push byte 20 + jmp isr_common_stub + +; 21: Reserved +isr21: + cli + push byte 0 + push byte 21 + jmp isr_common_stub + +; 22: Reserved +isr22: + cli + push byte 0 + push byte 22 + jmp isr_common_stub + +; 23: Reserved +isr23: + cli + push byte 0 + push byte 23 + jmp isr_common_stub + +; 24: Reserved +isr24: + cli + push byte 0 + push byte 24 + jmp isr_common_stub + +; 25: Reserved +isr25: + cli + push byte 0 + push byte 25 + jmp isr_common_stub + +; 26: Reserved +isr26: + cli + push byte 0 + push byte 26 + jmp isr_common_stub + +; 27: Reserved +isr27: + cli + push byte 0 + push byte 27 + jmp isr_common_stub + +; 28: Reserved +isr28: + cli + push byte 0 + push byte 28 + jmp isr_common_stub + +; 29: Reserved +isr29: + cli + push byte 0 + push byte 29 + jmp isr_common_stub + +; 30: Reserved +isr30: + cli + push byte 0 + push byte 30 + jmp isr_common_stub + +; 31: Reserved +isr31: + cli + push byte 0 + push byte 31 + jmp isr_common_stub + + +extern fault_handler + +; Stores the ISR in the stack and calls the C fault handler +isr_common_stub: + pusha + push ds + push es + push fs + push gs + mov ax, 0x10 + mov ds, ax + mov es, ax + mov fs, ax + mov gs, ax + mov eax, esp + push eax + mov eax, fault_handler + call eax + pop eax + pop gs + pop fs + pop es + pop ds + popa + add esp, 8 + iret + +global irq0 +global irq1 +global irq2 +global irq3 +global irq4 +global irq5 +global irq6 +global irq7 +global irq8 +global irq9 +global irq10 +global irq11 +global irq12 +global irq13 +global irq14 +global irq15 + +; 32: IRQ0 +irq0: + cli + push byte 0 + push byte 32 + jmp irq_common_stub + +; 33: IRQ1 +irq1: + cli + push byte 0 + push byte 33 + jmp irq_common_stub + +; 34: IRQ2 +irq2: + cli + push byte 0 + push byte 34 + jmp irq_common_stub + +; 35: IRQ3 +irq3: + cli + push byte 0 + push byte 35 + jmp irq_common_stub + +; 36: IRQ4 +irq4: + cli + push byte 0 + push byte 36 + jmp irq_common_stub + +; 37: IRQ5 +irq5: + cli + push byte 0 + push byte 37 + jmp irq_common_stub + +; 38: IRQ6 +irq6: + cli + push byte 0 + push byte 38 + jmp irq_common_stub + +; 39: IRQ7 +irq7: + cli + push byte 0 + push byte 39 + jmp irq_common_stub + +; 40: IRQ8 +irq8: + cli + push byte 0 + push byte 40 + jmp irq_common_stub + +; 41: IRQ9 +irq9: + cli + push byte 0 + push byte 41 + jmp irq_common_stub + +; 42: IRQ10 +irq10: + cli + push byte 0 + push byte 42 + jmp irq_common_stub + +; 43: IRQ11 +irq11: + cli + push byte 0 + push byte 43 + jmp irq_common_stub + +; 44: IRQ12 +irq12: + cli + push byte 0 + push byte 44 + jmp irq_common_stub + +; 45: IRQ13 +irq13: + cli + push byte 0 + push byte 45 + jmp irq_common_stub + +; 46: IRQ14 +irq14: + cli + push byte 0 + push byte 46 + jmp irq_common_stub + +; 47: IRQ15 +irq15: + cli + push byte 0 + push byte 47 + jmp irq_common_stub + +extern irq_handler + +irq_common_stub: + pusha + push ds + push es + push fs + push gs + + mov ax, 0x10 + mov ds, ax + mov es, ax + mov fs, ax + mov gs, ax + mov eax, esp + + push eax + mov eax, irq_handler + call eax + pop eax + + pop gs + pop fs + pop es + pop ds + popa + add esp, 8 + iret diff --git a/src/interrupts/isr.c b/src/interrupts/isr.c new file mode 100644 index 0000000..aea1ce6 --- /dev/null +++ b/src/interrupts/isr.c @@ -0,0 +1,155 @@ +#include "../graphics/vga.h" +#include "interrupts.h" + +// Defined in isr.asm +extern void isr0(); + +extern void isr1(); + +extern void isr2(); + +extern void isr3(); + +extern void isr4(); + +extern void isr5(); + +extern void isr6(); + +extern void isr7(); + +extern void isr8(); + +extern void isr9(); + +extern void isr10(); + +extern void isr11(); + +extern void isr12(); + +extern void isr13(); + +extern void isr14(); + +extern void isr15(); + +extern void isr16(); + +extern void isr17(); + +extern void isr18(); + +extern void isr19(); + +extern void isr20(); + +extern void isr21(); + +extern void isr22(); + +extern void isr23(); + +extern void isr24(); + +extern void isr25(); + +extern void isr26(); + +extern void isr27(); + +extern void isr28(); + +extern void isr29(); + +extern void isr30(); + +extern void isr31(); + +// 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); +} + +// Error exception messages +unsigned char *exception_messages[] = { + "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 handler - halt via endless loop +void fault_handler(struct regs *r) { + if (r->int_no < 32) { + terminal_write_string("\n"); + terminal_write_string(exception_messages[r->int_no]); + terminal_write_string(" Exception. System Halted!\n"); + for (;;); + } +} diff --git a/src/kernel.c b/src/kernel.c index 8e2f534..4f1281a 100644 --- a/src/kernel.c +++ b/src/kernel.c @@ -1,10 +1,12 @@ #include "graphics/vga.h" #include "gdt/gdt.h" -#include "idt/idt.h" +#include "interrupts/interrupts.h" void kernel_main(void) { gdt_install(); idt_install(); + isrs_install(); terminal_initialize(); terminal_write_string("Melvix loaded successfully!\nTest"); + // __asm__ ("div %0" :: "r"(0)); Exception testing x/0 }
\ No newline at end of file |