summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authornwhitehorn <nwhitehorn@FreeBSD.org>2012-04-22 17:58:30 +0000
committernwhitehorn <nwhitehorn@FreeBSD.org>2012-04-22 17:58:30 +0000
commitd39462116358678ad2e0504065654972410f8c9d (patch)
tree3464bcba47d02989c74be12e7eda89b7c6c7588d
parentb4a2957ede854f283170c49c1edf87dab79de82c (diff)
downloadFreeBSD-src-d39462116358678ad2e0504065654972410f8c9d.zip
FreeBSD-src-d39462116358678ad2e0504065654972410f8c9d.tar.gz
Avoid a lock order reversal in pmap_extract_and_hold() from relocking
the page. This PMAP requires an additional lock besides the PMAP lock in pmap_extract_and_hold(), which vm_page_pa_tryrelock() did not release. Suggested by: kib MFC after: 4 days
-rw-r--r--sys/powerpc/aim/mmu_oea64.c33
-rw-r--r--sys/vm/vm_page.c2
2 files changed, 33 insertions, 2 deletions
diff --git a/sys/powerpc/aim/mmu_oea64.c b/sys/powerpc/aim/mmu_oea64.c
index f010370..624e088 100644
--- a/sys/powerpc/aim/mmu_oea64.c
+++ b/sys/powerpc/aim/mmu_oea64.c
@@ -1333,6 +1333,37 @@ moea64_extract(mmu_t mmu, pmap_t pm, vm_offset_t va)
* pmap and virtual address pair if that mapping permits the given
* protection.
*/
+
+extern int pa_tryrelock_restart;
+
+static int
+vm_page_pa_tryrelock_moea64(pmap_t pmap, vm_paddr_t pa, vm_paddr_t *locked)
+{
+ /*
+ * This is a duplicate of vm_page_pa_tryrelock(), but with proper
+ * handling of the table lock
+ */
+ vm_paddr_t lockpa;
+
+ lockpa = *locked;
+ *locked = pa;
+ if (lockpa) {
+ PA_LOCK_ASSERT(lockpa, MA_OWNED);
+ if (PA_LOCKPTR(pa) == PA_LOCKPTR(lockpa))
+ return (0);
+ PA_UNLOCK(lockpa);
+ }
+ if (PA_TRYLOCK(pa))
+ return (0);
+ UNLOCK_TABLE_RD();
+ PMAP_UNLOCK(pmap);
+ atomic_add_int(&pa_tryrelock_restart, 1);
+ PA_LOCK(pa);
+ LOCK_TABLE_RD();
+ PMAP_LOCK(pmap);
+ return (EAGAIN);
+}
+
vm_page_t
moea64_extract_and_hold(mmu_t mmu, pmap_t pmap, vm_offset_t va, vm_prot_t prot)
{
@@ -1349,7 +1380,7 @@ retry:
if (pvo != NULL && (pvo->pvo_pte.lpte.pte_hi & LPTE_VALID) &&
((pvo->pvo_pte.lpte.pte_lo & LPTE_PP) == LPTE_RW ||
(prot & VM_PROT_WRITE) == 0)) {
- if (vm_page_pa_tryrelock(pmap,
+ if (vm_page_pa_tryrelock_moea64(pmap,
pvo->pvo_pte.lpte.pte_lo & LPTE_RPGN, &pa))
goto retry;
m = PHYS_TO_VM_PAGE(pvo->pvo_pte.lpte.pte_lo & LPTE_RPGN);
diff --git a/sys/vm/vm_page.c b/sys/vm/vm_page.c
index ee46503..0055732 100644
--- a/sys/vm/vm_page.c
+++ b/sys/vm/vm_page.c
@@ -131,7 +131,7 @@ TUNABLE_INT("vm.boot_pages", &boot_pages);
SYSCTL_INT(_vm, OID_AUTO, boot_pages, CTLFLAG_RD, &boot_pages, 0,
"number of pages allocated for bootstrapping the VM system");
-static int pa_tryrelock_restart;
+int pa_tryrelock_restart;
SYSCTL_INT(_vm, OID_AUTO, tryrelock_restart, CTLFLAG_RD,
&pa_tryrelock_restart, 0, "Number of tryrelock restarts");
OpenPOWER on IntegriCloud