blob: c8247e2e0a26a12d0d3eb5cd6c924a3661593094 [file] [log] [blame]
Eli Friedman7dff8a62012-02-28 03:32:48 +00001// RUN: %clang_cc1 -triple x86_64-apple-darwin10.0.0 -emit-llvm -o - %s -fexceptions -std=c++11 -fblocks -fobjc-arc | FileCheck -check-prefix=ARC %s
2// RUN: %clang_cc1 -triple x86_64-apple-darwin10.0.0 -emit-llvm -o - %s -fexceptions -std=c++11 -fblocks | FileCheck -check-prefix=MRC %s
3
4typedef int (^fp)();
5fp f() { auto x = []{ return 3; }; return x; }
6
Akira Hatanakad542ccf2016-09-16 00:02:06 +00007// ARC: %[[LAMBDACLASS:.*]] = type { i32 }
8
Saleem Abdulrasool3f307512016-09-18 16:12:14 +00009// MRC: @OBJC_METH_VAR_NAME{{.*}} = private unnamed_addr constant [5 x i8] c"copy\00"
10// MRC: @OBJC_METH_VAR_NAME{{.*}} = private unnamed_addr constant [12 x i8] c"autorelease\00"
Stephen Lin43622612013-08-15 06:47:53 +000011// MRC-LABEL: define i32 ()* @_Z1fv(
12// MRC-LABEL: define internal i32 ()* @"_ZZ1fvENK3$_0cvU13block_pointerFivEEv"
Eli Friedman7dff8a62012-02-28 03:32:48 +000013// MRC: store i8* bitcast (i8** @_NSConcreteStackBlock to i8*)
Fariborz Jahanian63628032012-06-26 16:06:38 +000014// MRC: store i8* bitcast (i32 (i8*)* @"___ZZ1fvENK3$_0cvU13block_pointerFivEEv_block_invoke" to i8*)
David Blaikied6c88ec2015-04-16 23:25:00 +000015// MRC: call i32 ()* bitcast (i8* (i8*, i8*, ...)* @objc_msgSend to i32 ()* (i8*, i8*)*)
16// MRC: call i32 ()* bitcast (i8* (i8*, i8*, ...)* @objc_msgSend to i32 ()* (i8*, i8*)*)
Eli Friedman7dff8a62012-02-28 03:32:48 +000017// MRC: ret i32 ()*
18
Stephen Lin43622612013-08-15 06:47:53 +000019// ARC-LABEL: define i32 ()* @_Z1fv(
20// ARC-LABEL: define internal i32 ()* @"_ZZ1fvENK3$_0cvU13block_pointerFivEEv"
Eli Friedman7dff8a62012-02-28 03:32:48 +000021// ARC: store i8* bitcast (i8** @_NSConcreteStackBlock to i8*)
Fariborz Jahanian63628032012-06-26 16:06:38 +000022// ARC: store i8* bitcast (i32 (i8*)* @"___ZZ1fvENK3$_0cvU13block_pointerFivEEv_block_invoke" to i8*)
Eli Friedman7dff8a62012-02-28 03:32:48 +000023// ARC: call i8* @objc_retainBlock
24// ARC: call i8* @objc_autoreleaseReturnValue
Eli Friedman98b01ed2012-03-01 04:01:32 +000025
26typedef int (^fp)();
27fp global;
28void f2() { global = []{ return 3; }; }
29
Bill Wendling2386bb12013-02-27 00:06:04 +000030// MRC: define void @_Z2f2v() [[NUW:#[0-9]+]] {
Fariborz Jahanian63628032012-06-26 16:06:38 +000031// MRC: store i8* bitcast (i32 (i8*)* @___Z2f2v_block_invoke to i8*),
Eli Friedman98b01ed2012-03-01 04:01:32 +000032// MRC-NOT: call
33// MRC: ret void
34// ("global" contains a dangling pointer after this function runs.)
35
Bill Wendling2386bb12013-02-27 00:06:04 +000036// ARC: define void @_Z2f2v() [[NUW:#[0-9]+]] {
Fariborz Jahanian63628032012-06-26 16:06:38 +000037// ARC: store i8* bitcast (i32 (i8*)* @___Z2f2v_block_invoke to i8*),
Eli Friedman98b01ed2012-03-01 04:01:32 +000038// ARC: call i8* @objc_retainBlock
39// ARC: call void @objc_release
Stephen Lin43622612013-08-15 06:47:53 +000040// ARC-LABEL: define internal i32 @___Z2f2v_block_invoke
Eli Friedman98b01ed2012-03-01 04:01:32 +000041// ARC: call i32 @"_ZZ2f2vENK3$_1clEv
Bill Wendlingc33fc4c2013-02-20 07:22:19 +000042
John McCalldec348f72013-05-03 07:33:41 +000043template <class T> void take_lambda(T &&lambda) { lambda(); }
44void take_block(void (^block)()) { block(); }
45
46// rdar://13800041
47@interface A
48- (void) test;
49@end
50@interface B : A @end
51@implementation B
52- (void) test {
53 take_block(^{
54 take_lambda([=]{
55 take_block(^{
56 take_lambda([=] {
57 [super test];
58 });
59 });
60 });
61 });
62}
63@end
64
Akira Hatanakad542ccf2016-09-16 00:02:06 +000065// ARC: define void @_ZN13LambdaCapture4foo1ERi(i32* dereferenceable(4) %{{.*}})
66// ARC: %[[CAPTURE0:.*]] = getelementptr inbounds %[[LAMBDACLASS]], %[[LAMBDACLASS]]* %{{.*}}, i32 0, i32 0
67// ARC: store i32 %{{.*}}, i32* %[[CAPTURE0]]
68
69// ARC: define internal void @"_ZZN13LambdaCapture4foo1ERiENK3$_3clEv"(%[[LAMBDACLASS]]* %{{.*}})
70// ARC: %[[BLOCK:.*]] = alloca <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, i32 }>
71// ARC: %[[CAPTURE1:.*]] = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, i32 }>, <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, i32 }>* %[[BLOCK]], i32 0, i32 5
72// ARC: store i32 %{{.*}}, i32* %[[CAPTURE1]]
73
Akira Hatanakaf1b3fc72017-02-14 06:46:55 +000074// ARC-LABEL: define internal void @"_ZZ10-[Foo foo]ENK3$_4clEv"(
75// ARC-NOT: @objc_storeStrong(
76// ARC: ret void
77
Akira Hatanakad542ccf2016-09-16 00:02:06 +000078// ARC: define internal void @"___ZZN13LambdaCapture4foo1ERiENK3$_3clEv_block_invoke"
79// ARC: %[[CAPTURE2:.*]] = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, i32 }>, <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, i32 }>* %{{.*}}, i32 0, i32 5
80// ARC: store i32 %{{.*}}, i32* %[[CAPTURE2]]
81
82// ARC: define internal void @"___ZZN13LambdaCapture4foo1ERiENK3$_3clEv_block_invoke_2"(i8* %{{.*}})
83// ARC: %[[CAPTURE3:.*]] = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, i32 }>, <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, i32 }>* %{{.*}}, i32 0, i32 5
84// ARC: %[[V1:.*]] = load i32, i32* %[[CAPTURE3]]
85// ARC: store i32 %[[V1]], i32* @_ZN13LambdaCapture1iE
86
87namespace LambdaCapture {
88 int i;
89 void foo1(int &a) {
90 auto lambda = [a]{
91 auto block1 = ^{
92 auto block2 = ^{
93 i = a;
94 };
95 block2();
96 };
97 block1();
98 };
99 lambda();
100 }
101}
102
Rafael Espindolae5df59f2015-01-22 00:24:57 +0000103// ARC-LABEL: define linkonce_odr i32 ()* @_ZZNK13StaticMembersIfE1fMUlvE_clEvENKUlvE_cvU13block_pointerFivEEv
104
Eli Friedmanc6036aa2013-07-12 22:05:26 +0000105// Check lines for BlockInLambda test below
Stephen Lin43622612013-08-15 06:47:53 +0000106// ARC-LABEL: define internal i32 @___ZZN13BlockInLambda1X1fEvENKUlvE_clEv_block_invoke
David Blaikie218b7832015-02-27 19:18:17 +0000107// ARC: [[Y:%.*]] = getelementptr inbounds %"struct.BlockInLambda::X", %"struct.BlockInLambda::X"* {{.*}}, i32 0, i32 1
David Blaikiea953f282015-02-27 21:19:58 +0000108// ARC-NEXT: [[YVAL:%.*]] = load i32, i32* [[Y]], align 4
Eli Friedmanc6036aa2013-07-12 22:05:26 +0000109// ARC-NEXT: ret i32 [[YVAL]]
110
Eli Friedmanef102822013-06-13 20:56:27 +0000111typedef int (^fptr)();
112template<typename T> struct StaticMembers {
113 static fptr f;
114};
115template<typename T>
116fptr StaticMembers<T>::f = [] { auto f = []{return 5;}; return fptr(f); }();
117template fptr StaticMembers<float>::f;
Bill Wendlingc33fc4c2013-02-20 07:22:19 +0000118
Eli Friedmanc6036aa2013-07-12 22:05:26 +0000119namespace BlockInLambda {
120 struct X {
121 int x,y;
122 void f() {
123 [this]{return ^{return y;}();}();
124 };
125 };
126 void g(X& x) {
127 x.f();
128 };
129}
130
Akira Hatanakaf1b3fc72017-02-14 06:46:55 +0000131@interface NSObject @end
132@interface Foo : NSObject @end
133@implementation Foo
134- (void)foo {
135 [&] {
136 ^{ (void)self; }();
137 }();
138}
139@end
Eli Friedmanef102822013-06-13 20:56:27 +0000140
Chandler Carruthfcd33142016-12-23 01:24:49 +0000141// ARC: attributes [[NUW]] = { noinline nounwind{{.*}} }
142// MRC: attributes [[NUW]] = { noinline nounwind{{.*}} }