kern
changeset 24:53588744382c
switched the vm to use recursive page tables
author | John Tsiombikas <nuclear@member.fsf.org> |
---|---|
date | Tue, 05 Apr 2011 02:09:02 +0300 (2011-04-04) |
parents | 5454cee245a3 |
children | 9939a6d7a45a |
files | src/vm.c |
diffstat | 1 files changed, 17 insertions(+), 9 deletions(-) [+] |
line diff
1.1 --- a/src/vm.c Mon Apr 04 23:34:06 2011 +0300 1.2 +++ b/src/vm.c Tue Apr 05 02:09:02 2011 +0300 1.3 @@ -11,6 +11,10 @@ 1.4 #define KMEM_START 0xc0000000 1.5 #define IDMAP_START 0xa0000 1.6 1.7 +#define PGDIR_ADDR 0xfffff000 1.8 +#define PGTBL_BASE (0xffffffff - 4096 * 1024 + 1) 1.9 +#define PGTBL(x) ((uint32_t*)(PGTBL_BASE + PGSIZE * (x))) 1.10 + 1.11 #define ATTR_PGDIR_MASK 0x3f 1.12 #define ATTR_PGTBL_MASK 0x1ff 1.13 #define ADDR_PGENT_MASK 0xfffff000 1.14 @@ -59,12 +63,16 @@ 1.15 /* setup the page tables */ 1.16 pgdir = (uint32_t*)alloc_phys_page(); 1.17 memset(pgdir, 0, PGSIZE); 1.18 - set_pgdir_addr((int32_t)pgdir); 1.19 + set_pgdir_addr((uint32_t)pgdir); 1.20 1.21 /* map the video memory and kernel code 1-1 */ 1.22 get_kernel_mem_range(0, &idmap_end); 1.23 map_mem_range(IDMAP_START, idmap_end - IDMAP_START, IDMAP_START, 0); 1.24 1.25 + /* make the last page directory entry point to the page directory */ 1.26 + pgdir[1023] = ((uint32_t)pgdir & ADDR_PGENT_MASK) | PG_PRESENT; 1.27 + pgdir = (uint32_t*)PGDIR_ADDR; 1.28 + 1.29 /* set the page fault handler */ 1.30 interrupt(PAGEFAULT, pgfault); 1.31 1.32 @@ -92,7 +100,6 @@ 1.33 int diridx, pgidx, pgon; 1.34 1.35 pgon = get_paging_status(); 1.36 - disable_paging(); 1.37 1.38 if(ppage < 0) { 1.39 uint32_t addr = alloc_phys_page(); 1.40 @@ -107,20 +114,21 @@ 1.41 1.42 if(!(pgdir[diridx] & PG_PRESENT)) { 1.43 uint32_t addr = alloc_phys_page(); 1.44 - pgtbl = (uint32_t*)addr; 1.45 + pgdir[diridx] = addr | (attr & ATTR_PGDIR_MASK) | PG_PRESENT; 1.46 + 1.47 + pgtbl = pgon ? PGTBL(diridx) : (uint32_t*)addr; 1.48 memset(pgtbl, 0, PGSIZE); 1.49 - 1.50 - pgdir[diridx] = addr | (attr & ATTR_PGDIR_MASK) | PG_PRESENT; 1.51 } else { 1.52 - pgtbl = (uint32_t*)(pgdir[diridx] & ADDR_PGENT_MASK); 1.53 + if(pgon) { 1.54 + pgtbl = PGTBL(diridx); 1.55 + } else { 1.56 + pgtbl = (uint32_t*)(pgdir[diridx] & ADDR_PGENT_MASK); 1.57 + } 1.58 } 1.59 1.60 pgtbl[pgidx] = PAGE_TO_ADDR(ppage) | (attr & ATTR_PGTBL_MASK) | PG_PRESENT; 1.61 flush_tlb_page(vpage); 1.62 1.63 - if(pgon) { 1.64 - enable_paging(); 1.65 - } 1.66 return 0; 1.67 } 1.68