Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch Merge x86 pmap changes from yamt-pagecache:



details:   https://anonhg.NetBSD.org/src/rev/57060ab30c2d
branches:  trunk
changeset: 847065:57060ab30c2d
user:      ad <ad%NetBSD.org@localhost>
date:      Sun Dec 08 20:42:48 2019 +0000

description:
Merge x86 pmap changes from yamt-pagecache:

- Deal better with the multi-level pmap object locking kludge.
- Handle uvm_pagealloc() being able to block.

diffstat:

 sys/arch/x86/include/pmap.h |    5 +-
 sys/arch/x86/x86/pmap.c     |  174 ++++++++++++++++++++-----------------------
 sys/arch/x86/x86/svs.c      |    6 +-
 sys/arch/xen/x86/xen_pmap.c |    6 +-
 4 files changed, 91 insertions(+), 100 deletions(-)

diffs (truncated from 611 to 300 lines):

diff -r 2296c678c05f -r 57060ab30c2d sys/arch/x86/include/pmap.h
--- a/sys/arch/x86/include/pmap.h       Sun Dec 08 20:35:23 2019 +0000
+++ b/sys/arch/x86/include/pmap.h       Sun Dec 08 20:42:48 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: pmap.h,v 1.105 2019/11/14 16:23:52 maxv Exp $  */
+/*     $NetBSD: pmap.h,v 1.106 2019/12/08 20:42:48 ad Exp $    */
 
 /*
  * Copyright (c) 1997 Charles D. Cranor and Washington University.
@@ -249,8 +249,7 @@
 
 struct pmap {
        struct uvm_object pm_obj[PTP_LEVELS-1]; /* objects for lvl >= 1) */
-#define        pm_lock pm_obj[0].vmobjlock
-       kmutex_t pm_obj_lock[PTP_LEVELS-1];     /* locks for pm_objs */
+       kmutex_t pm_lock;               /* locks for pm_objs */
        LIST_ENTRY(pmap) pm_list;       /* list (lck by pm_list lock) */
        pd_entry_t *pm_pdir;            /* VA of PD (lck by object lock) */
        paddr_t pm_pdirpa[PDP_SIZE];    /* PA of PDs (read-only after create) */
diff -r 2296c678c05f -r 57060ab30c2d sys/arch/x86/x86/pmap.c
--- a/sys/arch/x86/x86/pmap.c   Sun Dec 08 20:35:23 2019 +0000
+++ b/sys/arch/x86/x86/pmap.c   Sun Dec 08 20:42:48 2019 +0000
@@ -1,7 +1,7 @@
-/*     $NetBSD: pmap.c,v 1.342 2019/12/03 15:20:59 riastradh Exp $     */
+/*     $NetBSD: pmap.c,v 1.343 2019/12/08 20:42:48 ad Exp $    */
 
 /*
- * Copyright (c) 2008, 2010, 2016, 2017 The NetBSD Foundation, Inc.
+ * Copyright (c) 2008, 2010, 2016, 2017, 2019 The NetBSD Foundation, Inc.
  * All rights reserved.
  *
  * This code is derived from software contributed to The NetBSD Foundation
@@ -130,7 +130,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.342 2019/12/03 15:20:59 riastradh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.343 2019/12/08 20:42:48 ad Exp $");
 
 #include "opt_user_ldt.h"
 #include "opt_lockdebug.h"
@@ -261,24 +261,6 @@
 static vaddr_t pmap_maxkvaddr;
 
 /*
- * XXX kludge: dummy locking to make KASSERTs in uvm_page.c comfortable.
- * actual locking is done by pm_lock.
- */
-#if defined(DIAGNOSTIC)
-#define        PMAP_SUBOBJ_LOCK(pm, idx) \
-       KASSERT(mutex_owned((pm)->pm_lock)); \
-       if ((idx) != 0) \
-               mutex_enter((pm)->pm_obj[(idx)].vmobjlock)
-#define        PMAP_SUBOBJ_UNLOCK(pm, idx) \
-       KASSERT(mutex_owned((pm)->pm_lock)); \
-       if ((idx) != 0) \
-               mutex_exit((pm)->pm_obj[(idx)].vmobjlock)
-#else /* defined(DIAGNOSTIC) */
-#define        PMAP_SUBOBJ_LOCK(pm, idx)       /* nothing */
-#define        PMAP_SUBOBJ_UNLOCK(pm, idx)     /* nothing */
-#endif /* defined(DIAGNOSTIC) */
-
-/*
  * Misc. event counters.
  */
 struct evcnt pmap_iobmp_evcnt;
@@ -475,8 +457,8 @@
 static void pmap_remap_largepages(void);
 #endif
 
