t_tsan_vptr_race.sh revision 1.3
11.1Skamil# Copyright (c) 2018 The NetBSD Foundation, Inc. 21.1Skamil# All rights reserved. 31.1Skamil# 41.1Skamil# This code is derived from software contributed to The NetBSD Foundation 51.1Skamil# by Yang Zheng. 61.1Skamil# 71.1Skamil# Redistribution and use in source and binary forms, with or without 81.1Skamil# modification, are permitted provided that the following conditions 91.1Skamil# are met: 101.1Skamil# 1. Redistributions of source code must retain the above copyright 111.1Skamil# notice, this list of conditions and the following disclaimer. 121.1Skamil# 2. Redistributions in binary form must reproduce the above copyright 131.1Skamil# notice, this list of conditions and the following disclaimer in the 141.1Skamil# documentation and/or other materials provided with the distribution. 151.1Skamil# 161.1Skamil# THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 171.1Skamil# ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 181.1Skamil# TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 191.1Skamil# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 201.1Skamil# BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 211.1Skamil# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 221.1Skamil# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 231.1Skamil# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 241.1Skamil# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 251.1Skamil# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 261.1Skamil# POSSIBILITY OF SUCH DAMAGE. 271.1Skamil# 281.1Skamil 291.1Skamiltest_target() 301.1Skamil{ 311.1Skamil SUPPORT='n' 321.3Skamil # Detect address space larger than 32 bits 331.3Skamil maxaddress=`sysctl vm.maxaddress|awk '{print $3}'` 341.3Skamil if [ $maxaddress -gt 4294967295 ]; then 351.3Skamil if command -v cc >/dev/null 2>&1; then 361.3Skamil if ! echo __clang__ | cc -E - | grep -q __clang__; then 371.3Skamil SUPPORT='y' 381.3Skamil elif ! cc -v 2>&1 | awk '/gcc version/{print $3}' | \ 391.3Skamil awk -F '.' '($0+0) > 9 {exit 1}'; then 401.3Skamil SUPPORT='y' 411.3Skamil fi 421.1Skamil fi 431.1Skamil fi 441.1Skamil} 451.1Skamil 461.1Skamilatf_test_case vptr_race 471.1Skamilvptr_race_head() { 481.1Skamil atf_set "descr" "Test thread sanitizer for vptr race condition" 491.1Skamil atf_set "require.progs" "c++ paxctl" 501.1Skamil} 511.1Skamil 521.1Skamilatf_test_case vptr_race_profile 531.1Skamilvptr_race_profile_head() { 541.1Skamil atf_set "descr" "Test thread sanitizer for vptr race with profiling option" 551.1Skamil atf_set "require.progs" "c++ paxctl" 561.1Skamil} 571.1Skamilatf_test_case vptr_race_pic 581.1Skamilvptr_race_pic_head() { 591.1Skamil atf_set "descr" "Test thread sanitizer for vptr race with position independent code (PIC) flag" 601.1Skamil atf_set "require.progs" "c++ paxctl" 611.1Skamil} 621.1Skamilatf_test_case vptr_race_pie 631.1Skamilvptr_race_pie_head() { 641.1Skamil atf_set "descr" "Test thread sanitizer for vptr race with position independent execution (PIE) flag" 651.1Skamil atf_set "require.progs" "c++ paxctl" 661.1Skamil} 671.1Skamil 681.1Skamilvptr_race_body(){ 691.1Skamil cat > test.cc << EOF 701.1Skamil#include <pthread.h> 711.1Skamilpthread_barrier_t barrier; 721.1Skamilstruct A { 731.1Skamil volatile bool done; 741.1Skamil A(): done(false) { } 751.1Skamil virtual void Done() { done = true; } 761.1Skamil virtual ~A() { while (!done) ; } 771.1Skamil}; 781.1Skamilstruct B: A {}; 791.1SkamilA *obj = new B; 801.1Skamilvoid *Thread1(void *x) { 811.1Skamil pthread_barrier_wait(&barrier); 821.1Skamil obj->Done(); 831.1Skamil return NULL; 841.1Skamil} 851.1Skamilint main() { 861.1Skamil pthread_barrier_init(&barrier, NULL, 2); 871.1Skamil pthread_t t; 881.1Skamil pthread_create(&t, NULL, Thread1, NULL); 891.1Skamil pthread_barrier_wait(&barrier); 901.1Skamil delete obj; 911.1Skamil pthread_join(t, NULL); 921.1Skamil return 0; 931.1Skamil} 941.1SkamilEOF 951.1Skamil 961.1Skamil c++ -fsanitize=thread -o test test.cc 971.1Skamil paxctl +a test 981.1Skamil atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: data race on vptr \(ctor/dtor vs virtual call\)" ./test 991.1Skamil} 1001.1Skamil 1011.1Skamilvptr_race_profile_body(){ 1021.1Skamil cat > test.cc << EOF 1031.1Skamil#include <pthread.h> 1041.1Skamilpthread_barrier_t barrier; 1051.1Skamilstruct A { 1061.1Skamil volatile bool done; 1071.1Skamil A(): done(false) { } 1081.1Skamil virtual void Done() { done = true; } 1091.1Skamil virtual ~A() { while (!done) ; } 1101.1Skamil}; 1111.1Skamilstruct B: A {}; 1121.1SkamilA *obj = new B; 1131.1Skamilvoid *Thread1(void *x) { 1141.1Skamil pthread_barrier_wait(&barrier); 1151.1Skamil obj->Done(); 1161.1Skamil return NULL; 1171.1Skamil} 1181.1Skamilint main() { 1191.1Skamil pthread_barrier_init(&barrier, NULL, 2); 1201.1Skamil pthread_t t; 1211.1Skamil pthread_create(&t, NULL, Thread1, NULL); 1221.1Skamil pthread_barrier_wait(&barrier); 1231.1Skamil delete obj; 1241.1Skamil pthread_join(t, NULL); 1251.1Skamil return 0; 1261.1Skamil} 1271.1SkamilEOF 1281.1Skamil 1291.1Skamil c++ -fsanitize=thread -o test -pg test.cc 1301.1Skamil paxctl +a test 1311.1Skamil atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: data race on vptr \(ctor/dtor vs virtual call\)" ./test 1321.1Skamil} 1331.1Skamil 1341.1Skamilvptr_race_pic_body(){ 1351.1Skamil cat > test.cc << EOF 1361.1Skamil#include <stdio.h> 1371.1Skamil#include <stdlib.h> 1381.1Skamilint help(int); 1391.1Skamilint main(int argc, char **argv) {return help(argc);} 1401.1SkamilEOF 1411.1Skamil 1421.1Skamil cat > pic.cc << EOF 1431.1Skamil#include <pthread.h> 1441.1Skamilpthread_barrier_t barrier; 1451.1Skamilstruct A { 1461.1Skamil volatile bool done; 1471.1Skamil A(): done(false) { } 1481.1Skamil virtual void Done() { done = true; } 1491.1Skamil virtual ~A() { while (!done) ; } 1501.1Skamil}; 1511.1Skamilstruct B: A {}; 1521.1SkamilA *obj = new B; 1531.1Skamilvoid *Thread1(void *x) { 1541.1Skamil pthread_barrier_wait(&barrier); 1551.1Skamil obj->Done(); 1561.1Skamil return NULL; 1571.1Skamil} 1581.1Skamilint help(int argc) { 1591.1Skamil pthread_barrier_init(&barrier, NULL, 2); 1601.1Skamil pthread_t t; 1611.1Skamil pthread_create(&t, NULL, Thread1, NULL); 1621.1Skamil pthread_barrier_wait(&barrier); 1631.1Skamil delete obj; 1641.1Skamil pthread_join(t, NULL); 1651.1Skamil return 0; 1661.1Skamil} 1671.1SkamilEOF 1681.1Skamil 1691.1Skamil c++ -fsanitize=thread -fPIC -shared -o libtest.so pic.cc 1701.1Skamil c++ -o test test.cc -fsanitize=thread -L. -ltest 1711.1Skamil paxctl +a test 1721.1Skamil 1731.1Skamil export LD_LIBRARY_PATH=. 1741.1Skamil atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: data race on vptr \(ctor/dtor vs virtual call\)" ./test 1751.1Skamil} 1761.1Skamilvptr_race_pie_body(){ 1771.1Skamil 1781.1Skamil #check whether -pie flag is supported on this architecture 1791.1Skamil if ! c++ -pie -dM -E - < /dev/null 2>/dev/null >/dev/null; then 1801.1Skamil atf_set_skip "c++ -pie not supported on this architecture" 1811.1Skamil fi 1821.1Skamil cat > test.cc << EOF 1831.1Skamil#include <pthread.h> 1841.1Skamilpthread_barrier_t barrier; 1851.1Skamilstruct A { 1861.1Skamil volatile bool done; 1871.1Skamil A(): done(false) { } 1881.1Skamil virtual void Done() { done = true; } 1891.1Skamil virtual ~A() { while (!done) ; } 1901.1Skamil}; 1911.1Skamilstruct B: A {}; 1921.1SkamilA *obj = new B; 1931.1Skamilvoid *Thread1(void *x) { 1941.1Skamil pthread_barrier_wait(&barrier); 1951.1Skamil obj->Done(); 1961.1Skamil return NULL; 1971.1Skamil} 1981.1Skamilint main() { 1991.1Skamil pthread_barrier_init(&barrier, NULL, 2); 2001.1Skamil pthread_t t; 2011.1Skamil pthread_create(&t, NULL, Thread1, NULL); 2021.1Skamil pthread_barrier_wait(&barrier); 2031.1Skamil delete obj; 2041.1Skamil pthread_join(t, NULL); 2051.1Skamil return 0; 2061.1Skamil} 2071.1SkamilEOF 2081.1Skamil 2091.1Skamil c++ -fsanitize=thread -o test -fpie -pie test.cc 2101.1Skamil paxctl +a test 2111.1Skamil atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: data race on vptr \(ctor/dtor vs virtual call\)" ./test 2121.1Skamil} 2131.1Skamil 2141.1Skamil 2151.1Skamilatf_test_case target_not_supported 2161.1Skamiltarget_not_supported_head() 2171.1Skamil{ 2181.1Skamil atf_set "descr" "Test forced skip" 2191.1Skamil} 2201.1Skamil 2211.2Skamiltarget_not_supported_body() 2221.2Skamil{ 2231.2Skamil atf_skip "Target is not supported" 2241.2Skamil} 2251.2Skamil 2261.1Skamilatf_init_test_cases() 2271.1Skamil{ 2281.1Skamil test_target 2291.1Skamil test $SUPPORT = 'n' && { 2301.1Skamil atf_add_test_case target_not_supported 2311.1Skamil return 0 2321.1Skamil } 2331.1Skamil atf_add_test_case vptr_race 2341.1Skamil atf_add_test_case vptr_race_profile 2351.1Skamil atf_add_test_case vptr_race_pie 2361.1Skamil atf_add_test_case vptr_race_pic 2371.1Skamil} 238