Source-Changes-HG archive

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

[src/trunk]: src/sys/uvm Make the uvm_voaddr structure more compact, only occ...



details:   https://anonhg.NetBSD.org/src/rev/df36d666104f
branches:  trunk
changeset: 1009965:df36d666104f
user:      thorpej <thorpej%NetBSD.org@localhost>
date:      Sat May 09 15:13:19 2020 +0000

description:
Make the uvm_voaddr structure more compact, only occupying 2 pointers
worth of space, by encoding the type in the lower bits of the object
pointer.

diffstat:

 sys/uvm/uvm_extern.h |  14 ++-------
 sys/uvm/uvm_map.c    |  74 ++++++++++++++++++++++++++++++++++-----------------
 2 files changed, 52 insertions(+), 36 deletions(-)

diffs (175 lines):

diff -r 9a45869bac96 -r df36d666104f sys/uvm/uvm_extern.h
--- a/sys/uvm/uvm_extern.h      Sat May 09 13:16:41 2020 +0000
+++ b/sys/uvm/uvm_extern.h      Sat May 09 15:13:19 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: uvm_extern.h,v 1.225 2020/04/27 02:47:26 rin Exp $     */
+/*     $NetBSD: uvm_extern.h,v 1.226 2020/05/09 15:13:19 thorpej Exp $ */
 
 /*
  * Copyright (c) 1997 Charles D. Cranor and Washington University.
@@ -613,8 +613,7 @@
  *
  *     This structure encapsulates UVM's unique virtual object address
  *     for an individual byte inside a pageable page. Pageable pages can
- *     be owned by either a uvm_object (UVM_VOADDR_TYPE_OBJECT) or a
- *     vm_anon (UVM_VOADDR_TYPE_ANON).
+ *     be owned by either a uvm_object or a vm_anon.
  *
  *     In each case, the byte offset into the owning object
  *     (uvm_object or vm_anon) is included in the ID, so that
@@ -631,14 +630,7 @@
  *     use.
  */
 struct uvm_voaddr {
-       enum {
-               UVM_VOADDR_TYPE_OBJECT = 1,
-               UVM_VOADDR_TYPE_ANON = 2,
-       } type;
-       union {
-               struct uvm_object *uobj;
-               struct vm_anon *anon;
-       };
+       uintptr_t object;
        voff_t offset;
 };
 
