94 lines
3.1 KiB
C
94 lines
3.1 KiB
C
|
#include "idt.h"
|
||
|
#include "../libc/string.h"
|
||
|
|
||
|
// Internal method
|
||
|
extern void idt_flush(idt_ptr_t*); // defined on cpu/idt.asm
|
||
|
static void init_idt();
|
||
|
static void idt_set_gate(uint8_t idx, void(*base), uint16_t selector, idt_flags_t flags);
|
||
|
|
||
|
idt_entry_t idt_entries[256]; // 256 interrupts
|
||
|
idt_ptr_t idt_ptr;
|
||
|
|
||
|
// This method will be called by the kernel
|
||
|
void idt_setup() {
|
||
|
init_idt();
|
||
|
}
|
||
|
|
||
|
static void idt_set_gate(uint8_t idx, void(*base), uint16_t selector, idt_flags_t flags) {
|
||
|
idt_entries[idx] = (struct idt_entry) {
|
||
|
.base_low = (uint32_t)base & 0xFFFF,
|
||
|
.base_high = ((uint32_t)base >> 16) & 0xFFFF,
|
||
|
.seg_sel = selector,
|
||
|
.flags = flags
|
||
|
};
|
||
|
}
|
||
|
|
||
|
static void init_idt() {
|
||
|
idt_ptr.limit = sizeof(idt_entry_t) * 256 - 1;
|
||
|
idt_ptr.base = idt_entries;
|
||
|
|
||
|
memset(&idt_entries, 0, sizeof(idt_entry_t) * 256);
|
||
|
|
||
|
idt_flags_t flags = {
|
||
|
.reserved = IDT_FLAG_RESERVED, // Always 0x0E
|
||
|
.dpl = 0,
|
||
|
.p = 1
|
||
|
};
|
||
|
// Remap IDT with ISRs
|
||
|
idt_set_gate(0, isr0, 0x08, flags);
|
||
|
idt_set_gate(1, isr1, 0x08, flags);
|
||
|
idt_set_gate(2, isr2, 0x08, flags);
|
||
|
idt_set_gate(3, isr3, 0x08, flags);
|
||
|
idt_set_gate(4, isr4, 0x08, flags);
|
||
|
idt_set_gate(5, isr5, 0x08, flags);
|
||
|
idt_set_gate(6, isr6, 0x08, flags);
|
||
|
idt_set_gate(7, isr7, 0x08, flags);
|
||
|
idt_set_gate(8, isr8, 0x08, flags);
|
||
|
idt_set_gate(9, isr9, 0x08, flags);
|
||
|
idt_set_gate(10, isr10, 0x08, flags);
|
||
|
idt_set_gate(11, isr11, 0x08, flags);
|
||
|
idt_set_gate(12, isr12, 0x08, flags);
|
||
|
idt_set_gate(13, isr13, 0x08, flags);
|
||
|
idt_set_gate(14, isr14, 0x08, flags);
|
||
|
idt_set_gate(15, isr15, 0x08, flags);
|
||
|
idt_set_gate(16, isr16, 0x08, flags);
|
||
|
idt_set_gate(17, isr17, 0x08, flags);
|
||
|
idt_set_gate(18, isr18, 0x08, flags);
|
||
|
idt_set_gate(19, isr19, 0x08, flags);
|
||
|
idt_set_gate(20, isr20, 0x08, flags);
|
||
|
idt_set_gate(21, isr21, 0x08, flags);
|
||
|
idt_set_gate(22, isr22, 0x08, flags);
|
||
|
idt_set_gate(23, isr23, 0x08, flags);
|
||
|
idt_set_gate(24, isr24, 0x08, flags);
|
||
|
idt_set_gate(25, isr25, 0x08, flags);
|
||
|
idt_set_gate(26, isr26, 0x08, flags);
|
||
|
idt_set_gate(27, isr27, 0x08, flags);
|
||
|
idt_set_gate(28, isr28, 0x08, flags);
|
||
|
idt_set_gate(29, isr29, 0x08, flags);
|
||
|
idt_set_gate(30, isr30, 0x08, flags);
|
||
|
idt_set_gate(31, isr31, 0x08, flags);
|
||
|
|
||
|
// Also remap 15 entries for IRQs
|
||
|
idt_set_gate(32, irq0, 0x08, flags);
|
||
|
idt_set_gate(33, irq1, 0x08, flags);
|
||
|
idt_set_gate(34, irq2, 0x08, flags);
|
||
|
idt_set_gate(35, irq3, 0x08, flags);
|
||
|
idt_set_gate(36, irq4, 0x08, flags);
|
||
|
idt_set_gate(37, irq5, 0x08, flags);
|
||
|
idt_set_gate(38, irq6, 0x08, flags);
|
||
|
idt_set_gate(39, irq7, 0x08, flags);
|
||
|
idt_set_gate(40, irq8, 0x08, flags);
|
||
|
idt_set_gate(41, irq9, 0x08, flags);
|
||
|
idt_set_gate(42, irq10, 0x08, flags);
|
||
|
idt_set_gate(43, irq11, 0x08, flags);
|
||
|
idt_set_gate(44, irq12, 0x08, flags);
|
||
|
idt_set_gate(45, irq13, 0x08, flags);
|
||
|
idt_set_gate(46, irq14, 0x08, flags);
|
||
|
idt_set_gate(47, irq15, 0x08, flags);
|
||
|
|
||
|
idt_flush(&idt_ptr);
|
||
|
|
||
|
// Finally enable hardware interrupts with an assembly instruction
|
||
|
__asm__ __volatile__ ("sti");
|
||
|
|
||
|
}
|