blob: 94e1374c838fca52d3b09c848b3baa7c443df9ba [file] [log] [blame]
John McCall12cc42a2013-02-01 05:11:40 +00001// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -emit-llvm -fobjc-arc -fexceptions -fobjc-exceptions -fcxx-exceptions -fobjc-runtime-has-weak -o - -fobjc-arc-exceptions %s | FileCheck %s
John McCall97017312012-01-17 20:16:56 +00002
3@class Ety;
4
5// These first four tests are all PR11732 / rdar://problem/10667070.
6
7void test0_helper(void);
8void test0(void) {
9 @try {
10 test0_helper();
11 } @catch (Ety *e) {
12 }
13}
Stephen Lin43622612013-08-15 06:47:53 +000014// CHECK-LABEL: define void @_Z5test0v()
John McCall97017312012-01-17 20:16:56 +000015// CHECK: [[E:%.*]] = alloca [[ETY:%.*]]*, align 8
16// CHECK-NEXT: invoke void @_Z12test0_helperv()
17// CHECK: [[T0:%.*]] = call i8* @objc_begin_catch(
18// CHECK-NEXT: [[T1:%.*]] = bitcast i8* [[T0]] to [[ETY]]*
19// CHECK-NEXT: [[T2:%.*]] = bitcast [[ETY]]* [[T1]] to i8*
Pete Cooper2cd35962018-12-18 20:33:00 +000020// CHECK-NEXT: [[T3:%.*]] = call i8* @llvm.objc.retain(i8* [[T2]]) [[NUW:#[0-9]+]]
John McCall97017312012-01-17 20:16:56 +000021// CHECK-NEXT: [[T4:%.*]] = bitcast i8* [[T3]] to [[ETY]]*
22// CHECK-NEXT: store [[ETY]]* [[T4]], [[ETY]]** [[E]]
John McCalle68b8f42012-10-17 02:28:37 +000023// CHECK-NEXT: [[T0:%.*]] = bitcast [[ETY]]** [[E]] to i8**
Pete Cooper2cd35962018-12-18 20:33:00 +000024// CHECK-NEXT: call void @llvm.objc.storeStrong(i8** [[T0]], i8* null) [[NUW]]
Bill Wendlinge1c4a1b2013-02-22 09:10:20 +000025// CHECK-NEXT: call void @objc_end_catch() [[NUW]]
John McCall97017312012-01-17 20:16:56 +000026
27void test1_helper(void);
28void test1(void) {
29 @try {
30 test1_helper();
31 } @catch (__weak Ety *e) {
32 }
33}
Stephen Lin43622612013-08-15 06:47:53 +000034// CHECK-LABEL: define void @_Z5test1v()
John McCall97017312012-01-17 20:16:56 +000035// CHECK: [[E:%.*]] = alloca [[ETY:%.*]]*, align 8
36// CHECK-NEXT: invoke void @_Z12test1_helperv()
37// CHECK: [[T0:%.*]] = call i8* @objc_begin_catch(
38// CHECK-NEXT: [[T1:%.*]] = bitcast i8* [[T0]] to [[ETY]]*
39// CHECK-NEXT: [[T2:%.*]] = bitcast [[ETY]]** [[E]] to i8**
40// CHECK-NEXT: [[T3:%.*]] = bitcast [[ETY]]* [[T1]] to i8*
Pete Cooper2cd35962018-12-18 20:33:00 +000041// CHECK-NEXT: call i8* @llvm.objc.initWeak(i8** [[T2]], i8* [[T3]]) [[NUW]]
John McCall97017312012-01-17 20:16:56 +000042// CHECK-NEXT: [[T0:%.*]] = bitcast [[ETY]]** [[E]] to i8**
Pete Cooper2cd35962018-12-18 20:33:00 +000043// CHECK-NEXT: call void @llvm.objc.destroyWeak(i8** [[T0]]) [[NUW]]
Bill Wendlinge1c4a1b2013-02-22 09:10:20 +000044// CHECK-NEXT: call void @objc_end_catch() [[NUW]]
John McCall97017312012-01-17 20:16:56 +000045
46void test2_helper(void);
47void test2(void) {
48 try {
49 test2_helper();
50 } catch (Ety *e) {
51 }
52}
Stephen Lin43622612013-08-15 06:47:53 +000053// CHECK-LABEL: define void @_Z5test2v()
John McCall97017312012-01-17 20:16:56 +000054// CHECK: [[E:%.*]] = alloca [[ETY:%.*]]*, align 8
55// CHECK-NEXT: invoke void @_Z12test2_helperv()
56// CHECK: [[T0:%.*]] = call i8* @__cxa_begin_catch(
57// CHECK-NEXT: [[T1:%.*]] = bitcast i8* [[T0]] to [[ETY]]*
58// CHECK-NEXT: [[T2:%.*]] = bitcast [[ETY]]* [[T1]] to i8*
Pete Cooper2cd35962018-12-18 20:33:00 +000059// CHECK-NEXT: [[T3:%.*]] = call i8* @llvm.objc.retain(i8* [[T2]]) [[NUW]]
John McCall97017312012-01-17 20:16:56 +000060// CHECK-NEXT: [[T4:%.*]] = bitcast i8* [[T3]] to [[ETY]]*
61// CHECK-NEXT: store [[ETY]]* [[T4]], [[ETY]]** [[E]]
John McCalle68b8f42012-10-17 02:28:37 +000062// CHECK-NEXT: [[T0:%.*]] = bitcast [[ETY]]** [[E]] to i8**
Pete Cooper2cd35962018-12-18 20:33:00 +000063// CHECK-NEXT: call void @llvm.objc.storeStrong(i8** [[T0]], i8* null) [[NUW]]
Bill Wendlinge1c4a1b2013-02-22 09:10:20 +000064// CHECK-NEXT: call void @__cxa_end_catch() [[NUW]]
John McCall97017312012-01-17 20:16:56 +000065
66void test3_helper(void);
67void test3(void) {
68 try {
69 test3_helper();
70 } catch (Ety * __weak e) {
71 }
72}
Stephen Lin43622612013-08-15 06:47:53 +000073// CHECK-LABEL: define void @_Z5test3v()
John McCall97017312012-01-17 20:16:56 +000074// CHECK: [[E:%.*]] = alloca [[ETY:%.*]]*, align 8
75// CHECK-NEXT: invoke void @_Z12test3_helperv()
76// CHECK: [[T0:%.*]] = call i8* @__cxa_begin_catch(
77// CHECK-NEXT: [[T1:%.*]] = bitcast i8* [[T0]] to [[ETY]]*
78// CHECK-NEXT: [[T2:%.*]] = bitcast [[ETY]]** [[E]] to i8**
79// CHECK-NEXT: [[T3:%.*]] = bitcast [[ETY]]* [[T1]] to i8*
Pete Cooper2cd35962018-12-18 20:33:00 +000080// CHECK-NEXT: call i8* @llvm.objc.initWeak(i8** [[T2]], i8* [[T3]]) [[NUW]]
John McCall97017312012-01-17 20:16:56 +000081// CHECK-NEXT: [[T0:%.*]] = bitcast [[ETY]]** [[E]] to i8**
Pete Cooper2cd35962018-12-18 20:33:00 +000082// CHECK-NEXT: call void @llvm.objc.destroyWeak(i8** [[T0]]) [[NUW]]
Bill Wendlinge1c4a1b2013-02-22 09:10:20 +000083// CHECK-NEXT: call void @__cxa_end_catch() [[NUW]]
John McCall12cc42a2013-02-01 05:11:40 +000084
85namespace test4 {
86 struct A {
87 id single;
88 id array[2][3];
89
90 A();
91 };
92
93 A::A() {
94 throw 0;
95 }
Stephen Lin43622612013-08-15 06:47:53 +000096 // CHECK-LABEL: define void @_ZN5test41AC2Ev(
David Blaikiea953f282015-02-27 21:19:58 +000097 // CHECK: [[THIS:%.*]] = load [[A:%.*]]*, [[A:%.*]]** {{%.*}}
John McCall12cc42a2013-02-01 05:11:40 +000098 // Construct single.
David Blaikie218b7832015-02-27 19:18:17 +000099 // CHECK-NEXT: [[SINGLE:%.*]] = getelementptr inbounds [[A]], [[A]]* [[THIS]], i32 0, i32 0
John McCall12cc42a2013-02-01 05:11:40 +0000100 // CHECK-NEXT: store i8* null, i8** [[SINGLE]], align 8
101 // Construct array.
David Blaikie218b7832015-02-27 19:18:17 +0000102 // CHECK-NEXT: [[ARRAY:%.*]] = getelementptr inbounds [[A]], [[A]]* [[THIS]], i32 0, i32 1
John McCall12cc42a2013-02-01 05:11:40 +0000103 // CHECK-NEXT: [[T0:%.*]] = bitcast [2 x [3 x i8*]]* [[ARRAY]] to i8*
Daniel Neilson6e938ef2018-01-19 17:12:54 +0000104 // CHECK-NEXT: call void @llvm.memset.p0i8.i64(i8* align 8 [[T0]], i8 0, i64 48, i1 false)
John McCall12cc42a2013-02-01 05:11:40 +0000105 // throw 0;
106 // CHECK: invoke void @__cxa_throw(
107 // Landing pad from throw site:
108 // CHECK: landingpad
109 // - First, destroy all of array.
David Blaikie218b7832015-02-27 19:18:17 +0000110 // CHECK: [[ARRAYBEGIN:%.*]] = getelementptr inbounds [2 x [3 x i8*]], [2 x [3 x i8*]]* [[ARRAY]], i32 0, i32 0, i32 0
111 // CHECK-NEXT: [[ARRAYEND:%.*]] = getelementptr inbounds i8*, i8** [[ARRAYBEGIN]], i64 6
John McCall12cc42a2013-02-01 05:11:40 +0000112 // CHECK-NEXT: br label
113 // CHECK: [[AFTER:%.*]] = phi i8** [ [[ARRAYEND]], {{%.*}} ], [ [[ELT:%.*]], {{%.*}} ]
David Blaikie218b7832015-02-27 19:18:17 +0000114 // CHECK-NEXT: [[ELT]] = getelementptr inbounds i8*, i8** [[AFTER]], i64 -1
Pete Cooper2cd35962018-12-18 20:33:00 +0000115 // CHECK-NEXT: call void @llvm.objc.storeStrong(i8** [[ELT]], i8* null) [[NUW]]
John McCall12cc42a2013-02-01 05:11:40 +0000116 // CHECK-NEXT: [[DONE:%.*]] = icmp eq i8** [[ELT]], [[ARRAYBEGIN]]
117 // CHECK-NEXT: br i1 [[DONE]],
118 // - Next, destroy single.
Pete Cooper2cd35962018-12-18 20:33:00 +0000119 // CHECK: call void @llvm.objc.storeStrong(i8** [[SINGLE]], i8* null) [[NUW]]
John McCall12cc42a2013-02-01 05:11:40 +0000120 // CHECK: br label
121 // CHECK: resume
122}
Bill Wendlinge1c4a1b2013-02-22 09:10:20 +0000123
John McCallf0f0b7a2015-09-14 18:57:08 +0000124// rdar://21397946
125__attribute__((ns_returns_retained)) id test5_helper(unsigned);
126void test5(void) {
127 id array[][2] = {
128 test5_helper(0),
129 test5_helper(1),
130 test5_helper(2),
131 test5_helper(3)
132 };
133}
134// CHECK-LABEL: define void @_Z5test5v()
135// CHECK: [[ARRAY:%.*]] = alloca [2 x [2 x i8*]], align
136// CHECK: [[A0:%.*]] = getelementptr inbounds [2 x [2 x i8*]], [2 x [2 x i8*]]* [[ARRAY]], i64 0, i64 0
137// CHECK-NEXT: store [2 x i8*]* [[A0]],
138// CHECK-NEXT: [[A00:%.*]] = getelementptr inbounds [2 x i8*], [2 x i8*]* [[A0]], i64 0, i64 0
139// CHECK-NEXT: store i8** [[A00]],
140// CHECK-NEXT: [[T0:%.*]] = invoke i8* @_Z12test5_helperj(i32 0)
141// CHECK: store i8* [[T0]], i8** [[A00]], align
142// CHECK-NEXT: [[A01:%.*]] = getelementptr inbounds i8*, i8** [[A00]], i64 1
143// CHECK-NEXT: store i8** [[A01]],
144// CHECK-NEXT: [[T0:%.*]] = invoke i8* @_Z12test5_helperj(i32 1)
145// CHECK: store i8* [[T0]], i8** [[A01]], align
146// CHECK-NEXT: [[A1:%.*]] = getelementptr inbounds [2 x i8*], [2 x i8*]* [[A0]], i64 1
147// CHECK-NEXT: store [2 x i8*]* [[A1]],
148// CHECK-NEXT: [[A10:%.*]] = getelementptr inbounds [2 x i8*], [2 x i8*]* [[A1]], i64 0, i64 0
149// CHECK-NEXT: store i8** [[A10]],
150// CHECK-NEXT: [[T0:%.*]] = invoke i8* @_Z12test5_helperj(i32 2)
151// CHECK: store i8* [[T0]], i8** [[A10]], align
152// CHECK-NEXT: [[A11:%.*]] = getelementptr inbounds i8*, i8** [[A10]], i64 1
153// CHECK-NEXT: store i8** [[A11]],
154// CHECK-NEXT: [[T0:%.*]] = invoke i8* @_Z12test5_helperj(i32 3)
155// CHECK: store i8* [[T0]], i8** [[A11]], align
156
Bill Wendlinge1c4a1b2013-02-22 09:10:20 +0000157// CHECK: attributes [[NUW]] = { nounwind }