Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch/x86/x86 - Share common code between pmap_load() and...
details: https://anonhg.NetBSD.org/src/rev/1581dd117b49
branches: trunk
changeset: 967619:1581dd117b49
user: ad <ad%NetBSD.org@localhost>
date: Sun Dec 15 20:33:22 2019 +0000
description:
- Share common code between pmap_load() and pmap_map_ptes().
- Make pmap_map_ptes() better tolerate recovery from blocking.
diffstat:
sys/arch/x86/x86/pmap.c | 94 +++++++++++++++++++++++++-----------------------
1 files changed, 49 insertions(+), 45 deletions(-)
diffs (171 lines):
diff -r c5a459458469 -r 1581dd117b49 sys/arch/x86/x86/pmap.c
--- a/sys/arch/x86/x86/pmap.c Sun Dec 15 20:30:56 2019 +0000
+++ b/sys/arch/x86/x86/pmap.c Sun Dec 15 20:33:22 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: pmap.c,v 1.344 2019/12/15 19:24:11 ad Exp $ */
+/* $NetBSD: pmap.c,v 1.345 2019/12/15 20:33:22 ad Exp $ */
/*
* Copyright (c) 2008, 2010, 2016, 2017, 2019 The NetBSD Foundation, Inc.
@@ -130,7 +130,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.344 2019/12/15 19:24:11 ad Exp $");
+__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.345 2019/12/15 20:33:22 ad Exp $");
#include "opt_user_ldt.h"
#include "opt_lockdebug.h"
@@ -470,6 +470,7 @@
static void pmap_alloc_level(struct pmap *, vaddr_t, long *);
+static void pmap_load1(struct lwp *, struct pmap *, struct pmap *);
static void pmap_reactivate(struct pmap *);
/*
@@ -642,20 +643,12 @@
pmap_reactivate(pmap);
} else {
/*
- * Toss current pmap from CPU, but keep a reference to it.
- * The reference will be dropped by pmap_unmap_ptes().
- * Can happen if we block during exit().
+ * Toss current pmap from CPU and install new pmap, but keep
+ * a reference to the old one - dropping the reference can
+ * block, so we'll defer to pmap_unmap_ptes().
*/
- const cpuid_t cid = cpu_index(ci);
-
- kcpuset_atomic_clear(curpmap->pm_cpus, cid);
- 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);
+ pmap_reference(pmap);
+ pmap_load1(l, pmap, curpmap);
}
pmap->pm_ncsw = lwp_pctr();
*pmap2 = curpmap;
@@ -714,7 +707,7 @@
* Mark whatever's on the CPU now as lazy and unlock.
* If the pmap was already installed, we are done.
*/
- if (ci->ci_tlbstate == TLBSTATE_VALID) {
+ if (ci->ci_pmap != mypmap && ci->ci_tlbstate == TLBSTATE_VALID) {
ci->ci_tlbstate = TLBSTATE_LAZY;
ci->ci_want_pmapload = (mypmap != pmap_kernel());
} else {
@@ -722,21 +715,17 @@
* This can happen when undoing after pmap_get_ptp blocked.
*/
}
+
/* Now safe to free PTPs, with the pmap still locked. */
if (ptp_tofree != NULL) {
pmap_freepages(pmap, ptp_tofree);
}
mutex_exit(&pmap->pm_lock);
- if (pmap == pmap2) {
- return;
- }
-
- /*
- * We installed another pmap on the CPU. Grab a reference to
- * it and leave in place. Toss the evicted pmap (can block).
- */
- pmap_reference(pmap);
- pmap_destroy(pmap2);
+
+ /* Toss the pmap we evicted earlier (can block). */
+ if (pmap != pmap2) {
+ pmap_destroy(pmap2);
+ }
}
inline static void
@@ -2884,8 +2873,6 @@
struct cpu_info *ci;
struct pmap *pmap, *oldpmap;
struct lwp *l;
- struct pcb *pcb;
- cpuid_t cid;
uint64_t ncsw;
kpreempt_disable();
@@ -2912,7 +2899,6 @@
pmap = vm_map_pmap(&l->l_proc->p_vmspace->vm_map);
KASSERT(pmap != pmap_kernel());
oldpmap = ci->ci_pmap;
- pcb = lwp_getpcb(l);
if (pmap == oldpmap) {
pmap_reactivate(pmap);
@@ -2926,8 +2912,41 @@
*/
pmap_reference(pmap);
-
+ pmap_load1(l, pmap, oldpmap);
+ ci->ci_want_pmapload = 0;
+
+ /*
+ * we're now running with the new pmap. drop the reference
+ * to the old pmap. if we block, we need to go around again.
+ */
+
+ pmap_destroy(oldpmap);
+ __insn_barrier();
+ if (l->l_ncsw != ncsw) {
+ goto retry;
+ }
+
+ kpreempt_enable();
+}
+
+/*
+ * pmap_load1: the guts of pmap load, shared by pmap_map_ptes() and
+ * pmap_load(). It's critically important that this function does not
+ * block.
+ */
+static void
+pmap_load1(struct lwp *l, struct pmap *pmap, struct pmap *oldpmap)
+{
+ struct cpu_info *ci;
+ struct pcb *pcb;
+ cpuid_t cid;
+
+ KASSERT(kpreempt_disabled());
+
+ pcb = lwp_getpcb(l);
+ ci = l->l_cpu;
cid = cpu_index(ci);
+
kcpuset_atomic_clear(oldpmap->pm_cpus, cid);
kcpuset_atomic_clear(oldpmap->pm_kernel_cpus, cid);
@@ -2970,21 +2989,6 @@
lldt(pmap->pm_ldt_sel);
cpu_load_pmap(pmap, oldpmap);
-
- ci->ci_want_pmapload = 0;
-
- /*
- * we're now running with the new pmap. drop the reference
- * to the old pmap. if we block, we need to go around again.
- */
-
- pmap_destroy(oldpmap);
- __insn_barrier();
- if (l->l_ncsw != ncsw) {
- goto retry;
- }
-
- kpreempt_enable();
}
/*
Home |
Main Index |
Thread Index |
Old Index