diff options
author | Miod Vallat <miod@cvs.openbsd.org> | 2006-05-20 22:33:18 +0000 |
---|---|---|
committer | Miod Vallat <miod@cvs.openbsd.org> | 2006-05-20 22:33:18 +0000 |
commit | 7bfab854a934a1fd630f90fc609b2e75a1d06439 (patch) | |
tree | d8f6ab593a8232ef99758111d5f8479276e4b067 /sys | |
parent | 5ee84c2e258c4d0201c3d21a9d39ef93946fdefa (diff) |
Make sure pmap copes with address wraparounds when operating on ranges.
This removes the dependency for VM_MAX_ADDRESS to be aligned on a segment
boundary.
While there, remove a few internal pmap macros from mmu.h.
Diffstat (limited to 'sys')
-rw-r--r-- | sys/arch/m88k/include/mmu.h | 18 | ||||
-rw-r--r-- | sys/arch/m88k/m88k/pmap.c | 140 |
2 files changed, 73 insertions, 85 deletions
diff --git a/sys/arch/m88k/include/mmu.h b/sys/arch/m88k/include/mmu.h index ea750372ba3..2ddddbea2d5 100644 --- a/sys/arch/m88k/include/mmu.h +++ b/sys/arch/m88k/include/mmu.h @@ -1,4 +1,4 @@ -/* $OpenBSD: mmu.h,v 1.6 2005/12/02 21:16:45 miod Exp $ */ +/* $OpenBSD: mmu.h,v 1.7 2006/05/20 22:33:16 miod Exp $ */ /* * This file bears almost no resemblance to the original m68k file, @@ -183,22 +183,6 @@ typedef u_int32_t pt_ind_entry_t; #define SDTIDX(va) (((va) & SDT_MASK) >> SDT_SHIFT) #define PDTIDX(va) (((va) & PDT_MASK) >> PDT_SHIFT) -/* XXX uses knowledge of pmap structure */ -#define SDTENT(map, va) ((sdt_entry_t *)((map)->pm_stab + SDTIDX(va))) - -/* - * Va spaces mapped by tables and PDT table group. - */ - -#define PDT_VA_SPACE (PDT_ENTRIES * PAGE_SIZE) - -/* - * Number of sdt entries used to map user and kernel space. - */ - -#define USER_SDT_ENTRIES SDTIDX(VM_MIN_KERNEL_ADDRESS) -#define KERNEL_SDT_ENTRIES (SDT_ENTRIES - USER_SDT_ENTRIES) - /* * Parameters and macros for BATC */ diff --git a/sys/arch/m88k/m88k/pmap.c b/sys/arch/m88k/m88k/pmap.c index a008fdb5e5b..0d5aac1017c 100644 --- a/sys/arch/m88k/m88k/pmap.c +++ b/sys/arch/m88k/m88k/pmap.c @@ -1,4 +1,4 @@ -/* $OpenBSD: pmap.c,v 1.24 2006/05/20 15:41:09 miod Exp $ */ +/* $OpenBSD: pmap.c,v 1.25 2006/05/20 22:33:17 miod Exp $ */ /* * Copyright (c) 2001-2004, Miodrag Vallat * Copyright (c) 1998-2001 Steve Murphree, Jr. @@ -183,6 +183,8 @@ boolean_t pmap_testbit(struct vm_page *, int); #define m88k_protection(prot) ((prot) & VM_PROT_WRITE ? PG_RW : PG_RO) +#define SDTENT(map, va) ((sdt_entry_t *)((map)->pm_stab + SDTIDX(va))) + /* * Routine: FLUSH_ATC_ENTRY * @@ -238,6 +240,15 @@ flush_atc_entry(pmap_t pmap, vaddr_t va) * Otherwise the page table address is extracted from the segment table, * the page table index is added, and the result is returned. */ + +static __inline__ +pt_entry_t * +sdt_pte(sdt_entry_t *sdt, vaddr_t va) +{ + return ((pt_entry_t *) + (PG_PFNUM(*(sdt + SDT_ENTRIES)) << PDT_SHIFT) + PDTIDX(va)); +} + pt_entry_t * pmap_pte(pmap_t pmap, vaddr_t virt) { @@ -250,8 +261,7 @@ pmap_pte(pmap_t pmap, vaddr_t virt) if (!SDT_VALID(sdt)) return (NULL); - return (pt_entry_t *)(PG_PFNUM(*(sdt + SDT_ENTRIES)) << PDT_SHIFT) + - PDTIDX(virt); + return (sdt_pte(sdt, virt)); } /* @@ -457,7 +467,7 @@ pmap_cache_ctrl(pmap_t pmap, vaddr_t s, vaddr_t e, u_int mode) spl = splvm(); PMAP_LOCK(pmap); - for (va = s; va < e; va += PAGE_SIZE) { + for (va = s; va != e; va += PAGE_SIZE) { if ((pte = pmap_pte(pmap, va)) == NULL) continue; #ifdef DEBUG @@ -730,8 +740,7 @@ pmap_bootstrap(vaddr_t load_start) pmap_bootstrap_cpu(cpu_number()); #else cpu = cpu_number(); - cmmu_flush_tlb(cpu, TRUE, VM_MIN_KERNEL_ADDRESS, - btoc(VM_MAX_KERNEL_ADDRESS - VM_MIN_KERNEL_ADDRESS)); + cmmu_flush_tlb(cpu, TRUE, 0, -1); /* Load supervisor pointer to segment table. */ cmmu_set_sapr(cpu, kernel_pmap->pm_apr); #ifdef DEBUG @@ -748,8 +757,7 @@ pmap_bootstrap_cpu(cpuid_t cpu) if (cpu != master_cpu) { cmmu_initialize_cpu(cpu); } else { - cmmu_flush_tlb(cpu, TRUE, VM_MIN_KERNEL_ADDRESS, - btoc(VM_MAX_KERNEL_ADDRESS - VM_MIN_KERNEL_ADDRESS)); + cmmu_flush_tlb(cpu, TRUE, 0, -1); } /* Load supervisor pointer to segment table. */ cmmu_set_sapr(cpu, kernel_pmap->pm_apr); @@ -886,7 +894,7 @@ pmap_create(void) pmap->pm_apr = (atop(stpa) << PG_SHIFT) | CACHE_GLOBAL | APR_V; #ifdef DEBUG - if (!PAGE_ALIGNED(stpa)) + if (stpa & PAGE_MASK) panic("pmap_create: sdt_table 0x%x not aligned on page boundary", (int)stpa); @@ -940,13 +948,12 @@ pmap_create(void) * This routine sequences of through the user address space, releasing * all translation table space back to the system using uvm_km_free. * The loops are indexed by the virtual address space - * ranges represented by the table group sizes(PDT_VA_SPACE). - * + * ranges represented by the table group sizes (1 << SDT_SHIFT). */ void pmap_release(pmap_t pmap) { - unsigned long sdt_va; /* outer loop index */ + u_int sdt; /* outer loop index */ sdt_entry_t *sdttbl; /* ptr to first entry in the segment table */ pt_entry_t *gdttbl; /* ptr to first entry in a page table */ @@ -955,10 +962,10 @@ pmap_release(pmap_t pmap) printf("(pmap_release: %x) pmap %x\n", curproc, pmap); #endif - /* Segment table Loop */ - for (sdt_va = VM_MIN_ADDRESS; sdt_va < VM_MAX_ADDRESS; - sdt_va += PDT_VA_SPACE) { - if ((gdttbl = pmap_pte(pmap, (vaddr_t)sdt_va)) != NULL) { + /* segment table loop */ + for (sdt = VM_MIN_ADDRESS >> SDT_SHIFT; + sdt <= VM_MAX_ADDRESS >> SDT_SHIFT; sdt++) { + if ((gdttbl = pmap_pte(pmap, sdt << SDT_SHIFT)) != NULL) { #ifdef DEBUG if ((pmap_con_dbg & (CD_FREE | CD_FULL)) == (CD_FREE | CD_FULL)) printf("(pmap_release: %x) free page table = 0x%x\n", @@ -1212,7 +1219,8 @@ pmap_remove_range(pmap_t pmap, vaddr_t s, vaddr_t e) /* * Loop through the range in PAGE_SIZE increments. */ - for (va = s; va < e; va += PAGE_SIZE) { + va = s; + while (va != e) { sdt_entry_t *sdt; sdt = SDTENT(pmap, va); @@ -1220,11 +1228,13 @@ pmap_remove_range(pmap_t pmap, vaddr_t s, vaddr_t e) /* If no segment table, skip a whole segment */ if (!SDT_VALID(sdt)) { va &= SDT_MASK; - va += (1 << SDT_SHIFT) - PAGE_SIZE; - continue; + va += 1 << SDT_SHIFT; + if (va > e || va == 0) + va = e; + } else { + pmap_remove_pte(pmap, va, sdt_pte(sdt, va)); + va += PAGE_SIZE; } - - pmap_remove_pte(pmap, va, pmap_pte(pmap, va)); } } @@ -1405,11 +1415,6 @@ pmap_protect(pmap_t pmap, vaddr_t s, vaddr_t e, vm_prot_t prot) pt_entry_t *pte, ap; vaddr_t va; -#ifdef DEBUG - if (s >= e) - panic("pmap_protect: start grater than end address"); -#endif - if ((prot & VM_PROT_READ) == 0) { pmap_remove(pmap, s, e); return; @@ -1423,7 +1428,8 @@ pmap_protect(pmap_t pmap, vaddr_t s, vaddr_t e, vm_prot_t prot) /* * Loop through the range in PAGE_SIZE increments. */ - for (va = s; va < e; va += PAGE_SIZE) { + va = s; + while (va != e) { sdt_entry_t *sdt; sdt = SDTENT(pmap, va); @@ -1431,23 +1437,22 @@ pmap_protect(pmap_t pmap, vaddr_t s, vaddr_t e, vm_prot_t prot) /* If no segment table, skip a whole segment */ if (!SDT_VALID(sdt)) { va &= SDT_MASK; - va += (1 << SDT_SHIFT) - PAGE_SIZE; - continue; - } - - pte = pmap_pte(pmap, va); - if (pte == NULL || !PDT_VALID(pte)) { - continue; /* no page mapping */ + va += 1 << SDT_SHIFT; + if (va > e || va == 0) + va = e; + } else { + pte = sdt_pte(sdt, va); + if (pte != NULL && PDT_VALID(pte)) { + /* + * Invalidate pte temporarily to avoid the + * modified bit and/or the reference bit being + * written back by any other cpu. + */ + *pte = (invalidate_pte(pte) & ~PG_PROT) | ap; + flush_atc_entry(pmap, va); + } + va += PAGE_SIZE; } - - /* - * Invalidate pte temporarily to avoid the - * modified bit and/or the reference bit being - * written back by any other cpu. - */ - *pte = (invalidate_pte(pte) & ~PG_PROT) | ap; - flush_atc_entry(pmap, va); - pte++; } PMAP_UNLOCK(pmap); splx(spl); @@ -1909,7 +1914,8 @@ pmap_extract(pmap_t pmap, vaddr_t va, paddr_t *pap) void pmap_collect(pmap_t pmap) { - vaddr_t sdt_va; /* outer loop index */ + u_int sdt; /* outer loop index */ + vaddr_t sdt_va; sdt_entry_t *sdtp; /* ptr to index into segment table */ pt_entry_t *gdttbl; /* ptr to first entry in a page table */ pt_entry_t *gdttblend; /* ptr to byte after last entry in @@ -1929,9 +1935,10 @@ pmap_collect(pmap_t pmap) sdtp = pmap->pm_stab; /* addr of segment table */ - /* Segment table loop */ - for (sdt_va = VM_MIN_ADDRESS; sdt_va < VM_MAX_ADDRESS; - sdt_va += PDT_VA_SPACE, sdtp++) { + /* segment table loop */ + for (sdt = VM_MIN_ADDRESS >> SDT_SHIFT; + sdt <= VM_MAX_ADDRESS >> SDT_SHIFT; sdt++, sdtp++) { + sdt_va = sdt << SDT_SHIFT; gdttbl = pmap_pte(pmap, sdt_va); if (gdttbl == NULL) continue; /* no maps in this range */ @@ -1951,7 +1958,7 @@ pmap_collect(pmap_t pmap) continue; /* can't free this range */ /* invalidate all maps in this range */ - pmap_remove_range(pmap, sdt_va, sdt_va + PDT_VA_SPACE); + pmap_remove_range(pmap, sdt_va, sdt_va + (1 << SDT_SHIFT)); /* * we can safely deallocate the page map(s) @@ -2012,8 +2019,7 @@ pmap_activate(struct proc *p) PMAP_LOCK(pmap); cmmu_set_uapr(pmap->pm_apr); - cmmu_flush_tlb(cpu, FALSE, VM_MIN_ADDRESS, - btoc(VM_MAX_ADDRESS - VM_MIN_ADDRESS)); + cmmu_flush_tlb(cpu, FALSE, 0, -1); /* * Mark that this cpu is using the pmap. @@ -2515,7 +2521,7 @@ pmap_kremove(vaddr_t va, vsize_t len) PMAP_LOCK(kernel_pmap); e = va + len; - for (; va < e; va += PAGE_SIZE) { + while (va != e) { sdt_entry_t *sdt; pt_entry_t *pte; @@ -2524,23 +2530,21 @@ pmap_kremove(vaddr_t va, vsize_t len) /* If no segment table, skip a whole segment */ if (!SDT_VALID(sdt)) { va &= SDT_MASK; - va += (1 << SDT_SHIFT) - PAGE_SIZE; - continue; - } - - pte = pmap_pte(kernel_pmap, va); - if (pte == NULL || !PDT_VALID(pte)) { - continue; /* no page mapping */ + va += 1 << SDT_SHIFT; + if (va > e || va == 0) + va = e; + } else { + pte = sdt_pte(sdt, va); + if (pte != NULL && PDT_VALID(pte)) { + /* Update the counts */ + kernel_pmap->pm_stats.resident_count--; + kernel_pmap->pm_stats.wired_count--; + + invalidate_pte(pte); + flush_atc_entry(kernel_pmap, va); + } + va += PAGE_SIZE; } - - /* - * Update the counts - */ - kernel_pmap->pm_stats.resident_count--; - kernel_pmap->pm_stats.wired_count--; - - invalidate_pte(pte); - flush_atc_entry(kernel_pmap, va); } PMAP_UNLOCK(kernel_pmap); splx(spl); @@ -2558,7 +2562,7 @@ pmap_proc_iflush(struct proc *p, vaddr_t va, vsize_t len) eva = round_page(va + len); va = trunc_page(va); - while (va < eva) { + while (va != eva) { if (pmap_extract(pmap, va, &pa)) { users = pmap->pm_cpus; while ((cpu = ff1(users)) != 32) { |