2011-08-26 07:04:34 +08:00
|
|
|
// RUN: %clang_cc1 -triple i386-unknown-unknown %s -emit-llvm -o - -fblocks | FileCheck %s
|
2009-03-01 09:09:12 +08:00
|
|
|
void (^f)(void) = ^{};
|
2009-04-08 10:55:55 +08:00
|
|
|
|
|
|
|
// rdar://6768379
|
|
|
|
int f0(int (^a0)()) {
|
|
|
|
return a0(1, 2, 3);
|
|
|
|
}
|
2009-04-17 08:48:04 +08:00
|
|
|
|
|
|
|
// Verify that attributes on blocks are set correctly.
|
|
|
|
typedef struct s0 T;
|
|
|
|
struct s0 {
|
|
|
|
int a[64];
|
|
|
|
};
|
|
|
|
|
2013-02-15 13:25:49 +08:00
|
|
|
// CHECK: define internal void @__f2_block_invoke(%struct.s0* noalias sret {{%.*}}, i8* {{%.*}}, %struct.s0* byval align 4 {{.*}})
|
2009-04-17 08:48:04 +08:00
|
|
|
struct s0 f2(struct s0 a0) {
|
|
|
|
return ^(struct s0 a1){ return a1; }(a0);
|
|
|
|
}
|
|
|
|
|
2009-04-21 12:41:23 +08:00
|
|
|
// This should not crash: rdar://6808051
|
|
|
|
void *P = ^{
|
|
|
|
void *Q = __func__;
|
|
|
|
};
|
|
|
|
|
2009-05-01 09:31:57 +08:00
|
|
|
void (^test1)(void) = ^(void) {
|
|
|
|
__block int i;
|
|
|
|
^ { i = 1; }();
|
|
|
|
};
|
2009-04-21 12:41:23 +08:00
|
|
|
|
2010-06-05 03:02:56 +08:00
|
|
|
typedef double ftype(double);
|
|
|
|
// It's not clear that we *should* support this syntax, but until that decision
|
|
|
|
// is made, we should support it properly and not crash.
|
|
|
|
ftype ^test2 = ^ftype {
|
|
|
|
return 0;
|
|
|
|
};
|
2010-10-29 05:37:57 +08:00
|
|
|
|
|
|
|
// rdar://problem/8605032
|
|
|
|
void f3_helper(void (^)(void));
|
|
|
|
void f3() {
|
|
|
|
_Bool b = 0;
|
|
|
|
f3_helper(^{ if (b) {} });
|
|
|
|
}
|
2012-04-27 05:14:42 +08:00
|
|
|
|
|
|
|
// rdar://problem/11322251
|
2012-05-02 04:28:00 +08:00
|
|
|
// The bool can fill in between the header and the long long.
|
|
|
|
// Add the appropriate amount of padding between them.
|
2012-04-27 05:14:42 +08:00
|
|
|
void f4_helper(long long (^)(void));
|
2013-08-15 14:47:53 +08:00
|
|
|
// CHECK-LABEL: define void @f4()
|
2012-04-27 05:14:42 +08:00
|
|
|
void f4(void) {
|
|
|
|
_Bool b = 0;
|
|
|
|
long long ll = 0;
|
2012-05-02 04:28:00 +08:00
|
|
|
// CHECK: alloca <{ i8*, i32, i32, i8*, {{%.*}}*, i8, [3 x i8], i64 }>, align 8
|
2012-04-27 05:14:42 +08:00
|
|
|
f4_helper(^{ if (b) return ll; return 0LL; });
|
|
|
|
}
|
2012-05-02 04:28:00 +08:00
|
|
|
|
|
|
|
// rdar://problem/11354538
|
|
|
|
// The alignment after rounding up to the align of F5 is actually
|
|
|
|
// greater than the required alignment. Don't assert.
|
|
|
|
struct F5 {
|
|
|
|
char buffer[32] __attribute((aligned));
|
|
|
|
};
|
|
|
|
void f5_helper(void (^)(struct F5 *));
|
2013-08-15 14:47:53 +08:00
|
|
|
// CHECK-LABEL: define void @f5()
|
2012-05-02 04:28:00 +08:00
|
|
|
void f5(void) {
|
|
|
|
struct F5 value;
|
|
|
|
// CHECK: alloca <{ i8*, i32, i32, i8*, {{%.*}}*, [12 x i8], [[F5:%.*]] }>, align 16
|
|
|
|
f5_helper(^(struct F5 *slot) { *slot = value; });
|
|
|
|
}
|
2013-06-07 08:48:14 +08:00
|
|
|
|
|
|
|
// rdar://14085217
|
|
|
|
void (^b)() = ^{};
|
|
|
|
int main() {
|
|
|
|
(b?: ^{})();
|
|
|
|
}
|
2015-02-28 05:19:58 +08:00
|
|
|
// CHECK: [[ZERO:%.*]] = load void (...)*, void (...)** @b
|
2013-06-07 08:48:14 +08:00
|
|
|
// CHECK-NEXT: [[TB:%.*]] = icmp ne void (...)* [[ZERO]], null
|
|
|
|
// CHECK-NEXT: br i1 [[TB]], label [[CT:%.*]], label [[CF:%.*]]
|
|
|
|
// CHECK: [[ONE:%.*]] = bitcast void (...)* [[ZERO]] to void ()*
|
|
|
|
// CHECK-NEXT: br label [[CE:%.*]]
|
|
|
|
|