blob: 075b714f29e2a3bd6631512a7581198bfc765f32 [file] [log] [blame]
John McCall638d4f52013-04-03 00:56:07 +00001// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -fobjc-runtime=macosx-fragile-10.5 -emit-llvm -fobjc-exceptions -O2 -o - %s | FileCheck %s
Daniel Dunbare3883872010-04-23 19:12:32 +00002//
3// <rdar://problem/7471679> [irgen] [eh] Exception code built with clang (x86_64) crashes
4
5// Just check that we don't emit any dead blocks.
Daniel Dunbare3883872010-04-23 19:12:32 +00006@interface NSArray @end
7void f0() {
8 @try {
9 @try {
10 @throw @"a";
11 } @catch(NSArray *e) {
12 }
13 } @catch (id e) {
14 }
15}
John McCallbd309292010-07-06 01:34:17 +000016
Stephen Lin43622612013-08-15 06:47:53 +000017// CHECK-LABEL: define void @f1()
John McCallbd309292010-07-06 01:34:17 +000018void f1() {
19 extern void foo(void);
20
21 while (1) {
22 // CHECK: call void @objc_exception_try_enter
23 // CHECK-NEXT: getelementptr
24 // CHECK-NEXT: call i32 @_setjmp(
25 // CHECK-NEXT: icmp
26 // CHECK-NEXT: br i1
27 @try {
John McCall2dd7d442010-08-04 05:59:32 +000028 // CHECK: call void asm sideeffect "", "*m"
29 // CHECK-NEXT: call void @foo()
John McCallbd309292010-07-06 01:34:17 +000030 foo();
John McCall638d4f52013-04-03 00:56:07 +000031 // CHECK: call void @objc_exception_try_exit
John McCallbd309292010-07-06 01:34:17 +000032
John McCall2dd7d442010-08-04 05:59:32 +000033 // CHECK: call void asm sideeffect "", "=*m"
John McCallbd309292010-07-06 01:34:17 +000034 } @finally {
35 break;
36 }
37 }
38}
John McCall42227ed2010-07-31 23:20:56 +000039
40// Test that modifications to local variables are respected under
41// optimization. rdar://problem/8160285
42
Stephen Lin43622612013-08-15 06:47:53 +000043// CHECK-LABEL: define i32 @f2()
John McCall42227ed2010-07-31 23:20:56 +000044int f2() {
45 extern void foo(void);
46
47 // CHECK: [[X:%.*]] = alloca i32
John McCall2dd7d442010-08-04 05:59:32 +000048 // CHECK: store i32 5, i32* [[X]]
John McCall42227ed2010-07-31 23:20:56 +000049 int x = 0;
John McCall2dd7d442010-08-04 05:59:32 +000050 x += 5;
John McCall42227ed2010-07-31 23:20:56 +000051
52 // CHECK: [[SETJMP:%.*]] = call i32 @_setjmp
53 // CHECK-NEXT: [[CAUGHT:%.*]] = icmp eq i32 [[SETJMP]], 0
54 // CHECK-NEXT: br i1 [[CAUGHT]]
55 @try {
Nick Lewycky5f2899c2011-11-15 00:19:16 +000056 // CHECK: store i32 6, i32* [[X]]
John McCall42227ed2010-07-31 23:20:56 +000057 x++;
Philip Reamesc758ca32015-06-16 20:24:06 +000058 // CHECK-NEXT: call void asm sideeffect "", "*m,*m"(i32* nonnull [[X]]
John McCall42227ed2010-07-31 23:20:56 +000059 // CHECK-NEXT: call void @foo()
John McCall2dd7d442010-08-04 05:59:32 +000060 // CHECK-NEXT: call void @objc_exception_try_exit
David Blaikiea953f282015-02-27 21:19:58 +000061 // CHECK-NEXT: [[T:%.*]] = load i32, i32* [[X]]
John McCall42227ed2010-07-31 23:20:56 +000062 foo();
63 } @catch (id) {
John McCall2dd7d442010-08-04 05:59:32 +000064 // Landing pad. Note that we elide the re-enter.
Philip Reamesc758ca32015-06-16 20:24:06 +000065 // CHECK: call void asm sideeffect "", "=*m,=*m"(i32* nonnull [[X]]
John McCall42227ed2010-07-31 23:20:56 +000066 // CHECK-NEXT: call i8* @objc_exception_extract
David Blaikiea953f282015-02-27 21:19:58 +000067 // CHECK-NEXT: [[T1:%.*]] = load i32, i32* [[X]]
Chris Lattnerffda4522011-02-17 02:02:42 +000068 // CHECK-NEXT: [[T2:%.*]] = add nsw i32 [[T1]], -1
John McCall2dd7d442010-08-04 05:59:32 +000069
70 // This store is dead.
John McCall42227ed2010-07-31 23:20:56 +000071 // CHECK-NEXT: store i32 [[T2]], i32* [[X]]
John McCall42227ed2010-07-31 23:20:56 +000072 x--;
73 }
John McCallf26870c2011-01-28 06:05:16 +000074
John McCall42227ed2010-07-31 23:20:56 +000075 return x;
76}
John McCallcebe0ca2010-08-11 00:16:14 +000077
78// Test that the cleanup destination is saved when entering a finally
79// block. rdar://problem/8293901
Stephen Lin43622612013-08-15 06:47:53 +000080// CHECK-LABEL: define void @f3()
John McCallcebe0ca2010-08-11 00:16:14 +000081void f3() {
82 extern void f3_helper(int, int*);
83
84 // CHECK: [[X:%.*]] = alloca i32
David Majnemerdc012fa2015-04-22 21:38:15 +000085 // CHECK: [[XPTR:%.*]] = bitcast i32* [[X]] to i8*
86 // CHECK: call void @llvm.lifetime.start(i64 4, i8* [[XPTR]])
John McCallcebe0ca2010-08-11 00:16:14 +000087 // CHECK: store i32 0, i32* [[X]]
88 int x = 0;
89
90 // CHECK: call void @objc_exception_try_enter(
91 // CHECK: call i32 @_setjmp
92 // CHECK-NEXT: icmp eq
93 // CHECK-NEXT: br i1
94
95 @try {
Philip Reamesc758ca32015-06-16 20:24:06 +000096 // CHECK: call void @f3_helper(i32 0, i32* nonnull [[X]])
John McCallcebe0ca2010-08-11 00:16:14 +000097 // CHECK: call void @objc_exception_try_exit(
98 f3_helper(0, &x);
99 } @finally {
100 // CHECK: [[DEST1:%.*]] = phi i32 [ 0, {{%.*}} ], [ 3, {{%.*}} ]
101 // CHECK: call void @objc_exception_try_enter
102 // CHECK: call i32 @_setjmp
103 @try {
Philip Reamesc758ca32015-06-16 20:24:06 +0000104 // CHECK: call void @f3_helper(i32 1, i32* nonnull [[X]])
John McCallcebe0ca2010-08-11 00:16:14 +0000105 // CHECK: call void @objc_exception_try_exit(
106 f3_helper(1, &x);
107 } @finally {
108 // CHECK: [[DEST2:%.*]] = phi i32 [ 0, {{%.*}} ], [ 5, {{%.*}} ]
Philip Reamesc758ca32015-06-16 20:24:06 +0000109 // CHECK: call void @f3_helper(i32 2, i32* nonnull [[X]])
John McCallcebe0ca2010-08-11 00:16:14 +0000110 f3_helper(2, &x);
111
112 // This loop is large enough to dissuade the optimizer from just
113 // duplicating the finally block.
114 while (x) f3_helper(3, &x);
115
John McCallca7993f2010-08-11 02:06:44 +0000116 // This is a switch or maybe some chained branches, but relying
117 // on a specific result from the optimizer is really unstable.
118 // CHECK: [[DEST2]]
John McCallcebe0ca2010-08-11 00:16:14 +0000119 }
120
John McCallca7993f2010-08-11 02:06:44 +0000121 // This is a switch or maybe some chained branches, but relying
122 // on a specific result from the optimizer is really unstable.
123 // CHECK: [[DEST1]]
John McCallcebe0ca2010-08-11 00:16:14 +0000124 }
125
Philip Reamesc758ca32015-06-16 20:24:06 +0000126 // CHECK: call void @f3_helper(i32 4, i32* nonnull [[X]])
David Majnemerdc012fa2015-04-22 21:38:15 +0000127 // CHECK-NEXT: call void @llvm.lifetime.end(i64 4, i8* [[XPTR]])
John McCallcebe0ca2010-08-11 00:16:14 +0000128 // CHECK-NEXT: ret void
129 f3_helper(4, &x);
130}
John McCall9916e3f2010-10-04 23:42:51 +0000131
132// rdar://problem/8440970
133void f4() {
134 extern void f4_help(int);
135
Stephen Lin43622612013-08-15 06:47:53 +0000136 // CHECK-LABEL: define void @f4()
John McCall9916e3f2010-10-04 23:42:51 +0000137 // CHECK: [[EXNDATA:%.*]] = alloca [[EXNDATA_T:%.*]], align
Philip Reamesc758ca32015-06-16 20:24:06 +0000138 // CHECK: call void @objc_exception_try_enter([[EXNDATA_T]]* nonnull [[EXNDATA]])
John McCall9916e3f2010-10-04 23:42:51 +0000139 // CHECK: call i32 @_setjmp
140 @try {
141 // CHECK: call void @f4_help(i32 0)
142 f4_help(0);
143
144 // The finally cleanup has two threaded entrypoints after optimization:
145
146 // finally.no-call-exit: Predecessor is when the catch throws.
Philip Reamesc758ca32015-06-16 20:24:06 +0000147 // CHECK: call i8* @objc_exception_extract([[EXNDATA_T]]* nonnull [[EXNDATA]])
John McCall9916e3f2010-10-04 23:42:51 +0000148 // CHECK-NEXT: call void @f4_help(i32 2)
149 // CHECK-NEXT: br label
150 // -> rethrow
151
152 // finally.call-exit: Predecessors are the @try and @catch fallthroughs
153 // as well as the no-match case in the catch mechanism. The i1 is whether
154 // to rethrow and should be true only in the last case.
Chandler Carruthed2a2842012-09-24 01:10:25 +0000155 // CHECK: phi i8*
156 // CHECK-NEXT: phi i1
Philip Reamesc758ca32015-06-16 20:24:06 +0000157 // CHECK-NEXT: call void @objc_exception_try_exit([[EXNDATA_T]]* nonnull [[EXNDATA]])
John McCall9916e3f2010-10-04 23:42:51 +0000158 // CHECK-NEXT: call void @f4_help(i32 2)
159 // CHECK-NEXT: br i1
160 // -> ret, rethrow
161
162 // ret:
163 // CHECK: ret void
164
165 // Catch mechanism:
Philip Reamesc758ca32015-06-16 20:24:06 +0000166 // CHECK: call i8* @objc_exception_extract([[EXNDATA_T]]* nonnull [[EXNDATA]])
167 // CHECK-NEXT: call void @objc_exception_try_enter([[EXNDATA_T]]* nonnull [[EXNDATA]])
John McCall9916e3f2010-10-04 23:42:51 +0000168 // CHECK: call i32 @_setjmp
169 // -> next, finally.no-call-exit
170 // CHECK: call i32 @objc_exception_match
171 // -> finally.call-exit, match
172 } @catch (NSArray *a) {
173 // match:
174 // CHECK: call void @f4_help(i32 1)
175 // CHECK-NEXT: br label
176 // -> finally.call-exit
177 f4_help(1);
178 } @finally {
179 f4_help(2);
180 }
181
182 // rethrow:
183 // CHECK: phi i8*
184 // CHECK-NEXT: call void @objc_exception_throw(i8*
185 // CHECK-NEXT: unreachable
186}