Source-Changes-HG archive

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

[src/trunk]: src/sys/arch/x86 Replace the pv_hash_locks with atomic ops.



details:   https://anonhg.NetBSD.org/src/rev/e34179f708d3
branches:  trunk
changeset: 1006143:e34179f708d3
user:      ad <ad%NetBSD.org@localhost>
date:      Thu Jan 02 19:20:01 2020 +0000

description:
Replace the pv_hash_locks with atomic ops.

Leave the hash table at the same size for now: with the hash table size
doubled, system time for a build drops 10-15%, but user time starts to rise
suspiciously, presumably because the cache is wrecked.  Need to try another
data structure.

diffstat:

 sys/arch/x86/include/pmap_pv.h |    4 +-
 sys/arch/x86/x86/pmap.c        |  130 ++++++++++++++++++++++++----------------
 2 files changed, 81 insertions(+), 53 deletions(-)

diffs (235 lines):

diff -r 09448e24f7f3 -r e34179f708d3 sys/arch/x86/include/pmap_pv.h
--- a/sys/arch/x86/include/pmap_pv.h    Thu Jan 02 19:11:12 2020 +0000
+++ b/sys/arch/x86/include/pmap_pv.h    Thu Jan 02 19:20:01 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: pmap_pv.h,v 1.6 2019/11/13 12:55:10 maxv Exp $ */
+/*     $NetBSD: pmap_pv.h,v 1.7 2020/01/02 19:20:01 ad Exp $   */
 
 /*-
  * Copyright (c)2008 YAMAMOTO Takashi,
@@ -56,7 +56,7 @@
 struct pv_entry {
        struct pv_pte pve_pte;          /* should be the first member */
        LIST_ENTRY(pv_entry) pve_list;  /* on pv_head::pvh_list */
-       SLIST_ENTRY(pv_entry) pve_hash;
+       struct pv_entry *pve_hash;      /* hash table link */
 };
 #define        pve_next        pve_list.le_next
 
diff -r 09448e24f7f3 -r e34179f708d3 sys/arch/x86/x86/pmap.c
--- a/sys/arch/x86/x86/pmap.c   Thu Jan 02 19:11:12 2020 +0000
+++ b/sys/arch/x86/x86/pmap.c   Thu Jan 02 19:20:01 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: pmap.c,v 1.349 2019/12/31 12:40:27 ad Exp $    */
+/*     $NetBSD: pmap.c,v 1.350 2020/01/02 19:20:02 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.349 2019/12/31 12:40:27 ad Exp $");
+__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.350 2020/01/02 19:20:02 ad Exp $");
 
 #include "opt_user_ldt.h"
 #include "opt_lockdebug.h"
@@ -302,16 +302,8 @@
 #define        VM_PAGE_TO_PP(pg)       (&(pg)->mdpage.mp_pp)
 
 #define        PV_HASH_SIZE            32768
-#define        PV_HASH_LOCK_CNT        32
-
-struct pv_hash_lock {
-       kmutex_t lock;
-} __aligned(CACHE_LINE_SIZE) pv_hash_locks[PV_HASH_LOCK_CNT]
-    __aligned(CACHE_LINE_SIZE);
-
-struct pv_hash_head {
-       SLIST_HEAD(, pv_entry) hh_list;
-} pv_hash_heads[PV_HASH_SIZE];
+
+struct pv_entry        *pv_hash_heads[PV_HASH_SIZE];
 
 static u_int
 pvhash_hash(struct vm_page *ptp, vaddr_t va)
@@ -320,39 +312,78 @@
        return (uintptr_t)ptp / sizeof(*ptp) + (va >> PAGE_SHIFT);
 }
 
-static struct pv_hash_head *
+static struct pv_entry **
 pvhash_head(u_int hash)
 {
 
-       return &pv_hash_heads[hash % PV_HASH_SIZE];
-}
-
-static kmutex_t *
-pvhash_lock(u_int hash)
-{
-
-       return &pv_hash_locks[hash % PV_HASH_LOCK_CNT].lock;
+       return &pv_hash_heads[hash & (PV_HASH_SIZE - 1)];
 }
 
 static struct pv_entry *
-pvhash_remove(struct pv_hash_head *hh, struct vm_page *ptp, vaddr_t va)
+pvhash_remove(struct pv_entry **hh, struct vm_page *ptp, vaddr_t va)
 {
-       struct pv_entry *pve;
-       struct pv_entry *prev;
+       struct pv_entry *pve, *first, *unlocked, *locked, *prev;
+       int s, count;
+
+       count = SPINLOCK_BACKOFF_MIN;
+       for (;;) {
+               /*
+                * Entries can't be added behind us, only in front of us, so
+                * if this is the first entry in the bucket we can remove
+                * the entry and observe that nobody has the bucket locked
+                * in one pass.
+                *
+                * Otherwise we have to lock the bucket - use the low bit in
+                * the hash head to indicate this.  Go to splvm() while in
+                * possetion of the lock as we don't want to be interrupted
+                * while holding it, and we can't context switch.
+                */
+               unlocked = (struct pv_entry *)((uintptr_t)*hh & ~1L);
+               KASSERT(unlocked != NULL);
+               if (unlocked->pve_pte.pte_ptp == ptp &&
+                   unlocked->pve_pte.pte_va == va) {
+                       first = atomic_cas_ptr(hh, unlocked, unlocked->pve_hash);
+                       if (__predict_true(first == unlocked)) {
+                               /* Easy case - entry removed. */
+                               return first;
+                       }
+               } else {
+                       s = splvm();
+                       locked = (struct pv_entry *)((uintptr_t)unlocked | 1L);
+                       first = atomic_cas_ptr(hh, unlocked, locked);
+                       if (first == unlocked) {
+                               /* Got it! */
+                               break;
+                       }
+                       splx(s);
+               }
+               SPINLOCK_BACKOFF(count);
+       }
+
+       KASSERT(((uintptr_t)first & 1L) == 0);
+       KASSERT(((uintptr_t)*hh & 1L) == 1);
 
        prev = NULL;
