- add a kmutex_t ci_ctx_lock to struct cpu_info, and initialise it in
cpu_pmap_init() and replace pmap_ctx_lock usage with this new ci_ctx_lock - replace smp_dcache_flush_page_all() with smp_dcache_flush_page_cpuset(), that flushes only on the set of CPUs not, everyone. add new dcache_flush_page_cpuset() to flush this page from the D$ only on the specified set of CPUs. - add a cpuset to pmap_free_page() and use it when freeing PTE pages when a pmap is destroyed - introduce pmap_free_page_noflush(), and use it when we allocated a page for PTEs but didn't use it and don't need to flush it - don't bother with pmap_lock in pmap_extract(), the only potential issue is pseg_get() which is already safe tested on sb2000, sb2500 and ultra80 with a bunch of various heavy workloads, and seems to give a clear 1-2% speed up for high-forking / short lived processes, such as ./configure.
This commit is contained in:
parent
98a4c48317
commit
0edf2f9c9a
@ -1,4 +1,4 @@
|
||||
/* $NetBSD: cpu.h,v 1.89 2010/03/06 08:08:29 mrg Exp $ */
|
||||
/* $NetBSD: cpu.h,v 1.90 2010/03/28 05:24:00 mrg Exp $ */
|
||||
|
||||
/*
|
||||
* Copyright (c) 1992, 1993
|
||||
@ -151,7 +151,10 @@ struct cpu_info {
|
||||
* the right pointer and you get to the pmap segment tables. These are
|
||||
* physical addresses, of course.
|
||||
*
|
||||
* ci_ctx_lock protects this CPUs context allocation/free.
|
||||
* These are all allocated almost with in the same cacheline.
|
||||
*/
|
||||
kmutex_t ci_ctx_lock;
|
||||
int ci_pmap_next_ctx;
|
||||
int ci_numctx;
|
||||
paddr_t *ci_ctxbusy;
|
||||
|
@ -1,4 +1,4 @@
|
||||
/* $NetBSD: cache.h,v 1.17 2010/03/08 08:59:06 mrg Exp $ */
|
||||
/* $NetBSD: cache.h,v 1.18 2010/03/28 05:24:00 mrg Exp $ */
|
||||
|
||||
/*
|
||||
* Copyright (c) 1996
|
||||
@ -144,14 +144,16 @@ sp_tlb_flush_all(void)
|
||||
|
||||
#ifdef MULTIPROCESSOR
|
||||
void smp_tlb_flush_pte(vaddr_t, struct pmap *);
|
||||
void smp_dcache_flush_page_all(paddr_t pa);
|
||||
void smp_dcache_flush_page_cpuset(paddr_t pa, sparc64_cpuset_t);
|
||||
void smp_blast_dcache(sparc64_cpuset_t);
|
||||
#define tlb_flush_pte(va,pm ) smp_tlb_flush_pte(va, pm)
|
||||
#define dcache_flush_page_all(pa) smp_dcache_flush_page_all(pa)
|
||||
#define dcache_flush_page_all(pa) smp_dcache_flush_page_cpuset(pa, cpus_active)
|
||||
#define dcache_flush_page_cpuset(pa,cs) smp_dcache_flush_page_cpuset(pa, cs)
|
||||
#define blast_dcache() smp_blast_dcache(cpus_active)
|
||||
#else
|
||||
#define tlb_flush_pte(va,pm) sp_tlb_flush_pte(va, (pm)->pm_ctx[0])
|
||||
#define dcache_flush_page_all(pa) dcache_flush_page(pa)
|
||||
#define dcache_flush_page_cpuset(pa,cs) dcache_flush_page(pa)
|
||||
#define blast_dcache() sp_blast_dcache(dcache_size, \
|
||||
dcache_line_size)
|
||||
#endif
|
||||
|
@ -1,4 +1,4 @@
|
||||
/* $NetBSD: ipifuncs.c,v 1.35 2010/03/08 08:59:06 mrg Exp $ */
|
||||
/* $NetBSD: ipifuncs.c,v 1.36 2010/03/28 05:24:00 mrg Exp $ */
|
||||
|
||||
/*-
|
||||
* Copyright (c) 2004 The NetBSD Foundation, Inc.
|
||||
@ -27,7 +27,7 @@
|
||||
*/
|
||||
|
||||
#include <sys/cdefs.h>
|
||||
__KERNEL_RCSID(0, "$NetBSD: ipifuncs.c,v 1.35 2010/03/08 08:59:06 mrg Exp $");
|
||||
__KERNEL_RCSID(0, "$NetBSD: ipifuncs.c,v 1.36 2010/03/28 05:24:00 mrg Exp $");
|
||||
|
||||
#include "opt_ddb.h"
|
||||
|
||||
@ -412,10 +412,10 @@ smp_tlb_flush_pte(vaddr_t va, struct pmap * pm)
|
||||
}
|
||||
|
||||
/*
|
||||
* Make sure this page is flushed from all CPUs.
|
||||
* Make sure this page is flushed from all/some CPUs.
|
||||
*/
|
||||
void
|
||||
smp_dcache_flush_page_all(paddr_t pa)
|
||||
smp_dcache_flush_page_cpuset(paddr_t pa, sparc64_cpuset_t activecpus)
|
||||
{
|
||||
ipifunc_t func;
|
||||
|
||||
@ -424,7 +424,7 @@ smp_dcache_flush_page_all(paddr_t pa)
|
||||
else
|
||||
func = sparc64_ipi_dcache_flush_page_us;
|
||||
|
||||
sparc64_broadcast_ipi(func, pa, dcache_line_size);
|
||||
sparc64_multicast_ipi(activecpus, func, pa, dcache_line_size);
|
||||
dcache_flush_page(pa);
|
||||
}
|
||||
|
||||
|
@ -1,4 +1,4 @@
|
||||
/* $NetBSD: pmap.c,v 1.261 2010/03/21 22:38:08 mrg Exp $ */
|
||||
/* $NetBSD: pmap.c,v 1.262 2010/03/28 05:24:00 mrg Exp $ */
|
||||
/*
|
||||
*
|
||||
* Copyright (C) 1996-1999 Eduardo Horvath.
|
||||
@ -26,7 +26,7 @@
|
||||
*/
|
||||
|
||||
#include <sys/cdefs.h>
|
||||
__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.261 2010/03/21 22:38:08 mrg Exp $");
|
||||
__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.262 2010/03/28 05:24:00 mrg Exp $");
|
||||
|
||||
#undef NO_VCACHE /* Don't forget the locked TLB in dostart */
|
||||
#define HWREF
|
||||
@ -323,14 +323,14 @@ int pmap_pages_stolen = 0;
|
||||
|
||||
#define pv_check()
|
||||
|
||||
static int pmap_get_page(paddr_t *p);
|
||||
static void pmap_free_page(paddr_t pa);
|
||||
static int pmap_get_page(paddr_t *);
|
||||
static void pmap_free_page(paddr_t, sparc64_cpuset_t);
|
||||
static void pmap_free_page_noflush(paddr_t);
|
||||
|
||||
/*
|
||||
* Global pmap lock.
|
||||
* Global pmap locks.
|
||||
*/
|
||||
static kmutex_t pmap_lock;
|
||||
static kmutex_t pmap_ctx_lock;
|
||||
static bool lock_available = false;
|
||||
|
||||
/*
|
||||
@ -1226,6 +1226,7 @@ cpu_pmap_init(struct cpu_info *ci)
|
||||
{
|
||||
size_t ctxsize;
|
||||
|
||||
mutex_init(&ci->ci_ctx_lock, MUTEX_SPIN, IPL_VM);
|
||||
ci->ci_pmap_next_ctx = 1;
|
||||
#ifdef SUN4V
|
||||
#error find out if we have 16 or 13 bit context ids
|
||||
@ -1295,7 +1296,6 @@ pmap_init(void)
|
||||
vm_num_phys = avail_end - avail_start;
|
||||
|
||||
mutex_init(&pmap_lock, MUTEX_DEFAULT, IPL_NONE);
|
||||
mutex_init(&pmap_ctx_lock, MUTEX_SPIN, IPL_VM);
|
||||
#if defined(USE_LOCKSAFE_PSEG_GETSET)
|
||||
mutex_init(&pseg_lock, MUTEX_SPIN, IPL_VM);
|
||||
#endif
|
||||
@ -1410,6 +1410,9 @@ pmap_destroy(struct pmap *pm)
|
||||
{
|
||||
#ifdef MULTIPROCESSOR
|
||||
struct cpu_info *ci;
|
||||
sparc64_cpuset_t pmap_cpus_active;
|
||||
#else
|
||||
#define pmap_cpus_active 0
|
||||
#endif
|
||||
struct vm_page *pg, *nextpg;
|
||||
|
||||
@ -1417,26 +1420,36 @@ pmap_destroy(struct pmap *pm)
|
||||
return;
|
||||
}
|
||||
DPRINTF(PDB_DESTROY, ("pmap_destroy: freeing pmap %p\n", pm));
|
||||
mutex_enter(&pmap_ctx_lock);
|
||||
#ifdef MULTIPROCESSOR
|
||||
CPUSET_CLEAR(pmap_cpus_active);
|
||||
for (ci = cpus; ci != NULL; ci = ci->ci_next) {
|
||||
if (CPUSET_HAS(cpus_active, ci->ci_index))
|
||||
/* XXXMRG: Move the lock inside one or both tests? */
|
||||
mutex_enter(&ci->ci_ctx_lock);
|
||||
if (CPUSET_HAS(cpus_active, ci->ci_index)) {
|
||||
if (pm->pm_ctx[ci->ci_index] > 0) {
|
||||
CPUSET_ADD(pmap_cpus_active, ci->ci_index);
|
||||
ctx_free(pm, ci);
|
||||
}
|
||||
}
|
||||
mutex_exit(&ci->ci_ctx_lock);
|
||||
}
|
||||
#else
|
||||
if (pmap_ctx(pm)) {
|
||||
mutex_enter(&curcpu()->ci_ctx_lock);
|
||||
ctx_free(pm, curcpu());
|
||||
mutex_exit(&curcpu()->ci_ctx_lock);
|
||||
}
|
||||
#endif
|
||||
mutex_exit(&pmap_ctx_lock);
|
||||
|
||||
/* we could be a little smarter and leave pages zeroed */
|
||||
for (pg = TAILQ_FIRST(&pm->pm_obj.memq); pg != NULL; pg = nextpg) {
|
||||
nextpg = TAILQ_NEXT(pg, listq.queue);
|
||||
TAILQ_REMOVE(&pm->pm_obj.memq, pg, listq.queue);
|
||||
KASSERT(pg->mdpage.mdpg_pvh.pv_pmap == NULL);
|
||||
dcache_flush_page_all(VM_PAGE_TO_PHYS(pg));
|
||||
dcache_flush_page_cpuset(VM_PAGE_TO_PHYS(pg), pmap_cpus_active);
|
||||
uvm_pagefree(pg);
|
||||
}
|
||||
pmap_free_page((paddr_t)(u_long)pm->pm_segs);
|
||||
pmap_free_page((paddr_t)(u_long)pm->pm_segs, pmap_cpus_active);
|
||||
UVM_OBJ_DESTROY(&pm->pm_obj);
|
||||
pool_cache_put(&pmap_cache, pm);
|
||||
}
|
||||
@ -1555,7 +1568,7 @@ pmap_kenter_pa(vaddr_t va, paddr_t pa, vm_prot_t prot, u_int flags)
|
||||
/* We allocated a spare page but didn't use it. Free it. */
|
||||
printf("pmap_kenter_pa: freeing unused page %llx\n",
|
||||
(long long)ptp);
|
||||
pmap_free_page(ptp);
|
||||
pmap_free_page_noflush(ptp);
|
||||
}
|
||||
#ifdef DEBUG
|
||||
i = ptelookup_va(va);
|
||||
@ -1826,7 +1839,7 @@ pmap_enter(struct pmap *pm, vaddr_t va, paddr_t pa, vm_prot_t prot, u_int flags)
|
||||
/* We allocated a spare page but didn't use it. Free it. */
|
||||
printf("pmap_enter: freeing unused page %llx\n",
|
||||
(long long)ptp);
|
||||
pmap_free_page(ptp);
|
||||
pmap_free_page_noflush(ptp);
|
||||
}
|
||||
if (dopv) {
|
||||
pmap_enter_pv(pm, va, pa, pg, npv);
|
||||
@ -1924,22 +1937,36 @@ pmap_remove_all(struct pmap *pm)
|
||||
write_user_windows();
|
||||
pm->pm_refs = 0;
|
||||
|
||||
mutex_enter(&pmap_ctx_lock);
|
||||
/*
|
||||
* XXXMRG: pmap_destroy() does exactly the same dance here.
|
||||
* surely one of them isn't necessary?
|
||||
*/
|
||||
#ifdef MULTIPROCESSOR
|
||||
CPUSET_CLEAR(pmap_cpus_active);
|
||||
for (ci = cpus; ci != NULL; ci = ci->ci_next) {
|
||||
/* XXXMRG: Move the lock inside one or both tests? */
|
||||
mutex_enter(&ci->ci_ctx_lock);
|
||||
if (CPUSET_HAS(cpus_active, ci->ci_index)) {
|
||||
if (pm->pm_ctx[ci->ci_index] > 0)
|
||||
if (pm->pm_ctx[ci->ci_index] > 0) {
|
||||
CPUSET_ADD(pmap_cpus_active, ci->ci_index);
|
||||
ctx_free(pm, ci);
|
||||
}
|
||||
}
|
||||
mutex_exit(&ci->ci_ctx_lock);
|
||||
}
|
||||
#else
|
||||
if (pmap_ctx(pm)) {
|
||||
mutex_enter(&curcpu()->ci_ctx_lock);
|
||||
ctx_free(pm, curcpu());
|
||||
mutex_exit(&curcpu()->ci_ctx_lock);
|
||||
}
|
||||
#endif
|
||||
mutex_exit(&pmap_ctx_lock);
|
||||
|
||||
REMOVE_STAT(flushes);
|
||||
/*
|
||||
* XXXMRG: couldn't we do something less severe here, and
|
||||
* only flush the right context on each CPU?
|
||||
*/
|
||||
#ifdef MULTIPROCESSOR
|
||||
smp_blast_dcache(pmap_cpus_active);
|
||||
#else
|
||||
@ -2021,7 +2048,8 @@ pmap_remove(struct pmap *pm, vaddr_t va, vaddr_t endva)
|
||||
continue;
|
||||
|
||||
/*
|
||||
* if the pmap is being torn down, don't bother flushing.
|
||||
* if the pmap is being torn down, don't bother flushing,
|
||||
* we already have done so.
|
||||
*/
|
||||
|
||||
if (!pm->pm_refs)
|
||||
@ -2166,9 +2194,6 @@ pmap_extract(struct pmap *pm, vaddr_t va, paddr_t *pap)
|
||||
*pap = pa;
|
||||
return TRUE;
|
||||
} else {
|
||||
if (pm != pmap_kernel()) {
|
||||
mutex_enter(&pmap_lock);
|
||||
}
|
||||
data = pseg_get(pm, va);
|
||||
pa = data & TLB_PA_MASK;
|
||||
#ifdef DEBUG
|
||||
@ -2200,9 +2225,6 @@ pmap_extract(struct pmap *pm, vaddr_t va, paddr_t *pap)
|
||||
printf(" pseg_get: %lx\n", (long)pa);
|
||||
}
|
||||
#endif
|
||||
if (pm != pmap_kernel()) {
|
||||
mutex_exit(&pmap_lock);
|
||||
}
|
||||
}
|
||||
if ((data & TLB_V) == 0)
|
||||
return (FALSE);
|
||||
@ -3073,7 +3095,7 @@ ctx_alloc(struct pmap *pm)
|
||||
|
||||
KASSERT(pm != pmap_kernel());
|
||||
KASSERT(pm == curproc->p_vmspace->vm_map.pmap);
|
||||
mutex_enter(&pmap_ctx_lock);
|
||||
mutex_enter(&curcpu()->ci_ctx_lock);
|
||||
ctx = curcpu()->ci_pmap_next_ctx++;
|
||||
|
||||
/*
|
||||
@ -3108,7 +3130,7 @@ ctx_alloc(struct pmap *pm)
|
||||
curcpu()->ci_ctxbusy[ctx] = pm->pm_physaddr;
|
||||
LIST_INSERT_HEAD(&curcpu()->ci_pmap_ctxlist, pm, pm_list[cpu_number()]);
|
||||
pmap_ctx(pm) = ctx;
|
||||
mutex_exit(&pmap_ctx_lock);
|
||||
mutex_exit(&curcpu()->ci_ctx_lock);
|
||||
DPRINTF(PDB_CTX_ALLOC, ("ctx_alloc: cpu%d allocated ctx %d\n",
|
||||
cpu_number(), ctx));
|
||||
return ctx;
|
||||
@ -3123,7 +3145,7 @@ ctx_free(struct pmap *pm, struct cpu_info *ci)
|
||||
int oldctx;
|
||||
int cpunum;
|
||||
|
||||
KASSERT(mutex_owned(&pmap_ctx_lock));
|
||||
KASSERT(mutex_owned(&curcpu()->ci_ctx_lock));
|
||||
|
||||
#ifdef MULTIPROCESSOR
|
||||
cpunum = ci->ci_index;
|
||||
@ -3382,14 +3404,21 @@ pmap_get_page(paddr_t *p)
|
||||
}
|
||||
|
||||
static void
|
||||
pmap_free_page(paddr_t pa)
|
||||
pmap_free_page(paddr_t pa, sparc64_cpuset_t cs)
|
||||
{
|
||||
struct vm_page *pg = PHYS_TO_VM_PAGE(pa);
|
||||
|
||||
dcache_flush_page_all(pa);
|
||||
dcache_flush_page_cpuset(pa, cs);
|
||||
uvm_pagefree(pg);
|
||||
}
|
||||
|
||||
static void
|
||||
pmap_free_page_noflush(paddr_t pa)
|
||||
{
|
||||
struct vm_page *pg = PHYS_TO_VM_PAGE(pa);
|
||||
|
||||
uvm_pagefree(pg);
|
||||
}
|
||||
|
||||
#ifdef DDB
|
||||
|
||||
@ -3609,7 +3638,7 @@ pmap_testout(void)
|
||||
|
||||
pmap_remove(pmap_kernel(), va, va+1);
|
||||
pmap_update(pmap_kernel());
|
||||
pmap_free_page(pa);
|
||||
pmap_free_page(pa, cpus_active);
|
||||
}
|
||||
#endif
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user