1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
|
// 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.
//go:build linux || (freebsd && amd64)
package sanitizers_test
import (
"internal/testenv"
"os/exec"
"strings"
"testing"
)
func TestTSAN(t *testing.T) {
testenv.MustHaveGoBuild(t)
testenv.MustHaveCGO(t)
goos, err := goEnv("GOOS")
if err != nil {
t.Fatal(err)
}
goarch, err := goEnv("GOARCH")
if err != nil {
t.Fatal(err)
}
// The tsan tests require support for the -tsan option.
if !compilerRequiredTsanVersion(goos, goarch) {
t.Skipf("skipping on %s/%s; compiler version for -tsan option is too old.", goos, goarch)
}
if goarch == "riscv64" {
t.Skipf("skipping TestTSAN test on %s (see https://github.com/golang/go/issues/75553)", goarch)
}
t.Parallel()
requireOvercommit(t)
config := configure("thread")
config.skipIfCSanitizerBroken(t)
mustRun(t, config.goCmd("build", "std"))
cases := []struct {
src string
needsRuntime bool
}{
{src: "tsan.go"},
{src: "tsan2.go"},
{src: "tsan3.go"},
{src: "tsan4.go"},
{src: "tsan5.go", needsRuntime: true},
{src: "tsan6.go", needsRuntime: true},
{src: "tsan7.go", needsRuntime: true},
{src: "tsan8.go"},
{src: "tsan9.go"},
{src: "tsan10.go", needsRuntime: true},
{src: "tsan11.go", needsRuntime: true},
{src: "tsan12.go", needsRuntime: true},
{src: "tsan13.go", needsRuntime: true},
{src: "tsan14.go", needsRuntime: true},
{src: "tsan15.go", needsRuntime: true},
}
for _, tc := range cases {
tc := tc
name := strings.TrimSuffix(tc.src, ".go")
t.Run(name, func(t *testing.T) {
t.Parallel()
dir := newTempDir(t)
defer dir.RemoveAll(t)
outPath := dir.Join(name)
mustRun(t, config.goCmd("build", "-o", outPath, srcPath(tc.src)))
cmdArgs := []string{outPath}
if goos == "linux" {
// Disable ASLR for TSAN. See https://go.dev/issue/59418.
out, err := exec.Command("uname", "-m").Output()
if err != nil {
t.Fatalf("failed to run `uname -m`: %v", err)
}
arch := strings.TrimSpace(string(out))
if _, err := exec.Command("setarch", arch, "-R", "true").Output(); err != nil {
// Some systems don't have permission to run `setarch`.
// See https://go.dev/issue/70463.
t.Logf("failed to run `setarch %s -R true`: %v", arch, err)
} else {
cmdArgs = []string{"setarch", arch, "-R", outPath}
}
}
cmd := hangProneCmd(cmdArgs[0], cmdArgs[1:]...)
if tc.needsRuntime {
config.skipIfRuntimeIncompatible(t)
}
// If we don't see halt_on_error, the program
// will only exit non-zero if we call C.exit.
cmd.Env = append(cmd.Environ(), "TSAN_OPTIONS=halt_on_error=1")
mustRun(t, cmd)
})
}
}
|