Skip to content

Commit 442de98

Browse files
committed
cmd/compile,runtime: redo how map assignments work
To compile: m[k] = v instead of: mapassign(maptype, m, &k, &v), do do: *mapassign(maptype, m, &k) = v mapassign returns a pointer to the value slot in the map. It is just like mapaccess except that it will allocate a new slot if k is not already present in the map. This makes map accesses faster but potentially larger (codewise). It is faster because the write into the map is done when the compiler knows the concrete type, so it can be done with a few store instructions instead of calling typedmemmove. We also potentially avoid stack temporaries to hold v. The code can be larger when the map has pointers in its value type, since there is a write barrier call in addition to the mapassign call. That makes the code at the callsite a bit bigger (go binary is 0.3% bigger). This CL is in preparation for doing operations like m[k] += v with only a single runtime call. That will roughly double the speed of such operations. Update #17133 Update #5147 Change-Id: Ia435f032090a2ed905dac9234e693972fe8c2dc5 Reviewed-on: https://go-review.googlesource.com/30815 Run-TryBot: Keith Randall <[email protected]> Reviewed-by: Matthew Dempsky <[email protected]> TryBot-Result: Gobot Gobot <[email protected]>
1 parent 55ef67f commit 442de98

File tree

8 files changed

+137
-150
lines changed

8 files changed

+137
-150
lines changed

src/cmd/compile/internal/gc/builtin.go

