// RUN: %clang_cc1 -triple i386-unknown-unknown %s -emit-llvm -o - -fblocks | FileCheck %s void (^f)(void) = ^{}; // rdar://6768379 int f0(int (^a0)()) { return a0(1, 2, 3); } // Verify that attributes on blocks are set correctly. typedef struct s0 T; struct s0 { int a[64]; }; // CHECK: define internal void @__f2_block_invoke(%struct.s0* noalias sret {{%.*}}, i8* {{%.*}}, %struct.s0* byval align 4 {{.*}}) struct s0 f2(struct s0 a0) { return ^(struct s0 a1){ return a1; }(a0); } // This should not crash: rdar://6808051 void *P = ^{ void *Q = __func__; }; void (^test1)(void) = ^(void) { __block int i; ^ { i = 1; }(); }; 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; }; // rdar://problem/8605032 void f3_helper(void (^)(void)); void f3() { _Bool b = 0; f3_helper(^{ if (b) {} }); } // rdar://problem/11322251 // The bool can fill in between the header and the long long. // Add the appropriate amount of padding between them. void f4_helper(long long (^)(void)); // CHECK-LABEL: define void @f4() void f4(void) { _Bool b = 0; long long ll = 0; // CHECK: alloca <{ i8*, i32, i32, i8*, {{%.*}}*, i8, [3 x i8], i64 }>, align 8 f4_helper(^{ if (b) return ll; return 0LL; }); } // 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 *)); // CHECK-LABEL: define void @f5() 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; }); } // rdar://14085217 void (^b)() = ^{}; int main() { (b?: ^{})(); } // CHECK: [[ZERO:%.*]] = load void (...)*, void (...)** @b // 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:%.*]] // Ensure that we don't emit helper code in copy/dispose routines for variables // that are const-captured. void testConstCaptureInCopyAndDestroyHelpers() { const int x = 0; __block int i; (^ { i = x; })(); } // CHECK-LABEL: testConstCaptureInCopyAndDestroyHelpers_block_invoke // CHECK: @__copy_helper_block // CHECK: alloca // CHECK-NEXT: alloca // CHECK-NEXT: store // CHECK-NEXT: store // CHECK-NEXT: load // CHECK-NEXT: bitcast // CHECK-NEXT: load // CHECK-NEXT: bitcast // CHECK-NEXT: getelementptr // CHECK-NEXT: getelementptr // CHECK-NEXT: load // CHECK-NEXT: bitcast // CHECK-NEXT: call void @_Block_object_assign // CHECK-NEXT: ret // CHECK: @__destroy_helper_block // CHECK: alloca // CHECK-NEXT: store // CHECK-NEXT: load // CHECK-NEXT: bitcast // CHECK-NEXT: getelementptr // CHECK-NEXT: load // CHECK-NEXT: call void @_Block_object_dispose // CHECK-NEXT: ret