diff options
Diffstat (limited to 'src/kernel/interrupts')
-rw-r--r-- | src/kernel/interrupts/idt.c | 44 | ||||
-rw-r--r-- | src/kernel/interrupts/interrupts.h | 8 | ||||
-rw-r--r-- | src/kernel/interrupts/irq.c | 87 | ||||
-rw-r--r-- | src/kernel/interrupts/isr.c | 207 |
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 |