+46-46
Original file line numberDiff line numberDiff line change
@@ -57,52 +57,52 @@ const runtimeimport = "" +
5757
"ast64\x00\x06\x17\"\xc2\x01\x00\x00\x1d::\xc4\x01\x00\x00:\xc6\x01\x00\x00\x04\x17:\xb4\x01\x00\x00\x00\xc8\x01\x00\x00\t#m" +
5858
"apaccess2_faststr\x00\x06\x17\"\xc2\x01\x00\x00\x1d::\xc4\x01\x00\x00:\xc6\x01\x00\x00\x04\x17:" +
5959
"\xb4\x01\x00\x00\x00\xc8\x01\x00\x00\t\x1bmapaccess2_fat\x00\b\x17\"\xc2\x01\x00\x00\x1d::\xc4\x01\x00\x00" +
60-
"\x17:\xc6\x01\x00\x00\x17\"\rzero·6\x00\x00\x04\x17:\xb4\x01\x00\x00\x00\xc8\x01\x00\x00\t\x13mapassig" +
61-
"n1\x00\b\x17\"\x13mapType·1\x00\x00\x1d::\rhmap·2\x00\x00\x17:\vkey·" +
62-
"3\x00\x00\x17:\vval·4\x00\x00\x00\t\x15mapiterinit\x00\x06\x17\"\xd6\x01\x00\x00\x1d::\xd8" +
63-
"\x01\x00\x00\x17:\x0fhiter·3\x00\x00\x00\t\x11mapdelete\x00\x06\x17\"\xd6\x01\x00\x00\x1d::\xd8" +
64-
"\x01\x00\x00\x17:\xda\x01\x00\x00\x00\t\x15mapiternext\x00\x02\x17:\x0fhiter·1\x00\x00\x00\t" +
65-
"\x0fmakechan\x00\x04\x17\"\x15chanType·2\x00\x00\n\xa6\x01\x00\x00\x02\x1f\x06:\x0fhch" +
66-
"an·1\x00\x00\t\x11chanrecv1\x00\x06\x17\"\x15chanType·1\x00\x00\x1f\x02:\x0f" +
67-
"hchan·2\x00\x00\x17:j\x00\x00\x00\t\x11chanrecv2\x00\x06\x17\"\xea\x01\x00\x00\x1f\x02:\x0fh" +
68-
"chan·3\x00\x00\x17:\relem·4\x00\x00\x01\x00\x00\t\x11chansend1\x00\x06\x17\"\xf0" +
69-
"\x01\x00\x00\x1f\x04:\xf2\x01\x00\x00\x17:j\x00\x00\x00\t\x11closechan\x00\x02:\xec\x01\x00\x00\x00\a\x17wri" +
70-
"teBarrier\x00\x15\x06\renabled\x00\x00\x00\vneeded\x00\x00\x00\x05cgo\x00\x00\x00" +
71-
"\t\x1dwritebarrierptr\x00\x04\x17:\vdst·1\x00\x00:\vsrc·2\x00\x00" +
72-
"\x00\t\x17typedmemmove\x00\x06\x17\"t\x00\x00\x17:\vdst·2\x00\x00\x17:\vsrc\xc2" +
73-
"\xb73\x00\x00\x00\t\x1btypedslicecopy\x00\x06\x17\"\x06\x00\x00:\vdst·3\x00\x00:\v" +
74-
"src·4\x00\x00\x01\x02\x00\t\x17selectnbsend\x00\x06\x17\"\xea\x01\x00\x00\x1f\x04:\xf6\x01\x00\x00" +
75-
"\x17:\xf8\x01\x00\x00\x01\x00\x00\t\x17selectnbrecv\x00\x06\x17\"\xea\x01\x00\x00\x17:j\x00\x00\x1f\x02:\x0f" +
76-
"hchan·4\x00\x00\x01\x00\x00\t\x19selectnbrecv2\x00\b\x17\"\xea\x01\x00\x00\x17:j\x00" +
77-
"\x00\x17\x00\x15received·4\x00\x00\x1f\x02:\x0fhchan·5\x00\x00\x01\x00\x00\t\x11news" +
78-
"elect\x00\x06\x17\"\vsel·1\x00\x00\n\x13selsize·2\x00\x00\b\rsize·" +
79-
"3\x00\x00\x00\t\x13selectsend\x00\x06\x17\"\vsel·2\x00\x00\x1f\x04:\xf6\x01\x00\x00\x17:\xf8\x01" +
80-
"\x00\x00\x02\x00\x15selected·1\x00\x00\t\x13selectrecv\x00\x06\x17\"\xae\x02\x00\x00\x1f\x02" +
81-
":\xf6\x01\x00\x00\x17:\xf8\x01\x00\x00\x02\x00\xb0\x02\x00\x00\t\x15selectrecv2\x00\b\x17\"\xae\x02\x00\x00\x1f\x02" +
82-
":\xf6\x01\x00\x00\x17:\xf8\x01\x00\x00\x17\x00\x15received·5\x00\x00\x02\x00\xb0\x02\x00\x00\t\x19selec" +
83-
"tdefault\x00\x02\x17\"\xae\x02\x00\x00\x02\x00\xb0\x02\x00\x00\t\x0fselectgo\x00\x02\x17\"\xa6\x02\x00\x00" +
84-
"\x00\t\tblock\x00\x00\x00\t\x11makeslice\x00\x06\x17\"\x06\x00\x00\x02\vlen·3\x00\x00\x02" +
85-
"\vcap·4\x00\x00\x02\x11:\vary·1\x00\x00\t\x15makeslice64\x00\x06\x17\"\x06\x00" +
86-
"\x00\n\xc0\x02\x00\x00\n\xc2\x02\x00\x00\x02\x11:\xc4\x02\x00\x00\t\x11growslice\x00\x06\x17\"\x06\x00\x00\x11:\vo" +
87-
"ld·3\x00\x00\x02\xc2\x02\x00\x00\x02\x11:\xc4\x02\x00\x00\t\rmemmove\x00\x06\x17:\tto·1\x00\x00" +
88-
"\x17:\vfrm·2\x00\x00\x16\x11length·3\x00^\x00\t\vmemclr\x00\x04\x17\"\vpt" +
89-
"r·1\x00\x00\x16\x11length·2\x00^\x00\t\x0fmemequal\x00\x06\x17:\ax·2\x00" +
90-
"\x00\x17:\ay·3\x00\x00\x16\rsize·4\x00^\x01\x00\x00\t\x11memequal8\x00\x04\x17:\xdc" +
91-
"\x02\x00\x00\x17:\xde\x02\x00\x00\x01\x00\x00\t\x13memequal16\x00\x04\x17:\xdc\x02\x00\x00\x17:\xde\x02\x00\x00\x01\x00" +
92-
"\x00\t\x13memequal32\x00\x04\x17:\xdc\x02\x00\x00\x17:\xde\x02\x00\x00\x01\x00\x00\t\x13memequal" +
93-
"64\x00\x04\x17:\xdc\x02\x00\x00\x17:\xde\x02\x00\x00\x01\x00\x00\t\x15memequal128\x00\x04\x17:\xdc\x02\x00\x00" +
94-
"\x17:\xde\x02\x00\x00\x01\x00\x00\t\x0fint64div\x00\x03\n\x00\n\x00\x01\n\x00\t\x11uint64div\x00" +
95-
"\x03\x14\x00\x14\x00\x01\x14\x00\t\x0fint64mod\x00\x03\n\x00\n\x00\x01\n\x00\t\x11uint64mod\x00\x03" +
96-
"\x14\x00\x14\x00\x01\x14\x00\t\x1bfloat64toint64\x00\x01\x1a\x00\x01\n\x00\t\x1dfloat64t" +
97-
"ouint64\x00\x01\x1a\x00\x01\x14\x00\t\x1dfloat64touint32\x00\x01\x1a\x00\x01\x12\x00\t\x1b" +
98-
"int64tofloat64\x00\x01\n\x00\x01\x1a\x00\t\x1duint64tofloat64\x00\x01" +
99-
"\x14\x00\x01\x1a\x00\t\x1duint32tofloat64\x00\x01\x12\x00\x01\x1a\x00\t\x19complex12" +
100-
"8div\x00\x04\x1e\vnum·2\x00\x00\x1e\vden·3\x00\x00\x02\x1e\vquo·1\x00\x00\t\x19r" +
101-
"acefuncenter\x00\x01\x16^\x00\t\x17racefuncexit\x00\x00\x00\t\x0frace" +
102-
"read\x00\x01\x16^\x00\t\x11racewrite\x00\x01\x16^\x00\t\x19racereadrange" +
103-
"\x00\x04\x16\raddr·1\x00^\x16\rsize·2\x00^\x00\t\x1bracewriterang" +
104-
"e\x00\x04\x16\x92\x03\x00^\x16\x94\x03\x00^\x00\t\x0fmsanread\x00\x04\x16\x92\x03\x00^\x16\x94\x03\x00^\x00\t\x11m" +
105-
"sanwrite\x00\x04\x16\x92\x03\x00^\x16\x94\x03\x00^\x00\v\xf6\x01\v\x00\x01\x00\n$$\n"
60+
"\x17:\xc6\x01\x00\x00\x17\"\rzero·6\x00\x00\x04\x17:\xb4\x01\x00\x00\x00\xc8\x01\x00\x00\t\x11mapassig" +
61+
"n\x00\x06\x17\"\xa4\x01\x00\x00\x1d::\xb0\x01\x00\x00\x17:\xb2\x01\x00\x00\x02\x17:\xb4\x01\x00\x00\t\x15mapiterin" +
62+
"it\x00\x06\x17\"\x13mapType·1\x00\x00\x1d::\rhmap·2\x00\x00\x17:\x0fhiter" +
63+
"·3\x00\x00\x00\t\x11mapdelete\x00\x06\x17\"\xd8\x01\x00\x00\x1d::\xda\x01\x00\x00\x17:\vkey·" +
64+
"3\x00\x00\x00\t\x15mapiternext\x00\x02\x17:\x0fhiter·1\x00\x00\x00\t\x0fmakec" +
65+
"han\x00\x04\x17\"\x15chanType·2\x00\x00\n\xa6\x01\x00\x00\x02\x1f\x06:\x0fhchan·1\x00" +
66+
"\x00\t\x11chanrecv1\x00\x06\x17\"\x15chanType·1\x00\x00\x1f\x02:\x0fhchan\xc2" +
67+
"\xb72\x00\x00\x17:j\x00\x00\x00\t\x11chanrecv2\x00\x06\x17\"\xe8\x01\x00\x00\x1f\x02:\x0fhchan·" +
68+
"3\x00\x00\x17:\relem·4\x00\x00\x01\x00\x00\t\x11chansend1\x00\x06\x17\"\xee\x01\x00\x00\x1f\x04:" +
69+
"\xf0\x01\x00\x00\x17:j\x00\x00\x00\t\x11closechan\x00\x02:\xea\x01\x00\x00\x00\a\x17writeBarr" +
70+
"ier\x00\x15\x06\renabled\x00\x00\x00\vneeded\x00\x00\x00\x05cgo\x00\x00\x00\t\x1dwrit" +
71+
"ebarrierptr\x00\x04\x17:\vdst·1\x00\x00:\vsrc·2\x00\x00\x00\t\x17typ" +
72+
"edmemmove\x00\x06\x17\"t\x00\x00\x17:\vdst·2\x00\x00\x17:\vsrc·3\x00\x00\x00\t" +
73+
"\x1btypedslicecopy\x00\x06\x17\"\x06\x00\x00:\vdst·3\x00\x00:\vsrc·4" +
74+
"\x00\x00\x01\x02\x00\t\x17selectnbsend\x00\x06\x17\"\xe8\x01\x00\x00\x1f\x04:\xf4\x01\x00\x00\x17:\xf6\x01\x00\x00" +
75+
"\x01\x00\x00\t\x17selectnbrecv\x00\x06\x17\"\xe8\x01\x00\x00\x17:j\x00\x00\x1f\x02:\x0fhchan\xc2" +
76+
"\xb74\x00\x00\x01\x00\x00\t\x19selectnbrecv2\x00\b\x17\"\xe8\x01\x00\x00\x17:j\x00\x00\x17\x00\x15re" +
77+
"ceived·4\x00\x00\x1f\x02:\x0fhchan·5\x00\x00\x01\x00\x00\t\x11newselect\x00" +
78+
"\x06\x17\"\vsel·1\x00\x00\n\x13selsize·2\x00\x00\b\rsize·3\x00\x00\x00\t\x13" +
79+
"selectsend\x00\x06\x17\"\vsel·2\x00\x00\x1f\x04:\xf4\x01\x00\x00\x17:\xf6\x01\x00\x00\x02\x00\x15s" +
80+
"elected·1\x00\x00\t\x13selectrecv\x00\x06\x17\"\xac\x02\x00\x00\x1f\x02:\xf4\x01\x00\x00\x17" +
81+
":\xf6\x01\x00\x00\x02\x00\xae\x02\x00\x00\t\x15selectrecv2\x00\b\x17\"\xac\x02\x00\x00\x1f\x02:\xf4\x01\x00\x00\x17" +
82+
":\xf6\x01\x00\x00\x17\x00\x15received·5\x00\x00\x02\x00\xae\x02\x00\x00\t\x19selectdefau" +
83+
"lt\x00\x02\x17\"\xac\x02\x00\x00\x02\x00\xae\x02\x00\x00\t\x0fselectgo\x00\x02\x17\"\xa4\x02\x00\x00\x00\t\tblo" +
84+
"ck\x00\x00\x00\t\x11makeslice\x00\x06\x17\"\x06\x00\x00\x02\vlen·3\x00\x00\x02\vcap·" +
85+
"4\x00\x00\x02\x11:\vary·1\x00\x00\t\x15makeslice64\x00\x06\x17\"\x06\x00\x00\n\xbe\x02\x00\x00" +
86+
"\n\xc0\x02\x00\x00\x02\x11:\xc2\x02\x00\x00\t\x11growslice\x00\x06\x17\"\x06\x00\x00\x11:\vold·3\x00" +
87+
"\x00\x02\xc0\x02\x00\x00\x02\x11:\xc2\x02\x00\x00\t\rmemmove\x00\x06\x17:\tto·1\x00\x00\x17:\vfrm" +
88+
"·2\x00\x00\x16\x11length·3\x00^\x00\t\vmemclr\x00\x04\x17\"\vptr·1\x00\x00" +
89+
"\x16\x11length·2\x00^\x00\t\x0fmemequal\x00\x06\x17:\ax·2\x00\x00\x17:\ay\xc2" +
90+
"\xb73\x00\x00\x16\rsize·4\x00^\x01\x00\x00\t\x11memequal8\x00\x04\x17:\xda\x02\x00\x00\x17:\xdc" +
91+
"\x02\x00\x00\x01\x00\x00\t\x13memequal16\x00\x04\x17:\xda\x02\x00\x00\x17:\xdc\x02\x00\x00\x01\x00\x00\t\x13mem" +
92+
"equal32\x00\x04\x17:\xda\x02\x00\x00\x17:\xdc\x02\x00\x00\x01\x00\x00\t\x13memequal64\x00\x04\x17:" +
93+
"\xda\x02\x00\x00\x17:\xdc\x02\x00\x00\x01\x00\x00\t\x15memequal128\x00\x04\x17:\xda\x02\x00\x00\x17:\xdc\x02\x00\x00" +
94+
"\x01\x00\x00\t\x0fint64div\x00\x03\n\x00\n\x00\x01\n\x00\t\x11uint64div\x00\x03\x14\x00\x14\x00\x01" +
95+
"\x14\x00\t\x0fint64mod\x00\x03\n\x00\n\x00\x01\n\x00\t\x11uint64mod\x00\x03\x14\x00\x14\x00\x01\x14" +
96+
"\x00\t\x1bfloat64toint64\x00\x01\x1a\x00\x01\n\x00\t\x1dfloat64touint6" +
97+
"4\x00\x01\x1a\x00\x01\x14\x00\t\x1dfloat64touint32\x00\x01\x1a\x00\x01\x12\x00\t\x1bint64t" +
98+
"ofloat64\x00\x01\n\x00\x01\x1a\x00\t\x1duint64tofloat64\x00\x01\x14\x00\x01\x1a\x00\t" +
99+
"\x1duint32tofloat64\x00\x01\x12\x00\x01\x1a\x00\t\x19complex128div\x00\x04" +
100+
"\x1e\vnum·2\x00\x00\x1e\vden·3\x00\x00\x02\x1e\vquo·1\x00\x00\t\x19racefun" +
101+
"center\x00\x01\x16^\x00\t\x17racefuncexit\x00\x00\x00\t\x0fraceread\x00\x01" +
102+
"\x16^\x00\t\x11racewrite\x00\x01\x16^\x00\t\x19racereadrange\x00\x04\x16\rad" +
103+
"dr·1\x00^\x16\rsize·2\x00^\x00\t\x1bracewriterange\x00\x04\x16\x90\x03" +
104+
"\x00^\x16\x92\x03\x00^\x00\t\x0fmsanread\x00\x04\x16\x90\x03\x00^\x16\x92\x03\x00^\x00\t\x11msanwri" +
105+
"te\x00\x04\x16\x90\x03\x00^\x16\x92\x03\x00^\x00\v\xf6\x01\v\x00\x01\x00\n$$\n"
106106

