// 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") } }