Commit 3c6e60c0 authored by Josh Bleecher Snyder's avatar Josh Bleecher Snyder

cmd/compile: fix isStaticCompositeLiteral

Previously, isStaticCompositeLiteral would
return the wrong value for literals like:

[1]struct{ b []byte }{b: []byte{1}}

Note that the outermost component is an array,
but once we recurse into isStaticCompositeLiteral,
we never check again that arrays are actually arrays.

Instead of adding more logic to the guts of
isStaticCompositeLiteral, allow it to accept
any Node and return the correct answer.

Change-Id: I6af7814a9037bbc7043da9a96137fbee067bbe0e
Reviewed-on: https://go-review.googlesource.com/22247Reviewed-by: default avatarKeith Randall <khr@golang.org>
parent 562d398a
...@@ -564,8 +564,18 @@ func getdyn(n *Node, top int) initGenType { ...@@ -564,8 +564,18 @@ func getdyn(n *Node, top int) initGenType {
} }
// isStaticCompositeLiteral reports whether n is a compile-time constant. // isStaticCompositeLiteral reports whether n is a compile-time constant.
// n must be a struct or array literal.
func isStaticCompositeLiteral(n *Node) bool { func isStaticCompositeLiteral(n *Node) bool {
switch n.Op {
case OARRAYLIT:
if n.Type.IsSlice() {
return false
}
case OSTRUCTLIT:
case OLITERAL:
return true
default:
return false
}
for _, r := range n.List.Slice() { for _, r := range n.List.Slice() {
if r.Op != OKEY { if r.Op != OKEY {
Fatalf("isStaticCompositeLiteral: rhs not OKEY: %v", r) Fatalf("isStaticCompositeLiteral: rhs not OKEY: %v", r)
...@@ -575,15 +585,8 @@ func isStaticCompositeLiteral(n *Node) bool { ...@@ -575,15 +585,8 @@ func isStaticCompositeLiteral(n *Node) bool {
return false return false
} }
value := r.Right value := r.Right
switch value.Op { if !isStaticCompositeLiteral(value) {
case OSTRUCTLIT, OARRAYLIT: return false
if !isStaticCompositeLiteral(value) {
return false
}
default:
if value.Op != OLITERAL {
return false
}
} }
} }
return true return true
...@@ -1031,7 +1034,7 @@ func anylit(ctxt int, n *Node, var_ *Node, init *Nodes) { ...@@ -1031,7 +1034,7 @@ func anylit(ctxt int, n *Node, var_ *Node, init *Nodes) {
t := n.Type t := n.Type
switch n.Op { switch n.Op {
default: default:
Fatalf("anylit: not lit") Fatalf("anylit: not lit, op=%v node=%v", opnames[n.Op], n)
case OPTRLIT: case OPTRLIT:
if !t.IsPtr() { if !t.IsPtr() {
......
...@@ -1531,7 +1531,7 @@ opswitch: ...@@ -1531,7 +1531,7 @@ 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) { if isStaticCompositeLiteral(n) {
// n can be directly represented in the read-only data section. // n can be directly represented in the read-only data section.
// Make direct reference to the static data. See issue 12841. // Make direct reference to the static data. See issue 12841.
vstat := staticname(n.Type, 0) vstat := staticname(n.Type, 0)
......
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