Files
clang-p2996/clang/test/CodeGenCXX/blocks-cxx11.cpp
Hal Finkel a2347baaec Mark C++ reference parameters as dereferenceable
Because references must be initialized using some evaluated expression, they
must point to something, and a callee can assume the reference parameter is
dereferenceable. Taking advantage of a new attribute just added to LLVM, mark
them as such.

Because dereferenceability in addrspace(0) implies nonnull in the backend, we
don't need both attributes. However, we need to know the size of the object to
use the dereferenceable attribute, so for incomplete types we still emit only
nonnull.

llvm-svn: 213386
2014-07-18 15:52:10 +00:00

115 lines
3.8 KiB
C++

// RUN: %clang_cc1 %s -fblocks -triple x86_64-apple-darwin -std=c++11 -emit-llvm -o - | FileCheck %s
template <class T> void takeItByValue(T);
void takeABlock(void (^)());
// rdar://problem/11022704
namespace test_int {
void test() {
const int x = 100;
takeABlock(^{ takeItByValue(x); });
// CHECK: call void @_Z13takeItByValueIiEvT_(i32 100)
}
}
namespace test_int_ref {
void test() {
const int y = 200;
const int &x = y;
takeABlock(^{ takeItByValue(x); });
// TODO: there's no good reason that this isn't foldable.
// CHECK: call void @_Z13takeItByValueIiEvT_(i32 {{%.*}})
}
}
namespace test_float {
void test() {
const float x = 1;
takeABlock(^{ takeItByValue(x); });
// CHECK: call void @_Z13takeItByValueIfEvT_(float 1.0
}
}
namespace test_float_ref {
void test() {
const float y = 100;
const float &x = y;
takeABlock(^{ takeItByValue(x); });
// TODO: there's no good reason that this isn't foldable.
// CHECK: call void @_Z13takeItByValueIfEvT_(float {{%.*}})
}
}
namespace test_complex_int {
void test() {
constexpr _Complex int x = 500;
takeABlock(^{ takeItByValue(x); });
// CHECK: store { i32, i32 } { i32 500, i32 0 },
// CHECK: store i32 500,
// CHECK-NEXT: store i32 0,
// CHECK-NEXT: [[COERCE:%.*]] = bitcast
// CHECK-NEXT: [[CVAL:%.*]] = load i64* [[COERCE]]
// CHECK-NEXT: call void @_Z13takeItByValueICiEvT_(i64 [[CVAL]])
}
}
namespace test_complex_int_ref {
void test() {
const _Complex int y = 100;
const _Complex int &x = y;
takeABlock(^{ takeItByValue(x); });
// CHECK: call void @_Z13takeItByValueICiEvT_(i64
}
}
namespace test_complex_int_ref_mutable {
_Complex int y = 100;
void test() {
const _Complex int &x = y;
takeABlock(^{ takeItByValue(x); });
// CHECK: [[R:%.*]] = load i32* getelementptr inbounds ({ i32, i32 }* @_ZN28test_complex_int_ref_mutable1yE, i32 0, i32 0)
// CHECK-NEXT: [[I:%.*]] = load i32* getelementptr inbounds ({ i32, i32 }* @_ZN28test_complex_int_ref_mutable1yE, i32 0, i32 1)
// CHECK-NEXT: [[RSLOT:%.*]] = getelementptr inbounds { i32, i32 }* [[CSLOT:%.*]], i32 0, i32 0
// CHECK-NEXT: [[ISLOT:%.*]] = getelementptr inbounds { i32, i32 }* [[CSLOT]], i32 0, i32 1
// CHECK-NEXT: store i32 [[R]], i32* [[RSLOT]]
// CHECK-NEXT: store i32 [[I]], i32* [[ISLOT]]
// CHECK-NEXT: [[COERCE:%.*]] = bitcast { i32, i32 }* [[CSLOT]] to i64*
// CHECK-NEXT: [[CVAL:%.*]] = load i64* [[COERCE]],
// CHECK-NEXT: call void @_Z13takeItByValueICiEvT_(i64 [[CVAL]])
}
}
// rdar://13295759
namespace test_block_in_lambda {
void takeBlock(void (^block)());
// The captured variable has to be non-POD so that we have a copy expression.
struct A {
void *p;
A(const A &);
~A();
void use() const;
};
void test(A a) {
auto lambda = [a]() {
takeBlock(^{ a.use(); });
};
lambda(); // make sure we emit the invocation function
}
// CHECK-LABEL: define internal void @"_ZZN20test_block_in_lambda4testENS_1AEENK3$_0clEv"(
// CHECK: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], align 8
// CHECK: [[THIS:%.*]] = load [[LAMBDA_T:%.*]]**
// CHECK: [[TO_DESTROY:%.*]] = getelementptr inbounds [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5
// CHECK: [[T0:%.*]] = getelementptr inbounds [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5
// CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds [[LAMBDA_T]]* [[THIS]], i32 0, i32 0
// CHECK-NEXT: call void @_ZN20test_block_in_lambda1AC1ERKS0_({{.*}}* [[T0]], {{.*}}* dereferenceable({{[0-9]+}}) [[T1]])
// CHECK-NEXT: [[T0:%.*]] = bitcast [[BLOCK_T]]* [[BLOCK]] to void ()*
// CHECK-NEXT: call void @_ZN20test_block_in_lambda9takeBlockEU13block_pointerFvvE(void ()* [[T0]])
// CHECK-NEXT: call void @_ZN20test_block_in_lambda1AD1Ev({{.*}}* [[TO_DESTROY]])
// CHECK-NEXT: ret void
}