| 1 | // RUN: %clangxx_tsan -O1 %s -DBUILD_SO -fPIC -shared -o %t-so.so |
| 2 | // RUN: %clangxx_tsan -O1 %s %link_libcxx_tsan -o %t && %run %t 2>&1 | FileCheck %s |
| 3 | |
| 4 | // dl_iterate_phdr doesn't exist on OS X. |
| 5 | // UNSUPPORTED: darwin |
| 6 | |
| 7 | #ifdef BUILD_SO |
| 8 | |
| 9 | #include "test.h" |
| 10 | |
| 11 | int exported_var = 0; |
| 12 | |
| 13 | #else // BUILD_SO |
| 14 | |
| 15 | #include "test.h" |
| 16 | #include <dlfcn.h> |
| 17 | #include <link.h> |
| 18 | #include <string.h> |
| 19 | #include <string> |
| 20 | |
| 21 | static int callback(struct dl_phdr_info *info, size_t size, void *data) { |
| 22 | if (info->dlpi_name[0] == '\0') |
| 23 | info->dlpi_name = "/proc/self/exe" ; |
| 24 | return !strcmp(s1: info->dlpi_name, s2: "non existent module" ); |
| 25 | } |
| 26 | |
| 27 | void *thread(void *unused) { |
| 28 | for (int i = 0; i < 1000; i++) { |
| 29 | barrier_wait(barrier: &barrier); |
| 30 | dl_iterate_phdr(callback: callback, data: 0); |
| 31 | } |
| 32 | return 0; |
| 33 | } |
| 34 | |
| 35 | int main(int argc, char *argv[]) { |
| 36 | barrier_init(barrier: &barrier, count: 2); |
| 37 | std::string path = std::string(argv[0]) + std::string("-so.so" ); |
| 38 | pthread_t th; |
| 39 | pthread_create(newthread: &th, attr: 0, start_routine: thread, arg: 0); |
| 40 | for (int i = 0; i < 1000; i++) { |
| 41 | barrier_wait(barrier: &barrier); |
| 42 | void *lib = dlopen(path.c_str(), RTLD_NOW); |
| 43 | if (!lib) { |
| 44 | printf(format: "error in dlopen: %s\n" , dlerror()); |
| 45 | return 1; |
| 46 | } |
| 47 | dlclose(handle: lib); |
| 48 | } |
| 49 | pthread_join(th: th, thread_return: 0); |
| 50 | fprintf(stderr, format: "DONE\n" ); |
| 51 | return 0; |
| 52 | } |
| 53 | |
| 54 | #endif // BUILD_SO |
| 55 | |
| 56 | // CHECK-NOT: WARNING: ThreadSanitizer: data race |
| 57 | // CHECK: DONE |
| 58 | |