-       SLIST_FOREACH(pve, &hh->hh_list, pve_hash) {
+       for (pve = first; pve != NULL; pve = pve->pve_hash) {
                if (pve->pve_pte.pte_ptp == ptp &&
                    pve->pve_pte.pte_va == va) {
                        if (prev != NULL) {
-                               SLIST_REMOVE_AFTER(prev, pve_hash);
+                               /* Remove, then unlock below. */
+                               prev->pve_hash = pve->pve_hash;
+                               break;
                        } else {
-                               SLIST_REMOVE_HEAD(&hh->hh_list, pve_hash);
+                               /* Remove AND unlock below. */
+                               first = pve->pve_hash;
+                               break;
                        }
-                       break;
                }
                prev = pve;
        }
+
+       /* Lock release and SPL drop must be last, and in order. */
+       __insn_barrier();
+       *hh = first;
+       splx(s);
        return pve;
 }
 
@@ -1709,14 +1740,7 @@
 void
 pmap_init(void)
 {
-       int i, flags;
-
-       for (i = 0; i < PV_HASH_SIZE; i++) {
-               SLIST_INIT(&pv_hash_heads[i].hh_list);
-       }
-       for (i = 0; i < PV_HASH_LOCK_CNT; i++) {
-               mutex_init(&pv_hash_locks[i].lock, MUTEX_NODEBUG, IPL_VM);
-       }
+       int flags;
 
        /*
         * initialize caches.
@@ -1868,8 +1892,8 @@
  *   pmap_enter_pv: enter a mapping onto a pv_head list
  *   pmap_remove_pv: remove a mapping from a pv_head list
  *
- * NOTE: Both pmap_enter_pv and pmap_remove_pv expect the caller to lock
- *       the pvh before calling
+ * NOTE: Both pmap_enter_pv and pmap_remove_pv expect the caller to have
+ *      locked object that owns the page before calling
  */
 
 /*
@@ -1878,16 +1902,23 @@
 static void
 insert_pv(struct pmap_page *pp, struct pv_entry *pve)
 {
-       struct pv_hash_head *hh;
-       kmutex_t *lock;
-       u_int hash;
+       struct pv_entry **hh, *o, *v;
+       u_int hash, count;
 
        hash = pvhash_hash(pve->pve_pte.pte_ptp, pve->pve_pte.pte_va);
-       lock = pvhash_lock(hash);
        hh = pvhash_head(hash);
-       mutex_spin_enter(lock);
-       SLIST_INSERT_HEAD(&hh->hh_list, pve, pve_hash);
-       mutex_spin_exit(lock);
+
+       /* Observe the bucket unlocked and insert in one pass. */
+       count = SPINLOCK_BACKOFF_MIN;
+       for (v = *hh;;) {
+               o = (struct pv_entry *)((uintptr_t)v & ~1L);
+               pve->pve_hash = o;
+               v = atomic_cas_ptr(hh, o, pve);
+               if (o == v) {
+                       break;
+               }
+               SPINLOCK_BACKOFF(count);
+       }
 
        LIST_INSERT_HEAD(&pp->pp_head.pvh_list, pve, pve_list);
 }
@@ -1941,11 +1972,11 @@
  * => we return the removed pve
  */
 static struct pv_entry *
-pmap_remove_pv(struct pmap_page *pp, struct vm_page *ptp, vaddr_t va)
+pmap_remove_pv(struct pmap *pmap, struct pmap_page *pp, struct vm_page *ptp,
+    vaddr_t va)
 {
-       struct pv_hash_head *hh;
+       struct pv_entry **hh;
        struct pv_entry *pve;
-       kmutex_t *lock;
        u_int hash;
 
        KASSERT(ptp == NULL || ptp->uobject != NULL);
@@ -1962,11 +1993,8 @@
        }
 
        hash = pvhash_hash(ptp, va);
-       lock = pvhash_lock(hash);
        hh = pvhash_head(hash);
-       mutex_spin_enter(lock);
        pve = pvhash_remove(hh, ptp, va);
-       mutex_spin_exit(lock);
 
        LIST_REMOVE(pve, pve_list);
 



Home | Main Index | Thread Index | Old Index