Commit b024ed0d authored by Keith Randall's avatar Keith Randall

cmd/compile: eliminate copy for static literals

*p = [5]byte{1,2,3,4,5}

First we allocate a global containing the RHS.  Then we copy
that global to a local stack variable, and then copy that local
stack variable to *p.  The intermediate copy is unnecessary.

Note that this only works if the RHS is completely constant.
If the code was:
*p = [5]byte{1,2,x,4,5}
this optimization doesn't apply as we have to construct the
RHS on the stack before copying it to *p.

Fixes #12841

Change-Id: I7cd0404ecc7a2d1750cbd8fe1222dba0fa44611f
Reviewed-on: https://go-review.googlesource.com/22192Reviewed-by: default avatarJosh Bleecher Snyder <josharian@gmail.com>
parent f60fcca5
...@@ -563,6 +563,32 @@ func getdyn(n *Node, top int) initGenType { ...@@ -563,6 +563,32 @@ func getdyn(n *Node, top int) initGenType {
return mode return mode
} }
// isStaticCompositeLiteral reports whether n is a compile-time constant.
// n must be a struct or array literal.
func isStaticCompositeLiteral(n *Node) bool {
for _, r := range n.List.Slice() {
if r.Op != OKEY {
Fatalf("isStaticCompositeLiteral: rhs not OKEY: %v", r)
}
index := r.Left
if n.Op == OARRAYLIT && index.Op != OLITERAL {
return false
}
value := r.Right
switch value.Op {
case OSTRUCTLIT, OARRAYLIT:
if !isStaticCompositeLiteral(value) {
return false
}
default:
if value.Op != OLITERAL {
return false
}
}
}
return true
}
func structlit(ctxt int, pass int, n *Node, var_ *Node, init *Nodes) { func structlit(ctxt int, pass int, n *Node, var_ *Node, init *Nodes) {
for _, r := range n.List.Slice() { for _, r := range n.List.Slice() {
if r.Op != OKEY { if r.Op != OKEY {
......
...@@ -1531,6 +1531,19 @@ opswitch: ...@@ -1531,6 +1531,19 @@ opswitch:
n = r n = r
case OARRAYLIT, OMAPLIT, OSTRUCTLIT, OPTRLIT: case OARRAYLIT, OMAPLIT, OSTRUCTLIT, OPTRLIT:
if (n.Op == OSTRUCTLIT || (n.Op == OARRAYLIT && !n.Type.IsSlice())) && isStaticCompositeLiteral(n) {
// n can be directly represented in the read-only data section.
// Make direct reference to the static data. See issue 12841.
vstat := staticname(n.Type, 0)
if n.Op == OSTRUCTLIT {
structlit(0, 1, n, vstat, init)
} else {
arraylit(0, 1, n, vstat, init)
}
n = vstat
n = typecheck(n, Erv)
break
}
var_ := temp(n.Type) var_ := temp(n.Type)
anylit(0, n, var_, init) anylit(0, n, var_, init)
n = var_ n = var_
......
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