Commit c33d45a8 authored by Matthew Dempsky's avatar Matthew Dempsky

cmd/compile: don't statically copy string-typed variables

During package initialization, the compiler tries to optimize:

    var A = "foo"
    var B = A

into

    var A = "foo"
    var B = "foo"

so that we can statically initialize both A and B and skip emitting
dynamic initialization code to assign "B = A".

However, this isn't safe in the presence of cmd/link's -X flag, which
might overwrite an initialized string-typed variable at link time. In
particular, if cmd/link changes A's static initialization, it won't
know it also needs to change B's static initialization.

To address this, this CL disables this optimization for string-typed
variables.

Fixes #34675.

Change-Id: I1c18f3b855f6d7114aeb39f96aaaf1b452b88236
Reviewed-on: https://go-review.googlesource.com/c/go/+/198657
Run-TryBot: Matthew Dempsky <mdempsky@google.com>
Reviewed-by: default avatarCuong Manh Le <cuong.manhle.vn@gmail.com>
Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
parent 6b85fa80
...@@ -83,6 +83,9 @@ func (s *InitSchedule) staticcopy(l *Node, r *Node) bool { ...@@ -83,6 +83,9 @@ func (s *InitSchedule) staticcopy(l *Node, r *Node) bool {
if r.Name.Defn.Op != OAS { if r.Name.Defn.Op != OAS {
return false return false
} }
if r.Type.IsString() { // perhaps overwritten by cmd/link -X (#34675)
return false
}
orig := r orig := r
r = r.Name.Defn.Right r = r.Name.Defn.Right
......
...@@ -14,10 +14,19 @@ import "fmt" ...@@ -14,10 +14,19 @@ import "fmt"
var tbd string var tbd string
var overwrite string = "dibs" var overwrite string = "dibs"
var tbdcopy = tbd
var overwritecopy = overwrite
var arraycopy = [2]string{tbd, overwrite}
var b bool var b bool
var x int var x int
func main() { func main() {
fmt.Println(tbd) fmt.Println(tbd)
fmt.Println(tbdcopy)
fmt.Println(arraycopy[0])
fmt.Println(overwrite) fmt.Println(overwrite)
fmt.Println(overwritecopy)
fmt.Println(arraycopy[1])
} }
...@@ -36,7 +36,7 @@ func test(sep string) { ...@@ -36,7 +36,7 @@ func test(sep string) {
os.Exit(1) os.Exit(1)
} }
want := "hello\ntrumped\n" want := "hello\nhello\nhello\ntrumped\ntrumped\ntrumped\n"
got := out.String() got := out.String()
if got != want { if got != want {
fmt.Printf("got %q want %q\n", got, want) fmt.Printf("got %q want %q\n", got, want)
......
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