Subject: Re: replace pa_to_pvh() and pa_to_attribute() macro
To: None <port-mips@NetBSD.org>
From: Izumi Tsutsui <tsutsui@ceres.dti.ne.jp>
List: port-mips
Date: 12/10/2005 23:15:21
In article <051124004115.M0201326@mirage.ceres.dti.ne.jp>
I wrote:
> In article <F34BAF58-0A16-4C0C-A074-268073244A95@shagadelic.org>
> thorpej@shagadelic.org wrote:
>
> > > Some discussion is going on tech-kern, but is it OK
> > > to commit the attached diff against mips/pmap.c?
> >
> > A good first step. Next is to move those things into the optional
> > machine-dependent vm_page extensions like the Alpha port does :-)
>
> Hmm, like this?
Is it ok to commit the attached patch which moves pv_entry stuff
from MD pmap_physseg to MD vm_page? Tested on news3470 (R3000),
O2 (R5000), cobalt RaQ2 (Rm5200), Express5800/230 (R4400) and
EWS4800/360AD (R4400).
This change would also help future changes I discussed with Chuck Silvers,
adding a hint value (virtual cache colors) into MD vm_page to reduce
undesirable cache flush in pmap_zero_page(9) etc. on MIPS3 CPUs with
virtual-indexed-physical-tagged cache.
---
Izumi Tsutsui
---
Index: include/vmparam.h
===================================================================
RCS file: /cvsroot/src/sys/arch/mips/include/vmparam.h,v
retrieving revision 1.37
diff -u -r1.37 vmparam.h
--- include/vmparam.h 17 Jan 2005 05:26:02 -0000 1.37
+++ include/vmparam.h 10 Dec 2005 14:01:53 -0000
@@ -167,14 +167,21 @@
#define VM_PHYSSEG_STRAT VM_PSTRAT_BSEARCH
#define VM_PHYSSEG_NOADD /* can add RAM after vm_mem_init */
-#define __HAVE_PMAP_PHYSSEG
+#define __HAVE_VM_PAGE_MD
/*
- * pmap-specific data stored in the vm_physmem[] array.
+ * pmap-specific data stored in the vm_page structure.
*/
-struct pmap_physseg {
- struct pv_entry *pvent; /* pv table for this seg */
- char *attrs; /* page attributes for this seg */
+struct vm_page_md {
+ struct pv_entry *pvh_list; /* pv_entry list */
+ struct simplelock pvh_slock; /* lock on this head */
+ u_int pvh_attrs; /* page attributes */
};
+#define VM_MDPAGE_INIT(pg) \
+do { \
+ (pg)->mdpage.pvh_list = NULL; \
+ simple_lock_init(&(pg)->mdpage.pvh_slock); \
+} while (/* CONSTCOND */ 0)
+
#endif /* ! _MIPS_VMPARAM_H_ */
Index: mips/pmap.c
===================================================================
RCS file: /cvsroot/src/sys/arch/mips/mips/pmap.c,v
retrieving revision 1.162
diff -u -r1.162 pmap.c
--- mips/pmap.c 9 Dec 2005 20:50:25 -0000 1.162
+++ mips/pmap.c 10 Dec 2005 14:01:54 -0000
@@ -221,26 +221,10 @@
((pm) == pmap_kernel() || \
(pm) == curlwp->l_proc->p_vmspace->vm_map.pmap)
-#define pa_to_pvh(pa) \
-({ \
- int bank_, pg_; \
- \
- bank_ = vm_physseg_find(atop((pa)), &pg_); \
- &vm_physmem[bank_].pmseg.pvent[pg_]; \
-})
-
-#define pa_to_attribute(pa) \
-({ \
- int bank_, pg_; \
- \
- bank_ = vm_physseg_find(atop((pa)), &pg_); \
- &vm_physmem[bank_].pmseg.pvent[pg_].pv_flags; \
-})
-
/* Forward function declarations */
-void pmap_remove_pv(pmap_t pmap, vaddr_t va, paddr_t pa);
+void pmap_remove_pv(pmap_t, vaddr_t, struct vm_page *);
void pmap_asid_alloc(pmap_t pmap);
-void pmap_enter_pv(pmap_t, vaddr_t, paddr_t, u_int *);
+void pmap_enter_pv(pmap_t, vaddr_t, struct vm_page *, u_int *);
pt_entry_t *pmap_pte(pmap_t, vaddr_t);
/*
@@ -270,7 +254,20 @@
static void
mips_flushcache_allpvh(paddr_t pa)
{
- struct pv_entry *pv = pa_to_pvh(pa);
+ struct vm_page *pg;
+ struct pv_entry *pv;
+
+ pg = PHYS_TO_VM_PAGE(pa);
+ if (pg == NULL) {
+ /* page is unmanaged */
+#ifdef DIAGNOSTIC
+ printf("mips_flushcache_allpvh(): unmanged pa = %08lx\n",
+ (u_long)pa);
+#endif
+ return;
+ }
+
+ pv = pg->mdpage.pvh_list;
#if defined(MIPS3_NO_PV_UNCACHED)
/* No current mapping. Cache was flushed by pmap_remove_pv() */
@@ -492,7 +489,7 @@
pmap_init(void)
{
vsize_t s;
- int bank;
+ int bank, i;
pv_entry_t pv;
#ifdef DEBUG
@@ -508,8 +505,8 @@
pv = pv_table;
for (bank = 0; bank < vm_nphysseg; bank++) {
s = vm_physmem[bank].end - vm_physmem[bank].start;
- vm_physmem[bank].pmseg.pvent = pv;
- pv += s;
+ for (i = 0; i < s; i++)
+ vm_physmem[bank].pgs[i].mdpage.pvh_list = pv++;
}
/*
@@ -715,6 +712,7 @@
pmap_t pmap;
vaddr_t sva, eva;
{
+ struct vm_page *pg;
vaddr_t nssva;
pt_entry_t *pte;
unsigned entry;
@@ -739,7 +737,8 @@
if (mips_pg_wired(entry))
pmap->pm_stats.wired_count--;
pmap->pm_stats.resident_count--;
- pmap_remove_pv(pmap, sva, mips_tlbpfn_to_paddr(entry));
+ pg = PHYS_TO_VM_PAGE(mips_tlbpfn_to_paddr(entry));
+ pmap_remove_pv(pmap, sva, pg);
if (MIPS_HAS_R4K_MMU)
/* See above about G bit */
pte->pt_entry = MIPS3_PG_NV | MIPS3_PG_G;
@@ -797,7 +796,8 @@
if (mips_pg_wired(entry))
pmap->pm_stats.wired_count--;
pmap->pm_stats.resident_count--;
- pmap_remove_pv(pmap, sva, mips_tlbpfn_to_paddr(entry));
+ pg = PHYS_TO_VM_PAGE(mips_tlbpfn_to_paddr(entry));
+ pmap_remove_pv(pmap, sva, pg);
pte->pt_entry = mips_pg_nv_bit();
/*
* Flush the TLB for the given address.
@@ -822,14 +822,14 @@
struct vm_page *pg;
vm_prot_t prot;
{
- paddr_t pa = VM_PAGE_TO_PHYS(pg);
pv_entry_t pv;
vaddr_t va;
#ifdef DEBUG
if ((pmapdebug & (PDB_FOLLOW|PDB_PROTECT)) ||
(prot == VM_PROT_NONE && (pmapdebug & PDB_REMOVE)))
- printf("pmap_page_protect(%lx, %x)\n", (u_long)pa, prot);
+ printf("pmap_page_protect(%lx, %x)\n",
+ (u_long)VM_PAGE_TO_PHYS(pg), prot);
#endif
switch (prot) {
case VM_PROT_READ|VM_PROT_WRITE:
@@ -839,7 +839,7 @@
/* copy_on_write */
case VM_PROT_READ:
case VM_PROT_READ|VM_PROT_EXECUTE:
- pv = pa_to_pvh(pa);
+ pv = pg->mdpage.pvh_list;
/*
* Loop over all current mappings setting/clearing as appropos.
*/
@@ -855,7 +855,7 @@
/* remove_all */
default:
- pv = pa_to_pvh(pa);
+ pv = pg->mdpage.pvh_list;
while (pv->pv_pmap != NULL) {
pmap_remove(pv->pv_pmap, pv->pv_va,
pv->pv_va + PAGE_SIZE);
@@ -1044,7 +1044,7 @@
* Change all mappings of a managed page to cached/uncached.
*/
static void
-pmap_page_cache(paddr_t pa, int mode)
+pmap_page_cache(struct vm_page *pg, int mode)
{
pv_entry_t pv;
pt_entry_t *pte;
@@ -1054,10 +1054,10 @@
#ifdef DEBUG
if (pmapdebug & (PDB_FOLLOW|PDB_ENTER))
- printf("pmap_page_uncache(%lx)\n", (u_long)pa);
+ printf("pmap_page_uncache(%lx)\n", (u_long)VM_PAGE_TO_PHYS(pa));
#endif
newmode = mode & PV_UNCACHED ? MIPS3_PG_UNCACHED : MIPS3_PG_CACHED;
- pv = pa_to_pvh(pa);
+ pv = pg->mdpage.pvh_list;
asid = pv->pv_pmap->pm_asid;
needupdate = (pv->pv_pmap->pm_asidgen == pmap_asid_generation);
@@ -1114,10 +1114,9 @@
vm_prot_t prot;
int flags;
{
- int need_enter_pv;
pt_entry_t *pte;
u_int npte;
- struct vm_page *mem;
+ struct vm_page *pg, *mem;
unsigned asid;
boolean_t wired = (flags & PMAP_WIRED) != 0;
@@ -1149,9 +1148,10 @@
if (!(prot & VM_PROT_READ))
panic("pmap_enter: prot");
#endif
+ pg = PHYS_TO_VM_PAGE(pa);
- if (PAGE_IS_MANAGED(pa)) {
- int *attrs = pa_to_attribute(pa);
+ if (pg) {
+ int *attrs = &pg->mdpage.pvh_attrs;
/* Set page referenced/modified status based on flags */
if (flags & VM_PROT_WRITE)
@@ -1177,7 +1177,6 @@
#ifdef DEBUG
enter_stats.managed++;
#endif
- need_enter_pv = 1;
} else {
/*
* Assumption: if it is not part of our managed memory
@@ -1198,8 +1197,6 @@
(MIPS1_PG_D | MIPS1_PG_N) :
(MIPS1_PG_RO | MIPS1_PG_N);
}
-
- need_enter_pv = 0;
}
/*
@@ -1217,8 +1214,8 @@
#endif
if (pmap == pmap_kernel()) {
- if (need_enter_pv)
- pmap_enter_pv(pmap, va, pa, &npte);
+ if (pg)
+ pmap_enter_pv(pmap, va, pg, &npte);
/* enter entries into kernel pmap */
pte = kvtopte(va);
@@ -1274,8 +1271,8 @@
}
/* Done after case that may sleep/return. */
- if (need_enter_pv)
- pmap_enter_pv(pmap, va, pa, &npte);
+ if (pg)
+ pmap_enter_pv(pmap, va, pg, &npte);
pte += (va >> PGSHIFT) & (NPTEPG - 1);
@@ -1579,6 +1576,7 @@
{
vaddr_t va;
#if defined(MIPS3_PLUS)
+ struct vm_page *pg;
pv_entry_t pv;
#endif
@@ -1593,8 +1591,9 @@
va = MIPS_PHYS_TO_KSEG0(phys);
#if defined(MIPS3_PLUS) /* XXX mmu XXX */
+ pg = PHYS_TO_VM_PAGE(phys);
if (mips_cache_virtual_alias) {
- pv = pa_to_pvh(phys);
+ pv = pg->mdpage.pvh_list;
if ((pv->pv_flags & PV_UNCACHED) == 0 &&
mips_cache_indexof(pv->pv_va) != mips_cache_indexof(va))
mips_dcache_wbinv_range_index(pv->pv_va, PAGE_SIZE);
@@ -1690,15 +1689,15 @@
pmap_clear_reference(pg)
struct vm_page *pg;
{
- paddr_t pa = VM_PAGE_TO_PHYS(pg);
int *attrp;
boolean_t rv;
#ifdef DEBUG
if (pmapdebug & PDB_FOLLOW)
- printf("pmap_clear_reference(%lx)\n", (u_long)pa);
+ printf("pmap_clear_reference(%lx)\n",
+ (u_long)VM_PAGE_TO_PHYS(pg));
#endif
- attrp = pa_to_attribute(pa);
+ attrp = &pg->mdpage.pvh_attrs;
rv = *attrp & PV_REFERENCED;
*attrp &= ~PV_REFERENCED;
return rv;
@@ -1714,9 +1713,8 @@
pmap_is_referenced(pg)
struct vm_page *pg;
{
- paddr_t pa = VM_PAGE_TO_PHYS(pg);
- return (*pa_to_attribute(pa) & PV_REFERENCED);
+ return (pg->mdpage.pvh_attrs & PV_REFERENCED);
}
/*
@@ -1726,7 +1724,6 @@
pmap_clear_modify(pg)
struct vm_page *pg;
{
- paddr_t pa = VM_PAGE_TO_PHYS(pg);
struct pmap *pmap;
struct pv_entry *pv;
pt_entry_t *pte;
@@ -1739,13 +1736,13 @@
if (pmapdebug & PDB_FOLLOW)
printf("pmap_clear_modify(%lx)\n", (u_long)pa);
#endif
- attrp = pa_to_attribute(pa);
+ attrp = &pg->mdpage.pvh_attrs;
rv = *attrp & PV_MODIFIED;
*attrp &= ~PV_MODIFIED;
if (!rv) {
return rv;
}
- pv = pa_to_pvh(pa);
+ pv = pg->mdpage.pvh_list;
if (pv->pv_pmap == NULL) {
return TRUE;
}
@@ -1796,9 +1793,8 @@
pmap_is_modified(pg)
struct vm_page *pg;
{
- paddr_t pa = VM_PAGE_TO_PHYS(pg);
- return (*pa_to_attribute(pa) & PV_MODIFIED);
+ return (pg->mdpage.pvh_attrs & PV_MODIFIED);
}
/*
@@ -1810,7 +1806,10 @@
pmap_set_modified(pa)
paddr_t pa;
{
- *pa_to_attribute(pa) |= PV_MODIFIED | PV_REFERENCED;
+ struct vm_page *pg;
+
+ pg = PHYS_TO_VM_PAGE(pa);
+ pg->mdpage.pvh_attrs |= PV_MODIFIED | PV_REFERENCED;
}
paddr_t
@@ -1870,15 +1869,11 @@
* physical to virtual map table.
*/
void
-pmap_enter_pv(pmap, va, pa, npte)
- pmap_t pmap;
- vaddr_t va;
- paddr_t pa;
- u_int *npte;
+pmap_enter_pv(pmap_t pmap, vaddr_t va, struct vm_page *pg, u_int *npte)
{
pv_entry_t pv, npv;
- pv = pa_to_pvh(pa);
+ pv = pg->mdpage.pvh_list;
#ifdef DEBUG
if (pmapdebug & PDB_ENTER)
printf("pmap_enter: pv %p: was %lx/%p/%p\n",
@@ -1942,7 +1937,7 @@
*/
if (mips_cache_indexof(npv->pv_va) !=
mips_cache_indexof(va)) {
- pmap_page_cache(pa,PV_UNCACHED);
+ pmap_page_cache(pg,PV_UNCACHED);
mips_dcache_wbinv_range_index(
pv->pv_va, PAGE_SIZE);
*npte = (*npte & ~MIPS3_PG_CACHEMODE) | MIPS3_PG_UNCACHED;
@@ -1985,10 +1980,12 @@
entry = 0;
}
if (!mips_pg_v(entry) ||
- mips_tlbpfn_to_paddr(entry) != pa)
+ mips_tlbpfn_to_paddr(entry) !=
+ VM_PAGE_TO_PHYS(pg))
printf(
"pmap_enter: found va %lx pa %lx in pv_table but != %x\n",
- va, (u_long)pa, entry);
+ va, (u_long)VM_PAGE_TO_PHYS(pg),
+ entry);
#endif
return;
}
@@ -2021,28 +2018,18 @@
* at this point).
*/
void
-pmap_remove_pv(pmap, va, pa)
- pmap_t pmap;
- vaddr_t va;
- paddr_t pa;
+pmap_remove_pv(pmap_t pmap, vaddr_t va, struct vm_page *pg)
{
pv_entry_t pv, npv;
int last;
- int bank, off;
#ifdef DEBUG
if (pmapdebug & (PDB_FOLLOW|PDB_PVENTRY))
- printf("pmap_remove_pv(%p, %lx, %lx)\n", pmap, va, (u_long)pa);
+ printf("pmap_remove_pv(%p, %lx, %lx)\n", pmap, va,
+ (u_long)VM_PAGE_TO_PHYS(pg));
#endif
- /*
- * Remove page from the PV table.
- * Return immediately if the page is actually not managed.
- */
- bank = vm_physseg_find(atop(pa), &off);
- if (bank == -1)
- return;
- pv = &vm_physmem[bank].pmseg.pvent[off];
+ pv = pg->mdpage.pvh_list;
/*
* If it is the first entry on the list, it is actually
@@ -2094,13 +2081,13 @@
* removed. If it was, then reenable caching.
*/
- pv = pa_to_pvh(pa);
+ pv = pg->mdpage.pvh_list;
for (npv = pv->pv_next; npv; npv = npv->pv_next) {
if (mips_cache_indexof(pv->pv_va ^ npv->pv_va))
break;
}
if (npv == NULL)
- pmap_page_cache(pa, 0);
+ pmap_page_cache(pg, 0);
}
#endif
if (MIPS_HAS_R4K_MMU && last != 0)
@@ -2131,7 +2118,8 @@
va = MIPS_PHYS_TO_KSEG0(phys);
#if defined(MIPS3_PLUS)
if (mips_cache_virtual_alias) {
- pv = pa_to_pvh(phys);
+ pg = PHYS_TO_VM_PAGE(phys);
+ pv = pg->mdpage.pvh_list;
if ((pv->pv_flags & PV_UNCACHED) == 0 &&
mips_cache_indexof(pv->pv_va) != mips_cache_indexof(va))
mips_dcache_wbinv_range_index(pv->pv_va, PAGE_SIZE);
@@ -2201,13 +2189,15 @@
{
vaddr_t va;
#if defined(MIPS3_PLUS)
+ struct vm_page *pg;
pv_entry_t pv;
#endif
va = MIPS_PHYS_TO_KSEG0(pa);
#if defined(MIPS3_PLUS)
if (mips_cache_virtual_alias) {
- pv = pa_to_pvh(pa);
+ pg = PHYS_TO_VM_PAGE(pa);
+ pv = pg->mdpage.pvh_list;
if ((pv->pv_flags & PV_UNCACHED) == 0 &&
mips_cache_indexof(pv->pv_va) != mips_cache_indexof(va))
mips_dcache_wbinv_range_index(pv->pv_va, PAGE_SIZE);