aboutsummaryrefslogtreecommitdiff
path: root/src/kernel/interrupts/idt.c
blob: 3fb65b04d5b6a9b7b182773204d2a86667b68603 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
#include "../lib/lib.h"

struct idt_entry {
    unsigned short base_lo;
    unsigned short sel; // Kernel segment
    unsigned char always0; // Always 0
    unsigned char flags;
    unsigned short base_hi;
} __attribute__((packed));

struct idt_ptr {
    unsigned short limit;
    void *base;
} __attribute__((packed));

// Initialize IDT with 256 entries
struct idt_entry idt[256];
struct idt_ptr idtp;

// Defined in idt.asm
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_lo = (base & 0xFFFF);
    idt[num].base_hi = (base >> 16) & 0xFFFF;

    // Set selector/segment of IDT entry
    idt[num].sel = sel;
    idt[num].always0 = 0;
    idt[num].flags = flags;
}

// Install IDT
void idt_install() {
    // Set IDT pointer and limit
    idtp.limit = (sizeof(struct idt_entry) * 256) - 1;
    idtp.base = &idt;

    // Clear IDT by setting memory cells to 0
    memory_set(&idt, 0, sizeof(struct idt_entry) * 256);

    // TODO: Add method to add ISRs to IDT

    idt_load();
}