| // Copyright 2017 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package gc_test |
| |
| import ( |
| "bytes" |
| "internal/testenv" |
| "io/ioutil" |
| "os" |
| "os/exec" |
| "path/filepath" |
| "testing" |
| ) |
| |
| func TestReproducibleBuilds(t *testing.T) { |
| tests := []string{ |
| "issue20272.go", |
| "issue27013.go", |
| "issue30202.go", |
| } |
| |
| testenv.MustHaveGoBuild(t) |
| iters := 10 |
| if testing.Short() { |
| iters = 4 |
| } |
| t.Parallel() |
| for _, test := range tests { |
| test := test |
| t.Run(test, func(t *testing.T) { |
| t.Parallel() |
| var want []byte |
| tmp, err := ioutil.TempFile("", "") |
| if err != nil { |
| t.Fatalf("temp file creation failed: %v", err) |
| } |
| defer os.Remove(tmp.Name()) |
| defer tmp.Close() |
| for i := 0; i < iters; i++ { |
| // Note: use -c 2 to expose any nondeterminism which is the result |
| // of the runtime scheduler. |
| out, err := exec.Command(testenv.GoToolPath(t), "tool", "compile", "-c", "2", "-o", tmp.Name(), filepath.Join("testdata", "reproducible", test)).CombinedOutput() |
| if err != nil { |
| t.Fatalf("failed to compile: %v\n%s", err, out) |
| } |
| obj, err := ioutil.ReadFile(tmp.Name()) |
| if err != nil { |
| t.Fatalf("failed to read object file: %v", err) |
| } |
| if i == 0 { |
| want = obj |
| } else { |
| if !bytes.Equal(want, obj) { |
| t.Fatalf("builds produced different output after %d iters (%d bytes vs %d bytes)", i, len(want), len(obj)) |
| } |
| } |
| } |
| }) |
| } |
| } |
| |
| func TestIssue38068(t *testing.T) { |
| testenv.MustHaveGoBuild(t) |
| t.Parallel() |
| |
| // Compile a small package with and without the concurrent |
| // backend, then check to make sure that the resulting archives |
| // are identical. Note: this uses "go tool compile" instead of |
| // "go build" since the latter will generate differnent build IDs |
| // if it sees different command line flags. |
| scenarios := []struct { |
| tag string |
| args string |
| libpath string |
| }{ |
| {tag: "serial", args: "-c=1"}, |
| {tag: "concurrent", args: "-c=2"}} |
| |
| tmpdir, err := ioutil.TempDir("", "TestIssue38068") |
| if err != nil { |
| t.Fatal(err) |
| } |
| defer os.RemoveAll(tmpdir) |
| |
| src := filepath.Join("testdata", "reproducible", "issue38068.go") |
| for i := range scenarios { |
| s := &scenarios[i] |
| s.libpath = filepath.Join(tmpdir, s.tag+".a") |
| // Note: use of "-p" required in order for DWARF to be generated. |
| cmd := exec.Command(testenv.GoToolPath(t), "tool", "compile", "-trimpath", "-p=issue38068", "-buildid=", s.args, "-o", s.libpath, src) |
| out, err := cmd.CombinedOutput() |
| if err != nil { |
| t.Fatalf("%v: %v:\n%s", cmd.Args, err, out) |
| } |
| } |
| |
| readBytes := func(fn string) []byte { |
| payload, err := ioutil.ReadFile(fn) |
| if err != nil { |
| t.Fatalf("failed to read executable '%s': %v", fn, err) |
| } |
| return payload |
| } |
| |
| b1 := readBytes(scenarios[0].libpath) |
| b2 := readBytes(scenarios[1].libpath) |
| if !bytes.Equal(b1, b2) { |
| t.Fatalf("concurrent and serial builds produced different output") |
| } |
| } |