diff options
author | Dmitry Vyukov <dvyukov@google.com> | 2014-06-27 00:47:38 +0000 |
---|---|---|
committer | Dmitry Vyukov <dvyukov@google.com> | 2014-06-27 00:47:38 +0000 |
commit | fb251f51a9118a64376405a277ee6777abd7dca4 (patch) | |
tree | 5cf056ea8d02d032037086dec7025e6e79ab56ce /compiler-rt | |
parent | 73f72e15ac73a1a84bb5e36b5434815cc5273006 (diff) | |
download | bcm5719-llvm-fb251f51a9118a64376405a277ee6777abd7dca4.tar.gz bcm5719-llvm-fb251f51a9118a64376405a277ee6777abd7dca4.zip |
tsan: add __tsan_java_finalize interface function
It is required to prevent false positives between object ctor and finalizer,
as otherwise they look completely unsynchronized.
llvm-svn: 211829
Diffstat (limited to 'compiler-rt')
-rw-r--r-- | compiler-rt/lib/tsan/rtl/tsan_interface_java.cc | 6 | ||||
-rw-r--r-- | compiler-rt/lib/tsan/rtl/tsan_interface_java.h | 5 | ||||
-rw-r--r-- | compiler-rt/test/tsan/java.h | 1 | ||||
-rw-r--r-- | compiler-rt/test/tsan/java_finalizer.cc | 27 |
4 files changed, 39 insertions, 0 deletions
diff --git a/compiler-rt/lib/tsan/rtl/tsan_interface_java.cc b/compiler-rt/lib/tsan/rtl/tsan_interface_java.cc index ee610188292..e63b93f4139 100644 --- a/compiler-rt/lib/tsan/rtl/tsan_interface_java.cc +++ b/compiler-rt/lib/tsan/rtl/tsan_interface_java.cc @@ -142,6 +142,12 @@ void __tsan_java_move(jptr src, jptr dst, jptr size) { } } +void __tsan_java_finalize() { + SCOPED_JAVA_FUNC(__tsan_java_finalize); + DPrintf("#%d: java_mutex_finalize()\n", thr->tid); + AcquireGlobal(thr, 0); +} + void __tsan_java_mutex_lock(jptr addr) { SCOPED_JAVA_FUNC(__tsan_java_mutex_lock); DPrintf("#%d: java_mutex_lock(%p)\n", thr->tid, addr); diff --git a/compiler-rt/lib/tsan/rtl/tsan_interface_java.h b/compiler-rt/lib/tsan/rtl/tsan_interface_java.h index 9ac78e074bb..6a838851e43 100644 --- a/compiler-rt/lib/tsan/rtl/tsan_interface_java.h +++ b/compiler-rt/lib/tsan/rtl/tsan_interface_java.h @@ -52,6 +52,11 @@ void __tsan_java_free(jptr ptr, jptr size) INTERFACE_ATTRIBUTE; // Can be aggregated for several objects (preferably). // The ranges must not overlap. void __tsan_java_move(jptr src, jptr dst, jptr size) INTERFACE_ATTRIBUTE; +// This function must be called on the finalizer thread +// before executing a batch of finalizers. +// It ensures necessary synchronization between +// java object creation and finalization. +void __tsan_java_finalize() INTERFACE_ATTRIBUTE; // Mutex lock. // Addr is any unique address associated with the mutex. diff --git a/compiler-rt/test/tsan/java.h b/compiler-rt/test/tsan/java.h index 7aa0bca32ce..d986d08be06 100644 --- a/compiler-rt/test/tsan/java.h +++ b/compiler-rt/test/tsan/java.h @@ -11,6 +11,7 @@ int __tsan_java_fini(); void __tsan_java_alloc(jptr ptr, jptr size); void __tsan_java_free(jptr ptr, jptr size); void __tsan_java_move(jptr src, jptr dst, jptr size); +void __tsan_java_finalize(); void __tsan_java_mutex_lock(jptr addr); void __tsan_java_mutex_unlock(jptr addr); void __tsan_java_mutex_read_lock(jptr addr); diff --git a/compiler-rt/test/tsan/java_finalizer.cc b/compiler-rt/test/tsan/java_finalizer.cc new file mode 100644 index 00000000000..706b946be61 --- /dev/null +++ b/compiler-rt/test/tsan/java_finalizer.cc @@ -0,0 +1,27 @@ +// RUN: %clangxx_tsan -O1 %s -o %t && %run %t | FileCheck %s +#include "java.h" + +void *Thread(void *p) { + sleep(1); + __tsan_java_finalize(); + *(int*)p = 42; + return 0; +} + +int main() { + int const kHeapSize = 1024 * 1024; + void *jheap = (char*)malloc(kHeapSize + 8) + 8; + __tsan_java_init((jptr)jheap, kHeapSize); + const int kBlockSize = 16; + __tsan_java_alloc((jptr)jheap, kBlockSize); + pthread_t th; + pthread_create(&th, 0, Thread, jheap); + *(int*)jheap = 43; + pthread_join(th, 0); + __tsan_java_free((jptr)jheap, kBlockSize); + fprintf(stderr, "DONE\n"); + return __tsan_java_fini(); +} + +// CHECK-NOT: WARNING: ThreadSanitizer: data race +// CHECK: DONE |