/* RUN: %clang_msan -g -m64 %s -o %t RUN: %clang_msan -g -m64 %s -DBUILD_SO -fPIC -o %t-so.so -shared RUN: %run %t 2>&1 Regression test for a bug in msan/glibc integration, see https://sourceware.org/bugzilla/show_bug.cgi?id=16291 and https://code.google.com/p/memory-sanitizer/issues/detail?id=44 */ #ifndef BUILD_SO #include <assert.h> #include <dlfcn.h> #include <stdio.h> #include <stdlib.h> #include <pthread.h> typedef long *(* get_t)(); get_t GetTls; void *Thread1(void *unused) { long uninitialized; long *x = GetTls(); if (*x) fprintf(stderr, "bar\n"); *x = uninitialized; fprintf(stderr, "stack: %p dtls: %p\n", &x, x); return 0; } void *Thread2(void *unused) { long *x = GetTls(); fprintf(stderr, "stack: %p dtls: %p\n", &x, x); if (*x) fprintf(stderr, "foo\n"); // False negative here. return 0; } int main(int argc, char *argv[]) { char path[4096]; snprintf(path, sizeof(path), "%s-so.so", argv[0]); int i; void *handle = dlopen(path, RTLD_LAZY); if (!handle) fprintf(stderr, "%s\n", dlerror()); assert(handle != 0); GetTls = (get_t)dlsym(handle, "GetTls"); assert(dlerror() == 0); pthread_t t; pthread_create(&t, 0, Thread1, 0); pthread_join(t, 0); pthread_create(&t, 0, Thread2, 0); pthread_join(t, 0); return 0; } #else // BUILD_SO __thread long huge_thread_local_array[1 << 17]; long *GetTls() { return &huge_thread_local_array[0]; } #endif