diff options
author | Mark Kettenis <kettenis@cvs.openbsd.org> | 2010-03-28 18:00:52 +0000 |
---|---|---|
committer | Mark Kettenis <kettenis@cvs.openbsd.org> | 2010-03-28 18:00:52 +0000 |
commit | ff3cc83262d602dbc6c2b1274d24f4f1a9dae1cd (patch) | |
tree | a4185998fe95dc1c9efe7b15e0e5e4634b18b1c4 | |
parent | 5a000395f4ea51b06e810df0d6319442eccb9e3e (diff) |
Fix user-after-free bug in pmap_remove(). Page table pages are freed as soon
as the last page table entry that was actually used is cleared. So make sure
we check the page table page is still there for every page we remove.
Otherwise we will bring back the tlb entry and cache lines when we touch the
freed page, and we will create an illegal alias (non-equivalent mapping)
as soon as the page gets re-used.
Seems to fix the last remaining issue with fast page recycling (although I
need to do a bit more testing to be sure).
Help from drahn@, ok jsing@ and miod@ (for an earlier version of this diff)
-rw-r--r-- | sys/arch/hppa/hppa/pmap.c | 18 |
1 files changed, 9 insertions, 9 deletions
diff --git a/sys/arch/hppa/hppa/pmap.c b/sys/arch/hppa/hppa/pmap.c index 500afce5632..5303315aed8 100644 --- a/sys/arch/hppa/hppa/pmap.c +++ b/sys/arch/hppa/hppa/pmap.c @@ -1,4 +1,4 @@ -/* $OpenBSD: pmap.c,v 1.148 2010/03/16 17:12:50 miod Exp $ */ +/* $OpenBSD: pmap.c,v 1.149 2010/03/28 18:00:51 kettenis Exp $ */ /* * Copyright (c) 1998-2004 Michael Shalayeff @@ -886,14 +886,14 @@ pmap_remove(pmap, sva, eva) simple_lock(&pmap->pm_lock); - for (batch = 0, pdemask = 1; sva < eva; sva += PAGE_SIZE) { - if (pdemask != (sva & PDE_MASK)) { - pdemask = sva & PDE_MASK; - if (!(pde = pmap_pde_get(pmap->pm_pdir, sva))) { - sva = pdemask + (~PDE_MASK + 1) - PAGE_SIZE; - continue; - } - if (pdemask == sva && sva + (~PDE_MASK + 1) <= eva) + for (batch = 0; sva < eva; sva += PAGE_SIZE) { + pdemask = sva & PDE_MASK; + if (!(pde = pmap_pde_get(pmap->pm_pdir, sva))) { + sva = pdemask + (~PDE_MASK + 1) - PAGE_SIZE; + continue; + } + if (pdemask == sva) { + if (sva + (~PDE_MASK + 1) <= eva) batch = 1; else batch = 0; |