1*67e74705SXin Li // RUN: %clang_cc1 -triple x86_64-apple-darwin -emit-llvm -o - %s | FileCheck %s
2*67e74705SXin Li
3*67e74705SXin Li extern "C" int printf(...);
4*67e74705SXin Li
5*67e74705SXin Li int init = 100;
6*67e74705SXin Li
7*67e74705SXin Li struct M {
8*67e74705SXin Li int iM;
MM9*67e74705SXin Li M() : iM(init++) {}
10*67e74705SXin Li };
11*67e74705SXin Li
12*67e74705SXin Li struct N {
13*67e74705SXin Li int iN;
NN14*67e74705SXin Li N() : iN(200) {}
NN15*67e74705SXin Li N(N const & arg){this->iN = arg.iN; }
16*67e74705SXin Li };
17*67e74705SXin Li
18*67e74705SXin Li struct P {
19*67e74705SXin Li int iP;
PP20*67e74705SXin Li P() : iP(init++) {}
21*67e74705SXin Li };
22*67e74705SXin Li
23*67e74705SXin Li
24*67e74705SXin Li // CHECK-LABEL: define linkonce_odr void @_ZN1XC1ERKS_(%struct.X* %this, %struct.X* dereferenceable({{[0-9]+}})) unnamed_addr
25*67e74705SXin Li struct X : M, N, P { // ...
XX26*67e74705SXin Li X() : f1(1.0), d1(2.0), i1(3), name("HELLO"), bf1(0xff), bf2(0xabcd),
27*67e74705SXin Li au_i1(1234), au1_4("MASKED") {}
28*67e74705SXin Li P p0;
prX29*67e74705SXin Li void pr() {
30*67e74705SXin Li printf("iM = %d iN = %d, m1.iM = %d\n", iM, iN, m1.iM);
31*67e74705SXin Li printf("im = %d p0.iP = %d, p1.iP = %d\n", iP, p0.iP, p1.iP);
32*67e74705SXin Li printf("f1 = %f d1 = %f i1 = %d name(%s) \n", f1, d1, i1, name);
33*67e74705SXin Li printf("bf1 = %x bf2 = %x\n", bf1, bf2);
34*67e74705SXin Li printf("au_i2 = %d\n", au_i2);
35*67e74705SXin Li printf("au1_1 = %s\n", au1_1);
36*67e74705SXin Li }
37*67e74705SXin Li M m1;
38*67e74705SXin Li P p1;
39*67e74705SXin Li float f1;
40*67e74705SXin Li double d1;
41*67e74705SXin Li int i1;
42*67e74705SXin Li const char *name;
43*67e74705SXin Li unsigned bf1 : 8;
44*67e74705SXin Li unsigned bf2 : 16;
45*67e74705SXin Li int arr[2];
46*67e74705SXin Li _Complex float complex;
47*67e74705SXin Li
48*67e74705SXin Li union {
49*67e74705SXin Li int au_i1;
50*67e74705SXin Li int au_i2;
51*67e74705SXin Li };
52*67e74705SXin Li union {
53*67e74705SXin Li const char * au1_1;
54*67e74705SXin Li float au1_2;
55*67e74705SXin Li int au1_3;
56*67e74705SXin Li const char * au1_4;
57*67e74705SXin Li };
58*67e74705SXin Li };
59*67e74705SXin Li
60*67e74705SXin Li static int ix = 1;
61*67e74705SXin Li // class with user-defined copy constructor.
62*67e74705SXin Li struct S {
SS63*67e74705SXin Li S() : iS(ix++) { }
SS64*67e74705SXin Li S(const S& arg) { *this = arg; }
65*67e74705SXin Li int iS;
66*67e74705SXin Li };
67*67e74705SXin Li
68*67e74705SXin Li // class with trivial copy constructor.
69*67e74705SXin Li struct I {
II70*67e74705SXin Li I() : iI(ix++) { }
71*67e74705SXin Li int iI;
72*67e74705SXin Li };
73*67e74705SXin Li
74*67e74705SXin Li struct XM {
XMXM75*67e74705SXin Li XM() { }
76*67e74705SXin Li double dXM;
77*67e74705SXin Li S ARR_S[3][4][2];
prXM78*67e74705SXin Li void pr() {
79*67e74705SXin Li for (unsigned i = 0; i < 3; i++)
80*67e74705SXin Li for (unsigned j = 0; j < 4; j++)
81*67e74705SXin Li for (unsigned k = 0; k < 2; k++)
82*67e74705SXin Li printf("ARR_S[%d][%d][%d] = %d\n", i,j,k, ARR_S[i][j][k].iS);
83*67e74705SXin Li for (unsigned i = 0; i < 3; i++)
84*67e74705SXin Li for (unsigned k = 0; k < 2; k++)
85*67e74705SXin Li printf("ARR_I[%d][%d] = %d\n", i,k, ARR_I[i][k].iI);
86*67e74705SXin Li }
87*67e74705SXin Li I ARR_I[3][2];
88*67e74705SXin Li };
89*67e74705SXin Li
main()90*67e74705SXin Li int main() {
91*67e74705SXin Li X a;
92*67e74705SXin Li X b(a);
93*67e74705SXin Li b.pr();
94*67e74705SXin Li X x;
95*67e74705SXin Li X c(x);
96*67e74705SXin Li c.pr();
97*67e74705SXin Li
98*67e74705SXin Li XM m0;
99*67e74705SXin Li XM m1 = m0;
100*67e74705SXin Li m1.pr();
101*67e74705SXin Li }
102*67e74705SXin Li
103*67e74705SXin Li struct A {
104*67e74705SXin Li };
105*67e74705SXin Li
106*67e74705SXin Li struct B : A {
107*67e74705SXin Li A &a;
108*67e74705SXin Li };
109*67e74705SXin Li
f(const B & b1)110*67e74705SXin Li void f(const B &b1) {
111*67e74705SXin Li B b2(b1);
112*67e74705SXin Li }
113*67e74705SXin Li
114*67e74705SXin Li // PR6628
115*67e74705SXin Li namespace PR6628 {
116*67e74705SXin Li
117*67e74705SXin Li struct T {
118*67e74705SXin Li T();
119*67e74705SXin Li ~T();
120*67e74705SXin Li
121*67e74705SXin Li double d;
122*67e74705SXin Li };
123*67e74705SXin Li
124*67e74705SXin Li struct A {
125*67e74705SXin Li A(const A &other, const T &t = T(), const T& t2 = T());
126*67e74705SXin Li };
127*67e74705SXin Li
128*67e74705SXin Li struct B : A {
129*67e74705SXin Li A a1;
130*67e74705SXin Li A a2;
131*67e74705SXin Li A a[10];
132*67e74705SXin Li };
133*67e74705SXin Li
134*67e74705SXin Li // Force the copy constructor to be synthesized.
f(B b1)135*67e74705SXin Li void f(B b1) {
136*67e74705SXin Li B b2 = b1;
137*67e74705SXin Li }
138*67e74705SXin Li
139*67e74705SXin Li // CHECK: define linkonce_odr dereferenceable({{[0-9]+}}) [[A:%.*]]* @_ZN12rdar138169401AaSERKS0_(
140*67e74705SXin Li // CHECK: [[THIS:%.*]] = load [[A]]*, [[A]]**
141*67e74705SXin Li // CHECK-NEXT: [[T0:%.*]] = getelementptr inbounds [[A]], [[A]]* [[THIS]], i32 0, i32 1
142*67e74705SXin Li // CHECK-NEXT: [[OTHER:%.*]] = load [[A]]*, [[A]]**
143*67e74705SXin Li // CHECK-NEXT: [[T2:%.*]] = getelementptr inbounds [[A]], [[A]]* [[OTHER]], i32 0, i32 1
144*67e74705SXin Li // CHECK-NEXT: [[T4:%.*]] = bitcast i16* [[T0]] to i8*
145*67e74705SXin Li // CHECK-NEXT: [[T5:%.*]] = bitcast i16* [[T2]] to i8*
146*67e74705SXin Li // CHECK-NEXT: call void @llvm.memcpy.p0i8.p0i8.i64(i8* [[T4]], i8* [[T5]], i64 8, i32 8, i1 false)
147*67e74705SXin Li // CHECK-NEXT: ret [[A]]* [[THIS]]
148*67e74705SXin Li
149*67e74705SXin Li // CHECK-LABEL: define linkonce_odr void @_ZN6PR66281BC2ERKS0_(%"struct.PR6628::B"* %this, %"struct.PR6628::B"* dereferenceable({{[0-9]+}})) unnamed_addr
150*67e74705SXin Li // CHECK: call void @_ZN6PR66281TC1Ev
151*67e74705SXin Li // CHECK: call void @_ZN6PR66281TC1Ev
152*67e74705SXin Li // CHECK: call void @_ZN6PR66281AC2ERKS0_RKNS_1TES5_
153*67e74705SXin Li // CHECK: call void @_ZN6PR66281TD1Ev
154*67e74705SXin Li // CHECK: call void @_ZN6PR66281TD1Ev
155*67e74705SXin Li // CHECK: call void @_ZN6PR66281TC1Ev
156*67e74705SXin Li // CHECK: call void @_ZN6PR66281TC1Ev
157*67e74705SXin Li // CHECK: call void @_ZN6PR66281AC1ERKS0_RKNS_1TES5_
158*67e74705SXin Li // CHECK: call void @_ZN6PR66281TD1Ev
159*67e74705SXin Li // CHECK: call void @_ZN6PR66281TD1Ev
160*67e74705SXin Li // CHECK: call void @_ZN6PR66281TC1Ev
161*67e74705SXin Li // CHECK: call void @_ZN6PR66281TC1Ev
162*67e74705SXin Li // CHECK: call void @_ZN6PR66281AC1ERKS0_RKNS_1TES5_
163*67e74705SXin Li // CHECK: call void @_ZN6PR66281TD1Ev
164*67e74705SXin Li // CHECK: call void @_ZN6PR66281TD1Ev
165*67e74705SXin Li
166*67e74705SXin Li // CHECK-LABEL: define linkonce_odr void @_ZN12rdar138169401AC2ERKS0_(
167*67e74705SXin Li // CHECK: [[THIS:%.*]] = load [[A]]*, [[A]]**
168*67e74705SXin Li // CHECK-NEXT: [[T0:%.*]] = bitcast [[A]]* [[THIS]] to i32 (...)***
169*67e74705SXin Li // CHECK-NEXT: store i32 (...)** bitcast (i8** getelementptr inbounds ([4 x i8*], [4 x i8*]* @_ZTVN12rdar138169401AE, i32 0, i32 2) to i32 (...)**), i32 (...)*** [[T0]]
170*67e74705SXin Li // CHECK-NEXT: [[T0:%.*]] = getelementptr inbounds [[A]], [[A]]* [[THIS]], i32 0, i32 1
171*67e74705SXin Li // CHECK-NEXT: [[OTHER:%.*]] = load [[A]]*, [[A]]**
172*67e74705SXin Li // CHECK-NEXT: [[T2:%.*]] = getelementptr inbounds [[A]], [[A]]* [[OTHER]], i32 0, i32 1
173*67e74705SXin Li // CHECK-NEXT: [[T4:%.*]] = bitcast i16* [[T0]] to i8*
174*67e74705SXin Li // CHECK-NEXT: [[T5:%.*]] = bitcast i16* [[T2]] to i8*
175*67e74705SXin Li // CHECK-NEXT: call void @llvm.memcpy.p0i8.p0i8.i64(i8* [[T4]], i8* [[T5]], i64 8, i32 8, i1 false)
176*67e74705SXin Li // CHECK-NEXT: ret void
177*67e74705SXin Li }
178*67e74705SXin Li
179*67e74705SXin Li // rdar://13816940
180*67e74705SXin Li // Test above because things get weirdly re-ordered.
181*67e74705SXin Li namespace rdar13816940 {
182*67e74705SXin Li struct A {
183*67e74705SXin Li virtual ~A();
184*67e74705SXin Li unsigned short a : 1;
185*67e74705SXin Li unsigned short : 15;
186*67e74705SXin Li unsigned other;
187*67e74705SXin Li };
188*67e74705SXin Li
test(A & a)189*67e74705SXin Li void test(A &a) {
190*67e74705SXin Li A x = a; // force copy constructor into existence
191*67e74705SXin Li x = a; // also force the copy assignment operator
192*67e74705SXin Li }
193*67e74705SXin Li }
194