diff -r 9a45869bac96 -r df36d666104f sys/uvm/uvm_map.c
--- a/sys/uvm/uvm_map.c Sat May 09 13:16:41 2020 +0000
+++ b/sys/uvm/uvm_map.c Sat May 09 15:13:19 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: uvm_map.c,v 1.382 2020/04/30 04:18:07 thorpej Exp $    */
+/*     $NetBSD: uvm_map.c,v 1.383 2020/05/09 15:13:19 thorpej Exp $    */
 
 /*
  * Copyright (c) 1997 Charles D. Cranor and Washington University.
@@ -66,7 +66,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: uvm_map.c,v 1.382 2020/04/30 04:18:07 thorpej Exp $");
+__KERNEL_RCSID(0, "$NetBSD: uvm_map.c,v 1.383 2020/05/09 15:13:19 thorpej Exp $");
 
 #include "opt_ddb.h"
 #include "opt_pax.h"
@@ -4781,6 +4781,31 @@
        }
 }
 
+#define        UVM_VOADDR_TYPE_MASK    0x3UL
+#define        UVM_VOADDR_TYPE_UOBJ    0x1UL
+#define        UVM_VOADDR_TYPE_ANON    0x2UL
+#define        UVM_VOADDR_OBJECT_MASK  ~UVM_VOADDR_TYPE_MASK
+
+#define        UVM_VOADDR_GET_TYPE(voa)                                        \
+       ((voa)->object & UVM_VOADDR_TYPE_MASK)
+#define        UVM_VOADDR_GET_OBJECT(voa)                                      \
+       ((voa)->object & UVM_VOADDR_OBJECT_MASK)
+#define        UVM_VOADDR_SET_OBJECT(voa, obj, type)                           \
+do {                                                                   \
+       KASSERT(((uintptr_t)(obj) & UVM_VOADDR_TYPE_MASK) == 0);        \
+       (voa)->object = ((uintptr_t)(obj)) | (type);                    \
+} while (/*CONSTCOND*/0)
+
+#define        UVM_VOADDR_GET_UOBJ(voa)                                        \
+       ((struct uvm_object *)UVM_VOADDR_GET_OBJECT(voa))
+#define        UVM_VOADDR_SET_UOBJ(voa, uobj)                                  \
+       UVM_VOADDR_SET_OBJECT(voa, uobj, UVM_VOADDR_TYPE_UOBJ)
+
+#define        UVM_VOADDR_GET_ANON(voa)                                        \
+       ((struct vm_anon *)UVM_VOADDR_GET_OBJECT(voa))
+#define        UVM_VOADDR_SET_ANON(voa, anon)                                  \
+       UVM_VOADDR_SET_OBJECT(voa, anon, UVM_VOADDR_TYPE_ANON)
+
 /*
  * uvm_voaddr_acquire: returns the virtual object address corresponding
  * to the specified virtual address.
@@ -4936,8 +4961,7 @@
                        anon->an_ref++;
                        rw_obj_hold(anon->an_lock);
                        KASSERT(anon->an_ref != 0);
-                       voaddr->type = UVM_VOADDR_TYPE_ANON;
-                       voaddr->anon = anon;
+                       UVM_VOADDR_SET_ANON(voaddr, anon);
                        voaddr->offset = va & PAGE_MASK;
                        result = true;
                }
@@ -4950,8 +4974,7 @@
 
                KASSERT(uobj != NULL);
                (*uobj->pgops->pgo_reference)(uobj);
-               voaddr->type = UVM_VOADDR_TYPE_OBJECT;
-               voaddr->uobj = uobj;
+               UVM_VOADDR_SET_UOBJ(voaddr, uobj);
                voaddr->offset = entry->offset + (va - entry->start);
                result = true;
        }
@@ -4961,7 +4984,9 @@
        if (result) {
                UVMHIST_LOG(maphist,
                    "<- done OK (type=%jd,owner=#%jx,offset=%jx)",
-                   voaddr->type, (uintptr_t)voaddr->uobj, voaddr->offset, 0);
+                   UVM_VOADDR_GET_TYPE(voaddr),
+                   UVM_VOADDR_GET_OBJECT(voaddr),
+                   voaddr->offset, 0);
        } else {
                UVMHIST_LOG(maphist,"<- done (failed)",0,0,0,0);
        }
@@ -4977,9 +5002,9 @@
 uvm_voaddr_release(struct uvm_voaddr * const voaddr)
 {
 
-       switch (voaddr->type) {
-       case UVM_VOADDR_TYPE_OBJECT: {
-               struct uvm_object * const uobj = voaddr->uobj;
+       switch (UVM_VOADDR_GET_TYPE(voaddr)) {
+       case UVM_VOADDR_TYPE_UOBJ: {
+               struct uvm_object * const uobj = UVM_VOADDR_GET_UOBJ(voaddr);
 
                KASSERT(uobj != NULL);
                KASSERT(uobj->pgops->pgo_detach != NULL);
@@ -4987,7 +5012,7 @@
                break;
            }
        case UVM_VOADDR_TYPE_ANON: {
-               struct vm_anon * const anon = voaddr->anon;
+               struct vm_anon * const anon = UVM_VOADDR_GET_ANON(voaddr);
                krwlock_t *lock;
 
                KASSERT(anon != NULL);
@@ -5015,23 +5040,22 @@
 uvm_voaddr_compare(const struct uvm_voaddr * const voaddr1,
     const struct uvm_voaddr * const voaddr2)
 {
-
-       KASSERT(voaddr1->type == UVM_VOADDR_TYPE_OBJECT ||
-               voaddr1->type == UVM_VOADDR_TYPE_ANON);
-
-       KASSERT(voaddr2->type == UVM_VOADDR_TYPE_OBJECT ||
-               voaddr2->type == UVM_VOADDR_TYPE_ANON);
-
-       if (voaddr1->type < voaddr2->type)
+       const uintptr_t type1 = UVM_VOADDR_GET_TYPE(voaddr1);
+       const uintptr_t type2 = UVM_VOADDR_GET_TYPE(voaddr2);
+
+       KASSERT(type1 == UVM_VOADDR_TYPE_UOBJ ||
+               type1 == UVM_VOADDR_TYPE_ANON);
+
+       KASSERT(type2 == UVM_VOADDR_TYPE_UOBJ ||
+               type2 == UVM_VOADDR_TYPE_ANON);
+
+       if (type1 < type2)
                return -1;
-       if (voaddr1->type > voaddr2->type)
+       if (type1 > type2)
                return 1;
 
-       /* These fields are unioned together. */
-       CTASSERT(offsetof(struct uvm_voaddr, uobj) ==
-                offsetof(struct uvm_voaddr, anon));
-       const uintptr_t addr1 = (uintptr_t)voaddr1->uobj;
-       const uintptr_t addr2 = (uintptr_t)voaddr2->uobj;
+       const uintptr_t addr1 = UVM_VOADDR_GET_OBJECT(voaddr1);
+       const uintptr_t addr2 = UVM_VOADDR_GET_OBJECT(voaddr2);
 
        if (addr1 < addr2)
                return -1;



Home | Main Index | Thread Index | Old Index