-static struct vm_page *pmap_get_ptp(struct pmap *, vaddr_t,
-    pd_entry_t * const *, int);
+static int pmap_get_ptp(struct pmap *, vaddr_t,
+    pd_entry_t * const *, int, struct vm_page **);
 static struct vm_page *pmap_find_ptp(struct pmap *, vaddr_t, paddr_t, int);
 static void pmap_freepage(struct pmap *, struct vm_page *, int);
 static void pmap_free_ptp(struct pmap *, struct vm_page *, vaddr_t,
@@ -502,7 +484,7 @@
                atomic_add_long(&pmap->pm_stats.resident_count, resid_diff);
                atomic_add_long(&pmap->pm_stats.wired_count, wired_diff);
        } else {
-               KASSERT(mutex_owned(pmap->pm_lock));
+               KASSERT(mutex_owned(&pmap->pm_lock));
                pmap->pm_stats.resident_count += resid_diff;
                pmap->pm_stats.wired_count += wired_diff;
        }
@@ -640,13 +622,13 @@
 
        l = curlwp;
  retry:
-       mutex_enter(pmap->pm_lock);
+       mutex_enter(&pmap->pm_lock);
        ci = curcpu();
        curpmap = ci->ci_pmap;
        if (vm_map_pmap(&l->l_proc->p_vmspace->vm_map) == pmap) {
                /* Our own pmap so just load it: easy. */
                if (__predict_false(ci->ci_want_pmapload)) {
-                       mutex_exit(pmap->pm_lock);
+                       mutex_exit(&pmap->pm_lock);
                        pmap_load();
                        goto retry;
                }
@@ -670,6 +652,7 @@
                kcpuset_atomic_clear(curpmap->pm_kernel_cpus, cid);
                ci->ci_pmap = pmap;
                ci->ci_tlbstate = TLBSTATE_VALID;
+               ci->ci_want_pmapload = 0;
                kcpuset_atomic_set(pmap->pm_cpus, cid);
                kcpuset_atomic_set(pmap->pm_kernel_cpus, cid);
                cpu_load_pmap(pmap, curpmap);
@@ -717,7 +700,7 @@
        KASSERT(pmap->pm_ncsw == curlwp->l_ncsw);
        mypmap = vm_map_pmap(&curproc->p_vmspace->vm_map);
        if (pmap == mypmap) {
-               mutex_exit(pmap->pm_lock);
+               mutex_exit(&pmap->pm_lock);
                return;
        }
 
@@ -725,9 +708,15 @@
         * Mark whatever's on the CPU now as lazy and unlock.
         * If the pmap was already installed, we are done.
         */
-       ci->ci_tlbstate = TLBSTATE_LAZY;
-       ci->ci_want_pmapload = (mypmap != pmap_kernel());
-       mutex_exit(pmap->pm_lock);
+       if (ci->ci_tlbstate == TLBSTATE_VALID) {
+               ci->ci_tlbstate = TLBSTATE_LAZY;
+               ci->ci_want_pmapload = (mypmap != pmap_kernel());
+       } else {
+               /*
+                * This can happen when undoing after pmap_get_ptp blocked.
+                */ 
+       }
+       mutex_exit(&pmap->pm_lock);
        if (pmap == pmap2) {
                return;
        }
@@ -1089,10 +1078,10 @@
         * tables (fast user-level vtophys?). This may or may not be useful.
         */
        kpm = pmap_kernel();
+       mutex_init(&kpm->pm_lock, MUTEX_DEFAULT, IPL_NONE);
        for (i = 0; i < PTP_LEVELS - 1; i++) {
-               mutex_init(&kpm->pm_obj_lock[i], MUTEX_DEFAULT, IPL_NONE);
                uvm_obj_init(&kpm->pm_obj[i], NULL, false, 1);
-               uvm_obj_setlock(&kpm->pm_obj[i], &kpm->pm_obj_lock[i]);
+               uvm_obj_setlock(&kpm->pm_obj[i], &kpm->pm_lock);
                kpm->pm_ptphint[i] = NULL;
        }
        memset(&kpm->pm_list, 0, sizeof(kpm->pm_list));  /* pm_list not used */
@@ -1992,15 +1981,13 @@
        int lidx = level - 1;
        struct vm_page *pg;
 
-       KASSERT(mutex_owned(pmap->pm_lock));
+       KASSERT(mutex_owned(&pmap->pm_lock));
 
        if (pa != (paddr_t)-1 && pmap->pm_ptphint[lidx] &&
            pa == VM_PAGE_TO_PHYS(pmap->pm_ptphint[lidx])) {
                return (pmap->pm_ptphint[lidx]);
        }
-       PMAP_SUBOBJ_LOCK(pmap, lidx);
        pg = uvm_pagelookup(&pmap->pm_obj[lidx], ptp_va2o(va, level));
-       PMAP_SUBOBJ_UNLOCK(pmap, lidx);
 
        KASSERT(pg == NULL || pg->wire_count >= 1);
        return pg;
@@ -2019,8 +2006,6 @@
 
        obj = &pmap->pm_obj[lidx];
        pmap_stats_update(pmap, -1, 0);
-       if (lidx != 0)
-               mutex_enter(obj->vmobjlock);
        if (pmap->pm_ptphint[lidx] == ptp)
                pmap->pm_ptphint[lidx] = TAILQ_FIRST(&obj->memq);
        ptp->wire_count = 0;
@@ -2029,8 +2014,6 @@
        KASSERT((l->l_pflag & LP_INTR) == 0);
        VM_PAGE_TO_PP(ptp)->pp_link = l->l_md.md_gc_ptp;
        l->l_md.md_gc_ptp = ptp;
-       if (lidx != 0)
-               mutex_exit(obj->vmobjlock);
 }
 
 static void
