2012-02-28 11:32:48 +08:00
|
|
|
// 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
|
|
|
|
// RUN: %clang_cc1 -triple x86_64-apple-darwin10.0.0 -emit-llvm -o - %s -fexceptions -std=c++11 -fblocks | FileCheck -check-prefix=MRC %s
|
|
|
|
|
|
|
|
typedef int (^fp)();
|
|
|
|
fp f() { auto x = []{ return 3; }; return x; }
|
|
|
|
|
2014-02-28 03:01:11 +08:00
|
|
|
// MRC: @"\01L_OBJC_METH_VAR_NAME{{.*}}" = private global [5 x i8] c"copy\00"
|
|
|
|
// MRC: @"\01L_OBJC_METH_VAR_NAME{{.*}}" = private global [12 x i8] c"autorelease\00"
|
2013-08-15 14:47:53 +08:00
|
|
|
// MRC-LABEL: define i32 ()* @_Z1fv(
|
|
|
|
// MRC-LABEL: define internal i32 ()* @"_ZZ1fvENK3$_0cvU13block_pointerFivEEv"
|
2012-02-28 11:32:48 +08:00
|
|
|
// MRC: store i8* bitcast (i8** @_NSConcreteStackBlock to i8*)
|
2012-06-27 00:06:38 +08:00
|
|
|
// MRC: store i8* bitcast (i32 (i8*)* @"___ZZ1fvENK3$_0cvU13block_pointerFivEEv_block_invoke" to i8*)
|
2012-02-28 11:32:48 +08:00
|
|
|
// MRC: call i32 ()* (i8*, i8*)* bitcast (i8* (i8*, i8*, ...)* @objc_msgSend to i32 ()* (i8*, i8*)*)
|
|
|
|
// MRC: call i32 ()* (i8*, i8*)* bitcast (i8* (i8*, i8*, ...)* @objc_msgSend to i32 ()* (i8*, i8*)*)
|
|
|
|
// MRC: ret i32 ()*
|
|
|
|
|
2013-08-15 14:47:53 +08:00
|
|
|
// ARC-LABEL: define i32 ()* @_Z1fv(
|
|
|
|
// ARC-LABEL: define internal i32 ()* @"_ZZ1fvENK3$_0cvU13block_pointerFivEEv"
|
2012-02-28 11:32:48 +08:00
|
|
|
// ARC: store i8* bitcast (i8** @_NSConcreteStackBlock to i8*)
|
2012-06-27 00:06:38 +08:00
|
|
|
// ARC: store i8* bitcast (i32 (i8*)* @"___ZZ1fvENK3$_0cvU13block_pointerFivEEv_block_invoke" to i8*)
|
2012-02-28 11:32:48 +08:00
|
|
|
// ARC: call i8* @objc_retainBlock
|
|
|
|
// ARC: call i8* @objc_autoreleaseReturnValue
|
2012-03-01 12:01:32 +08:00
|
|
|
|
|
|
|
typedef int (^fp)();
|
|
|
|
fp global;
|
|
|
|
void f2() { global = []{ return 3; }; }
|
|
|
|
|
2013-02-27 08:06:04 +08:00
|
|
|
// MRC: define void @_Z2f2v() [[NUW:#[0-9]+]] {
|
2012-06-27 00:06:38 +08:00
|
|
|
// MRC: store i8* bitcast (i32 (i8*)* @___Z2f2v_block_invoke to i8*),
|
2012-03-01 12:01:32 +08:00
|
|
|
// MRC-NOT: call
|
|
|
|
// MRC: ret void
|
|
|
|
// ("global" contains a dangling pointer after this function runs.)
|
|
|
|
|
2013-02-27 08:06:04 +08:00
|
|
|
// ARC: define void @_Z2f2v() [[NUW:#[0-9]+]] {
|
2012-06-27 00:06:38 +08:00
|
|
|
// ARC: store i8* bitcast (i32 (i8*)* @___Z2f2v_block_invoke to i8*),
|
2012-03-01 12:01:32 +08:00
|
|
|
// ARC: call i8* @objc_retainBlock
|
|
|
|
// ARC: call void @objc_release
|
2013-08-15 14:47:53 +08:00
|
|
|
// ARC-LABEL: define internal i32 @___Z2f2v_block_invoke
|
2012-03-01 12:01:32 +08:00
|
|
|
// ARC: call i32 @"_ZZ2f2vENK3$_1clEv
|
2013-02-20 15:22:19 +08:00
|
|
|
|
2013-05-03 15:33:41 +08:00
|
|
|
template <class T> void take_lambda(T &&lambda) { lambda(); }
|
|
|
|
void take_block(void (^block)()) { block(); }
|
|
|
|
|
|
|
|
// rdar://13800041
|
|
|
|
@interface A
|
|
|
|
- (void) test;
|
|
|
|
@end
|
|
|
|
@interface B : A @end
|
|
|
|
@implementation B
|
|
|
|
- (void) test {
|
|
|
|
take_block(^{
|
|
|
|
take_lambda([=]{
|
|
|
|
take_block(^{
|
|
|
|
take_lambda([=] {
|
|
|
|
[super test];
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
|
2013-07-13 06:05:26 +08:00
|
|
|
// Check lines for BlockInLambda test below
|
2013-08-15 14:47:53 +08:00
|
|
|
// ARC-LABEL: define internal i32 @___ZZN13BlockInLambda1X1fEvENKUlvE_clEv_block_invoke
|
2013-07-13 06:05:26 +08:00
|
|
|
// ARC: [[Y:%.*]] = getelementptr inbounds %"struct.BlockInLambda::X"* {{.*}}, i32 0, i32 1
|
|
|
|
// ARC-NEXT: [[YVAL:%.*]] = load i32* [[Y]], align 4
|
|
|
|
// ARC-NEXT: ret i32 [[YVAL]]
|
|
|
|
|
2013-06-14 04:56:27 +08:00
|
|
|
typedef int (^fptr)();
|
|
|
|
template<typename T> struct StaticMembers {
|
|
|
|
static fptr f;
|
|
|
|
};
|
|
|
|
template<typename T>
|
|
|
|
fptr StaticMembers<T>::f = [] { auto f = []{return 5;}; return fptr(f); }();
|
|
|
|
template fptr StaticMembers<float>::f;
|
2013-08-15 14:47:53 +08:00
|
|
|
// ARC-LABEL: define linkonce_odr i32 ()* @_ZZNK13StaticMembersIfE1fMUlvE_clEvENKUlvE_cvU13block_pointerFivEEv
|
2013-06-14 04:56:27 +08:00
|
|
|
|
2013-07-13 06:05:26 +08:00
|
|
|
namespace BlockInLambda {
|
|
|
|
struct X {
|
|
|
|
int x,y;
|
|
|
|
void f() {
|
|
|
|
[this]{return ^{return y;}();}();
|
|
|
|
};
|
|
|
|
};
|
|
|
|
void g(X& x) {
|
|
|
|
x.f();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2013-02-20 15:22:19 +08:00
|
|
|
|
2013-06-14 04:56:27 +08:00
|
|
|
// ARC: attributes [[NUW]] = { nounwind{{.*}} }
|
2013-02-27 08:06:04 +08:00
|
|
|
// MRC: attributes [[NUW]] = { nounwind{{.*}} }
|