mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
New test case for issue 38068, which deals with build reproducibility: do a pair of compilations, the first with the concurrent back end turned on, and the second with -c=1, then check to make sure we get the same output (using a test case that triggers late inlining into wrapper methods). Updates #38068. Change-Id: I4afaf78898706a66985f09d18f6f6f29876c9017 Reviewed-on: https://go-review.googlesource.com/c/go/+/234417 Reviewed-by: Keith Randall <khr@golang.org> Reviewed-by: Cherry Zhang <cherryyz@google.com> Run-TryBot: Keith Randall <khr@golang.org> TryBot-Result: Gobot Gobot <gobot@golang.org>
112 lines
2.9 KiB
Go
112 lines
2.9 KiB
Go
// 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")
|
|
}
|
|
}
|