xref: /aosp_15_r20/external/clang/test/CodeGenCXX/constructor-init.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li // RUN: %clang_cc1 -std=c++11 -triple x86_64-apple-darwin10 %s -emit-llvm -o %t
2*67e74705SXin Li // RUN: FileCheck %s < %t
3*67e74705SXin Li // RUN: FileCheck -check-prefix=CHECK-PR10720 %s < %t
4*67e74705SXin Li 
5*67e74705SXin Li extern "C" int printf(...);
6*67e74705SXin Li 
7*67e74705SXin Li struct M {
MM8*67e74705SXin Li   M() { printf("M()\n"); }
MM9*67e74705SXin Li   M(int i) { iM = i; printf("M(%d)\n", i); }
10*67e74705SXin Li   int iM;
MPRM11*67e74705SXin Li   void MPR() {printf("iM = %d\n", iM); };
12*67e74705SXin Li };
13*67e74705SXin Li 
14*67e74705SXin Li struct P {
PP15*67e74705SXin Li   P() { printf("P()\n"); }
PP16*67e74705SXin Li   P(int i) { iP = i; printf("P(%d)\n", i); }
17*67e74705SXin Li   int iP;
PPRP18*67e74705SXin Li   void PPR() {printf("iP = %d\n", iP); };
19*67e74705SXin Li };
20*67e74705SXin Li 
21*67e74705SXin Li struct Q {
QQ22*67e74705SXin Li   Q() { printf("Q()\n"); }
QQ23*67e74705SXin Li   Q(int i) { iQ = i; printf("Q(%d)\n", i); }
24*67e74705SXin Li   int iQ;
QPRQ25*67e74705SXin Li   void QPR() {printf("iQ = %d\n", iQ); };
26*67e74705SXin Li };
27*67e74705SXin Li 
28*67e74705SXin Li struct N : M , P, Q {
NN29*67e74705SXin Li   N() : f1(1.314), P(2000), ld(00.1234+f1), M(1000), Q(3000),
30*67e74705SXin Li         d1(3.4567), i1(1234), m1(100) { printf("N()\n"); }
31*67e74705SXin Li   M m1;
32*67e74705SXin Li   M m2;
33*67e74705SXin Li   float f1;
34*67e74705SXin Li   int i1;
35*67e74705SXin Li   float d1;
PRN36*67e74705SXin Li   void PR() {
37*67e74705SXin Li     printf("f1 = %f d1 = %f i1 = %d ld = %f \n", f1,d1,i1, ld);
38*67e74705SXin Li     MPR();
39*67e74705SXin Li     PPR();
40*67e74705SXin Li     QPR();
41*67e74705SXin Li     printf("iQ = %d\n", iQ);
42*67e74705SXin Li     printf("iP = %d\n", iP);
43*67e74705SXin Li     printf("iM = %d\n", iM);
44*67e74705SXin Li     // FIXME. We don't yet support this syntax.
45*67e74705SXin Li     // printf("iQ = %d\n", (*this).iQ);
46*67e74705SXin Li     printf("iQ = %d\n", this->iQ);
47*67e74705SXin Li     printf("iP = %d\n", this->iP);
48*67e74705SXin Li     printf("iM = %d\n", this->iM);
49*67e74705SXin Li   }
50*67e74705SXin Li   float ld;
51*67e74705SXin Li   float ff;
52*67e74705SXin Li   M arr_m[3];
53*67e74705SXin Li   P arr_p[1][3];
54*67e74705SXin Li   Q arr_q[2][3][4];
55*67e74705SXin Li };
56*67e74705SXin Li 
main()57*67e74705SXin Li int main() {
58*67e74705SXin Li   M m1;
59*67e74705SXin Li 
60*67e74705SXin Li   N n1;
61*67e74705SXin Li   n1.PR();
62*67e74705SXin Li }
63*67e74705SXin Li 
64*67e74705SXin Li // PR5826
65*67e74705SXin Li template <class T> struct A {
AA66*67e74705SXin Li   A() {}
AA67*67e74705SXin Li   A(int) {}
AA68*67e74705SXin Li   A(const A&) {}
~AA69*67e74705SXin Li   ~A() {}
operator intA70*67e74705SXin Li   operator int() {return 0;}
71*67e74705SXin Li };
72*67e74705SXin Li 
73*67e74705SXin Li // CHECK-LABEL: define void @_Z1fv()
f()74*67e74705SXin Li void f() {
75*67e74705SXin Li   // CHECK: call void @_ZN1AIsEC1Ei
76*67e74705SXin Li   A<short> a4 = 97;
77*67e74705SXin Li 
78*67e74705SXin Li   // CHECK-NEXT: store i32 17
79*67e74705SXin Li   int i = 17;
80*67e74705SXin Li 
81*67e74705SXin Li   // CHECK-NEXT: call void @_ZN1AIsED1Ev
82*67e74705SXin Li   // CHECK-NOT: call void @_ZN1AIsED1Ev
83*67e74705SXin Li   // CHECK: ret void
84*67e74705SXin Li }
85*67e74705SXin Li 
86*67e74705SXin Li // Make sure we initialize the vtable pointer if it's required by a
87*67e74705SXin Li // base initializer.
88*67e74705SXin Li namespace InitVTable {
89*67e74705SXin Li   struct A { A(int); };
90*67e74705SXin Li   struct B : A {
91*67e74705SXin Li     virtual int foo();
92*67e74705SXin Li     B();
93*67e74705SXin Li     B(int);
94*67e74705SXin Li   };
95*67e74705SXin Li 
96*67e74705SXin Li   // CHECK-LABEL: define void @_ZN10InitVTable1BC2Ev(%"struct.InitVTable::B"* %this) unnamed_addr
97*67e74705SXin Li   // CHECK:      [[T0:%.*]] = bitcast [[B:%.*]]* [[THIS:%.*]] to i32 (...)***
98*67e74705SXin Li   // CHECK-NEXT: store i32 (...)** bitcast (i8** getelementptr inbounds ([3 x i8*], [3 x i8*]* @_ZTVN10InitVTable1BE, i32 0, i32 2) to i32 (...)**), i32 (...)*** [[T0]]
99*67e74705SXin Li   // CHECK:      [[VTBL:%.*]] = load i32 ([[B]]*)**, i32 ([[B]]*)*** {{%.*}}
100*67e74705SXin Li   // CHECK-NEXT: [[FNP:%.*]] = getelementptr inbounds i32 ([[B]]*)*, i32 ([[B]]*)** [[VTBL]], i64 0
101*67e74705SXin Li   // CHECK-NEXT: [[FN:%.*]] = load i32 ([[B]]*)*, i32 ([[B]]*)** [[FNP]]
102*67e74705SXin Li   // CHECK-NEXT: [[ARG:%.*]] = call i32 [[FN]]([[B]]* [[THIS]])
103*67e74705SXin Li   // CHECK-NEXT: call void @_ZN10InitVTable1AC2Ei({{.*}}* {{%.*}}, i32 [[ARG]])
104*67e74705SXin Li   // CHECK-NEXT: [[T0:%.*]] = bitcast [[B]]* [[THIS]] to i32 (...)***
105*67e74705SXin Li   // CHECK-NEXT: store i32 (...)** bitcast (i8** getelementptr inbounds ([3 x i8*], [3 x i8*]* @_ZTVN10InitVTable1BE, i32 0, i32 2) to i32 (...)**), i32 (...)*** [[T0]]
106*67e74705SXin Li   // CHECK-NEXT: ret void
B()107*67e74705SXin Li   B::B() : A(foo()) {}
108*67e74705SXin Li 
109*67e74705SXin Li   // CHECK-LABEL: define void @_ZN10InitVTable1BC2Ei(%"struct.InitVTable::B"* %this, i32 %x) unnamed_addr
110*67e74705SXin Li   // CHECK:      [[ARG:%.*]] = add nsw i32 {{%.*}}, 5
111*67e74705SXin Li   // CHECK-NEXT: call void @_ZN10InitVTable1AC2Ei({{.*}}* {{%.*}}, i32 [[ARG]])
112*67e74705SXin Li   // CHECK-NEXT: [[T0:%.*]] = bitcast [[B]]* {{%.*}} to i32 (...)***
113*67e74705SXin Li   // CHECK-NEXT: store i32 (...)** bitcast (i8** getelementptr inbounds ([3 x i8*], [3 x i8*]* @_ZTVN10InitVTable1BE, i32 0, i32 2) to i32 (...)**), i32 (...)*** [[T0]]
114*67e74705SXin Li   // CHECK-NEXT: ret void
B(int x)115*67e74705SXin Li   B::B(int x) : A(x + 5) {}
116*67e74705SXin Li }
117*67e74705SXin Li 
118*67e74705SXin Li namespace rdar9694300 {
119*67e74705SXin Li   struct X {
120*67e74705SXin Li     int x;
121*67e74705SXin Li   };
122*67e74705SXin Li 
123*67e74705SXin Li   // CHECK-LABEL: define void @_ZN11rdar96943001fEv
f()124*67e74705SXin Li   void f() {
125*67e74705SXin Li     // CHECK: alloca
126*67e74705SXin Li     X x;
127*67e74705SXin Li     // CHECK-NEXT: [[I:%.*]] = alloca i32
128*67e74705SXin Li     // CHECK-NEXT: store i32 17, i32* [[I]]
129*67e74705SXin Li     int i = 17;
130*67e74705SXin Li     // CHECK-NEXT: ret void
131*67e74705SXin Li   }
132*67e74705SXin Li }
133*67e74705SXin Li 
134*67e74705SXin Li // Check that we emit a zero initialization step for list-value-initialization
135*67e74705SXin Li // which calls a trivial default constructor.
136*67e74705SXin Li namespace PR13273 {
137*67e74705SXin Li   struct U {
138*67e74705SXin Li     int t;
139*67e74705SXin Li     U() = default;
140*67e74705SXin Li   };
141*67e74705SXin Li 
142*67e74705SXin Li   struct S : U {
143*67e74705SXin Li     S() = default;
144*67e74705SXin Li   };
145*67e74705SXin Li 
146*67e74705SXin Li   // CHECK: define {{.*}}@_ZN7PR132731fEv(
f()147*67e74705SXin Li   int f() {
148*67e74705SXin Li     // CHECK-NOT: }
149*67e74705SXin Li     // CHECK: llvm.memset{{.*}}i8 0
150*67e74705SXin Li     return (new S{})->t;
151*67e74705SXin Li   }
152*67e74705SXin Li }
153*67e74705SXin Li 
154*67e74705SXin Li template<typename T>
155*67e74705SXin Li struct X {
156*67e74705SXin Li   X(const X &);
157*67e74705SXin Li 
158*67e74705SXin Li   T *start;
159*67e74705SXin Li   T *end;
160*67e74705SXin Li };
161*67e74705SXin Li 
162*67e74705SXin Li template<typename T> struct X;
163*67e74705SXin Li 
164*67e74705SXin Li // Make sure that the instantiated constructor initializes start and
165*67e74705SXin Li // end properly.
166*67e74705SXin Li // CHECK-LABEL: define linkonce_odr void @_ZN1XIiEC2ERKS0_(%struct.X* %this, %struct.X* dereferenceable({{[0-9]+}}) %other) unnamed_addr
167*67e74705SXin Li // CHECK: {{store.*null}}
168*67e74705SXin Li // CHECK: {{store.*null}}
169*67e74705SXin Li // CHECK: ret
170*67e74705SXin Li template<typename T>
X(const X & other)171*67e74705SXin Li X<T>::X(const X &other) : start(0), end(0) { }
172*67e74705SXin Li 
get_X(X<int> x)173*67e74705SXin Li X<int> get_X(X<int> x) { return x; }
174*67e74705SXin Li 
175*67e74705SXin Li namespace PR10720 {
176*67e74705SXin Li   struct X {
177*67e74705SXin Li     X(const X&);
178*67e74705SXin Li     X(X&&);
179*67e74705SXin Li     X& operator=(const X&);
180*67e74705SXin Li     X& operator=(X&&);
181*67e74705SXin Li     ~X();
182*67e74705SXin Li   };
183*67e74705SXin Li 
184*67e74705SXin Li   struct pair2 {
185*67e74705SXin Li     X second[4];
186*67e74705SXin Li 
187*67e74705SXin Li     // CHECK-PR10720: define linkonce_odr {{.*}} @_ZN7PR107205pair2aSERKS0_
188*67e74705SXin Li     // CHECK-PR10720: load
189*67e74705SXin Li     // CHECK-PR10720: icmp ne
190*67e74705SXin Li     // CHECK-PR10720-NEXT: br i1
191*67e74705SXin Li     // CHECK-PR10720: call {{.*}} @_ZN7PR107201XaSERKS0_
192*67e74705SXin Li     // CHECK-PR10720: ret
193*67e74705SXin Li     pair2 &operator=(const pair2&) = default;
194*67e74705SXin Li 
195*67e74705SXin Li     // CHECK-PR10720: define linkonce_odr {{.*}} @_ZN7PR107205pair2aSEOS0_
196*67e74705SXin Li     // CHECK-PR10720: load
197*67e74705SXin Li     // CHECK-PR10720: icmp ne
198*67e74705SXin Li     // CHECK-PR10720-NEXT: br i1
199*67e74705SXin Li     // CHECK-PR10720: call {{.*}} @_ZN7PR107201XaSEOS0_
200*67e74705SXin Li     // CHECK-PR10720: ret
201*67e74705SXin Li     pair2 &operator=(pair2&&) = default;
202*67e74705SXin Li 
203*67e74705SXin Li     // CHECK-PR10720-LABEL: define linkonce_odr void @_ZN7PR107204pairC2ERKS0_
204*67e74705SXin Li     // CHECK-PR10720-NOT: ret
205*67e74705SXin Li     // CHECK-PR10720: call void @llvm.memcpy
206*67e74705SXin Li     // CHECK-PR10720-NEXT: ret void
207*67e74705SXin Li 
208*67e74705SXin Li     // CHECK-PR10720-LABEL: define linkonce_odr void @_ZN7PR107205pair2C2ERKS0_
209*67e74705SXin Li     // CHECK-PR10720-NOT: ret
210*67e74705SXin Li     // CHECK-PR10720: load
211*67e74705SXin Li     // CHECK-PR10720: icmp ult
212*67e74705SXin Li     // CHECK-PR10720-NEXT: br i1
213*67e74705SXin Li     // CHECK-PR10720: call void @_ZN7PR107201XC1ERKS0_
214*67e74705SXin Li     // CHECK-PR10720-NEXT: br label
215*67e74705SXin Li     // CHECK-PR10720: ret void
216*67e74705SXin Li 
217*67e74705SXin Li     // CHECK-PR10720-LABEL: define linkonce_odr void @_ZN7PR107205pair2C2EOS0_
218*67e74705SXin Li     // CHECK-PR10720-NOT: ret
219*67e74705SXin Li     // CHECK-PR10720: load
220*67e74705SXin Li     // CHECK-PR10720: icmp ult
221*67e74705SXin Li     // CHECK-PR10720-NEXT: br i1
222*67e74705SXin Li     // CHECK-PR10720: call void @_ZN7PR107201XC1EOS0_
223*67e74705SXin Li     // CHECK-PR10720-NEXT: br label
224*67e74705SXin Li     // CHECK-PR10720: ret void
225*67e74705SXin Li     pair2(pair2&&) = default;
226*67e74705SXin Li 
227*67e74705SXin Li     pair2(const pair2&) = default;
228*67e74705SXin Li   };
229*67e74705SXin Li 
230*67e74705SXin Li   struct pair : X { // Make the copy constructor non-trivial, so we actually generate it.
231*67e74705SXin Li     int second[4];
232*67e74705SXin Li     pair(const pair&) = default;
233*67e74705SXin Li   };
234*67e74705SXin Li 
foo(const pair & x,const pair2 & x2)235*67e74705SXin Li   void foo(const pair &x, const pair2 &x2) {
236*67e74705SXin Li     pair y(x);
237*67e74705SXin Li     pair2 y2(x2);
238*67e74705SXin Li     pair2 y2m(static_cast<pair2&&>(y2));
239*67e74705SXin Li 
240*67e74705SXin Li     y2 = x2;
241*67e74705SXin Li     y2m = static_cast<pair2&&>(y2);
242*67e74705SXin Li   }
243*67e74705SXin Li 
244*67e74705SXin Li }
245