summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--lib/tsan/rtl/tsan_rtl.cc8
-rw-r--r--test/tsan/exceptions.cc185
-rw-r--r--test/tsan/test.h1
3 files changed, 194 insertions, 0 deletions
diff --git a/lib/tsan/rtl/tsan_rtl.cc b/lib/tsan/rtl/tsan_rtl.cc
index 9c23b51c9..b341c414e 100644
--- a/lib/tsan/rtl/tsan_rtl.cc
+++ b/lib/tsan/rtl/tsan_rtl.cc
@@ -1002,6 +1002,14 @@ void ThreadIgnoreEnd(ThreadState *thr, uptr pc) {
}
}
+#if !SANITIZER_GO
+extern "C" SANITIZER_INTERFACE_ATTRIBUTE
+uptr __tsan_testonly_shadow_stack_current_size() {
+ ThreadState *thr = cur_thread();
+ return thr->shadow_stack_pos - thr->shadow_stack;
+}
+#endif
+
void ThreadIgnoreSyncBegin(ThreadState *thr, uptr pc) {
DPrintf("#%d: ThreadIgnoreSyncBegin\n", thr->tid);
thr->ignore_sync++;
diff --git a/test/tsan/exceptions.cc b/test/tsan/exceptions.cc
new file mode 100644
index 000000000..193e115f7
--- /dev/null
+++ b/test/tsan/exceptions.cc
@@ -0,0 +1,185 @@
+// RUN: %clangxx_tsan -O0 %s -o %t && %run %t 2>&1 | FileCheck %s
+// RUN: %clangxx_tsan -O1 %s -o %t && %run %t 2>&1 | FileCheck %s
+
+#include "test.h"
+#include <setjmp.h>
+
+__attribute__((noinline)) void throws_int() {
+ throw 42;
+}
+
+__attribute__((noinline)) void callee_throws() {
+ try {
+ throws_int();
+ } catch (int) { // NOLINT
+ fprintf(stderr, "callee_throws caught exception\n");
+ }
+}
+
+__attribute__((noinline)) void throws_catches_rethrows() {
+ try {
+ throws_int();
+ } catch (int) { // NOLINT
+ fprintf(stderr, "throws_catches_rethrows caught exception\n");
+ throw;
+ }
+}
+
+__attribute__((noinline)) void callee_rethrows() {
+ try {
+ throws_catches_rethrows();
+ } catch (int) { // NOLINT
+ fprintf(stderr, "callee_rethrows caught exception\n");
+ }
+}
+
+__attribute__((noinline)) void throws_and_catches() {
+ try {
+ throws_int();
+ } catch (int) { // NOLINT
+ fprintf(stderr, "throws_and_catches caught exception\n");
+ }
+}
+
+__attribute__((noinline)) void nested_try() {
+ try {
+ try {
+ throws_int();
+ } catch (double) { // NOLINT
+ fprintf(stderr, "nested_try inner block caught exception\n");
+ }
+ } catch (int) { // NOLINT
+ fprintf(stderr, "nested_try outer block caught exception\n");
+ }
+}
+
+__attribute__((noinline)) void nested_try2() {
+ try {
+ try {
+ throws_int();
+ } catch (int) { // NOLINT
+ fprintf(stderr, "nested_try inner block caught exception\n");
+ }
+ } catch (double) { // NOLINT
+ fprintf(stderr, "nested_try outer block caught exception\n");
+ }
+}
+
+class ClassWithDestructor {
+ public:
+ ClassWithDestructor() {
+ fprintf(stderr, "ClassWithDestructor\n");
+ }
+ ~ClassWithDestructor() {
+ fprintf(stderr, "~ClassWithDestructor\n");
+ }
+};
+
+__attribute__((noinline)) void local_object_then_throw() {
+ ClassWithDestructor obj;
+ throws_int();
+}
+
+__attribute__((noinline)) void cpp_object_with_destructor() {
+ try {
+ local_object_then_throw();
+ } catch (int) { // NOLINT
+ fprintf(stderr, "cpp_object_with_destructor caught exception\n");
+ }
+}
+
+__attribute__((noinline)) void recursive_call(long n) {
+ if (n > 0) {
+ recursive_call(n - 1);
+ } else {
+ throws_int();
+ }
+}
+
+__attribute__((noinline)) void multiframe_unwind() {
+ try {
+ recursive_call(5);
+ } catch (int) { // NOLINT
+ fprintf(stderr, "multiframe_unwind caught exception\n");
+ }
+}
+
+__attribute__((noinline)) void longjmp_unwind() {
+ jmp_buf env;
+ int i = setjmp(env);
+ if (i != 0) {
+ fprintf(stderr, "longjmp_unwind jumped\n");
+ return;
+ }
+
+ try {
+ longjmp(env, 42);
+ } catch (int) { // NOLINT
+ fprintf(stderr, "longjmp_unwind caught exception\n");
+ }
+}
+
+__attribute__((noinline)) void recursive_call_longjmp(jmp_buf env, long n) {
+ if (n > 0) {
+ recursive_call_longjmp(env, n - 1);
+ } else {
+ longjmp(env, 42);
+ }
+}
+
+__attribute__((noinline)) void longjmp_unwind_multiple_frames() {
+ jmp_buf env;
+ int i = setjmp(env);
+ if (i != 0) {
+ fprintf(stderr, "longjmp_unwind_multiple_frames jumped\n");
+ return;
+ }
+
+ try {
+ recursive_call_longjmp(env, 5);
+ } catch (int) { // NOLINT
+ fprintf(stderr, "longjmp_unwind_multiple_frames caught exception\n");
+ }
+}
+
+#define CHECK_SHADOW_STACK(val) \
+ fprintf(stderr, (val == __tsan_testonly_shadow_stack_current_size() \
+ ? "OK.\n" \
+ : "Shadow stack leak!\n"));
+
+int main(int argc, const char * argv[]) {
+ fprintf(stderr, "Hello, World!\n");
+ unsigned long shadow_stack_size = __tsan_testonly_shadow_stack_current_size();
+
+ throws_and_catches();
+ CHECK_SHADOW_STACK(shadow_stack_size);
+
+ callee_throws();
+ CHECK_SHADOW_STACK(shadow_stack_size);
+
+ callee_rethrows();
+ CHECK_SHADOW_STACK(shadow_stack_size);
+
+ nested_try();
+ CHECK_SHADOW_STACK(shadow_stack_size);
+
+ nested_try2();
+ CHECK_SHADOW_STACK(shadow_stack_size);
+
+ cpp_object_with_destructor();
+ CHECK_SHADOW_STACK(shadow_stack_size);
+
+ multiframe_unwind();
+ CHECK_SHADOW_STACK(shadow_stack_size);
+
+ longjmp_unwind();
+ CHECK_SHADOW_STACK(shadow_stack_size);
+
+ longjmp_unwind_multiple_frames();
+ CHECK_SHADOW_STACK(shadow_stack_size);
+
+ return 0;
+}
+
+// CHECK: Hello, World!
+// CHECK-NOT: Shadow stack leak
diff --git a/test/tsan/test.h b/test/tsan/test.h
index ed300dbbe..6b981c09f 100644
--- a/test/tsan/test.h
+++ b/test/tsan/test.h
@@ -23,6 +23,7 @@ extern "C" {
void __tsan_testonly_barrier_init(invisible_barrier_t *barrier,
unsigned count);
void __tsan_testonly_barrier_wait(invisible_barrier_t *barrier);
+unsigned long __tsan_testonly_shadow_stack_current_size();
#ifdef __cplusplus
}
#endif