1 | // RUN: %clang_cc1 -std=c++98 %s -triple x86_64-apple-darwin10 -emit-llvm -o - | FileCheck %s --check-prefix=CHECK --check-prefix=CHECK-CXX98 |
2 | // RUN: %clang_cc1 -std=c++17 %s -triple x86_64-apple-darwin10 -emit-llvm -o - | FileCheck %s --check-prefix=CHECK --check-prefix=CHECK-CXX17 |
3 | |
4 | struct A { |
5 | virtual ~A(); |
6 | }; |
7 | |
8 | struct B : A { }; |
9 | |
10 | struct C { |
11 | int i; |
12 | B b; |
13 | }; |
14 | |
15 | // CHECK: _Z15test_value_initv |
16 | void test_value_init() { |
17 | // This value initialization requires zero initialization of the 'B' |
18 | // subobject followed by a call to its constructor. |
19 | // PR5800 |
20 | |
21 | // CHECK: store i32 17 |
22 | // CHECK: call void @llvm.memset.p0i8.i64 |
23 | // CHECK: call void @_ZN1BC1Ev |
24 | C c = { 17 } ; |
25 | // CHECK: call void @_ZN1CD1Ev |
26 | } |
27 | |
28 | enum enum_type { negative_number = -1, magic_number = 42 }; |
29 | |
30 | class enum_holder |
31 | { |
32 | enum_type m_enum; |
33 | |
34 | public: |
35 | enum_holder() : m_enum(magic_number) { } |
36 | }; |
37 | |
38 | struct enum_holder_and_int |
39 | { |
40 | enum_holder e; |
41 | int i; |
42 | }; |
43 | |
44 | // CHECK: _Z24test_enum_holder_and_intv() |
45 | void test_enum_holder_and_int() { |
46 | // CHECK: alloca |
47 | // CHECK-NEXT: bitcast |
48 | // CHECK-NEXT: call void @llvm.memset |
49 | // CHECK-NEXT: call void @_ZN19enum_holder_and_intC1Ev |
50 | enum_holder_and_int(); |
51 | // CHECK-NEXT: ret void |
52 | } |
53 | |
54 | // PR7834: don't crash. |
55 | namespace test1 { |
56 | struct A { |
57 | int A::*f; |
58 | A(); |
59 | A(const A&); |
60 | A &operator=(const A &); |
61 | }; |
62 | |
63 | struct B { |
64 | A base; |
65 | }; |
66 | |
67 | void foo() { |
68 | B(); |
69 | } |
70 | } |
71 | |
72 | namespace ptrmem { |
73 | struct S { |
74 | int mem1; |
75 | int S::*mem2; |
76 | }; |
77 | |
78 | // CHECK-LABEL: define i32 @_ZN6ptrmem4testEPNS_1SE |
79 | int test(S *s) { |
80 | // CHECK: call void @llvm.memcpy.p0i8.p0i8.i64 |
81 | // CHECK: getelementptr |
82 | // CHECK: ret |
83 | return s->*S().mem2; |
84 | } |
85 | } |
86 | |
87 | namespace PR9801 { |
88 | |
89 | struct Test { |
90 | Test() : i(10) {} |
91 | Test(int i) : i(i) {} |
92 | int i; |
93 | private: |
94 | int j; |
95 | }; |
96 | |
97 | struct Test2 { |
98 | Test t; |
99 | }; |
100 | |
101 | struct Test3 : public Test { }; |
102 | |
103 | // CHECK-LABEL: define void @_ZN6PR98011fEv |
104 | void f() { |
105 | // CHECK-NOT: call void @llvm.memset.p0i8.i64 |
106 | // CHECK: call void @_ZN6PR98014TestC1Ei |
107 | // CHECK-NOT: call void @llvm.memset.p0i8.i64 |
108 | // CHECK: call void @_ZN6PR98014TestC1Ev |
109 | Test partial[3] = { 1 }; |
110 | |
111 | // CHECK-NOT: call void @llvm.memset.p0i8.i64 |
112 | // CHECK: call void @_ZN6PR98014TestC1Ev |
113 | // CHECK-NOT: call void @_ZN6PR98014TestC1Ev |
114 | Test empty[3] = {}; |
115 | |
116 | // CHECK: call void @llvm.memset.p0i8.i64 |
117 | // CHECK-NOT: call void @llvm.memset.p0i8.i64 |
118 | // CHECK-CXX98: call void @_ZN6PR98015Test2C1Ev |
119 | // CHECK-CXX17: call void @_ZN6PR98014TestC1Ev |
120 | // CHECK-NOT: call void @_ZN6PR98015Test2C1Ev |
121 | Test2 empty2[3] = {}; |
122 | |
123 | // CHECK: call void @llvm.memset.p0i8.i64 |
124 | // CHECK-NOT: call void @llvm.memset.p0i8.i64 |
125 | // CHECK-CXX98: call void @_ZN6PR98015Test3C1Ev |
126 | // CHECK-CXX17: call void @_ZN6PR98014TestC2Ev |
127 | // CHECK-NOT: call void @llvm.memset.p0i8.i64 |
128 | // CHECK-NOT: call void @_ZN6PR98015Test3C1Ev |
129 | Test3 empty3[3] = {}; |
130 | } |
131 | |
132 | } |
133 | |
134 | namespace zeroinit { |
135 | struct S { int i; }; |
136 | |
137 | // CHECK-LABEL: define i32 @_ZN8zeroinit4testEv() |
138 | int test() { |
139 | // CHECK: call void @llvm.memset.p0i8.i64 |
140 | // CHECK: ret i32 0 |
141 | return S().i; |
142 | } |
143 | |
144 | struct X0 { |
145 | X0() { } |
146 | int x; |
147 | }; |
148 | |
149 | struct X1 : X0 { |
150 | int x1; |
151 | void f(); |
152 | }; |
153 | |
154 | // CHECK-LABEL: define void @_ZN8zeroinit9testX0_X1Ev |
155 | void testX0_X1() { |
156 | // CHECK: call void @llvm.memset.p0i8.i64 |
157 | // CHECK-NEXT: call void @_ZN8zeroinit2X1C1Ev |
158 | // CHECK-NEXT: call void @_ZN8zeroinit2X11fEv |
159 | X1().f(); |
160 | } |
161 | |
162 | template<typename> |
163 | struct X2 : X0 { |
164 | int x2; |
165 | void f(); |
166 | }; |
167 | |
168 | template<typename> |
169 | struct X3 : X2<int> { |
170 | X3() : X2<int>() { } |
171 | int i; |
172 | }; |
173 | |
174 | |
175 | // CHECK-LABEL: define void @_ZN8zeroinit9testX0_X3Ev |
176 | void testX0_X3() { |
177 | // CHECK-NOT: call void @llvm.memset |
178 | // CHECK: call void @_ZN8zeroinit2X3IiEC1Ev |
179 | // CHECK: call void @_ZN8zeroinit2X2IiE1fEv |
180 | // CHECK-NEXT: ret void |
181 | X3<int>().f(); |
182 | } |
183 | |
184 | // More checks at EOF |
185 | } |
186 | |
187 | namespace PR8726 { |
188 | class C; |
189 | struct S { |
190 | const C &c1; |
191 | int i; |
192 | const C &c2; |
193 | }; |
194 | void f(const C& c) { |
195 | S s = {c, 42, c}; |
196 | } |
197 | |
198 | } |
199 | |
200 | // rdar://problem/9355931 |
201 | namespace test6 { |
202 | struct A { A(); A(int); }; |
203 | |
204 | void test() { |
205 | A arr[10][20] = { 5 }; |
206 | }; |
207 | // CHECK-LABEL: define void @_ZN5test64testEv() |
208 | // CHECK: [[ARR:%.*]] = alloca [10 x [20 x [[A:%.*]]]], |
209 | |
210 | // CHECK-NEXT: [[INNER:%.*]] = getelementptr inbounds [10 x [20 x [[A]]]], [10 x [20 x [[A]]]]* [[ARR]], i64 0, i64 0 |
211 | // CHECK-NEXT: [[T0:%.*]] = getelementptr inbounds [20 x [[A]]], [20 x [[A]]]* [[INNER]], i64 0, i64 0 |
212 | // CHECK-NEXT: call void @_ZN5test61AC1Ei([[A]]* [[T0]], i32 5) |
213 | // CHECK-NEXT: [[BEGIN:%.*]] = getelementptr inbounds [[A]], [[A]]* [[T0]], i64 1 |
214 | // CHECK-NEXT: [[END:%.*]] = getelementptr inbounds [[A]], [[A]]* [[T0]], i64 20 |
215 | // CHECK-NEXT: br label |
216 | // CHECK: [[CUR:%.*]] = phi [[A]]* [ [[BEGIN]], {{%.*}} ], [ [[NEXT:%.*]], {{%.*}} ] |
217 | // CHECK-NEXT: call void @_ZN5test61AC1Ev([[A]]* [[CUR]]) |
218 | // CHECK-NEXT: [[NEXT]] = getelementptr inbounds [[A]], [[A]]* [[CUR]], i64 1 |
219 | // CHECK-NEXT: [[T0:%.*]] = icmp eq [[A]]* [[NEXT]], [[END]] |
220 | // CHECK-NEXT: br i1 |
221 | |
222 | // CHECK: [[BEGIN:%.*]] = getelementptr inbounds [20 x [[A]]], [20 x [[A]]]* [[INNER]], i64 1 |
223 | // CHECK-NEXT: [[END:%.*]] = getelementptr inbounds [20 x [[A]]], [20 x [[A]]]* [[INNER]], i64 10 |
224 | // CHECK-NEXT: br label |
225 | // CHECK: [[CUR:%.*]] = phi [20 x [[A]]]* [ [[BEGIN]], {{%.*}} ], [ [[NEXT:%.*]], {{%.*}} ] |
226 | |
227 | // Inner loop. |
228 | // CHECK-NEXT: [[IBEGIN:%.*]] = getelementptr inbounds [20 x [[A]]], [20 x [[A]]]* [[CUR]], i{{32|64}} 0, i{{32|64}} 0 |
229 | // CHECK-NEXT: [[IEND:%.*]] = getelementptr inbounds [[A]], [[A]]* [[IBEGIN]], i64 20 |
230 | // CHECK-NEXT: br label |
231 | // CHECK: [[ICUR:%.*]] = phi [[A]]* [ [[IBEGIN]], {{%.*}} ], [ [[INEXT:%.*]], {{%.*}} ] |
232 | // CHECK-NEXT: call void @_ZN5test61AC1Ev([[A]]* [[ICUR]]) |
233 | // CHECK-NEXT: [[INEXT:%.*]] = getelementptr inbounds [[A]], [[A]]* [[ICUR]], i64 1 |
234 | // CHECK-NEXT: [[T0:%.*]] = icmp eq [[A]]* [[INEXT]], [[IEND]] |
235 | // CHECK-NEXT: br i1 [[T0]], |
236 | |
237 | // CHECK: [[NEXT]] = getelementptr inbounds [20 x [[A]]], [20 x [[A]]]* [[CUR]], i64 1 |
238 | // CHECK-NEXT: [[T0:%.*]] = icmp eq [20 x [[A]]]* [[NEXT]], [[END]] |
239 | // CHECK-NEXT: br i1 [[T0]] |
240 | // CHECK: ret void |
241 | } |
242 | |
243 | namespace PR11124 { |
244 | // Make sure C::C doesn't overwrite parts of A while it is zero-initializing B |
245 | struct A { int a; A(); A(int); }; |
246 | struct B : virtual A { int b; }; |
247 | struct C : B { C(); }; |
248 | C::C() : A(3), B() {} |
249 | // CHECK-LABEL: define void @_ZN7PR111241CC1Ev |
250 | // CHECK: call void @llvm.memset.p0i8.i64(i8* align 8 {{.*}}, i8 0, i64 12, i1 false) |
251 | // CHECK-NEXT: call void @_ZN7PR111241BC2Ev |
252 | // Make sure C::C doesn't overwrite parts of A while it is zero-initializing B |
253 | |
254 | struct B2 : virtual A { int B::*b; }; |
255 | struct C2 : B2 { C2(); }; |
256 | C2::C2() : A(3), B2() {} |
257 | // CHECK-LABEL: define void @_ZN7PR111242C2C1Ev |
258 | // CHECK: call void @llvm.memcpy.p0i8.p0i8.i64(i8* align 8 %{{.*}}, i8* align 8 {{.*}}, i64 16, i1 false) |
259 | // CHECK-NEXT: call void @_ZN7PR111242B2C2Ev |
260 | } |
261 | |
262 | // Ensure we produce an i1 here, and don't assert. |
263 | // CHECK-LABEL: define void @_Z9r170806_bv( |
264 | // CHECK: call void @_Z9r170806_ab(i1 zeroext false) |
265 | void r170806_a(bool b = bool()); |
266 | void r170806_b() { r170806_a(); } |
267 | |
268 | namespace PR20256 { |
269 | struct data { int i; }; |
270 | |
271 | template<typename T = int> |
272 | data g() { |
273 | data d; // not value-init |
274 | return d; |
275 | } |
276 | template data g(); |
277 | // CHECK-LABEL: define {{.*}} @_ZN7PR202561gIiEENS_4dataEv( |
278 | // CHECK-NOT: store |
279 | // CHECK-NOT: memset |
280 | // CHECK: } |
281 | |
282 | template<typename ...T> |
283 | data h(T ...t) { |
284 | data d(t...); // value-init |
285 | return d; |
286 | } |
287 | template data h(); |
288 | // CHECK-LABEL: define {{.*}} @_ZN7PR202561hIJEEENS_4dataEDpT_( |
289 | // CHECK: call void @llvm.memset |
290 | // CHECK: } |
291 | |
292 | |
293 | template<typename T = int> |
294 | data j() { |
295 | data d = {}; // value-init |
296 | return d; |
297 | } |
298 | template data j(); |
299 | // CHECK-LABEL: define {{.*}} @_ZN7PR202561jIiEENS_4dataEv( |
300 | // CHECK: call void @llvm.memset |
301 | // CHECK: } |
302 | |
303 | data f() { |
304 | data d; // not value-init |
305 | return d; |
306 | } |
307 | // CHECK-LABEL: define {{.*}} @_ZN7PR202561fEv( |
308 | // CHECK-NOT: store |
309 | // CHECK-NOT: memset |
310 | // CHECK: } |
311 | |
312 | data i() { |
313 | data d = {}; // value-init |
314 | return d; |
315 | } |
316 | // CHECK-LABEL: define {{.*}} @_ZN7PR202561iEv( |
317 | // CHECK: call void @llvm.memset |
318 | // CHECK: } |
319 | } |
320 | |
321 | // CHECK-LABEL: define {{.*}}@_Z20explicitly_defaultedv |
322 | int explicitly_defaulted() { |
323 | struct A { A() = default; int n; }; |
324 | // CHECK: call void @llvm.memset |
325 | A a = A(); |
326 | return a.n; |
327 | } // CHECK-LABEL: } |
328 | |
329 | // CHECK-LABEL: define linkonce_odr void @_ZN8zeroinit2X3IiEC2Ev(%"struct.zeroinit::X3"* %this) unnamed_addr |
330 | // CHECK: call void @llvm.memset.p0i8.i64 |
331 | // CHECK-NEXT: call void @_ZN8zeroinit2X2IiEC2Ev |
332 | // CHECK-NEXT: ret void |
333 | |
334 | #if __cplusplus >= 201103L |
335 | namespace transparent_init_list { |
336 | struct optional_assign_base {}; |
337 | struct optional_data_dtor_base { char dummy_[24]; }; |
338 | struct optional : optional_data_dtor_base, optional_assign_base {}; |
339 | optional f(optional a) { return {optional(a)}; } |
340 | } |
341 | #endif |
342 | |