2015-09-29 22:24:13 -06:00
|
|
|
#!/usr/bin/env bash
|
|
|
|
# Copyright 2015 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.
|
|
|
|
|
|
|
|
# This directory is intended to test the use of Go with sanitizers
|
|
|
|
# like msan, asan, etc. See https://github.com/google/sanitizers .
|
|
|
|
|
|
|
|
set -e
|
|
|
|
|
|
|
|
# The sanitizers were originally developed with clang, so prefer it.
|
|
|
|
CC=cc
|
2015-11-09 13:22:52 -07:00
|
|
|
if test -x "$(type -p clang)"; then
|
2015-09-29 22:24:13 -06:00
|
|
|
CC=clang
|
|
|
|
fi
|
|
|
|
export CC
|
|
|
|
|
2015-12-03 20:17:21 -07:00
|
|
|
msan=yes
|
|
|
|
|
2015-11-03 08:14:22 -07:00
|
|
|
TMPDIR=${TMPDIR:-/tmp}
|
2016-02-29 10:31:12 -07:00
|
|
|
echo 'int main() { return 0; }' > ${TMPDIR}/testsanitizers$$.c
|
2015-11-03 18:15:30 -07:00
|
|
|
if $CC -fsanitize=memory -c ${TMPDIR}/testsanitizers$$.c -o ${TMPDIR}/testsanitizers$$.o 2>&1 | grep "unrecognized" >& /dev/null; then
|
2015-12-03 20:17:21 -07:00
|
|
|
echo "skipping msan tests: -fsanitize=memory not supported"
|
|
|
|
msan=no
|
2015-09-29 22:24:13 -06:00
|
|
|
fi
|
2015-11-03 08:14:22 -07:00
|
|
|
rm -f ${TMPDIR}/testsanitizers$$.*
|
2015-10-02 08:04:34 -06:00
|
|
|
|
2016-02-29 11:05:46 -07:00
|
|
|
tsan=yes
|
|
|
|
|
|
|
|
# The memory and thread sanitizers in versions of clang before 3.6
|
|
|
|
# don't work with Go.
|
2015-12-03 20:17:21 -07:00
|
|
|
if test "$msan" = "yes" && $CC --version | grep clang >& /dev/null; then
|
2015-10-02 08:04:34 -06:00
|
|
|
ver=$($CC --version | sed -e 's/.* version \([0-9.-]*\).*/\1/')
|
|
|
|
major=$(echo $ver | sed -e 's/\([0-9]*\).*/\1/')
|
|
|
|
minor=$(echo $ver | sed -e 's/[0-9]*\.\([0-9]*\).*/\1/')
|
2015-11-30 12:50:21 -07:00
|
|
|
if test "$major" -lt 3 || test "$major" -eq 3 -a "$minor" -lt 6; then
|
2016-02-29 11:05:46 -07:00
|
|
|
echo "skipping msan/tsan tests: clang version $major.$minor (older than 3.6)"
|
2015-12-03 20:17:21 -07:00
|
|
|
msan=no
|
2016-02-29 11:05:46 -07:00
|
|
|
tsan=no
|
2015-10-02 08:04:34 -06:00
|
|
|
fi
|
2015-12-03 12:11:44 -07:00
|
|
|
|
2015-11-30 12:50:21 -07:00
|
|
|
# Clang before 3.8 does not work with Linux at or after 4.1.
|
|
|
|
# golang.org/issue/12898.
|
2015-12-03 20:17:21 -07:00
|
|
|
if test "$msan" = "yes" -a "$major" -lt 3 || test "$major" -eq 3 -a "$minor" -lt 8; then
|
2015-11-30 12:50:21 -07:00
|
|
|
if test "$(uname)" = Linux; then
|
|
|
|
linuxver=$(uname -r)
|
2015-12-03 12:11:44 -07:00
|
|
|
linuxmajor=$(echo $linuxver | sed -e 's/\([0-9]*\).*/\1/')
|
|
|
|
linuxminor=$(echo $linuxver | sed -e 's/[0-9]*\.\([0-9]*\).*/\1/')
|
2015-11-30 12:50:21 -07:00
|
|
|
if test "$linuxmajor" -gt 4 || test "$linuxmajor" -eq 4 -a "$linuxminor" -ge 1; then
|
2016-02-29 11:05:46 -07:00
|
|
|
echo "skipping msan/tsan tests: clang version $major.$minor (older than 3.8) incompatible with linux version $linuxmajor.$linuxminor (4.1 or newer)"
|
2015-12-03 20:17:21 -07:00
|
|
|
msan=no
|
2016-02-29 11:05:46 -07:00
|
|
|
tsan=no
|
2015-11-30 12:50:21 -07:00
|
|
|
fi
|
|
|
|
fi
|
|
|
|
fi
|
2015-10-02 08:04:34 -06:00
|
|
|
fi
|
|
|
|
|
2015-10-21 13:33:56 -06:00
|
|
|
status=0
|
|
|
|
|
2015-12-03 20:17:21 -07:00
|
|
|
if test "$msan" = "yes"; then
|
|
|
|
if ! go build -msan std; then
|
|
|
|
echo "FAIL: build -msan std"
|
|
|
|
status=1
|
|
|
|
fi
|
2015-10-26 14:58:23 -06:00
|
|
|
|
2015-12-03 20:17:21 -07:00
|
|
|
if ! go run -msan msan.go; then
|
|
|
|
echo "FAIL: msan"
|
|
|
|
status=1
|
|
|
|
fi
|
2015-10-21 13:33:56 -06:00
|
|
|
|
2015-12-03 20:17:21 -07:00
|
|
|
if ! CGO_LDFLAGS="-fsanitize=memory" CGO_CPPFLAGS="-fsanitize=memory" go run -msan -a msan2.go; then
|
|
|
|
echo "FAIL: msan2 with -fsanitize=memory"
|
|
|
|
status=1
|
|
|
|
fi
|
2016-01-05 15:06:58 -07:00
|
|
|
|
2015-12-03 20:17:21 -07:00
|
|
|
if ! go run -msan -a msan2.go; then
|
|
|
|
echo "FAIL: msan2"
|
|
|
|
status=1
|
|
|
|
fi
|
2015-10-21 13:33:56 -06:00
|
|
|
|
2015-12-03 20:17:21 -07:00
|
|
|
if ! go run -msan msan3.go; then
|
|
|
|
echo "FAIL: msan3"
|
|
|
|
status=1
|
|
|
|
fi
|
|
|
|
|
|
|
|
if ! go run -msan msan4.go; then
|
|
|
|
echo "FAIL: msan4"
|
|
|
|
status=1
|
|
|
|
fi
|
|
|
|
|
2016-07-06 16:30:33 -06:00
|
|
|
if ! go run -msan msan5.go; then
|
|
|
|
echo "FAIL: msan5"
|
|
|
|
status=1
|
|
|
|
fi
|
|
|
|
|
2015-12-03 20:17:21 -07:00
|
|
|
if go run -msan msan_fail.go 2>/dev/null; then
|
|
|
|
echo "FAIL: msan_fail"
|
|
|
|
status=1
|
|
|
|
fi
|
2015-11-03 18:15:30 -07:00
|
|
|
fi
|
|
|
|
|
2016-02-29 11:05:46 -07:00
|
|
|
if test "$tsan" = "yes"; then
|
|
|
|
echo 'int main() { return 0; }' > ${TMPDIR}/testsanitizers$$.c
|
|
|
|
ok=yes
|
|
|
|
if ! $CC -fsanitize=thread ${TMPDIR}/testsanitizers$$.c -o ${TMPDIR}/testsanitizers$$ &> ${TMPDIR}/testsanitizers$$.err; then
|
|
|
|
ok=no
|
|
|
|
fi
|
|
|
|
if grep "unrecognized" ${TMPDIR}/testsanitizers$$.err >& /dev/null; then
|
|
|
|
echo "skipping tsan tests: -fsanitize=thread not supported"
|
|
|
|
tsan=no
|
|
|
|
elif test "$ok" != "yes"; then
|
|
|
|
cat ${TMPDIR}/testsanitizers$$.err
|
|
|
|
echo "skipping tsan tests: -fsanitizer=thread build failed"
|
|
|
|
tsan=no
|
|
|
|
fi
|
|
|
|
rm -f ${TMPDIR}/testsanitizers$$*
|
2015-11-04 13:30:30 -07:00
|
|
|
fi
|
2015-12-03 20:17:21 -07:00
|
|
|
|
2016-06-03 11:49:24 -06:00
|
|
|
# Run a TSAN test.
|
|
|
|
# $1 test name
|
|
|
|
# $2 environment variables
|
|
|
|
# $3 go run args
|
|
|
|
testtsan() {
|
2015-12-03 20:17:21 -07:00
|
|
|
err=${TMPDIR}/tsanerr$$.out
|
2016-06-03 11:49:24 -06:00
|
|
|
if ! env $2 go run $3 $1 2>$err; then
|
2016-05-19 11:07:41 -06:00
|
|
|
cat $err
|
2016-06-03 11:49:24 -06:00
|
|
|
echo "FAIL: $1"
|
2016-05-19 11:07:41 -06:00
|
|
|
status=1
|
|
|
|
elif grep -i warning $err >/dev/null 2>&1; then
|
|
|
|
cat $err
|
2016-06-03 11:49:24 -06:00
|
|
|
echo "FAIL: $1"
|
2016-05-19 11:07:41 -06:00
|
|
|
status=1
|
|
|
|
fi
|
2016-06-03 11:49:24 -06:00
|
|
|
rm -f $err
|
|
|
|
}
|
2016-05-19 11:07:41 -06:00
|
|
|
|
2016-06-03 11:49:24 -06:00
|
|
|
if test "$tsan" = "yes"; then
|
|
|
|
testtsan tsan.go
|
|
|
|
testtsan tsan2.go
|
|
|
|
testtsan tsan3.go
|
|
|
|
testtsan tsan4.go
|
2016-05-19 17:27:23 -06:00
|
|
|
|
2016-06-06 22:44:24 -06:00
|
|
|
# These tests are only reliable using clang or GCC version 7 or later.
|
|
|
|
# Otherwise runtime/cgo/libcgo.h can't tell whether TSAN is in use.
|
|
|
|
ok=false
|
|
|
|
if ${CC} --version | grep clang >/dev/null 2>&1; then
|
|
|
|
ok=true
|
|
|
|
else
|
|
|
|
ver=$($CC -dumpversion)
|
|
|
|
major=$(echo $ver | sed -e 's/\([0-9]*\).*/\1/')
|
|
|
|
if test "$major" -lt 7; then
|
|
|
|
echo "skipping remaining TSAN tests: GCC version $major (older than 7)"
|
|
|
|
else
|
|
|
|
ok=true
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
|
|
|
|
if test "$ok" = "true"; then
|
|
|
|
# This test requires rebuilding os/user with -fsanitize=thread.
|
|
|
|
testtsan tsan5.go "CGO_CFLAGS=-fsanitize=thread CGO_LDFLAGS=-fsanitize=thread" "-installsuffix=tsan"
|
2016-05-26 18:47:03 -06:00
|
|
|
|
2016-06-06 22:44:24 -06:00
|
|
|
# This test requires rebuilding runtime/cgo with -fsanitize=thread.
|
|
|
|
testtsan tsan6.go "CGO_CFLAGS=-fsanitize=thread CGO_LDFLAGS=-fsanitize=thread" "-installsuffix=tsan"
|
2016-08-02 12:42:53 -06:00
|
|
|
|
|
|
|
# This test requires rebuilding runtime/cgo with -fsanitize=thread.
|
|
|
|
testtsan tsan7.go "CGO_CFLAGS=-fsanitize=thread CGO_LDFLAGS=-fsanitize=thread" "-installsuffix=tsan"
|
2016-06-06 22:44:24 -06:00
|
|
|
fi
|
2015-10-21 13:33:56 -06:00
|
|
|
fi
|
|
|
|
|
|
|
|
exit $status
|