Commit 6c102e14 authored by Martin Möhrmann's avatar Martin Möhrmann

cmd/compile: avoid stack allocation of a map bucket for large constant hints

runtime.makemap will allocate map buckets on the heap for hints larger
than the number of elements a single map bucket can hold.

Do not allocate any map bucket on the stack if it is known at compile time
that hint is larger than the number of elements one map bucket can hold.
This avoids zeroing and reserving memory on the stack that will not be used.

Change-Id: I1a5ab853fb16f6a18d67674a77701bf0cf29b550
Reviewed-on: https://go-review.googlesource.com/60450
Run-TryBot: Martin Möhrmann <moehrmann@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarKeith Randall <khr@golang.org>
parent 034d825e
...@@ -1445,11 +1445,12 @@ opswitch: ...@@ -1445,11 +1445,12 @@ opswitch:
case OMAKEMAP: case OMAKEMAP:
t := n.Type t := n.Type
hmapType := hmap(t) hmapType := hmap(t)
hint := n.Left
// var h *hmap // var h *hmap
var h *Node var h *Node
if n.Esc == EscNone { if n.Esc == EscNone {
// Allocate hmap and one bucket on stack. // Allocate hmap on stack.
// var hv hmap // var hv hmap
hv := temp(hmapType) hv := temp(hmapType)
...@@ -1459,26 +1460,29 @@ opswitch: ...@@ -1459,26 +1460,29 @@ opswitch:
// h = &hv // h = &hv
h = nod(OADDR, hv, nil) h = nod(OADDR, hv, nil)
// Allocate one bucket pointed to by hmap.buckets on stack. // Allocate one bucket pointed to by hmap.buckets on stack if hint
// Maximum key/value size is 128 bytes, larger objects // is not larger than BUCKETSIZE. In case hint is larger than
// BUCKETSIZE runtime.makemap will allocate the buckets on the heap.
// Maximum key and value size is 128 bytes, larger objects
// are stored with an indirection. So max bucket size is 2048+eps. // are stored with an indirection. So max bucket size is 2048+eps.
if !Isconst(hint, CTINT) ||
!(hint.Val().U.(*Mpint).CmpInt64(BUCKETSIZE) > 0) {
// var bv bmap
bv := temp(bmap(t))
// var bv bmap zero = nod(OAS, bv, nil)
bv := temp(bmap(t)) zero = typecheck(zero, Etop)
init.Append(zero)
zero = nod(OAS, bv, nil)
zero = typecheck(zero, Etop)
init.Append(zero)
// b = &bv // b = &bv
b := nod(OADDR, bv, nil) b := nod(OADDR, bv, nil)
// h.buckets = b
bsym := hmapType.Field(5).Sym // hmap.buckets see reflect.go:hmap
na := nod(OAS, nodSym(ODOT, h, bsym), b)
na = typecheck(na, Etop)
init.Append(na)
// h.buckets = b
bsym := hmapType.Field(5).Sym // hmap.buckets see reflect.go:hmap
na := nod(OAS, nodSym(ODOT, h, bsym), b)
na = typecheck(na, Etop)
init.Append(na)
}
} else { } else {
// h = nil // h = nil
h = nodnil() h = nodnil()
...@@ -1486,7 +1490,6 @@ opswitch: ...@@ -1486,7 +1490,6 @@ opswitch:
// When hint fits into int, use makemap instead of // When hint fits into int, use makemap instead of
// makemap64, which is faster and shorter on 32 bit platforms. // makemap64, which is faster and shorter on 32 bit platforms.
hint := n.Left
fnname := "makemap64" fnname := "makemap64"
argtype := types.Types[TINT64] argtype := types.Types[TINT64]
......
...@@ -646,14 +646,31 @@ func BenchmarkMapPop100(b *testing.B) { benchmarkMapPop(b, 100) } ...@@ -646,14 +646,31 @@ func BenchmarkMapPop100(b *testing.B) { benchmarkMapPop(b, 100) }
func BenchmarkMapPop1000(b *testing.B) { benchmarkMapPop(b, 1000) } func BenchmarkMapPop1000(b *testing.B) { benchmarkMapPop(b, 1000) }
func BenchmarkMapPop10000(b *testing.B) { benchmarkMapPop(b, 10000) } func BenchmarkMapPop10000(b *testing.B) { benchmarkMapPop(b, 10000) }
var testNonEscapingMapVariable int = 8
func TestNonEscapingMap(t *testing.T) { func TestNonEscapingMap(t *testing.T) {
n := testing.AllocsPerRun(1000, func() { n := testing.AllocsPerRun(1000, func() {
m := make(map[int]int) m := make(map[int]int)
m[0] = 0 m[0] = 0
}) })
if n != 0 { if n != 0 {
t.Fatalf("want 0 allocs, got %v", n) t.Fatalf("no hint: want 0 allocs, got %v", n)
}
n = testing.AllocsPerRun(1000, func() {
m := make(map[int]int, 8)
m[0] = 0
})
if n != 0 {
t.Fatalf("with small hint: want 0 allocs, got %v", n)
} }
n = testing.AllocsPerRun(1000, func() {
m := make(map[int]int, testNonEscapingMapVariable)
m[0] = 0
})
if n != 0 {
t.Fatalf("with variable hint: want 0 allocs, got %v", n)
}
} }
func benchmarkMapAssignInt32(b *testing.B, n int) { func benchmarkMapAssignInt32(b *testing.B, n int) {
......
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