107107
const unsafeimport = "" +
108108
"version 2\n\n\x00\x00\x01\vunsafe\x00\t\x0fOffsetof\x00\x01:\x00\x01\x16\x00\t" +

src/cmd/compile/internal/gc/builtin/runtime.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -93,7 +93,7 @@ func mapaccess2_fast32(mapType *byte, hmap map[any]any, key any) (val *any, pres
9393
func mapaccess2_fast64(mapType *byte, hmap map[any]any, key any) (val *any, pres bool)
9494
func mapaccess2_faststr(mapType *byte, hmap map[any]any, key any) (val *any, pres bool)
9595
func mapaccess2_fat(mapType *byte, hmap map[any]any, key *any, zero *byte) (val *any, pres bool)
96-
func mapassign1(mapType *byte, hmap map[any]any, key *any, val *any)
96+
func mapassign(mapType *byte, hmap map[any]any, key *any) (val *any)
9797
func mapiterinit(mapType *byte, hmap map[any]any, hiter *any)
9898
func mapdelete(mapType *byte, hmap map[any]any, key *any)
9999
func mapiternext(hiter *any)

src/cmd/compile/internal/gc/order.go

+17-15
Original file line numberDiff line numberDiff line change
@@ -188,9 +188,9 @@ func isaddrokay(n *Node) bool {
188188
return islvalue(n) && (n.Op != ONAME || n.Class == PEXTERN || istemp(n))
189189
}
190190

191-
// Orderaddrtemp ensures that *np is okay to pass by address to runtime routines.
192-
// If the original argument *np is not okay, orderaddrtemp creates a tmp, emits
193-
// tmp = *np, and then sets *np to the tmp variable.
191+
// Orderaddrtemp ensures that n is okay to pass by address to runtime routines.
192+
// If the original argument n is not okay, orderaddrtemp creates a tmp, emits
193+
// tmp = n, and then returns tmp.
194194
func orderaddrtemp(n *Node, order *Order) *Node {
195195
if isaddrokay(n) {
196196
return n
@@ -395,13 +395,8 @@ func ordercall(n *Node, order *Order) {
395395
}
396396

397397
// Ordermapassign appends n to order->out, introducing temporaries
398-
// to make sure that all map assignments have the form m[k] = x,
399-
// where x is addressable.
400-
// (Orderexpr has already been called on n, so we know k is addressable.)
401-
//
402-
// If n is m[k] = x where x is not addressable, the rewrite is:
403-
// tmp = x
404-
// m[k] = tmp
398+
// to make sure that all map assignments have the form m[k] = x.
399+
// (Note: orderexpr has already been called on n, so we know k is addressable.)
405400
//
406401
// If n is the multiple assignment form ..., m[k], ... = ..., the rewrite is
407402
// t1 = m
@@ -428,7 +423,7 @@ func ordermapassign(n *Node, order *Order) {
428423
// We call writebarrierfat only for values > 4 pointers long. See walk.go.
429424
// TODO(mdempsky): writebarrierfat doesn't exist anymore, but removing that
430425
// logic causes net/http's tests to become flaky; see CL 21242.
431-
if (n.Left.Op == OINDEXMAP || (needwritebarrier(n.Left, n.Right) && n.Left.Type.Width > int64(4*Widthptr))) && !isaddrokay(n.Right) {
426+
if needwritebarrier(n.Left, n.Right) && n.Left.Type.Width > int64(4*Widthptr) && !isaddrokay(n.Right) {
432427
m := n.Left
433428
n.Left = ordertemp(m.Type, order, false)
434429
a := nod(OAS, m, n.Left)
@@ -1061,8 +1056,14 @@ func orderexpr(n *Node, order *Order, lhs *Node) *Node {
10611056
// key must be addressable
10621057
case OINDEXMAP:
10631058
n.Left = orderexpr(n.Left, order, nil)
1064-
10651059
n.Right = orderexpr(n.Right, order, nil)
1060+
needCopy := false
1061+
1062+
if n.Etype == 0 && instrumenting {
1063+
// Race detector needs the copy so it can
1064+
// call treecopy on the result.
1065+
needCopy = true
1066+
}
10661067

10671068
// For x = m[string(k)] where k is []byte, the allocation of
10681069
// backing bytes for the string can be avoided by reusing
@@ -1076,12 +1077,13 @@ func orderexpr(n *Node, order *Order, lhs *Node) *Node {
10761077
// conversion (by the ordercopyexpr a few lines below).
10771078
if n.Etype == 0 && n.Right.Op == OARRAYBYTESTR {
10781079
n.Right.Op = OARRAYBYTESTRTMP
1080+
needCopy = true
10791081
}
10801082

1083+
// Map calls need to take the address of the key.
10811084
n.Right = orderaddrtemp(n.Right, order)
1082-
if n.Etype == 0 {
1083-
// use of value (not being assigned);
1084-
// make copy in temporary.
1085+
1086+
if needCopy {
10851087
n = ordercopyexpr(n, n.Type, order, 0)
10861088
}
10871089

src/cmd/compile/internal/gc/racewalk.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -474,8 +474,8 @@ func isartificial(n *Node) bool {
474474
func callinstr(np **Node, init *Nodes, wr int, skip int) bool {
475475
n := *np
476476

477-
//print("callinstr for %+N [ %O ] etype=%E class=%d\n",
478-
// n, n->op, n->type ? n->type->etype : -1, n->class);
477+
//fmt.Printf("callinstr for %v [ %v ] etype=%v class=%v\n",
478+
// n, n.Op, n.Type.Etype, n.Class)
479479

480480
if skip != 0 || n.Type == nil || n.Type.Etype >= TIDEAL {
481481
return false

src/cmd/compile/internal/gc/syntax.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -52,7 +52,7 @@ type Node struct {
5252
Op Op
5353
Ullman uint8 // sethi/ullman number
5454
Addable bool // addressable
55-
Etype EType // op for OASOP, etype for OTYPE, exclam for export, 6g saved reg, ChanDir for OTCHAN
55+
Etype EType // op for OASOP, etype for OTYPE, exclam for export, 6g saved reg, ChanDir for OTCHAN, for OINDEXMAP 1=LHS,0=RHS
5656
Bounded bool // bounds check unnecessary
5757
NonNil bool // guaranteed to be non-nil
5858
Class Class // PPARAM, PAUTO, PEXTERN, etc

src/cmd/compile/internal/gc/walk.go

+36-48
Original file line numberDiff line numberDiff line change
@@ -1292,44 +1292,48 @@ opswitch:
12921292
}
12931293

12941294
case OINDEXMAP:
1295-
if n.Etype == 1 {
1296-
break
1297-
}
1295+
// Replace m[k] with *map{access1,assign}(maptype, m, &k)
12981296
n.Left = walkexpr(n.Left, init)
12991297
n.Right = walkexpr(n.Right, init)
1300-
1301-
t := n.Left.Type
1302-
p := ""
1303-
if t.Val().Width <= 128 { // Check ../../runtime/hashmap.go:maxValueSize before changing.
1304-
switch algtype(t.Key()) {
1305-
case AMEM32:
1306-
p = "mapaccess1_fast32"
1307-
case AMEM64:
1308-
p = "mapaccess1_fast64"
1309-
case ASTRING:
1310-
p = "mapaccess1_faststr"
1298+
map_ := n.Left
1299+
key := n.Right
1300+
t := map_.Type
1301+
if n.Etype == 1 {
1302+
// This m[k] expression is on the left-hand side of an assignment.
1303+
// orderexpr made sure key is addressable.
1304+
key = nod(OADDR, key, nil)
1305+
n = mkcall1(mapfn("mapassign", t), nil, init, typename(t), map_, key)
1306+
} else {
1307+
// m[k] is not the target of an assignment.
1308+
p := ""
1309+
if t.Val().Width <= 128 { // Check ../../runtime/hashmap.go:maxValueSize before changing.
1310+
switch algtype(t.Key()) {
1311+
case AMEM32:
1312+
p = "mapaccess1_fast32"
1313+
case AMEM64:
1314+
p = "mapaccess1_fast64"
1315+
case ASTRING:
1316+
p = "mapaccess1_faststr"
1317+
}
13111318
}
1312-
}
13131319

1314-
var key *Node
1315-
if p != "" {
1316-
// fast versions take key by value
1317-
key = n.Right
1318-
} else {
1319-
// standard version takes key by reference.
1320-
// orderexpr made sure key is addressable.
1321-
key = nod(OADDR, n.Right, nil)
1322-
p = "mapaccess1"
1323-
}
1320+
if p == "" {
1321+
// standard version takes key by reference.
1322+
// orderexpr made sure key is addressable.
1323+
key = nod(OADDR, key, nil)
1324+
p = "mapaccess1"
1325+
}
13241326

1325-
if w := t.Val().Width; w <= 1024 { // 1024 must match ../../../../runtime/hashmap.go:maxZero
1326-
n = mkcall1(mapfn(p, t), ptrto(t.Val()), init, typename(t), n.Left, key)
1327-
} else {
1328-
p = "mapaccess1_fat"
1329-
z := zeroaddr(w)
1330-
n = mkcall1(mapfn(p, t), ptrto(t.Val()), init, typename(t), n.Left, key, z)
1327+
if w := t.Val().Width; w <= 1024 { // 1024 must match ../../../../runtime/hashmap.go:maxZero
1328+
n = mkcall1(mapfn(p, t), ptrto(t.Val()), init, typename(t), map_, key)
1329+
} else {
1330+
p = "mapaccess1_fat"
1331+
z := zeroaddr(w)
1332+
n = mkcall1(mapfn(p, t), ptrto(t.Val()), init, typename(t), map_, key, z)
1333+
}
13311334
}
1332-
n.NonNil = true // mapaccess always returns a non-nil pointer
1335+
n.Type = ptrto(t.Val())
1336+
n.NonNil = true // mapaccess1* and mapassign always return non-nil pointers.
13331337
n = nod(OIND, n, nil)
13341338
n.Type = t.Val()
13351339
n.Typecheck = 1
@@ -2306,22 +2310,6 @@ func convas(n *Node, init *Nodes) *Node {
23062310
goto out
23072311
}
23082312

2309-
if n.Left.Op == OINDEXMAP {
2310-
map_ := n.Left.Left
2311-
key := n.Left.Right
2312-
val := n.Right
2313-
map_ = walkexpr(map_, init)
2314-
key = walkexpr(key, init)
2315-
val = walkexpr(val, init)
2316-
2317-
// orderexpr made sure key and val are addressable.
2318-
key = nod(OADDR, key, nil)
2319-
2320-
val = nod(OADDR, val, nil)
2321-
n = mkcall1(mapfn("mapassign1", map_.Type), nil, init, typename(map_.Type), map_, key, val)
2322-
goto out
2323-
}
2324-
23252313
if !eqtype(lt, rt) {
23262314
n.Right = assignconv(n.Right, lt, "assignment")
23272315
n.Right = walkexpr(n.Right, init)

0 commit comments

Comments
 (0)