@@ -2043,7 +2026,7 @@
        pd_entry_t opde;
 
        KASSERT(pmap != pmap_kernel());
-       KASSERT(mutex_owned(pmap->pm_lock));
+       KASSERT(mutex_owned(&pmap->pm_lock));
        KASSERT(kpreempt_disabled());
 
        level = 1;
@@ -2092,23 +2075,26 @@
  * => pmap should be locked
  * => preemption should be disabled
  */
-static struct vm_page *
-pmap_get_ptp(struct pmap *pmap, vaddr_t va, pd_entry_t * const *pdes, int flags)
+static int
+pmap_get_ptp(struct pmap *pmap, vaddr_t va, pd_entry_t * const *pdes, int flags,
+            struct vm_page **resultp)
 {
        struct vm_page *ptp;
        struct {
                struct vm_page *pg;
                bool new;
        } pt[PTP_LEVELS + 1];
-       int i, aflags;
+       int i, aflags, error;
        unsigned long index;
        pd_entry_t *pva;
        paddr_t pa;
        struct uvm_object *obj;
        voff_t off;
+       lwp_t *l;
+       uint64_t ncsw;
 
        KASSERT(pmap != pmap_kernel());
-       KASSERT(mutex_owned(pmap->pm_lock));
+       KASSERT(mutex_owned(&pmap->pm_lock));
        KASSERT(kpreempt_disabled());
 
        /*
@@ -2122,16 +2108,24 @@
                obj = &pmap->pm_obj[i - 2];
                off = ptp_va2o(va, i - 1);
 
-               PMAP_SUBOBJ_LOCK(pmap, i - 2);
                pt[i].pg = uvm_pagelookup(obj, off);
                if (pt[i].pg == NULL) {
+                       l = curlwp;
+                       ncsw = l->l_ncsw;
                        pt[i].pg = uvm_pagealloc(obj, off, NULL, aflags);
                        pt[i].new = true;
+                       if (__predict_false(ncsw != l->l_ncsw)) {
+                               /* uvm_pagealloc can block. */
+                               /* XXX silence assertion in pmap_unmap_ptes */
+                               pmap->pm_ncsw = l->l_ncsw;
+                               error = EAGAIN;
+                               goto fail;
+                       }
                }
-               PMAP_SUBOBJ_UNLOCK(pmap, i - 2);
-
-               if (pt[i].pg == NULL)
+               if (pt[i].pg == NULL) {
+                       error = ENOMEM;
                        goto fail;
+               }
        }
 
        /*
@@ -2183,7 +2177,8 @@
        ptp = pt[2].pg;
        KASSERT(ptp != NULL);
        pmap->pm_ptphint[0] = ptp;
-       return ptp;
+       *resultp = ptp;
+       return 0;
 
        /*
         * Allocation of a PTP failed, free any others that we just allocated.
@@ -2197,11 +2192,9 @@
                        continue;
                }
                obj = &pmap->pm_obj[i - 2];
-               PMAP_SUBOBJ_LOCK(pmap, i - 2);
                uvm_pagefree(pt[i].pg);
-               PMAP_SUBOBJ_UNLOCK(pmap, i - 2);
-       }
-       return NULL;
+       }
+       return error;
 }
 
 /*
@@ -2384,10 +2377,10 @@
        pmap = pool_cache_get(&pmap_cache, PR_WAITOK);
 
        /* init uvm_object */
+       mutex_init(&pmap->pm_lock, MUTEX_DEFAULT, IPL_NONE);
        for (i = 0; i < PTP_LEVELS - 1; i++) {
-               mutex_init(&pmap->pm_obj_lock[i], MUTEX_DEFAULT, IPL_NONE);
                uvm_obj_init(&pmap->pm_obj[i], NULL, false, 1);
-               uvm_obj_setlock(&pmap->pm_obj[i], &pmap->pm_obj_lock[i]);
+               uvm_obj_setlock(&pmap->pm_obj[i], &pmap->pm_lock);
                pmap->pm_ptphint[i] = NULL;
        }
        pmap->pm_stats.wired_count = 0;
@@ -2585,8 +2578,8 @@
 
        for (i = 0; i < PTP_LEVELS - 1; i++) {
                uvm_obj_destroy(&pmap->pm_obj[i], false);
-               mutex_destroy(&pmap->pm_obj_lock[i]);
-       }
+       }
+       mutex_destroy(&pmap->pm_lock);



Home | Main Index | Thread Index | Old Index