forked from OSchip/llvm-project
75 lines
1.7 KiB
C++
75 lines
1.7 KiB
C++
// Check that if the list of shared libraries changes between the two race
|
|
// reports, the second report occurring in a new shared library is still
|
|
// symbolized correctly.
|
|
|
|
// RUN: %clangxx_tsan -O1 %s -DBUILD_SO -fPIC -shared -o %t-so.so
|
|
// RUN: %clangxx_tsan -O1 %s -o %t -rdynamic && %deflake %run %t | FileCheck %s
|
|
|
|
#ifdef BUILD_SO
|
|
|
|
#include "test.h"
|
|
|
|
int GLOB_SHARED = 0;
|
|
|
|
extern "C"
|
|
void init_so() {
|
|
barrier_init(&barrier, 2);
|
|
}
|
|
|
|
extern "C"
|
|
void *write_from_so(void *unused) {
|
|
if (unused == 0)
|
|
barrier_wait(&barrier);
|
|
GLOB_SHARED++;
|
|
if (unused != 0)
|
|
barrier_wait(&barrier);
|
|
return NULL;
|
|
}
|
|
|
|
#else // BUILD_SO
|
|
|
|
#include "test.h"
|
|
#include <dlfcn.h>
|
|
#include <string>
|
|
|
|
int GLOB = 0;
|
|
|
|
void *write_glob(void *unused) {
|
|
if (unused == 0)
|
|
barrier_wait(&barrier);
|
|
GLOB++;
|
|
if (unused != 0)
|
|
barrier_wait(&barrier);
|
|
return NULL;
|
|
}
|
|
|
|
void race_two_threads(void *(*access_callback)(void *unused)) {
|
|
pthread_t t1, t2;
|
|
pthread_create(&t1, NULL, access_callback, (void*)1);
|
|
pthread_create(&t2, NULL, access_callback, NULL);
|
|
pthread_join(t1, NULL);
|
|
pthread_join(t2, NULL);
|
|
}
|
|
|
|
int main(int argc, char *argv[]) {
|
|
barrier_init(&barrier, 2);
|
|
std::string path = std::string(argv[0]) + std::string("-so.so");
|
|
race_two_threads(write_glob);
|
|
// CHECK: write_glob
|
|
void *lib = dlopen(path.c_str(), RTLD_NOW);
|
|
if (!lib) {
|
|
printf("error in dlopen(): %s\n", dlerror());
|
|
return 1;
|
|
}
|
|
void (*init_so)();
|
|
*(void **)&init_so = dlsym(lib, "init_so");
|
|
init_so();
|
|
void *(*write_from_so)(void *unused);
|
|
*(void **)&write_from_so = dlsym(lib, "write_from_so");
|
|
race_two_threads(write_from_so);
|
|
// CHECK: write_from_so
|
|
return 0;
|
|
}
|
|
|
|
#endif // BUILD_SO
|