Home | History | Annotate | Line # | Download | only in c++
      1 # Copyright (c) 2018 The NetBSD Foundation, Inc.
      2 # All rights reserved.
      3 #
      4 # This code is derived from software contributed to The NetBSD Foundation
      5 # by Yang Zheng.
      6 #
      7 # Redistribution and use in source and binary forms, with or without
      8 # modification, are permitted provided that the following conditions
      9 # are met:
     10 # 1. Redistributions of source code must retain the above copyright
     11 #    notice, this list of conditions and the following disclaimer.
     12 # 2. Redistributions in binary form must reproduce the above copyright
     13 #    notice, this list of conditions and the following disclaimer in the
     14 #    documentation and/or other materials provided with the distribution.
     15 #
     16 # THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     17 # ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     18 # TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     19 # PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     20 # BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     21 # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     22 # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     23 # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     24 # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     25 # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     26 # POSSIBILITY OF SUCH DAMAGE.
     27 #
     28 
     29 
     30 test_target()
     31 {
     32 	atf_set "require.arch" "x86_64"
     33 }
     34 
     35 atf_test_case vptr_race
     36 vptr_race_head() {
     37 	atf_set "descr" "Test thread sanitizer for vptr race condition"
     38 	atf_set "require.progs" "c++ paxctl"
     39 	test_target
     40 }
     41 
     42 atf_test_case vptr_race_profile
     43 vptr_race_profile_head() {
     44 	atf_set "descr" "Test thread sanitizer for vptr race with profiling option"
     45 	atf_set "require.progs" "c++ paxctl"
     46 	test_target
     47 }
     48 atf_test_case vptr_race_pic
     49 vptr_race_pic_head() {
     50 	atf_set "descr" "Test thread sanitizer for vptr race with position independent code (PIC) flag"
     51 	atf_set "require.progs" "c++ paxctl"
     52 	test_target
     53 }
     54 atf_test_case vptr_race_pie
     55 vptr_race_pie_head() {
     56 	atf_set "descr" "Test thread sanitizer for vptr race with position independent execution (PIE) flag"
     57 	atf_set "require.progs" "c++ paxctl"
     58 	test_target
     59 }
     60 
     61 vptr_race_body(){
     62 	cat > test.cc << EOF
     63 #include <pthread.h>
     64 pthread_barrier_t barrier;
     65 struct A {
     66   volatile bool done;
     67   A(): done(false) { }
     68   virtual void Done() { done = true; }
     69   virtual ~A() { while (!done) ; }
     70 };
     71 struct B: A {};
     72 A *obj = new B;
     73 void *Thread1(void *x) {
     74   pthread_barrier_wait(&barrier);
     75   obj->Done();
     76   return NULL;
     77 }
     78 int main() {
     79   pthread_barrier_init(&barrier, NULL, 2);
     80   pthread_t t;
     81   pthread_create(&t, NULL, Thread1, NULL);
     82   pthread_barrier_wait(&barrier);
     83   delete obj;
     84   pthread_join(t, NULL);
     85   return 0;
     86 }
     87 EOF
     88 
     89 	c++ -fsanitize=thread -o test test.cc
     90 	paxctl +a test
     91 	atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: data race" ./test
     92 }
     93 
     94 vptr_race_profile_body(){
     95 	atf_expect_fail "PR toolchain/55760"
     96 	cat > test.cc << EOF
     97 #include <pthread.h>
     98 pthread_barrier_t barrier;
     99 struct A {
    100   volatile bool done;
    101   A(): done(false) { }
    102   virtual void Done() { done = true; }
    103   virtual ~A() { while (!done) ; }
    104 };
    105 struct B: A {};
    106 A *obj = new B;
    107 void *Thread1(void *x) {
    108   pthread_barrier_wait(&barrier);
    109   obj->Done();
    110   return NULL;
    111 }
    112 int main() {
    113   pthread_barrier_init(&barrier, NULL, 2);
    114   pthread_t t;
    115   pthread_create(&t, NULL, Thread1, NULL);
    116   pthread_barrier_wait(&barrier);
    117   delete obj;
    118   pthread_join(t, NULL);
    119   return 0;
    120 }
    121 EOF
    122 	c++ -fsanitize=thread -o test -pg test.cc
    123 	paxctl +a test
    124 	atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: data race" ./test
    125 }
    126 
    127 vptr_race_pic_body(){
    128 	cat > test.cc << EOF
    129 #include <stdio.h>
    130 #include <stdlib.h>
    131 int help(int);
    132 int main(int argc, char **argv) {return help(argc);}
    133 EOF
    134 
    135 	cat > pic.cc << EOF
    136 #include <pthread.h>
    137 pthread_barrier_t barrier;
    138 struct A {
    139   volatile bool done;
    140   A(): done(false) { }
    141   virtual void Done() { done = true; }
    142   virtual ~A() { while (!done) ; }
    143 };
    144 struct B: A {};
    145 A *obj = new B;
    146 void *Thread1(void *x) {
    147   pthread_barrier_wait(&barrier);
    148   obj->Done();
    149   return NULL;
    150 }
    151 int help(int argc) {
    152   pthread_barrier_init(&barrier, NULL, 2);
    153   pthread_t t;
    154   pthread_create(&t, NULL, Thread1, NULL);
    155   pthread_barrier_wait(&barrier);
    156   delete obj;
    157   pthread_join(t, NULL);
    158   return 0;
    159 }
    160 EOF
    161 
    162 	c++ -fsanitize=thread -fPIC -shared -o libtest.so pic.cc
    163 	c++ -o test test.cc -fsanitize=thread -L. -ltest
    164 	paxctl +a test
    165 
    166 	export LD_LIBRARY_PATH=.
    167 	atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: data race" ./test
    168 }
    169 vptr_race_pie_body(){
    170 
    171 	#check whether -pie flag is supported on this architecture
    172 	if ! c++ -pie -dM -E - < /dev/null 2>/dev/null >/dev/null; then
    173 		atf_set_skip "c++ -pie not supported on this architecture"
    174 	fi
    175 	cat > test.cc << EOF
    176 #include <pthread.h>
    177 pthread_barrier_t barrier;
    178 struct A {
    179   volatile bool done;
    180   A(): done(false) { }
    181   virtual void Done() { done = true; }
    182   virtual ~A() { while (!done) ; }
    183 };
    184 struct B: A {};
    185 A *obj = new B;
    186 void *Thread1(void *x) {
    187   pthread_barrier_wait(&barrier);
    188   obj->Done();
    189   return NULL;
    190 }
    191 int main() {
    192   pthread_barrier_init(&barrier, NULL, 2);
    193   pthread_t t;
    194   pthread_create(&t, NULL, Thread1, NULL);
    195   pthread_barrier_wait(&barrier);
    196   delete obj;
    197   pthread_join(t, NULL);
    198   return 0;
    199 }
    200 EOF
    201 
    202 	c++ -fsanitize=thread -o test -fpie -pie test.cc
    203 	paxctl +a test
    204 	atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: data race" ./test
    205 }
    206 
    207 
    208 atf_test_case target_not_supported
    209 target_not_supported_head()
    210 {
    211 	atf_set "descr" "Test forced skip"
    212 }
    213 
    214 target_not_supported_body()
    215 {
    216 	atf_skip "Target is not supported"
    217 }
    218 
    219 atf_init_test_cases()
    220 {
    221 	atf_add_test_case vptr_race
    222 	atf_add_test_case vptr_race_profile
    223 	atf_add_test_case vptr_race_pie
    224 	atf_add_test_case vptr_race_pic
    225 }
    226