diff options
Diffstat (limited to 'test/tsan/load_shared_lib.cc')
| -rw-r--r-- | test/tsan/load_shared_lib.cc | 66 | 
1 files changed, 66 insertions, 0 deletions
| diff --git a/test/tsan/load_shared_lib.cc b/test/tsan/load_shared_lib.cc new file mode 100644 index 000000000000..a27dc1cc6ffb --- /dev/null +++ b/test/tsan/load_shared_lib.cc @@ -0,0 +1,66 @@ +// 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 && %deflake %run %t | FileCheck %s + +#ifdef BUILD_SO + +#include <stddef.h> +#include <unistd.h> + +int GLOB_SHARED = 0; + +extern "C" +void *write_from_so(void *unused) { +  if (unused) +    sleep(1); +  GLOB_SHARED++; +  return NULL; +} + +#else  // BUILD_SO + +#include <dlfcn.h> +#include <pthread.h> +#include <stdio.h> +#include <stddef.h> +#include <unistd.h> + +#include <string> + +int GLOB = 0; + +void *write_glob(void *unused) { +  if (unused) +    sleep(1); +  GLOB++; +  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[]) { +  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 *(*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 | 
