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