Commit 77a9cb9b authored by Josh Bleecher Snyder's avatar Josh Bleecher Snyder

runtime: refactor evacuate x/y handling

This change unifies the x and y cases.

It shrinks evacuate's machine code by ~25% and its stack size by ~15%.

It also eliminates a critical branch.
Whether an entry should go to x or y is designed to be unpredictable.
As a result, half of the branch predictions for useX were wrong.
Mispredicting that branch can easily incur an expensive cache miss.
Switching to an xy array allows elimination of that branch,
which in turn reduces cache misses.

Change-Id: Ie9cef53744b96c724c377ac0985b487fc50b49b1
Reviewed-on: https://go-review.googlesource.com/54653
Run-TryBot: Josh Bleecher Snyder <josharian@gmail.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarKeith Randall <khr@golang.org>
parent 589fc314
...@@ -298,6 +298,10 @@ func makemap(t *maptype, hint int64, h *hmap, bucket unsafe.Pointer) *hmap { ...@@ -298,6 +298,10 @@ func makemap(t *maptype, hint int64, h *hmap, bucket unsafe.Pointer) *hmap {
if dataOffset%uintptr(t.elem.align) != 0 { if dataOffset%uintptr(t.elem.align) != 0 {
throw("need padding in bucket (value)") throw("need padding in bucket (value)")
} }
if evacuatedX+1 != evacuatedY {
// evacuate relies on this relationship
throw("bad evacuatedN")
}
// find size parameter which will hold the requested # of elements // find size parameter which will hold the requested # of elements
B := uint8(0) B := uint8(0)
...@@ -1044,24 +1048,30 @@ func evacuate(t *maptype, h *hmap, oldbucket uintptr) { ...@@ -1044,24 +1048,30 @@ func evacuate(t *maptype, h *hmap, oldbucket uintptr) {
// TODO: reuse overflow buckets instead of using new ones, if there // TODO: reuse overflow buckets instead of using new ones, if there
// is no iterator using the old buckets. (If !oldIterator.) // is no iterator using the old buckets. (If !oldIterator.)
var ( // evacDst is an evacuation destination.
x, y *bmap // current low/high buckets in new map type evacDst struct {
xi, yi int // key/val indices into x and y b *bmap // current destination bucket
xk, yk unsafe.Pointer // pointers to current x and y key storage i int // key/val index into b
xv, yv unsafe.Pointer // pointers to current x and y value storage k unsafe.Pointer // pointer to current key storage
) v unsafe.Pointer // pointer to current value storage
x = (*bmap)(add(h.buckets, oldbucket*uintptr(t.bucketsize))) }
xi = 0
xk = add(unsafe.Pointer(x), dataOffset) // xy contains the x and y (low and high) evacuation destinations.
xv = add(xk, bucketCnt*uintptr(t.keysize)) var xy [2]evacDst
x := &xy[0]
x.b = (*bmap)(add(h.buckets, oldbucket*uintptr(t.bucketsize)))
x.k = add(unsafe.Pointer(x.b), dataOffset)
x.v = add(x.k, bucketCnt*uintptr(t.keysize))
if !h.sameSizeGrow() { if !h.sameSizeGrow() {
// Only calculate y pointers if we're growing bigger. // Only calculate y pointers if we're growing bigger.
// Otherwise GC can see bad pointers. // Otherwise GC can see bad pointers.
y = (*bmap)(add(h.buckets, (oldbucket+newbit)*uintptr(t.bucketsize))) y := &xy[1]
yi = 0 y.b = (*bmap)(add(h.buckets, (oldbucket+newbit)*uintptr(t.bucketsize)))
yk = add(unsafe.Pointer(y), dataOffset) y.k = add(unsafe.Pointer(y.b), dataOffset)
yv = add(yk, bucketCnt*uintptr(t.keysize)) y.v = add(y.k, bucketCnt*uintptr(t.keysize))
} }
for ; b != nil; b = b.overflow(t) { for ; b != nil; b = b.overflow(t) {
k := add(unsafe.Pointer(b), dataOffset) k := add(unsafe.Pointer(b), dataOffset)
v := add(k, bucketCnt*uintptr(t.keysize)) v := add(k, bucketCnt*uintptr(t.keysize))
...@@ -1078,7 +1088,7 @@ func evacuate(t *maptype, h *hmap, oldbucket uintptr) { ...@@ -1078,7 +1088,7 @@ func evacuate(t *maptype, h *hmap, oldbucket uintptr) {
if t.indirectkey { if t.indirectkey {
k2 = *((*unsafe.Pointer)(k2)) k2 = *((*unsafe.Pointer)(k2))
} }
useX := true var useY uint8
if !h.sameSizeGrow() { if !h.sameSizeGrow() {
// Compute hash to make our evacuation decision (whether we need // Compute hash to make our evacuation decision (whether we need
// to send this key/value to bucket x or bucket y). // to send this key/value to bucket x or bucket y).
...@@ -1105,54 +1115,37 @@ func evacuate(t *maptype, h *hmap, oldbucket uintptr) { ...@@ -1105,54 +1115,37 @@ func evacuate(t *maptype, h *hmap, oldbucket uintptr) {
top += minTopHash top += minTopHash
} }
} }
useX = hash&newbit == 0 if hash&newbit != 0 {
} useY = 1
if useX {
b.tophash[i] = evacuatedX
if xi == bucketCnt {
newx := h.newoverflow(t, x)
x = newx
xi = 0
xk = add(unsafe.Pointer(x), dataOffset)
xv = add(xk, bucketCnt*uintptr(t.keysize))
} }
x.tophash[xi] = top
if t.indirectkey {
*(*unsafe.Pointer)(xk) = k2 // copy pointer
} else {
typedmemmove(t.key, xk, k) // copy value
} }
if t.indirectvalue {
*(*unsafe.Pointer)(xv) = *(*unsafe.Pointer)(v) b.tophash[i] = evacuatedX + useY // evacuatedX + 1 == evacuatedY, enforced in makemap
} else { dst := &xy[useY] // evacuation destination
typedmemmove(t.elem, xv, v)
if dst.i == bucketCnt {
dst.b = h.newoverflow(t, dst.b)
dst.i = 0
dst.k = add(unsafe.Pointer(dst.b), dataOffset)
dst.v = add(dst.k, bucketCnt*uintptr(t.keysize))
} }
xi++ dst.b.tophash[dst.i] = top
xk = add(xk, uintptr(t.keysize))
xv = add(xv, uintptr(t.valuesize))
} else {
b.tophash[i] = evacuatedY
if yi == bucketCnt {
newy := h.newoverflow(t, y)
y = newy
yi = 0
yk = add(unsafe.Pointer(y), dataOffset)
yv = add(yk, bucketCnt*uintptr(t.keysize))
}
y.tophash[yi] = top
if t.indirectkey { if t.indirectkey {
*(*unsafe.Pointer)(yk) = k2 *(*unsafe.Pointer)(dst.k) = k2 // copy pointer
} else { } else {
typedmemmove(t.key, yk, k) typedmemmove(t.key, dst.k, k) // copy value
} }
if t.indirectvalue { if t.indirectvalue {
*(*unsafe.Pointer)(yv) = *(*unsafe.Pointer)(v) *(*unsafe.Pointer)(dst.v) = *(*unsafe.Pointer)(v)
} else { } else {
typedmemmove(t.elem, yv, v) typedmemmove(t.elem, dst.v, v)
} }
yi++ dst.i++
yk = add(yk, uintptr(t.keysize)) // If we're at the end of the bucket, don't update k/v,
yv = add(yv, uintptr(t.valuesize)) // to avoid pointers pointing past the end of the bucket.
if dst.i < bucketCnt {
dst.k = add(dst.k, uintptr(t.keysize))
dst.v = add(dst.v, uintptr(t.valuesize))
} }
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment