Viewing: gdt.c
📄 gdt.c (Read Only) ⬅ To go back
#include "arch/x86/gdt.h"
#include "arch/x86/smp.h"

#include "console.h"
 #include "utils.h"

struct gdt_entry {
    uint16_t limit_low;
    uint16_t base_low;
    uint8_t base_middle;
    uint8_t access;
    uint8_t granularity;
    uint8_t base_high;
} __attribute__((packed));

struct tss_entry {
    uint32_t prev_tss;
    uint32_t esp0;
    uint32_t ss0;
    uint32_t esp1;
    uint32_t ss1;
    uint32_t esp2;
    uint32_t ss2;
    uint32_t cr3;
    uint32_t eip;
    uint32_t eflags;
    uint32_t eax;
    uint32_t ecx;
    uint32_t edx;
    uint32_t ebx;
    uint32_t esp;
    uint32_t ebp;
    uint32_t esi;
    uint32_t edi;
    uint32_t es;
    uint32_t cs;
    uint32_t ss;
    uint32_t ds;
    uint32_t fs;
    uint32_t gs;
    uint32_t ldt;
    uint16_t trap;
    uint16_t iomap_base;
} __attribute__((packed));

extern void gdt_flush(uint32_t gdt_ptr_addr);
extern void tss_flush(uint16_t tss_selector);

/* 6 base + 16 percpu GS + 1 user TLS + 16 per-CPU TSS = 39 max */
#define GDT_MAX_ENTRIES 40
/* AP TSS entries start at GDT slot 23 (after user TLS at 22) */
#define TSS_AP_GDT_BASE 23

static struct gdt_entry gdt[GDT_MAX_ENTRIES];
struct gdt_ptr gp;
static struct tss_entry tss_array[SMP_MAX_CPUS];

static void gdt_set_gate(int num, uint32_t base, uint32_t limit, uint8_t access, uint8_t gran) {
    gdt[num].base_low = (base & 0xFFFF);
    gdt[num].base_middle = (base >> 16) & 0xFF;
    gdt[num].base_high = (base >> 24) & 0xFF;

    gdt[num].limit_low = (limit & 0xFFFF);
    gdt[num].granularity = ((limit >> 16) & 0x0F);

    gdt[num].granularity |= (gran & 0xF0);
    gdt[num].access = access;
}

void gdt_set_gate_ext(int num, uint32_t base, uint32_t limit, uint8_t access, uint8_t gran) {
    if (num < 0 || num >= GDT_MAX_ENTRIES) return;
    gdt_set_gate(num, base, limit, access, gran);
    /* Reload GDT limit to include new entries */
    gp.limit = (uint16_t)(sizeof(struct gdt_entry) * GDT_MAX_ENTRIES - 1);
    __asm__ volatile("lgdt %0" : : "m"(gp));
}

static void tss_write(uint32_t gdt_idx, uint32_t cpu, uint16_t kernel_ss, uint32_t kernel_esp) {
    struct tss_entry* t = &tss_array[cpu];
    uintptr_t base = (uintptr_t)t;
    uint32_t limit = (uint32_t)(sizeof(*t) - 1);

    gdt_set_gate((int)gdt_idx, (uint32_t)base, limit, 0x89, 0x00);

    for (size_t i = 0; i < sizeof(*t); i++) {
        ((uint8_t*)t)[i] = 0;
    }

    t->ss0 = kernel_ss;
    t->esp0 = kernel_esp;
    t->iomap_base = (uint16_t)sizeof(*t);
}

extern void x86_sysenter_set_kernel_stack(uintptr_t esp0);

void tss_set_kernel_stack(uintptr_t esp0) {
    /* Determine which CPU we're on and update that CPU's TSS */
    extern uint32_t lapic_get_id(void);
    extern int lapic_is_enabled(void);
    uint32_t cpu = 0;
    if (lapic_is_enabled()) {
        extern uint32_t smp_current_cpu(void);
        cpu = smp_current_cpu();
    }
    if (cpu >= SMP_MAX_CPUS) cpu = 0;
    tss_array[cpu].esp0 = (uint32_t)esp0;
    x86_sysenter_set_kernel_stack(esp0);
}

void tss_init_ap(uint32_t cpu_index) {
    if (cpu_index == 0 || cpu_index >= SMP_MAX_CPUS) return;
    uint32_t gdt_idx = TSS_AP_GDT_BASE + (cpu_index - 1);
    tss_write(gdt_idx, cpu_index, 0x10, 0);
    uint16_t sel = (uint16_t)(gdt_idx * 8);
    tss_flush(sel);
}

void gdt_init(void) {
    kprintf("[GDT] Initializing GDT/TSS...\n");

    gp.limit = (uint16_t)(sizeof(struct gdt_entry) * GDT_MAX_ENTRIES - 1);
    gp.base = (uint32_t)(uintptr_t)&gdt;

    gdt_set_gate(0, 0, 0, 0, 0);

    gdt_set_gate(1, 0, 0xFFFFFFFF, 0x9A, 0xCF);
    gdt_set_gate(2, 0, 0xFFFFFFFF, 0x92, 0xCF);

    gdt_set_gate(3, 0, 0xFFFFFFFF, 0xFA, 0xCF);
    gdt_set_gate(4, 0, 0xFFFFFFFF, 0xF2, 0xCF);

    tss_write(5, 0, 0x10, 0);

    gdt_flush((uint32_t)(uintptr_t)&gp);
    tss_flush(0x28);

    kprintf("[GDT] Loaded.\n");
}