3 #define _POSIX_SOURCE 1
5 #include <minix/callnr.h>
7 #include <minix/config.h>
8 #include <minix/const.h>
10 #include <minix/endpoint.h>
11 #include <minix/keymap.h>
12 #include <minix/minlib.h>
13 #include <minix/type.h>
14 #include <minix/ipc.h>
15 #include <minix/sysutil.h>
16 #include <minix/syslib.h>
17 #include <minix/safecopies.h>
18 #include <minix/cpufeature.h>
19 #include <minix/bitmap.h>
20 #include <minix/debug.h>
35 #include "sanitycheck.h"
39 /* PDE used to map in kernel, kernel physical address. */
40 PRIVATE
int id_map_high_pde
= -1, pagedir_pde
= -1;
41 PRIVATE u32_t global_bit
= 0, pagedir_pde_val
;
43 PRIVATE
int proc_pde
= 0;
45 /* 4MB page size available in hardware? */
46 PRIVATE
int bigpage_ok
= 0;
48 /* Our process table entry. */
49 struct vmproc
*vmprocess
= &vmproc
[VM_PROC_NR
];
51 /* Spare memory, ready to go after initialization, to avoid a
52 * circular dependency on allocating memory and writing it into VM's
56 int missing_spares
= SPAREPAGES
;
60 } sparepages
[SPAREPAGES
];
62 #define MAX_KERNMAPPINGS 10
64 phys_bytes phys_addr
; /* Physical addr. */
65 phys_bytes len
; /* Length in bytes. */
66 vir_bytes lin_addr
; /* Offset in page table. */
68 } kern_mappings
[MAX_KERNMAPPINGS
];
71 /* Clicks must be pages, as
72 * - they must be page aligned to map them
73 * - they must be a multiple of the page size
74 * - it's inconvenient to have them bigger than pages, because we often want
76 * May as well require them to be equal then.
78 #if CLICK_SIZE != I386_PAGE_SIZE
79 #error CLICK_SIZE must be page size.
82 /* Bytes of virtual address space one pde controls. */
83 #define BYTESPERPDE (I386_VM_PT_ENTRIES * I386_PAGE_SIZE)
85 /* Nevertheless, introduce these macros to make the code readable. */
86 #define CLICK2PAGE(c) ((c) / CLICKSPERPAGE)
88 /* Page table that contains pointers to all page directories. */
89 u32_t page_directories_phys
, *page_directories
= NULL
;
92 /*===========================================================================*
94 *===========================================================================*/
95 PUBLIC
void pt_sanitycheck(pt_t
*pt
, char *file
, int line
)
97 /* Basic pt sanity check. */
102 MYASSERT(pt
->pt_dir
);
103 MYASSERT(pt
->pt_dir_phys
);
105 for(slot
= 0; slot
< ELEMENTS(vmproc
); slot
++) {
106 if(pt
== &vmproc
[slot
].vm_pt
)
110 if(slot
>= ELEMENTS(vmproc
)) {
111 panic("pt_sanitycheck: passed pt not in any proc");
114 MYASSERT(usedpages_add(pt
->pt_dir_phys
, I386_PAGE_SIZE
) == OK
);
116 for(i
= proc_pde
; i
< I386_VM_DIR_ENTRIES
; i
++) {
119 MYASSERT(vm_addrok(pt
->pt_pt
[i
], 1));
120 if(!(pt
->pt_dir
[i
] & I386_VM_PRESENT
)) {
121 printf("slot %d: pt->pt_pt[%d] = 0x%lx, but pt_dir entry 0x%lx\n",
122 slot
, i
, pt
->pt_pt
[i
], pt
->pt_dir
[i
]);
124 MYASSERT(pt
->pt_dir
[i
] & I386_VM_PRESENT
);
125 MYASSERT(usedpages_add(I386_VM_PFA(pt
->pt_dir
[i
]),
126 I386_PAGE_SIZE
) == OK
);
128 MYASSERT(!(pt
->pt_dir
[i
] & I386_VM_PRESENT
));
134 /*===========================================================================*
136 *===========================================================================*/
137 PRIVATE
void *aalloc(size_t bytes
)
139 /* Page-aligned malloc(). only used if vm_allocpage can't be used. */
142 b
= (u32_t
) malloc(I386_PAGE_SIZE
+ bytes
);
143 if(!b
) panic("aalloc: out of memory: %d", bytes
);
144 b
+= I386_PAGE_SIZE
- (b
% I386_PAGE_SIZE
);
149 /*===========================================================================*
151 *===========================================================================*/
152 PRIVATE u32_t
findhole(pt_t
*pt
, u32_t vmin
, u32_t vmax
)
154 /* Find a space in the virtual address space of pageteble 'pt',
155 * between page-aligned BYTE offsets vmin and vmax, to fit
156 * a page in. Return byte offset.
158 u32_t freefound
= 0, curv
;
159 int pde
= 0, try_restart
;
160 static u32_t lastv
= 0;
162 /* Input sanity check. */
163 assert(vmin
+ I386_PAGE_SIZE
>= vmin
);
164 assert(vmax
>= vmin
+ I386_PAGE_SIZE
);
165 assert((vmin
% I386_PAGE_SIZE
) == 0);
166 assert((vmax
% I386_PAGE_SIZE
) == 0);
169 curv
= ((u32_t
) random()) % ((vmax
- vmin
)/I386_PAGE_SIZE
);
170 curv
*= I386_PAGE_SIZE
;
174 if(curv
< vmin
|| curv
>= vmax
)
179 /* Start looking for a free page starting at vmin. */
183 assert(curv
>= vmin
);
186 pde
= I386_VM_PDE(curv
);
187 pte
= I386_VM_PTE(curv
);
189 if(!(pt
->pt_dir
[pde
] & I386_VM_PRESENT
) ||
190 !(pt
->pt_pt
[pde
][pte
] & I386_VM_PRESENT
)) {
195 curv
+=I386_PAGE_SIZE
;
197 if(curv
>= vmax
&& try_restart
) {
203 printf("VM: out of virtual address space in vm\n");
208 /*===========================================================================*
210 *===========================================================================*/
211 PRIVATE
void vm_freepages(vir_bytes vir
, vir_bytes phys
, int pages
, int reason
)
213 assert(reason
>= 0 && reason
< VMP_CATEGORIES
);
214 if(vir
>= vmprocess
->vm_stacktop
) {
215 assert(!(vir
% I386_PAGE_SIZE
));
216 assert(!(phys
% I386_PAGE_SIZE
));
217 free_mem(ABS2CLICK(phys
), pages
);
218 if(pt_writemap(&vmprocess
->vm_pt
, arch_vir2map(vmprocess
, vir
),
219 MAP_NONE
, pages
*I386_PAGE_SIZE
, 0, WMF_OVERWRITE
) != OK
)
220 panic("vm_freepages: pt_writemap failed");
222 printf("VM: vm_freepages not freeing VM heap pages (%d)\n",
227 /* If SANITYCHECKS are on, flush tlb so accessing freed pages is
228 * always trapped, also if not in tlb.
230 if((sys_vmctl(SELF
, VMCTL_FLUSHTLB
, 0)) != OK
) {
231 panic("VMCTL_FLUSHTLB failed");
236 /*===========================================================================*
238 *===========================================================================*/
239 PRIVATE
void *vm_getsparepage(u32_t
*phys
)
242 assert(missing_spares
>= 0 && missing_spares
<= SPAREPAGES
);
243 for(s
= 0; s
< SPAREPAGES
; s
++) {
244 if(sparepages
[s
].page
) {
246 sp
= sparepages
[s
].page
;
247 *phys
= sparepages
[s
].phys
;
248 sparepages
[s
].page
= NULL
;
250 assert(missing_spares
>= 0 && missing_spares
<= SPAREPAGES
);
257 /*===========================================================================*
259 *===========================================================================*/
260 PRIVATE
void *vm_checkspares(void)
263 static int total
= 0, worst
= 0;
264 assert(missing_spares
>= 0 && missing_spares
<= SPAREPAGES
);
265 for(s
= 0; s
< SPAREPAGES
&& missing_spares
> 0; s
++)
266 if(!sparepages
[s
].page
) {
268 if((sparepages
[s
].page
= vm_allocpage(&sparepages
[s
].phys
,
271 assert(missing_spares
>= 0);
272 assert(missing_spares
<= SPAREPAGES
);
274 printf("VM: warning: couldn't get new spare page\n");
277 if(worst
< n
) worst
= n
;
283 /*===========================================================================*
285 *===========================================================================*/
286 PUBLIC
void *vm_allocpage(phys_bytes
*phys
, int reason
)
288 /* Allocate a page for use by VM itself. */
293 static int level
= 0;
296 pt
= &vmprocess
->vm_pt
;
297 assert(reason
>= 0 && reason
< VMP_CATEGORIES
);
304 if(level
> 1 || !(vmprocess
->vm_flags
& VMF_HASPT
) || !meminit_done
) {
307 s
=vm_getsparepage(phys
);
311 printf("VM: warning: out of spare pages\n");
316 /* VM does have a pagetable, so get a page and map it in there.
317 * Where in our virtual address space can we put it?
319 loc
= findhole(pt
, arch_vir2map(vmprocess
, vmprocess
->vm_stacktop
),
320 vmprocess
->vm_arch
.vm_data_top
);
323 printf("VM: vm_allocpage: findhole failed\n");
327 /* Allocate page of memory for use by VM. As VM
328 * is trusted, we don't have to pre-clear it.
330 if((newpage
= alloc_mem(CLICKSPERPAGE
, 0)) == NO_MEM
) {
332 printf("VM: vm_allocpage: alloc_mem failed\n");
336 *phys
= CLICK2ABS(newpage
);
338 /* Map this page into our address space. */
339 if((r
=pt_writemap(pt
, loc
, *phys
, I386_PAGE_SIZE
,
340 I386_VM_PRESENT
| I386_VM_USER
| I386_VM_WRITE
, 0)) != OK
) {
341 free_mem(newpage
, CLICKSPERPAGE
);
342 printf("vm_allocpage writemap failed\n");
347 if((r
=sys_vmctl(SELF
, VMCTL_FLUSHTLB
, 0)) != OK
) {
348 panic("VMCTL_FLUSHTLB failed: %d", r
);
353 /* Return user-space-ready pointer to it. */
354 ret
= (void *) arch_map2vir(vmprocess
, loc
);
359 /*===========================================================================*
361 *===========================================================================*/
362 PUBLIC
void vm_pagelock(void *vir
, int lockflag
)
364 /* Mark a page allocated by vm_allocpage() unwritable, i.e. only for VM. */
367 u32_t flags
= I386_VM_PRESENT
| I386_VM_USER
;
370 pt
= &vmprocess
->vm_pt
;
371 m
= arch_vir2map(vmprocess
, (vir_bytes
) vir
);
373 assert(!(m
% I386_PAGE_SIZE
));
376 flags
|= I386_VM_WRITE
;
379 if((r
=pt_writemap(pt
, m
, 0, I386_PAGE_SIZE
,
380 flags
, WMF_OVERWRITE
| WMF_WRITEFLAGSONLY
)) != OK
) {
381 panic("vm_lockpage: pt_writemap failed");
384 if((r
=sys_vmctl(SELF
, VMCTL_FLUSHTLB
, 0)) != OK
) {
385 panic("VMCTL_FLUSHTLB failed: %d", r
);
391 /*===========================================================================*
393 *===========================================================================*/
394 PUBLIC
int vm_addrok(void *vir
, int writeflag
)
396 /* Mark a page allocated by vm_allocpage() unwritable, i.e. only for VM. */
397 pt_t
*pt
= &vmprocess
->vm_pt
;
399 vir_bytes v
= arch_vir2map(vmprocess
, (vir_bytes
) vir
);
401 /* No PT yet? Don't bother looking. */
402 if(!(vmprocess
->vm_flags
& VMF_HASPT
)) {
406 pde
= I386_VM_PDE(v
);
407 pte
= I386_VM_PTE(v
);
409 if(!(pt
->pt_dir
[pde
] & I386_VM_PRESENT
)) {
410 printf("addr not ok: missing pde %d\n", pde
);
415 !(pt
->pt_dir
[pde
] & I386_VM_WRITE
)) {
416 printf("addr not ok: pde %d present but pde unwritable\n", pde
);
420 if(!(pt
->pt_pt
[pde
][pte
] & I386_VM_PRESENT
)) {
421 printf("addr not ok: missing pde %d / pte %d\n",
427 !(pt
->pt_pt
[pde
][pte
] & I386_VM_WRITE
)) {
428 printf("addr not ok: pde %d / pte %d present but unwritable\n",
436 /*===========================================================================*
438 *===========================================================================*/
439 PRIVATE
int pt_ptalloc(pt_t
*pt
, int pde
, u32_t flags
)
441 /* Allocate a page table and write its address into the page directory. */
445 /* Argument must make sense. */
446 assert(pde
>= 0 && pde
< I386_VM_DIR_ENTRIES
);
447 assert(!(flags
& ~(PTF_ALLFLAGS
)));
449 /* We don't expect to overwrite page directory entry, nor
450 * storage for the page table.
452 assert(!(pt
->pt_dir
[pde
] & I386_VM_PRESENT
));
453 assert(!pt
->pt_pt
[pde
]);
455 /* Get storage for the page table. */
456 if(!(pt
->pt_pt
[pde
] = vm_allocpage(&pt_phys
, VMP_PAGETABLE
)))
459 for(i
= 0; i
< I386_VM_PT_ENTRIES
; i
++)
460 pt
->pt_pt
[pde
][i
] = 0; /* Empty entry. */
462 /* Make page directory entry.
463 * The PDE is always 'present,' 'writable,' and 'user accessible,'
464 * relying on the PTE for protection.
466 pt
->pt_dir
[pde
] = (pt_phys
& I386_VM_ADDR_MASK
) | flags
467 | I386_VM_PRESENT
| I386_VM_USER
| I386_VM_WRITE
;
472 PRIVATE
char *ptestr(u32_t pte
)
474 #define FLAG(constant, name) { \
475 if(pte & (constant)) { strcat(str, name); strcat(str, " "); } \
479 if(!(pte
& I386_VM_PRESENT
)) {
480 return "not present";
483 FLAG(I386_VM_WRITE
, "W");
484 FLAG(I386_VM_USER
, "U");
485 FLAG(I386_VM_PWT
, "PWT");
486 FLAG(I386_VM_PCD
, "PCD");
487 FLAG(I386_VM_ACC
, "ACC");
488 FLAG(I386_VM_DIRTY
, "DIRTY");
489 FLAG(I386_VM_PS
, "PS");
490 FLAG(I386_VM_GLOBAL
, "G");
491 FLAG(I386_VM_PTAVAIL1
, "AV1");
492 FLAG(I386_VM_PTAVAIL2
, "AV2");
493 FLAG(I386_VM_PTAVAIL3
, "AV3");
498 /*===========================================================================*
500 *===========================================================================*/
501 PUBLIC
int pt_writemap(pt_t
*pt
, vir_bytes v
, phys_bytes physaddr
,
502 size_t bytes
, u32_t flags
, u32_t writemapflags
)
504 /* Write mapping into page table. Allocate a new page table if necessary. */
505 /* Page directory and table entries for this virtual address. */
506 int p
, pages
, pdecheck
;
510 if(writemapflags
& WMF_VERIFY
)
513 assert(!(bytes
% I386_PAGE_SIZE
));
514 assert(!(flags
& ~(PTF_ALLFLAGS
)));
516 pages
= bytes
/ I386_PAGE_SIZE
;
518 /* MAP_NONE means to clear the mapping. It doesn't matter
519 * what's actually written into the PTE if I386_VM_PRESENT
520 * isn't on, so we can just write MAP_NONE into it.
522 assert(physaddr
== MAP_NONE
|| (flags
& I386_VM_PRESENT
));
523 assert(physaddr
!= MAP_NONE
|| !flags
);
525 finalpde
= I386_VM_PDE(v
+ I386_PAGE_SIZE
* pages
);
527 /* First make sure all the necessary page tables are allocated,
528 * before we start writing in any of them, because it's a pain
529 * to undo our work properly. Walk the range in page-directory-entry
532 for(pdecheck
= I386_VM_PDE(v
); pdecheck
<= finalpde
; pdecheck
++) {
533 assert(pdecheck
>= 0 && pdecheck
< I386_VM_DIR_ENTRIES
);
534 assert(!(pt
->pt_dir
[pdecheck
] & I386_VM_BIGPAGE
));
535 if(!(pt
->pt_dir
[pdecheck
] & I386_VM_PRESENT
)) {
538 printf("pt_writemap verify: no pde %d\n", pdecheck
);
541 assert(!pt
->pt_dir
[pdecheck
]);
542 if((r
=pt_ptalloc(pt
, pdecheck
, flags
)) != OK
) {
543 /* Couldn't do (complete) mapping.
544 * Don't bother freeing any previously
545 * allocated page tables, they're
546 * still writable, don't point to nonsense,
547 * and pt_ptalloc leaves the directory
548 * and other data in a consistent state.
550 printf("pt_writemap: pt_ptalloc failed\n", pdecheck
);
554 assert(pt
->pt_dir
[pdecheck
] & I386_VM_PRESENT
);
557 /* Now write in them. */
558 for(p
= 0; p
< pages
; p
++) {
560 int pde
= I386_VM_PDE(v
);
561 int pte
= I386_VM_PTE(v
);
563 assert(!(v
% I386_PAGE_SIZE
));
564 assert(pte
>= 0 && pte
< I386_VM_PT_ENTRIES
);
565 assert(pde
>= 0 && pde
< I386_VM_DIR_ENTRIES
);
567 /* Page table has to be there. */
568 assert(pt
->pt_dir
[pde
] & I386_VM_PRESENT
);
570 /* Make sure page directory entry for this page table
571 * is marked present and page table entry is available.
573 assert((pt
->pt_dir
[pde
] & I386_VM_PRESENT
));
574 assert(pt
->pt_pt
[pde
]);
577 /* We don't expect to overwrite a page. */
578 if(!(writemapflags
& (WMF_OVERWRITE
|WMF_VERIFY
)))
579 assert(!(pt
->pt_pt
[pde
][pte
] & I386_VM_PRESENT
));
581 if(writemapflags
& (WMF_WRITEFLAGSONLY
|WMF_FREE
)) {
582 physaddr
= pt
->pt_pt
[pde
][pte
] & I386_VM_ADDR_MASK
;
585 if(writemapflags
& WMF_FREE
) {
586 free_mem(ABS2CLICK(physaddr
), 1);
589 /* Entry we will write. */
590 entry
= (physaddr
& I386_VM_ADDR_MASK
) | flags
;
594 maskedentry
= pt
->pt_pt
[pde
][pte
];
595 maskedentry
&= ~(I386_VM_ACC
|I386_VM_DIRTY
);
596 /* Verify pagetable entry. */
597 if(maskedentry
!= entry
) {
598 printf("pt_writemap: mismatch: ");
599 if((entry
& I386_VM_ADDR_MASK
) !=
600 (maskedentry
& I386_VM_ADDR_MASK
)) {
601 printf("pt_writemap: physaddr mismatch (0x%lx, 0x%lx); ", entry
, maskedentry
);
602 } else printf("phys ok; ");
603 printf(" flags: found %s; ",
604 ptestr(pt
->pt_pt
[pde
][pte
]));
605 printf(" masked %s; ",
606 ptestr(maskedentry
));
607 printf(" expected %s\n", ptestr(entry
));
611 /* Write pagetable entry. */
613 assert(vm_addrok(pt
->pt_pt
[pde
], 1));
615 pt
->pt_pt
[pde
][pte
] = entry
;
618 physaddr
+= I386_PAGE_SIZE
;
625 /*===========================================================================*
627 *===========================================================================*/
628 PUBLIC
int pt_checkrange(pt_t
*pt
, vir_bytes v
, size_t bytes
,
633 assert(!(bytes
% I386_PAGE_SIZE
));
635 pages
= bytes
/ I386_PAGE_SIZE
;
637 for(p
= 0; p
< pages
; p
++) {
639 int pde
= I386_VM_PDE(v
);
640 int pte
= I386_VM_PTE(v
);
642 assert(!(v
% I386_PAGE_SIZE
));
643 assert(pte
>= 0 && pte
< I386_VM_PT_ENTRIES
);
644 assert(pde
>= 0 && pde
< I386_VM_DIR_ENTRIES
);
646 /* Page table has to be there. */
647 if(!(pt
->pt_dir
[pde
] & I386_VM_PRESENT
))
650 /* Make sure page directory entry for this page table
651 * is marked present and page table entry is available.
653 assert((pt
->pt_dir
[pde
] & I386_VM_PRESENT
) && pt
->pt_pt
[pde
]);
655 if(!(pt
->pt_pt
[pde
][pte
] & I386_VM_PRESENT
)) {
659 if(write
&& !(pt
->pt_pt
[pde
][pte
] & I386_VM_WRITE
)) {
669 /*===========================================================================*
671 *===========================================================================*/
672 PUBLIC
int pt_new(pt_t
*pt
)
674 /* Allocate a pagetable root. On i386, allocate a page-aligned page directory
675 * and set them to 0 (indicating no page tables are allocated). Lookup
676 * its physical address as we'll need that in the future. Verify it's
681 /* Don't ever re-allocate/re-move a certain process slot's
682 * page directory once it's been created. This is a fraction
683 * faster, but also avoids having to invalidate the page
684 * mappings from in-kernel page tables pointing to
685 * the page directories (the page_directories data).
688 !(pt
->pt_dir
= vm_allocpage(&pt
->pt_dir_phys
, VMP_PAGEDIR
))) {
692 for(i
= 0; i
< I386_VM_DIR_ENTRIES
; i
++) {
693 pt
->pt_dir
[i
] = 0; /* invalid entry (I386_VM_PRESENT bit = 0) */
697 /* Where to start looking for free virtual address space? */
701 if(pt_mapkernel(pt
) != OK
)
702 panic("pt_new: pt_mapkernel failed");
707 /*===========================================================================*
709 *===========================================================================*/
710 PUBLIC
void pt_init(phys_bytes usedlimit
)
712 /* By default, the kernel gives us a data segment with pre-allocated
713 * memory that then can't grow. We want to be able to allocate memory
714 * dynamically, however. So here we copy the part of the page table
715 * that's ours, so we get a private page table. Then we increase the
716 * hardware segment size so we can allocate memory above our stack.
720 vir_bytes v
, kpagedir
;
722 vir_bytes extra_clicks
;
724 int global_bit_ok
= 0;
727 struct vm_ep_data ep_data
;
728 vir_bytes sparepages_mem
;
729 phys_bytes sparepages_ph
;
732 newpt
= &vmprocess
->vm_pt
;
734 /* Get ourselves spare pages. */
735 if(!(sparepages_mem
= (vir_bytes
) aalloc(I386_PAGE_SIZE
*SPAREPAGES
)))
736 panic("pt_init: aalloc for spare failed");
737 if((r
=sys_umap(SELF
, VM_D
, (vir_bytes
) sparepages_mem
,
738 I386_PAGE_SIZE
*SPAREPAGES
, &sparepages_ph
)) != OK
)
739 panic("pt_init: sys_umap failed: %d", r
);
741 for(s
= 0; s
< SPAREPAGES
; s
++) {
742 sparepages
[s
].page
= (void *) (sparepages_mem
+ s
*I386_PAGE_SIZE
);
743 sparepages
[s
].phys
= sparepages_ph
+ s
*I386_PAGE_SIZE
;
748 /* global bit and 4MB pages available? */
749 global_bit_ok
= _cpufeature(_CPUF_I386_PGE
);
750 bigpage_ok
= _cpufeature(_CPUF_I386_PSE
);
752 /* Set bit for PTE's and PDE's if available. */
754 global_bit
= I386_VM_GLOBAL
;
756 /* The kernel and boot time processes need an identity mapping.
757 * We use full PDE's for this without separate page tables.
758 * Figure out which pde we can start using for other purposes.
760 id_map_high_pde
= usedlimit
/ I386_BIG_PAGE_SIZE
;
762 /* We have to make mappings up till here. */
763 free_pde
= id_map_high_pde
+1;
765 /* Initial (current) range of our virtual address space. */
766 lo
= CLICK2ABS(vmprocess
->vm_arch
.vm_seg
[T
].mem_phys
);
767 hi
= CLICK2ABS(vmprocess
->vm_arch
.vm_seg
[S
].mem_phys
+
768 vmprocess
->vm_arch
.vm_seg
[S
].mem_len
);
770 assert(!(lo
% I386_PAGE_SIZE
));
771 assert(!(hi
% I386_PAGE_SIZE
));
773 if(lo
< VM_PROCSTART
) {
774 moveup
= VM_PROCSTART
- lo
;
775 assert(!(VM_PROCSTART
% I386_PAGE_SIZE
));
776 assert(!(lo
% I386_PAGE_SIZE
));
777 assert(!(moveup
% I386_PAGE_SIZE
));
780 /* Make new page table for ourselves, partly copied
781 * from the current one.
783 if(pt_new(newpt
) != OK
)
784 panic("pt_init: pt_new failed");
786 /* Set up mappings for VM process. */
787 for(v
= lo
; v
< hi
; v
+= I386_PAGE_SIZE
) {
791 /* We have to write the new position in the PT,
792 * so we can move our segments.
794 if(pt_writemap(newpt
, v
+moveup
, v
, I386_PAGE_SIZE
,
795 I386_VM_PRESENT
|I386_VM_WRITE
|I386_VM_USER
, 0) != OK
)
796 panic("pt_init: pt_writemap failed");
799 /* Move segments up too. */
800 vmprocess
->vm_arch
.vm_seg
[T
].mem_phys
+= ABS2CLICK(moveup
);
801 vmprocess
->vm_arch
.vm_seg
[D
].mem_phys
+= ABS2CLICK(moveup
);
802 vmprocess
->vm_arch
.vm_seg
[S
].mem_phys
+= ABS2CLICK(moveup
);
804 /* Allocate us a page table in which to remember page directory
807 if(!(page_directories
= vm_allocpage(&page_directories_phys
,
809 panic("no virt addr for vm mappings");
811 memset(page_directories
, 0, I386_PAGE_SIZE
);
813 /* Increase our hardware data segment to create virtual address
814 * space above our stack. We want to increase it to VM_DATATOP,
815 * like regular processes have.
817 extra_clicks
= ABS2CLICK(VM_DATATOP
- hi
);
818 vmprocess
->vm_arch
.vm_seg
[S
].mem_len
+= extra_clicks
;
820 /* We pretend to the kernel we have a huge stack segment to
821 * increase our data segment.
823 vmprocess
->vm_arch
.vm_data_top
=
824 (vmprocess
->vm_arch
.vm_seg
[S
].mem_vir
+
825 vmprocess
->vm_arch
.vm_seg
[S
].mem_len
) << CLICK_SHIFT
;
827 /* Where our free virtual address space starts.
828 * This is only a hint to the VM system.
830 newpt
->pt_virtop
= 0;
832 /* Let other functions know VM now has a private page table. */
833 vmprocess
->vm_flags
|= VMF_HASPT
;
835 /* Now reserve another pde for kernel's own mappings. */
838 phys_bytes addr
, len
;
839 int flags
, index
= 0;
842 kernmap_pde
= free_pde
++;
843 offset
= kernmap_pde
* I386_BIG_PAGE_SIZE
;
845 while(sys_vmctl_get_mapping(index
, &addr
, &len
,
848 if(index
>= MAX_KERNMAPPINGS
)
849 panic("VM: too many kernel mappings: %d", index
);
850 kern_mappings
[index
].phys_addr
= addr
;
851 kern_mappings
[index
].len
= len
;
852 kern_mappings
[index
].flags
= flags
;
853 kern_mappings
[index
].lin_addr
= offset
;
854 kern_mappings
[index
].flags
=
855 I386_VM_PRESENT
| I386_VM_USER
| I386_VM_WRITE
|
857 if(flags
& VMMF_UNCACHED
)
858 kern_mappings
[index
].flags
|=
859 I386_VM_PWT
| I386_VM_PCD
;
860 if(addr
% I386_PAGE_SIZE
)
861 panic("VM: addr unaligned: %d", addr
);
862 if(len
% I386_PAGE_SIZE
)
863 panic("VM: len unaligned: %d", len
);
864 vir
= arch_map2vir(&vmproc
[VMP_SYSTEM
], offset
);
865 if(sys_vmctl_reply_mapping(index
, vir
) != OK
)
866 panic("VM: reply failed");
873 /* Find a PDE below processes available for mapping in the
874 * page directories (readonly).
876 pagedir_pde
= free_pde
++;
877 pagedir_pde_val
= (page_directories_phys
& I386_VM_ADDR_MASK
) |
878 I386_VM_PRESENT
| I386_VM_USER
| I386_VM_WRITE
;
880 /* Tell kernel about free pde's. */
881 while(free_pde
*I386_BIG_PAGE_SIZE
< VM_PROCSTART
) {
882 if((r
=sys_vmctl(SELF
, VMCTL_I386_FREEPDE
, free_pde
++)) != OK
) {
883 panic("VMCTL_I386_FREEPDE failed: %d", r
);
887 /* first pde in use by process. */
890 kpagedir
= arch_map2vir(&vmproc
[VMP_SYSTEM
],
891 pagedir_pde
*I386_BIG_PAGE_SIZE
);
893 /* Tell kernel how to get at the page directories. */
894 if((r
=sys_vmctl(SELF
, VMCTL_I386_PAGEDIRS
, kpagedir
)) != OK
) {
895 panic("VMCTL_I386_PAGEDIRS failed: %d", r
);
898 /* Give our process the new, copied, private page table. */
899 pt_mapkernel(newpt
); /* didn't know about vm_dir pages earlier */
900 pt_bind(newpt
, vmprocess
);
902 /* new segment limit for the kernel after paging is enabled */
903 ep_data
.data_seg_limit
= free_pde
*I386_BIG_PAGE_SIZE
;
904 /* the memory map which must be installed after paging is enabled */
905 ep_data
.mem_map
= vmprocess
->vm_arch
.vm_seg
;
907 /* Now actually enable paging. */
908 if(sys_vmctl_enable_paging(&ep_data
) != OK
)
909 panic("pt_init: enable paging failed");
911 /* Back to reality - this is where the stack actually is. */
912 vmprocess
->vm_arch
.vm_seg
[S
].mem_len
-= extra_clicks
;
919 /*===========================================================================*
921 *===========================================================================*/
922 PUBLIC
int pt_bind(pt_t
*pt
, struct vmproc
*who
)
927 /* Basic sanity checks. */
929 assert(who
->vm_flags
& VMF_INUSE
);
934 assert(slot
< ELEMENTS(vmproc
));
935 assert(slot
< I386_VM_PT_ENTRIES
);
937 phys
= pt
->pt_dir_phys
& I386_VM_ADDR_MASK
;
938 assert(pt
->pt_dir_phys
== phys
);
940 /* Update "page directory pagetable." */
941 page_directories
[slot
] = phys
| I386_VM_PRESENT
|I386_VM_WRITE
;
944 printf("VM: slot %d has pde val 0x%lx\n", slot
, page_directories
[slot
]);
946 /* Tell kernel about new page table root. */
947 return sys_vmctl(who
->vm_endpoint
, VMCTL_I386_SETCR3
,
948 pt
? pt
->pt_dir_phys
: 0);
951 /*===========================================================================*
953 *===========================================================================*/
954 PUBLIC
void pt_free(pt_t
*pt
)
956 /* Free memory associated with this pagetable. */
959 for(i
= 0; i
< I386_VM_DIR_ENTRIES
; i
++)
961 vm_freepages((vir_bytes
) pt
->pt_pt
[i
],
962 I386_VM_PFA(pt
->pt_dir
[i
]), 1, VMP_PAGETABLE
);
967 /*===========================================================================*
969 *===========================================================================*/
970 PUBLIC
int pt_mapkernel(pt_t
*pt
)
974 /* Any i386 page table needs to map in the kernel address space. */
975 assert(vmproc
[VMP_SYSTEM
].vm_flags
& VMF_INUSE
);
979 for(pde
= 0; pde
<= id_map_high_pde
; pde
++) {
981 addr
= pde
* I386_BIG_PAGE_SIZE
;
982 assert((addr
& I386_VM_ADDR_MASK
) == addr
);
983 pt
->pt_dir
[pde
] = addr
| I386_VM_PRESENT
|
984 I386_VM_BIGPAGE
| I386_VM_USER
|
985 I386_VM_WRITE
| global_bit
;
988 panic("VM: pt_mapkernel: no bigpage");
991 if(pagedir_pde
>= 0) {
992 /* Kernel also wants to know about all page directories. */
993 pt
->pt_dir
[pagedir_pde
] = pagedir_pde_val
;
996 for(i
= 0; i
< kernmappings
; i
++) {
998 kern_mappings
[i
].lin_addr
,
999 kern_mappings
[i
].phys_addr
,
1000 kern_mappings
[i
].len
,
1001 kern_mappings
[i
].flags
, 0) != OK
) {
1002 panic("pt_mapkernel: pt_writemap failed");
1009 /*===========================================================================*
1011 *===========================================================================*/
1012 PUBLIC
void pt_cycle(void)