From f2a0c7595bf805cd53001984495e4db0268f5f98 Mon Sep 17 00:00:00 2001 From: dzwdz Date: Mon, 19 Jul 2021 18:17:50 +0200 Subject: very basic paging --- src/kernel/arch/i386/pagedir.c | 98 ++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 98 insertions(+) create mode 100644 src/kernel/arch/i386/pagedir.c (limited to 'src/kernel/arch/i386/pagedir.c') diff --git a/src/kernel/arch/i386/pagedir.c b/src/kernel/arch/i386/pagedir.c new file mode 100644 index 0000000..264508b --- /dev/null +++ b/src/kernel/arch/i386/pagedir.c @@ -0,0 +1,98 @@ +#include +#include +#include + +/* nitpick: I highly recommend you dont use bitfields for paging + * structures + * you can't change them atomically and the way they're layed out + * in memory is implementation defined iirc + */ +struct pagetable_entry { + uint32_t present : 1; + uint32_t writeable : 1; + uint32_t user : 1; + uint32_t writethru : 1; + uint32_t uncached : 1; + uint32_t dirty : 1; + uint32_t always0 : 1; // memory type thing? + uint32_t global : 1; + uint32_t _unused : 3; + uint32_t address : 21; +}; + +struct pagedir_entry { + uint32_t present : 1; + uint32_t writeable : 1; + uint32_t user : 1; + uint32_t writethru : 1; + uint32_t uncached : 1; + uint32_t accessed : 1; + uint32_t always0 : 1; + uint32_t large : 1; // 4 MiB instead of 4 KiB + uint32_t _unused : 3; + uint32_t address : 21; +} __attribute__((packed)); + +struct pagedir { + struct pagedir_entry e[1024]; +} __attribute__((packed)); + + +struct pagedir *pagedir_new() { + struct pagedir *dir = page_alloc(1); + for (int i = 0; i < 1024; i++) + dir->e[i].present = 0; + return dir; +} + +void pagedir_map(struct pagedir *dir, void *virt, void *phys, + bool user, bool writeable) +{ + uintptr_t virt_casted = (uintptr_t) virt; + uint32_t pd_idx = virt_casted >> 22; + uint32_t pt_idx = virt_casted >> 12 & 0x03FF; + struct pagetable_entry *pagetable; + + if (dir->e[pd_idx].present) { + pagetable = (void*) (dir->e[pd_idx].address << 11); + } else { + pagetable = page_alloc(1); + for (int i = 0; i < 1024; i++) + pagetable[i].present = 0; + + dir->e[pd_idx] = (struct pagedir_entry) { + .present = 1, + .writeable = 1, + .user = 1, + .writethru = 1, + .uncached = 0, + .accessed = 0, + .always0 = 0, + .large = 0, + ._unused = 0, + .address = (uintptr_t) pagetable >> 11 + }; + } + + pagetable[pt_idx] = (struct pagetable_entry) { + .present = 1, + .writeable = writeable, + .user = user, + .writethru = 1, + .uncached = 0, + .dirty = 0, + .always0 = 0, + .global = 0, + ._unused = 0, + .address = (uintptr_t) phys >> 11 + }; +} + +void pagedir_use(struct pagedir *dir) { + asm volatile("mov %0, %%cr3;" + + "mov %%cr0, %%eax;" + "or $0x80000001, %%eax;" + "mov %%eax, %%cr0;" + : : "r" (dir) : "eax"); +} -- cgit v1.2.3