1 | // RUN: %clang_tsan -O1 %s -o %t |
2 | // RUN: %deflake %run %t | FileCheck %s --check-prefix=CHECK-NOZUPP |
3 | // RUN: %env_tsan_opts=suppressions='%s.supp':print_suppressions=1 %run %t 2>&1 | FileCheck %s --check-prefix=CHECK-SUPP |
4 | |
5 | #include "test.h" |
6 | |
7 | int *mem; |
8 | pthread_mutex_t mtx; |
9 | |
10 | void *Thread1(void *x) { |
11 | pthread_mutex_lock(mutex: &mtx); |
12 | free(ptr: mem); |
13 | pthread_mutex_unlock(mutex: &mtx); |
14 | barrier_wait(barrier: &barrier); |
15 | return NULL; |
16 | } |
17 | |
18 | __attribute__((noinline)) void *Thread2(void *x) { |
19 | barrier_wait(barrier: &barrier); |
20 | pthread_mutex_lock(mutex: &mtx); |
21 | mem[0] = 42; |
22 | pthread_mutex_unlock(mutex: &mtx); |
23 | return NULL; |
24 | } |
25 | |
26 | int main() { |
27 | barrier_init(barrier: &barrier, count: 2); |
28 | mem = (int*)malloc(size: 100); |
29 | pthread_mutex_init(mutex: &mtx, mutexattr: 0); |
30 | pthread_t t; |
31 | pthread_create(newthread: &t, NULL, start_routine: Thread1, NULL); |
32 | Thread2(x: 0); |
33 | pthread_join(th: t, NULL); |
34 | pthread_mutex_destroy(mutex: &mtx); |
35 | return 0; |
36 | } |
37 | |
38 | // CHECK-NOZUPP: WARNING: ThreadSanitizer: heap-use-after-free |
39 | // CHECK-NOZUPP: Write of size 4 at {{.*}} by main thread{{.*}}: |
40 | // CHECK-NOZUPP: #0 Thread2 |
41 | // CHECK-NOZUPP: #1 main |
42 | // CHECK-NOZUPP: Previous write of size 8 at {{.*}} by thread T1{{.*}}: |
43 | // CHECK-NOZUPP: #0 free |
44 | // CHECK-NOZUPP: #{{(1|2)}} Thread1 |
45 | // CHECK-NOZUPP: SUMMARY: ThreadSanitizer: heap-use-after-free{{.*}}Thread2 |
46 | // CHECK-SUPP: ThreadSanitizer: Matched 1 suppressions |
47 | // CHECK-SUPP: 1 race:^Thread2$ |
48 | |