#include u32int page_directory[1024] __attribute__((aligned(4096))); u32int first_page_table[1024] __attribute__((aligned(4096))); page_directory_t *current_directory = NULL; page_directory_t *kernel_directory = NULL; u32int nframes; u32int *frames; u32int full_end = 0; u32int max(u32int a, u32int b) { return a > b ? a : b; } u32int min(u32int a, u32int b) { return a < b ? a : b; } u32int c_log_phys(u32int logical) { page_t *page = get_page(logical, 0, current_directory); u32int logical_frame = logical & 0xFFFFF000; u32int offset = logical - logical_frame; u32int real_address = page->frame * 0x1000 + offset; return real_address; } void initialise_paging2() { u32int mem_end_page = mem_upper * 1024; nframes = mem_end_page / 0x1000; printf("Mem upper: 0x%xu\n", mem_upper); printf("Available memory: %uMB, NFRAMES: %u\n", (u32int)(mem_upper / 1024 ), nframes); full_end = (u32int)&end; multiboot_module_t *modules = (multiboot_module_t *)m->mods_addr; full_end = max(full_end,modules[0].mod_end); u32int tables = nframes / 1024; full_end = ((full_end - 1) / 0x1000 + 1) * 0x1000; printf("Full end: 0x%x\n", full_end); u32int full_end_pages = full_end + 0x1000 * tables; printf("Full end with pages: 0x%x\n", full_end_pages); u32int frames_size = ((nframes - 1) / 32 + 1 * 32) * sizeof(u32int); u32int frames_start = full_end_pages; u32int frames_end = frames_start + frames_size; frames = (u32int *)frames_start; memset((unsigned char *)frames, 0, frames_size); u32int pd_start = ((frames_end - 1) / 0x1000 + 1) * 0x1000; u32int pd_size = sizeof(page_directory_t); u32int pd_end = pd_start + pd_size; page_directory_t *page_directory = (page_directory_t *)pd_start; memset((unsigned char *)page_directory, 0, sizeof(page_directory_t)); log("page_directory: 0x%x\r\n", page_directory); unsigned int i = 0; for (i = 0; i < 1024; i++) { page_directory->tablesPhysical[i] = 0 | 2; } page_directory->physicalAddr = (u32int)page_directory->tablesPhysical; init_heap(&kernel_heap, pd_end, min(mem_end_page, pd_end + 0x5000), mem_end_page, 1,1); printf("Inicjalizacja\n"); dump_heap(&kernel_heap); unsigned int address = 0; current_directory = page_directory; kernel_directory = page_directory; //we will fill all 1024 entries, mapping 4 megabytes log("Tworzenie mapy 1-1 dla stronnicowania\r\n"); for(address = 0; address < kernel_heap.max_address; address += 0x1000) { page_t *page = get_page(address, 1, page_directory); page->frame = address >> 12; page->present = 1;; // attributes: supervisor level, page->rw = 1; page->user = 0; // read/write, present. } log("Oznaczenie ramek kernel + reserved as used\r\n"); log("Ramek pamięci jest: 0x%x\n", full_end_pages / 0x1000); dump_frames(); for (u32int i = 0; i < kernel_heap.max_address; i += 0x1000) { set_frame(i); } log("Zakonczono oznaczenie\r\n"); dump_frames(); /** zapisujemy, które ramki adresów fizycznych są już zajęte */ idt_set_gate(14, (unsigned)page_fault, 0x08, 0X8E); log("Ustawiono nowy handler page fault\r\n"); switch_page_directory(page_directory); log("Uaktywniono stronnicowanie\r\n"); printf("Slotow: 0x%x\n", kernel_heap.max_id); } void switch_page_directory(page_directory_t *dir) { current_directory = dir; asm volatile("mov %0, %%cr3":: "r"(&dir->tablesPhysical)); u32int cr0; asm volatile("mov %%cr0, %0": "=r"(cr0)); cr0 |= 0x80000000; // uaktywniamy stronnicowanie asm volatile("mov %0, %%cr0":: "r"(cr0)); } page_t *get_page(u32int address, int make, page_directory_t *dir) { // przekształcamy adres w indeks // log("get_page, address: 0x%x, make: %d, dir: 0x%x\r\n", address, // make, dir); address /= 0x1000; // znajdujemy tablicę stron zawierający dany adres u32int table_idx = address / 1024; if (dir->table[table_idx]) { // jeżeli tablica jest już aktualnie przypisana return &dir->table[table_idx]->pages[address % 1024]; } else if(make) { // adres podany nie jest zmapowany, ale możemy go utworzyć dir->table[table_idx] = (page_table_t *)(full_end + 0x1000 * table_idx); memset((unsigned char *)dir->table[table_idx], 0, sizeof(page_table_t)); dir->tablesPhysical[table_idx] = (u32int)dir->table[table_idx] | 0x3; // PRESENT | RW | US // log("Table: 0x%x, page: 0x%x\n", table_idx, address % 1024); return &dir->table[table_idx]->pages[address % 1024]; } else { return NULL; } } int page_fault(struct regs *r) { u32int faulting_address; asm volatile("mov %%cr2, %0" : "=r" (faulting_address)); int present = !(r->err_code & 0x1); // strona nie istnieje int rw = r->err_code & 0x2; // czy strona zapisywalna int us = r->err_code & 0x4; // czy procesor w user-mode int reserved = r->err_code & 0x8; // nadpisywane przez cpu zarezerwowane // bity int id = r->err_code & 0x10; // czy spowodowane przez // pobranie rozkazu log("Page fault! ( "); if (present) { log("present"); } if (rw) { log("read-only "); } if (us) { log("user-mode "); } if (reserved) { log("reserved "); } if (id) { log("instr_fetch "); } log(") at 0x%x\r\n", faulting_address); printf("Page fault! ( "); if (present) { printf("present"); } if (rw) { printf("read-only "); } if (us) { printf("user-mode "); } if (reserved) { printf("reserved "); } if (id) { printf("instr_fetch "); } printf(") at 0x%x\r\n", faulting_address); PANIC("PAGE FAULT"); } u32int* init_paging3() { for (int i = 0; i < 1024; i++) { page_directory[i] = 0x00000002; } for (int i = 0; i < 1024; i++) { first_page_table[i] = (i * 0x1000) | 3; // attributes - supervisor, read/write, present } page_directory[0] = ((unsigned int)first_page_table) | 3; idt_set_gate(14, (unsigned)page_fault, 0x08, 0X8E); log("Ustawiono nowy handler page fault\r\n"); //switch_page_directory(page_directory); log("Uaktywniono stronnicowanie\r\n"); loadPageDirectory(page_directory); enablePaging(); return page_directory; }