xref: /aosp_15_r20/external/compiler-rt/test/tsan/vptr_benign_race.cc (revision 7c3d14c8b49c529e04be81a3ce6f5cc23712e4c6)
1*7c3d14c8STreehugger Robot // RUN: %clangxx_tsan -O1 %s -o %t && %run %t 2>&1 | FileCheck %s
2*7c3d14c8STreehugger Robot #include <pthread.h>
3*7c3d14c8STreehugger Robot #include <stdio.h>
4*7c3d14c8STreehugger Robot 
5*7c3d14c8STreehugger Robot struct A {
AA6*7c3d14c8STreehugger Robot   A() {
7*7c3d14c8STreehugger Robot     pthread_mutex_init(&m, 0);
8*7c3d14c8STreehugger Robot     pthread_cond_init(&c, 0);
9*7c3d14c8STreehugger Robot     signaled = false;
10*7c3d14c8STreehugger Robot   }
FA11*7c3d14c8STreehugger Robot   virtual void F() {
12*7c3d14c8STreehugger Robot   }
DoneA13*7c3d14c8STreehugger Robot   void Done() {
14*7c3d14c8STreehugger Robot     pthread_mutex_lock(&m);
15*7c3d14c8STreehugger Robot     signaled = true;
16*7c3d14c8STreehugger Robot     pthread_cond_signal(&c);
17*7c3d14c8STreehugger Robot     pthread_mutex_unlock(&m);
18*7c3d14c8STreehugger Robot   }
~AA19*7c3d14c8STreehugger Robot   virtual ~A() {
20*7c3d14c8STreehugger Robot   }
21*7c3d14c8STreehugger Robot   pthread_mutex_t m;
22*7c3d14c8STreehugger Robot   pthread_cond_t c;
23*7c3d14c8STreehugger Robot   bool signaled;
24*7c3d14c8STreehugger Robot };
25*7c3d14c8STreehugger Robot 
26*7c3d14c8STreehugger Robot struct B : A {
FB27*7c3d14c8STreehugger Robot   virtual void F() {
28*7c3d14c8STreehugger Robot   }
~BB29*7c3d14c8STreehugger Robot   virtual ~B() {
30*7c3d14c8STreehugger Robot     pthread_mutex_lock(&m);
31*7c3d14c8STreehugger Robot     while (!signaled)
32*7c3d14c8STreehugger Robot       pthread_cond_wait(&c, &m);
33*7c3d14c8STreehugger Robot     pthread_mutex_unlock(&m);
34*7c3d14c8STreehugger Robot   }
35*7c3d14c8STreehugger Robot };
36*7c3d14c8STreehugger Robot 
37*7c3d14c8STreehugger Robot static A *obj = new B;
38*7c3d14c8STreehugger Robot 
Thread1(void * x)39*7c3d14c8STreehugger Robot void *Thread1(void *x) {
40*7c3d14c8STreehugger Robot   obj->F();
41*7c3d14c8STreehugger Robot   obj->Done();
42*7c3d14c8STreehugger Robot   return NULL;
43*7c3d14c8STreehugger Robot }
44*7c3d14c8STreehugger Robot 
Thread2(void * x)45*7c3d14c8STreehugger Robot void *Thread2(void *x) {
46*7c3d14c8STreehugger Robot   delete obj;
47*7c3d14c8STreehugger Robot   return NULL;
48*7c3d14c8STreehugger Robot }
49*7c3d14c8STreehugger Robot 
main()50*7c3d14c8STreehugger Robot int main() {
51*7c3d14c8STreehugger Robot   pthread_t t[2];
52*7c3d14c8STreehugger Robot   pthread_create(&t[0], NULL, Thread1, NULL);
53*7c3d14c8STreehugger Robot   pthread_create(&t[1], NULL, Thread2, NULL);
54*7c3d14c8STreehugger Robot   pthread_join(t[0], NULL);
55*7c3d14c8STreehugger Robot   pthread_join(t[1], NULL);
56*7c3d14c8STreehugger Robot   fprintf(stderr, "PASS\n");
57*7c3d14c8STreehugger Robot }
58*7c3d14c8STreehugger Robot // CHECK: PASS
59*7c3d14c8STreehugger Robot // CHECK-NOT: WARNING: ThreadSanitizer: data race
60