xref: /llvm-project/compiler-rt/test/tsan/load_shared_lib.cpp (revision bcaeed49cb063de9fe504aa29e1cadff8a7be710)
1*bcaeed49SFangrui Song // Check that if the list of shared libraries changes between the two race
2*bcaeed49SFangrui Song // reports, the second report occurring in a new shared library is still
3*bcaeed49SFangrui Song // symbolized correctly.
4*bcaeed49SFangrui Song 
5*bcaeed49SFangrui Song // RUN: %clangxx_tsan -O1 %s -DBUILD_SO -fPIC -shared -o %t-so.so
6*bcaeed49SFangrui Song // RUN: %clangxx_tsan -O1 %s %link_libcxx_tsan -o %t -rdynamic && %deflake %run %t | FileCheck %s
7*bcaeed49SFangrui Song 
8*bcaeed49SFangrui Song #ifdef BUILD_SO
9*bcaeed49SFangrui Song 
10*bcaeed49SFangrui Song #include "test.h"
11*bcaeed49SFangrui Song 
12*bcaeed49SFangrui Song int GLOB_SHARED = 0;
13*bcaeed49SFangrui Song 
14*bcaeed49SFangrui Song extern "C"
init_so()15*bcaeed49SFangrui Song void init_so() {
16*bcaeed49SFangrui Song   barrier_init(&barrier, 2);
17*bcaeed49SFangrui Song }
18*bcaeed49SFangrui Song 
19*bcaeed49SFangrui Song extern "C"
write_from_so(void * unused)20*bcaeed49SFangrui Song void *write_from_so(void *unused) {
21*bcaeed49SFangrui Song   if (unused == 0)
22*bcaeed49SFangrui Song     barrier_wait(&barrier);
23*bcaeed49SFangrui Song   GLOB_SHARED++;
24*bcaeed49SFangrui Song   if (unused != 0)
25*bcaeed49SFangrui Song     barrier_wait(&barrier);
26*bcaeed49SFangrui Song   return NULL;
27*bcaeed49SFangrui Song }
28*bcaeed49SFangrui Song 
29*bcaeed49SFangrui Song #else  // BUILD_SO
30*bcaeed49SFangrui Song 
31*bcaeed49SFangrui Song #include "test.h"
32*bcaeed49SFangrui Song #include <dlfcn.h>
33*bcaeed49SFangrui Song #include <string>
34*bcaeed49SFangrui Song 
35*bcaeed49SFangrui Song int GLOB = 0;
36*bcaeed49SFangrui Song 
write_glob(void * unused)37*bcaeed49SFangrui Song void *write_glob(void *unused) {
38*bcaeed49SFangrui Song   if (unused == 0)
39*bcaeed49SFangrui Song     barrier_wait(&barrier);
40*bcaeed49SFangrui Song   GLOB++;
41*bcaeed49SFangrui Song   if (unused != 0)
42*bcaeed49SFangrui Song     barrier_wait(&barrier);
43*bcaeed49SFangrui Song   return NULL;
44*bcaeed49SFangrui Song }
45*bcaeed49SFangrui Song 
race_two_threads(void * (* access_callback)(void * unused))46*bcaeed49SFangrui Song void race_two_threads(void *(*access_callback)(void *unused)) {
47*bcaeed49SFangrui Song   pthread_t t1, t2;
48*bcaeed49SFangrui Song   pthread_create(&t1, NULL, access_callback, (void*)1);
49*bcaeed49SFangrui Song   pthread_create(&t2, NULL, access_callback, NULL);
50*bcaeed49SFangrui Song   pthread_join(t1, NULL);
51*bcaeed49SFangrui Song   pthread_join(t2, NULL);
52*bcaeed49SFangrui Song }
53*bcaeed49SFangrui Song 
main(int argc,char * argv[])54*bcaeed49SFangrui Song int main(int argc, char *argv[]) {
55*bcaeed49SFangrui Song   barrier_init(&barrier, 2);
56*bcaeed49SFangrui Song   std::string path = std::string(argv[0]) + std::string("-so.so");
57*bcaeed49SFangrui Song   race_two_threads(write_glob);
58*bcaeed49SFangrui Song   // CHECK: write_glob
59*bcaeed49SFangrui Song   void *lib = dlopen(path.c_str(), RTLD_NOW);
60*bcaeed49SFangrui Song     if (!lib) {
61*bcaeed49SFangrui Song     printf("error in dlopen(): %s\n", dlerror());
62*bcaeed49SFangrui Song     return 1;
63*bcaeed49SFangrui Song   }
64*bcaeed49SFangrui Song   void (*init_so)();
65*bcaeed49SFangrui Song   *(void **)&init_so = dlsym(lib, "init_so");
66*bcaeed49SFangrui Song   init_so();
67*bcaeed49SFangrui Song   void *(*write_from_so)(void *unused);
68*bcaeed49SFangrui Song   *(void **)&write_from_so = dlsym(lib, "write_from_so");
69*bcaeed49SFangrui Song   race_two_threads(write_from_so);
70*bcaeed49SFangrui Song   // CHECK: write_from_so
71*bcaeed49SFangrui Song   return 0;
72*bcaeed49SFangrui Song }
73*bcaeed49SFangrui Song 
74*bcaeed49SFangrui Song #endif  // BUILD_SO
75