xref: /llvm-project/compiler-rt/test/tsan/exceptions.cpp (revision c0fa632236308100e1031fc86edfde13ddc4bcef)
1bcaeed49SFangrui Song // RUN: %clangxx_tsan -O0 %s -o %t && %run %t 2>&1 | FileCheck %s
2bcaeed49SFangrui Song // RUN: %clangxx_tsan -O1 %s -o %t && %run %t 2>&1 | FileCheck %s
3bcaeed49SFangrui Song 
4bcaeed49SFangrui Song #include "test.h"
5bcaeed49SFangrui Song #include <setjmp.h>
6bcaeed49SFangrui Song 
throws_int()7bcaeed49SFangrui Song __attribute__((noinline)) void throws_int() {
8bcaeed49SFangrui Song   throw 42;
9bcaeed49SFangrui Song }
10bcaeed49SFangrui Song 
callee_throws()11bcaeed49SFangrui Song __attribute__((noinline)) void callee_throws() {
12bcaeed49SFangrui Song   try {
13bcaeed49SFangrui Song     throws_int();
14*c0fa6322SVitaly Buka   } catch (int) {
15bcaeed49SFangrui Song     fprintf(stderr, "callee_throws caught exception\n");
16bcaeed49SFangrui Song   }
17bcaeed49SFangrui Song }
18bcaeed49SFangrui Song 
throws_catches_rethrows()19bcaeed49SFangrui Song __attribute__((noinline)) void throws_catches_rethrows() {
20bcaeed49SFangrui Song   try {
21bcaeed49SFangrui Song     throws_int();
22*c0fa6322SVitaly Buka   } catch (int) {
23bcaeed49SFangrui Song     fprintf(stderr, "throws_catches_rethrows caught exception\n");
24bcaeed49SFangrui Song     throw;
25bcaeed49SFangrui Song   }
26bcaeed49SFangrui Song }
27bcaeed49SFangrui Song 
callee_rethrows()28bcaeed49SFangrui Song __attribute__((noinline)) void callee_rethrows() {
29bcaeed49SFangrui Song   try {
30bcaeed49SFangrui Song     throws_catches_rethrows();
31*c0fa6322SVitaly Buka   } catch (int) {
32bcaeed49SFangrui Song     fprintf(stderr, "callee_rethrows caught exception\n");
33bcaeed49SFangrui Song   }
34bcaeed49SFangrui Song }
35bcaeed49SFangrui Song 
throws_and_catches()36bcaeed49SFangrui Song __attribute__((noinline)) void throws_and_catches() {
37bcaeed49SFangrui Song   try {
38bcaeed49SFangrui Song     throws_int();
39*c0fa6322SVitaly Buka   } catch (int) {
40bcaeed49SFangrui Song     fprintf(stderr, "throws_and_catches caught exception\n");
41bcaeed49SFangrui Song   }
42bcaeed49SFangrui Song }
43bcaeed49SFangrui Song 
nested_try()44bcaeed49SFangrui Song __attribute__((noinline)) void nested_try() {
45bcaeed49SFangrui Song   try {
46bcaeed49SFangrui Song     try {
47bcaeed49SFangrui Song       throws_int();
48*c0fa6322SVitaly Buka     } catch (double) {
49bcaeed49SFangrui Song       fprintf(stderr, "nested_try inner block caught exception\n");
50bcaeed49SFangrui Song     }
51*c0fa6322SVitaly Buka   } catch (int) {
52bcaeed49SFangrui Song     fprintf(stderr, "nested_try outer block caught exception\n");
53bcaeed49SFangrui Song   }
54bcaeed49SFangrui Song }
55bcaeed49SFangrui Song 
nested_try2()56bcaeed49SFangrui Song __attribute__((noinline)) void nested_try2() {
57bcaeed49SFangrui Song   try {
58bcaeed49SFangrui Song     try {
59bcaeed49SFangrui Song       throws_int();
60*c0fa6322SVitaly Buka     } catch (int) {
61bcaeed49SFangrui Song       fprintf(stderr, "nested_try inner block caught exception\n");
62bcaeed49SFangrui Song     }
63*c0fa6322SVitaly Buka   } catch (double) {
64bcaeed49SFangrui Song     fprintf(stderr, "nested_try outer block caught exception\n");
65bcaeed49SFangrui Song   }
66bcaeed49SFangrui Song }
67bcaeed49SFangrui Song 
68bcaeed49SFangrui Song class ClassWithDestructor {
69bcaeed49SFangrui Song  public:
ClassWithDestructor()70bcaeed49SFangrui Song   ClassWithDestructor() {
71bcaeed49SFangrui Song     fprintf(stderr, "ClassWithDestructor\n");
72bcaeed49SFangrui Song   }
~ClassWithDestructor()73bcaeed49SFangrui Song   ~ClassWithDestructor() {
74bcaeed49SFangrui Song     fprintf(stderr, "~ClassWithDestructor\n");
75bcaeed49SFangrui Song   }
76bcaeed49SFangrui Song };
77bcaeed49SFangrui Song 
local_object_then_throw()78bcaeed49SFangrui Song __attribute__((noinline)) void local_object_then_throw() {
79bcaeed49SFangrui Song   ClassWithDestructor obj;
80bcaeed49SFangrui Song   throws_int();
81bcaeed49SFangrui Song }
82bcaeed49SFangrui Song 
cpp_object_with_destructor()83bcaeed49SFangrui Song __attribute__((noinline)) void cpp_object_with_destructor() {
84bcaeed49SFangrui Song   try {
85bcaeed49SFangrui Song     local_object_then_throw();
86*c0fa6322SVitaly Buka   } catch (int) {
87bcaeed49SFangrui Song     fprintf(stderr, "cpp_object_with_destructor caught exception\n");
88bcaeed49SFangrui Song   }
89bcaeed49SFangrui Song }
90bcaeed49SFangrui Song 
recursive_call(long n)91bcaeed49SFangrui Song __attribute__((noinline)) void recursive_call(long n) {
92bcaeed49SFangrui Song   if (n > 0) {
93bcaeed49SFangrui Song     recursive_call(n - 1);
94bcaeed49SFangrui Song   } else {
95bcaeed49SFangrui Song     throws_int();
96bcaeed49SFangrui Song   }
97bcaeed49SFangrui Song }
98bcaeed49SFangrui Song 
multiframe_unwind()99bcaeed49SFangrui Song __attribute__((noinline)) void multiframe_unwind() {
100bcaeed49SFangrui Song   try {
101bcaeed49SFangrui Song     recursive_call(5);
102*c0fa6322SVitaly Buka   } catch (int) {
103bcaeed49SFangrui Song     fprintf(stderr, "multiframe_unwind caught exception\n");
104bcaeed49SFangrui Song   }
105bcaeed49SFangrui Song }
106bcaeed49SFangrui Song 
longjmp_unwind()107bcaeed49SFangrui Song __attribute__((noinline)) void longjmp_unwind() {
108bcaeed49SFangrui Song   jmp_buf env;
109bcaeed49SFangrui Song   int i = setjmp(env);
110bcaeed49SFangrui Song   if (i != 0) {
111bcaeed49SFangrui Song     fprintf(stderr, "longjmp_unwind jumped\n");
112bcaeed49SFangrui Song     return;
113bcaeed49SFangrui Song   }
114bcaeed49SFangrui Song 
115bcaeed49SFangrui Song   try {
116bcaeed49SFangrui Song     longjmp(env, 42);
117*c0fa6322SVitaly Buka   } catch (int) {
118bcaeed49SFangrui Song     fprintf(stderr, "longjmp_unwind caught exception\n");
119bcaeed49SFangrui Song   }
120bcaeed49SFangrui Song }
121bcaeed49SFangrui Song 
recursive_call_longjmp(jmp_buf env,long n)122bcaeed49SFangrui Song __attribute__((noinline)) void recursive_call_longjmp(jmp_buf env, long n) {
123bcaeed49SFangrui Song   if (n > 0) {
124bcaeed49SFangrui Song     recursive_call_longjmp(env, n - 1);
125bcaeed49SFangrui Song   } else {
126bcaeed49SFangrui Song     longjmp(env, 42);
127bcaeed49SFangrui Song   }
128bcaeed49SFangrui Song }
129bcaeed49SFangrui Song 
longjmp_unwind_multiple_frames()130bcaeed49SFangrui Song __attribute__((noinline)) void longjmp_unwind_multiple_frames() {
131bcaeed49SFangrui Song   jmp_buf env;
132bcaeed49SFangrui Song   int i = setjmp(env);
133bcaeed49SFangrui Song   if (i != 0) {
134bcaeed49SFangrui Song     fprintf(stderr, "longjmp_unwind_multiple_frames jumped\n");
135bcaeed49SFangrui Song     return;
136bcaeed49SFangrui Song   }
137bcaeed49SFangrui Song 
138bcaeed49SFangrui Song   try {
139bcaeed49SFangrui Song     recursive_call_longjmp(env, 5);
140*c0fa6322SVitaly Buka   } catch (int) {
141bcaeed49SFangrui Song     fprintf(stderr, "longjmp_unwind_multiple_frames caught exception\n");
142bcaeed49SFangrui Song   }
143bcaeed49SFangrui Song }
144bcaeed49SFangrui Song 
145bcaeed49SFangrui Song #define CHECK_SHADOW_STACK(val)                                                \
146bcaeed49SFangrui Song   fprintf(stderr, (val == __tsan_testonly_shadow_stack_current_size()          \
147bcaeed49SFangrui Song                        ? "OK.\n"                                               \
148bcaeed49SFangrui Song                        : "Shadow stack leak!\n"));
149bcaeed49SFangrui Song 
main(int argc,const char * argv[])150bcaeed49SFangrui Song int main(int argc, const char * argv[]) {
151bcaeed49SFangrui Song   fprintf(stderr, "Hello, World!\n");
152bcaeed49SFangrui Song   unsigned long shadow_stack_size = __tsan_testonly_shadow_stack_current_size();
153bcaeed49SFangrui Song 
154bcaeed49SFangrui Song   throws_and_catches();
155bcaeed49SFangrui Song   CHECK_SHADOW_STACK(shadow_stack_size);
156bcaeed49SFangrui Song 
157bcaeed49SFangrui Song   callee_throws();
158bcaeed49SFangrui Song   CHECK_SHADOW_STACK(shadow_stack_size);
159bcaeed49SFangrui Song 
160bcaeed49SFangrui Song   callee_rethrows();
161bcaeed49SFangrui Song   CHECK_SHADOW_STACK(shadow_stack_size);
162bcaeed49SFangrui Song 
163bcaeed49SFangrui Song   nested_try();
164bcaeed49SFangrui Song   CHECK_SHADOW_STACK(shadow_stack_size);
165bcaeed49SFangrui Song 
166bcaeed49SFangrui Song   nested_try2();
167bcaeed49SFangrui Song   CHECK_SHADOW_STACK(shadow_stack_size);
168bcaeed49SFangrui Song 
169bcaeed49SFangrui Song   cpp_object_with_destructor();
170bcaeed49SFangrui Song   CHECK_SHADOW_STACK(shadow_stack_size);
171bcaeed49SFangrui Song 
172bcaeed49SFangrui Song   multiframe_unwind();
173bcaeed49SFangrui Song   CHECK_SHADOW_STACK(shadow_stack_size);
174bcaeed49SFangrui Song 
175bcaeed49SFangrui Song   longjmp_unwind();
176bcaeed49SFangrui Song   CHECK_SHADOW_STACK(shadow_stack_size);
177bcaeed49SFangrui Song 
178bcaeed49SFangrui Song   longjmp_unwind_multiple_frames();
179bcaeed49SFangrui Song   CHECK_SHADOW_STACK(shadow_stack_size);
180bcaeed49SFangrui Song 
181bcaeed49SFangrui Song   return 0;
182bcaeed49SFangrui Song }
183bcaeed49SFangrui Song 
184bcaeed49SFangrui Song // CHECK: Hello, World!
185bcaeed49SFangrui Song // CHECK-NOT: Shadow stack leak
186