mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-29 10:46:06 +00:00

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
108 lines
2.1 KiB
C++
108 lines
2.1 KiB
C++
// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -emit-llvm -o - %s -std=c++11 | FileCheck %s
|
|
|
|
struct A {
|
|
A();
|
|
A(const A&);
|
|
A(A&);
|
|
~A();
|
|
};
|
|
|
|
struct B {
|
|
B();
|
|
B(B&);
|
|
};
|
|
|
|
struct C {
|
|
C() {}
|
|
C(C& other, A a = A());
|
|
int i, j;
|
|
};
|
|
|
|
struct POD {
|
|
int array[3][4];
|
|
};
|
|
|
|
struct D : A, B, virtual C {
|
|
D();
|
|
int scalar;
|
|
int scalar_array[2][3];
|
|
B class_member;
|
|
C class_member_array[2][3];
|
|
POD pod_array[2][3];
|
|
|
|
union {
|
|
int x;
|
|
float f[3];
|
|
};
|
|
};
|
|
|
|
void f(D d) {
|
|
D d2(d);
|
|
}
|
|
|
|
// CHECK-LABEL: define linkonce_odr void @_ZN1DC1ERS_(%struct.D* %this, %struct.D* dereferenceable({{[0-9]+}})) unnamed_addr
|
|
// CHECK: call void @_ZN1AC1Ev
|
|
// CHECK: call void @_ZN1CC2ERS_1A
|
|
// CHECK: call void @_ZN1AD1Ev
|
|
// CHECK: call void @_ZN1AC2ERS_
|
|
// CHECK: call void @_ZN1BC2ERS_
|
|
// CHECK: {{call void @llvm.memcpy.p0i8.p0i8.i64.*i64 28}}
|
|
// CHECK: call void @_ZN1BC1ERS_
|
|
// CHECK: br
|
|
// CHECK: {{icmp ult.*, 2}}
|
|
// CHECK: {{icmp ult.*, 3}}
|
|
// CHECK: call void @_ZN1AC1Ev
|
|
// CHECK: call void @_ZN1CC1ERS_1A
|
|
// CHECK: call void @_ZN1AD1Ev
|
|
// CHECK: {{call void @llvm.memcpy.p0i8.p0i8.i64.*i64 300}}
|
|
// CHECK: ret void
|
|
|
|
|
|
template<class T> struct X0 { void f0(T * ) { } };
|
|
template <class > struct X1 { X1( X1& , int = 0 ) { } };
|
|
struct X2 { X1<int> result; };
|
|
void test_X2()
|
|
{
|
|
typedef X2 impl;
|
|
typedef X0<impl> pimpl;
|
|
impl* i;
|
|
pimpl pdata;
|
|
pdata.f0( new impl(*i));
|
|
}
|
|
|
|
// rdar://problem/9598341
|
|
namespace test3 {
|
|
struct A { A(const A&); A&operator=(const A&); };
|
|
struct B { A a; unsigned : 0; };
|
|
void test(const B &x) {
|
|
B y = x;
|
|
y = x;
|
|
}
|
|
}
|
|
|
|
namespace test4 {
|
|
// When determining whether to implement an array copy as a memcpy, look at
|
|
// whether the *selected* constructor is trivial.
|
|
struct S {
|
|
int arr[5][5];
|
|
S(S &);
|
|
S(const S &) = default;
|
|
};
|
|
// CHECK: @_ZN5test42f1
|
|
void f1(S a) {
|
|
// CHECK-NOT: memcpy
|
|
// CHECK: call void @_ZN5test41SC1ERS0_
|
|
// CHECK-NOT: memcpy
|
|
S b(a);
|
|
// CHECK: }
|
|
}
|
|
// CHECK: @_ZN5test42f2
|
|
void f2(const S a) {
|
|
// CHECK-NOT: call void @_ZN5test41SC1ERS0_
|
|
// CHECK: memcpy
|
|
// CHECK-NOT: call void @_ZN5test41SC1ERS0_
|
|
S b(a);
|
|
// CHECK: }
|
|
}
|
|
}
|