xref: /llvm-project/compiler-rt/test/tsan/atomic_store.cpp (revision bcaeed49cb063de9fe504aa29e1cadff8a7be710)
1*bcaeed49SFangrui Song // RUN: %clangxx_tsan -O1 %s -o %t && %deflake %run %t 2>&1 | FileCheck %s
2*bcaeed49SFangrui Song #include "test.h"
3*bcaeed49SFangrui Song 
4*bcaeed49SFangrui Song long long Data;
5*bcaeed49SFangrui Song long long Sync;
6*bcaeed49SFangrui Song 
Thread1(void * x)7*bcaeed49SFangrui Song void *Thread1(void *x) {
8*bcaeed49SFangrui Song   Data++;
9*bcaeed49SFangrui Song   __atomic_store_n(&Sync, 1, __ATOMIC_RELEASE);
10*bcaeed49SFangrui Song   barrier_wait(&barrier);
11*bcaeed49SFangrui Song   barrier_wait(&barrier);
12*bcaeed49SFangrui Song   return NULL;
13*bcaeed49SFangrui Song }
14*bcaeed49SFangrui Song 
Thread2(void * x)15*bcaeed49SFangrui Song void *Thread2(void *x) {
16*bcaeed49SFangrui Song   barrier_wait(&barrier);
17*bcaeed49SFangrui Song   if (__atomic_load_n(&Sync, __ATOMIC_RELAXED) != 1)
18*bcaeed49SFangrui Song     exit(0);
19*bcaeed49SFangrui Song   // This store must terminate release sequence of the store in Thread1,
20*bcaeed49SFangrui Song   // thus tsan must detect race between Thread1 and main on Data.
21*bcaeed49SFangrui Song   __atomic_store_n(&Sync, 2, __ATOMIC_RELEASE);
22*bcaeed49SFangrui Song   barrier_wait(&barrier);
23*bcaeed49SFangrui Song   return NULL;
24*bcaeed49SFangrui Song }
25*bcaeed49SFangrui Song 
main()26*bcaeed49SFangrui Song int main() {
27*bcaeed49SFangrui Song   barrier_init(&barrier, 3);
28*bcaeed49SFangrui Song   pthread_t t[2];
29*bcaeed49SFangrui Song   pthread_create(&t[0], NULL, Thread1, NULL);
30*bcaeed49SFangrui Song   pthread_create(&t[1], NULL, Thread2, NULL);
31*bcaeed49SFangrui Song   barrier_wait(&barrier);
32*bcaeed49SFangrui Song   barrier_wait(&barrier);
33*bcaeed49SFangrui Song   if (__atomic_load_n(&Sync, __ATOMIC_ACQUIRE) != 2)
34*bcaeed49SFangrui Song     exit(0);
35*bcaeed49SFangrui Song   if (Data != 1)
36*bcaeed49SFangrui Song     exit(0);
37*bcaeed49SFangrui Song   pthread_join(t[0], NULL);
38*bcaeed49SFangrui Song   pthread_join(t[1], NULL);
39*bcaeed49SFangrui Song   fprintf(stderr, "DONE\n");
40*bcaeed49SFangrui Song   return 0;
41*bcaeed49SFangrui Song }
42*bcaeed49SFangrui Song 
43*bcaeed49SFangrui Song // CHECK: WARNING: ThreadSanitizer: data race
44*bcaeed49SFangrui Song // CHECK:   Read
45*bcaeed49SFangrui Song // CHECK:     #0 main
46*bcaeed49SFangrui Song // CHECK:   Previous write
47*bcaeed49SFangrui Song // CHECK:     #0 Thread1
48*bcaeed49SFangrui Song // CHECK:   Location is global 'Data'
49*bcaeed49SFangrui Song // CHECK: DONE
50