1*67e74705SXin Li // RUN: %clang_cc1 %s -triple=x86_64-apple-darwin10 -emit-llvm-only -fdump-vtable-layouts > %t
2*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-1 %s < %t
3*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-2 %s < %t
4*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-3 %s < %t
5*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-4 %s < %t
6*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-5 %s < %t
7*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-6 %s < %t
8*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-7 %s < %t
9*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-8 %s < %t
10*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-9 %s < %t
11*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-10 %s < %t
12*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-11 %s < %t
13*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-12 %s < %t
14*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-13 %s < %t
15*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-14 %s < %t
16*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-15 %s < %t
17*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-16 %s < %t
18*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-17 %s < %t
19*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-18 %s < %t
20*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-19 %s < %t
21*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-20 %s < %t
22*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-21 %s < %t
23*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-22 %s < %t
24*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-23 %s < %t
25*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-24 %s < %t
26*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-25 %s < %t
27*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-26 %s < %t
28*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-27 %s < %t
29*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-28 %s < %t
30*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-29 %s < %t
31*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-30 %s < %t
32*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-31 %s < %t
33*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-32 %s < %t
34*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-33 %s < %t
35*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-34 %s < %t
36*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-35 %s < %t
37*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-36 %s < %t
38*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-37 %s < %t
39*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-38 %s < %t
40*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-39 %s < %t
41*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-40 %s < %t
42*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-41 %s < %t
43*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-42 %s < %t
44*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-43 %s < %t
45*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-44 %s < %t
46*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-45 %s < %t
47*67e74705SXin Li // RUN: FileCheck --check-prefix=CHECK-46 %s < %t
48*67e74705SXin Li
49*67e74705SXin Li // For now, just verify this doesn't crash.
50*67e74705SXin Li namespace test0 {
51*67e74705SXin Li struct Obj {};
52*67e74705SXin Li
53*67e74705SXin Li struct Base { virtual const Obj *foo() = 0; };
footest0::Derived54*67e74705SXin Li struct Derived : Base { virtual Obj *foo() { return new Obj(); } };
55*67e74705SXin Li
test(Derived * D)56*67e74705SXin Li void test(Derived *D) { D->foo(); }
57*67e74705SXin Li }
58*67e74705SXin Li
59*67e74705SXin Li namespace Test1 {
60*67e74705SXin Li // CHECK-1: Vtable for 'Test1::A' (3 entries).
61*67e74705SXin Li // CHECK-1-NEXT: 0 | offset_to_top (0)
62*67e74705SXin Li // CHECK-1-NEXT: 1 | Test1::A RTTI
63*67e74705SXin Li // CHECK-1-NEXT: -- (Test1::A, 0) vtable address --
64*67e74705SXin Li // CHECK-1-NEXT: 2 | void Test1::A::f()
65*67e74705SXin Li //
66*67e74705SXin Li // CHECK-1: VTable indices for 'Test1::A' (1 entries).
67*67e74705SXin Li // CHECK-1-NEXT: 0 | void Test1::A::f()
68*67e74705SXin Li struct A {
69*67e74705SXin Li virtual void f();
70*67e74705SXin Li };
f()71*67e74705SXin Li void A::f() { }
72*67e74705SXin Li
73*67e74705SXin Li }
74*67e74705SXin Li
75*67e74705SXin Li namespace Test2 {
76*67e74705SXin Li
77*67e74705SXin Li // This is a smoke test of the vtable dumper.
78*67e74705SXin Li // CHECK-2: Vtable for 'Test2::A' (9 entries).
79*67e74705SXin Li // CHECK-2-NEXT: 0 | offset_to_top (0)
80*67e74705SXin Li // CHECK-2-NEXT: 1 | Test2::A RTTI
81*67e74705SXin Li // CHECK-2-NEXT: -- (Test2::A, 0) vtable address --
82*67e74705SXin Li // CHECK-2-NEXT: 2 | void Test2::A::f()
83*67e74705SXin Li // CHECK-2-NEXT: 3 | void Test2::A::f() const
84*67e74705SXin Li // CHECK-2-NEXT: 4 | Test2::A *Test2::A::g(int)
85*67e74705SXin Li // CHECK-2-NEXT: 5 | Test2::A::~A() [complete]
86*67e74705SXin Li // CHECK-2-NEXT: 6 | Test2::A::~A() [deleting]
87*67e74705SXin Li // CHECK-2-NEXT: 7 | void Test2::A::h()
88*67e74705SXin Li // CHECK-2-NEXT: 8 | Test2::A &Test2::A::operator=(const Test2::A &)
89*67e74705SXin Li //
90*67e74705SXin Li // CHECK-2: VTable indices for 'Test2::A' (7 entries).
91*67e74705SXin Li // CHECK-2-NEXT: 0 | void Test2::A::f()
92*67e74705SXin Li // CHECK-2-NEXT: 1 | void Test2::A::f() const
93*67e74705SXin Li // CHECK-2-NEXT: 2 | Test2::A *Test2::A::g(int)
94*67e74705SXin Li // CHECK-2-NEXT: 3 | Test2::A::~A() [complete]
95*67e74705SXin Li // CHECK-2-NEXT: 4 | Test2::A::~A() [deleting]
96*67e74705SXin Li // CHECK-2-NEXT: 5 | void Test2::A::h()
97*67e74705SXin Li // CHECK-2-NEXT: 6 | Test2::A &Test2::A::operator=(const Test2::A &)
98*67e74705SXin Li struct A {
99*67e74705SXin Li virtual void f();
100*67e74705SXin Li virtual void f() const;
101*67e74705SXin Li
102*67e74705SXin Li virtual A* g(int a);
103*67e74705SXin Li virtual ~A();
104*67e74705SXin Li virtual void h();
105*67e74705SXin Li virtual A& operator=(const A&);
106*67e74705SXin Li };
f()107*67e74705SXin Li void A::f() { }
108*67e74705SXin Li
109*67e74705SXin Li // Another simple vtable dumper test.
110*67e74705SXin Li
111*67e74705SXin Li // CHECK-3: Vtable for 'Test2::B' (6 entries).
112*67e74705SXin Li // CHECK-3-NEXT: 0 | offset_to_top (0)
113*67e74705SXin Li // CHECK-3-NEXT: 1 | Test2::B RTTI
114*67e74705SXin Li // CHECK-3-NEXT: -- (Test2::B, 0) vtable address --
115*67e74705SXin Li // CHECK-3-NEXT: 2 | void Test2::B::f()
116*67e74705SXin Li // CHECK-3-NEXT: 3 | void Test2::B::g() [pure]
117*67e74705SXin Li // CHECK-3-NEXT: 4 | Test2::B::~B() [complete] [pure]
118*67e74705SXin Li // CHECK-3-NEXT: 5 | Test2::B::~B() [deleting] [pure]
119*67e74705SXin Li //
120*67e74705SXin Li // CHECK-3: VTable indices for 'Test2::B' (4 entries).
121*67e74705SXin Li // CHECK-3-NEXT: 0 | void Test2::B::f()
122*67e74705SXin Li // CHECK-3-NEXT: 1 | void Test2::B::g()
123*67e74705SXin Li // CHECK-3-NEXT: 2 | Test2::B::~B() [complete]
124*67e74705SXin Li // CHECK-3-NEXT: 3 | Test2::B::~B() [deleting]
125*67e74705SXin Li struct B {
126*67e74705SXin Li virtual void f();
127*67e74705SXin Li virtual void g() = 0;
128*67e74705SXin Li virtual ~B() = 0;
129*67e74705SXin Li };
f()130*67e74705SXin Li void B::f() { }
131*67e74705SXin Li
132*67e74705SXin Li }
133*67e74705SXin Li
134*67e74705SXin Li namespace Test3 {
135*67e74705SXin Li
136*67e74705SXin Li // If a function in a derived class overrides a function in a primary base,
137*67e74705SXin Li // then the function should not have an entry in the derived class (unless the return
138*67e74705SXin Li // value requires adjusting).
139*67e74705SXin Li
140*67e74705SXin Li // CHECK-4: Vtable for 'Test3::A' (3 entries).
141*67e74705SXin Li // CHECK-4-NEXT: 0 | offset_to_top (0)
142*67e74705SXin Li // CHECK-4-NEXT: 1 | Test3::A RTTI
143*67e74705SXin Li // CHECK-4-NEXT: -- (Test3::A, 0) vtable address --
144*67e74705SXin Li // CHECK-4-NEXT: 2 | void Test3::A::f()
145*67e74705SXin Li //
146*67e74705SXin Li // CHECK-4: VTable indices for 'Test3::A' (1 entries).
147*67e74705SXin Li // CHECK-4-NEXT: 0 | void Test3::A::f()
148*67e74705SXin Li struct A {
149*67e74705SXin Li virtual void f();
150*67e74705SXin Li };
f()151*67e74705SXin Li void A::f() { }
152*67e74705SXin Li
153*67e74705SXin Li // CHECK-5: Vtable for 'Test3::B' (4 entries).
154*67e74705SXin Li // CHECK-5-NEXT: 0 | offset_to_top (0)
155*67e74705SXin Li // CHECK-5-NEXT: 1 | Test3::B RTTI
156*67e74705SXin Li // CHECK-5-NEXT: -- (Test3::A, 0) vtable address --
157*67e74705SXin Li // CHECK-5-NEXT: -- (Test3::B, 0) vtable address --
158*67e74705SXin Li // CHECK-5-NEXT: 2 | void Test3::B::f()
159*67e74705SXin Li // CHECK-5-NEXT: 3 | void Test3::B::g()
160*67e74705SXin Li //
161*67e74705SXin Li // CHECK-5: VTable indices for 'Test3::B' (2 entries).
162*67e74705SXin Li // CHECK-5-NEXT: 0 | void Test3::B::f()
163*67e74705SXin Li // CHECK-5-NEXT: 1 | void Test3::B::g()
164*67e74705SXin Li struct B : A {
165*67e74705SXin Li virtual void f();
166*67e74705SXin Li virtual void g();
167*67e74705SXin Li };
f()168*67e74705SXin Li void B::f() { }
169*67e74705SXin Li
170*67e74705SXin Li // CHECK-6: Vtable for 'Test3::C' (5 entries).
171*67e74705SXin Li // CHECK-6-NEXT: 0 | offset_to_top (0)
172*67e74705SXin Li // CHECK-6-NEXT: 1 | Test3::C RTTI
173*67e74705SXin Li // CHECK-6-NEXT: -- (Test3::A, 0) vtable address --
174*67e74705SXin Li // CHECK-6-NEXT: -- (Test3::C, 0) vtable address --
175*67e74705SXin Li // CHECK-6-NEXT: 2 | void Test3::A::f()
176*67e74705SXin Li // CHECK-6-NEXT: 3 | void Test3::C::g()
177*67e74705SXin Li // CHECK-6-NEXT: 4 | void Test3::C::h()
178*67e74705SXin Li //
179*67e74705SXin Li // CHECK-6: VTable indices for 'Test3::C' (2 entries).
180*67e74705SXin Li // CHECK-6-NEXT: 1 | void Test3::C::g()
181*67e74705SXin Li // CHECK-6-NEXT: 2 | void Test3::C::h()
182*67e74705SXin Li struct C : A {
183*67e74705SXin Li virtual void g();
184*67e74705SXin Li virtual void h();
185*67e74705SXin Li };
g()186*67e74705SXin Li void C::g() { }
187*67e74705SXin Li
188*67e74705SXin Li // CHECK-7: Vtable for 'Test3::D' (5 entries).
189*67e74705SXin Li // CHECK-7-NEXT: 0 | offset_to_top (0)
190*67e74705SXin Li // CHECK-7-NEXT: 1 | Test3::D RTTI
191*67e74705SXin Li // CHECK-7-NEXT: -- (Test3::A, 0) vtable address --
192*67e74705SXin Li // CHECK-7-NEXT: -- (Test3::B, 0) vtable address --
193*67e74705SXin Li // CHECK-7-NEXT: -- (Test3::D, 0) vtable address --
194*67e74705SXin Li // CHECK-7-NEXT: 2 | void Test3::D::f()
195*67e74705SXin Li // CHECK-7-NEXT: 3 | void Test3::D::g()
196*67e74705SXin Li // CHECK-7-NEXT: 4 | void Test3::D::h()
197*67e74705SXin Li //
198*67e74705SXin Li // CHECK-7: VTable indices for 'Test3::D' (3 entries).
199*67e74705SXin Li // CHECK-7-NEXT: 0 | void Test3::D::f()
200*67e74705SXin Li // CHECK-7-NEXT: 1 | void Test3::D::g()
201*67e74705SXin Li // CHECK-7-NEXT: 2 | void Test3::D::h()
202*67e74705SXin Li struct D : B {
203*67e74705SXin Li virtual void f();
204*67e74705SXin Li virtual void g();
205*67e74705SXin Li virtual void h();
206*67e74705SXin Li };
207*67e74705SXin Li
f()208*67e74705SXin Li void D::f() { }
209*67e74705SXin Li }
210*67e74705SXin Li
211*67e74705SXin Li namespace Test4 {
212*67e74705SXin Li
213*67e74705SXin Li // Test non-virtual result adjustments.
214*67e74705SXin Li
215*67e74705SXin Li struct R1 { int r1; };
216*67e74705SXin Li struct R2 { int r2; };
217*67e74705SXin Li struct R3 : R1, R2 { int r3; };
218*67e74705SXin Li
219*67e74705SXin Li struct A {
220*67e74705SXin Li virtual R2 *f();
221*67e74705SXin Li };
222*67e74705SXin Li
223*67e74705SXin Li // CHECK-8: Vtable for 'Test4::B' (4 entries).
224*67e74705SXin Li // CHECK-8-NEXT: 0 | offset_to_top (0)
225*67e74705SXin Li // CHECK-8-NEXT: 1 | Test4::B RTTI
226*67e74705SXin Li // CHECK-8-NEXT: -- (Test4::A, 0) vtable address --
227*67e74705SXin Li // CHECK-8-NEXT: -- (Test4::B, 0) vtable address --
228*67e74705SXin Li // CHECK-8-NEXT: 2 | Test4::R3 *Test4::B::f()
229*67e74705SXin Li // CHECK-8-NEXT: [return adjustment: 4 non-virtual]
230*67e74705SXin Li // CHECK-8-NEXT: 3 | Test4::R3 *Test4::B::f()
231*67e74705SXin Li //
232*67e74705SXin Li // CHECK-8: VTable indices for 'Test4::B' (1 entries).
233*67e74705SXin Li // CHECK-8-NEXT: 1 | Test4::R3 *Test4::B::f()
234*67e74705SXin Li struct B : A {
235*67e74705SXin Li virtual R3 *f();
236*67e74705SXin Li };
f()237*67e74705SXin Li R3 *B::f() { return 0; }
238*67e74705SXin Li
239*67e74705SXin Li // Test virtual result adjustments.
240*67e74705SXin Li struct V1 { int v1; };
241*67e74705SXin Li struct V2 : virtual V1 { int v1; };
242*67e74705SXin Li
243*67e74705SXin Li struct C {
244*67e74705SXin Li virtual V1 *f();
245*67e74705SXin Li };
246*67e74705SXin Li
247*67e74705SXin Li // CHECK-9: Vtable for 'Test4::D' (4 entries).
248*67e74705SXin Li // CHECK-9-NEXT: 0 | offset_to_top (0)
249*67e74705SXin Li // CHECK-9-NEXT: 1 | Test4::D RTTI
250*67e74705SXin Li // CHECK-9-NEXT: -- (Test4::C, 0) vtable address --
251*67e74705SXin Li // CHECK-9-NEXT: -- (Test4::D, 0) vtable address --
252*67e74705SXin Li // CHECK-9-NEXT: 2 | Test4::V2 *Test4::D::f()
253*67e74705SXin Li // CHECK-9-NEXT: [return adjustment: 0 non-virtual, -24 vbase offset offset]
254*67e74705SXin Li // CHECK-9-NEXT: 3 | Test4::V2 *Test4::D::f()
255*67e74705SXin Li //
256*67e74705SXin Li // CHECK-9: VTable indices for 'Test4::D' (1 entries).
257*67e74705SXin Li // CHECK-9-NEXT: 1 | Test4::V2 *Test4::D::f()
258*67e74705SXin Li struct D : C {
259*67e74705SXin Li virtual V2 *f();
260*67e74705SXin Li };
f()261*67e74705SXin Li V2 *D::f() { return 0; };
262*67e74705SXin Li
263*67e74705SXin Li // Virtual result adjustments with an additional non-virtual adjustment.
264*67e74705SXin Li struct V3 : virtual R3 { int r3; };
265*67e74705SXin Li
266*67e74705SXin Li // CHECK-10: Vtable for 'Test4::E' (4 entries).
267*67e74705SXin Li // CHECK-10-NEXT: 0 | offset_to_top (0)
268*67e74705SXin Li // CHECK-10-NEXT: 1 | Test4::E RTTI
269*67e74705SXin Li // CHECK-10-NEXT: -- (Test4::A, 0) vtable address --
270*67e74705SXin Li // CHECK-10-NEXT: -- (Test4::E, 0) vtable address --
271*67e74705SXin Li // CHECK-10-NEXT: 2 | Test4::V3 *Test4::E::f()
272*67e74705SXin Li // CHECK-10-NEXT: [return adjustment: 4 non-virtual, -24 vbase offset offset]
273*67e74705SXin Li // CHECK-10-NEXT: 3 | Test4::V3 *Test4::E::f()
274*67e74705SXin Li //
275*67e74705SXin Li // CHECK-10: VTable indices for 'Test4::E' (1 entries).
276*67e74705SXin Li // CHECK-10-NEXT: 1 | Test4::V3 *Test4::E::f()
277*67e74705SXin Li struct E : A {
278*67e74705SXin Li virtual V3 *f();
279*67e74705SXin Li };
f()280*67e74705SXin Li V3 *E::f() { return 0;}
281*67e74705SXin Li
282*67e74705SXin Li // Test that a pure virtual member doesn't get a thunk.
283*67e74705SXin Li
284*67e74705SXin Li // CHECK-11: Vtable for 'Test4::F' (5 entries).
285*67e74705SXin Li // CHECK-11-NEXT: 0 | offset_to_top (0)
286*67e74705SXin Li // CHECK-11-NEXT: 1 | Test4::F RTTI
287*67e74705SXin Li // CHECK-11-NEXT: -- (Test4::A, 0) vtable address --
288*67e74705SXin Li // CHECK-11-NEXT: -- (Test4::F, 0) vtable address --
289*67e74705SXin Li // CHECK-11-NEXT: 2 | Test4::R3 *Test4::F::f() [pure]
290*67e74705SXin Li // CHECK-11-NEXT: 3 | void Test4::F::g()
291*67e74705SXin Li // CHECK-11-NEXT: 4 | Test4::R3 *Test4::F::f() [pure]
292*67e74705SXin Li //
293*67e74705SXin Li // CHECK-11: VTable indices for 'Test4::F' (2 entries).
294*67e74705SXin Li // CHECK-11-NEXT: 1 | void Test4::F::g()
295*67e74705SXin Li // CHECK-11-NEXT: 2 | Test4::R3 *Test4::F::f()
296*67e74705SXin Li struct F : A {
297*67e74705SXin Li virtual void g();
298*67e74705SXin Li virtual R3 *f() = 0;
299*67e74705SXin Li };
g()300*67e74705SXin Li void F::g() { }
301*67e74705SXin Li
302*67e74705SXin Li }
303*67e74705SXin Li
304*67e74705SXin Li namespace Test5 {
305*67e74705SXin Li
306*67e74705SXin Li // Simple secondary vtables without 'this' pointer adjustments.
307*67e74705SXin Li struct A {
308*67e74705SXin Li virtual void f();
309*67e74705SXin Li virtual void g();
310*67e74705SXin Li int a;
311*67e74705SXin Li };
312*67e74705SXin Li
313*67e74705SXin Li struct B1 : A {
314*67e74705SXin Li virtual void f();
315*67e74705SXin Li int b1;
316*67e74705SXin Li };
317*67e74705SXin Li
318*67e74705SXin Li struct B2 : A {
319*67e74705SXin Li virtual void g();
320*67e74705SXin Li int b2;
321*67e74705SXin Li };
322*67e74705SXin Li
323*67e74705SXin Li // CHECK-12: Vtable for 'Test5::C' (9 entries).
324*67e74705SXin Li // CHECK-12-NEXT: 0 | offset_to_top (0)
325*67e74705SXin Li // CHECK-12-NEXT: 1 | Test5::C RTTI
326*67e74705SXin Li // CHECK-12-NEXT: -- (Test5::A, 0) vtable address --
327*67e74705SXin Li // CHECK-12-NEXT: -- (Test5::B1, 0) vtable address --
328*67e74705SXin Li // CHECK-12-NEXT: -- (Test5::C, 0) vtable address --
329*67e74705SXin Li // CHECK-12-NEXT: 2 | void Test5::B1::f()
330*67e74705SXin Li // CHECK-12-NEXT: 3 | void Test5::A::g()
331*67e74705SXin Li // CHECK-12-NEXT: 4 | void Test5::C::h()
332*67e74705SXin Li // CHECK-12-NEXT: 5 | offset_to_top (-16)
333*67e74705SXin Li // CHECK-12-NEXT: 6 | Test5::C RTTI
334*67e74705SXin Li // CHECK-12-NEXT: -- (Test5::A, 16) vtable address --
335*67e74705SXin Li // CHECK-12-NEXT: -- (Test5::B2, 16) vtable address --
336*67e74705SXin Li // CHECK-12-NEXT: 7 | void Test5::A::f()
337*67e74705SXin Li // CHECK-12-NEXT: 8 | void Test5::B2::g()
338*67e74705SXin Li //
339*67e74705SXin Li // CHECK-12: VTable indices for 'Test5::C' (1 entries).
340*67e74705SXin Li // CHECK-12-NEXT: 2 | void Test5::C::h()
341*67e74705SXin Li struct C : B1, B2 {
342*67e74705SXin Li virtual void h();
343*67e74705SXin Li };
h()344*67e74705SXin Li void C::h() { }
345*67e74705SXin Li }
346*67e74705SXin Li
347*67e74705SXin Li namespace Test6 {
348*67e74705SXin Li
349*67e74705SXin Li // Simple non-virtual 'this' pointer adjustments.
350*67e74705SXin Li struct A1 {
351*67e74705SXin Li virtual void f();
352*67e74705SXin Li int a;
353*67e74705SXin Li };
354*67e74705SXin Li
355*67e74705SXin Li struct A2 {
356*67e74705SXin Li virtual void f();
357*67e74705SXin Li int a;
358*67e74705SXin Li };
359*67e74705SXin Li
360*67e74705SXin Li // CHECK-13: Vtable for 'Test6::C' (6 entries).
361*67e74705SXin Li // CHECK-13-NEXT: 0 | offset_to_top (0)
362*67e74705SXin Li // CHECK-13-NEXT: 1 | Test6::C RTTI
363*67e74705SXin Li // CHECK-13-NEXT: -- (Test6::A1, 0) vtable address --
364*67e74705SXin Li // CHECK-13-NEXT: -- (Test6::C, 0) vtable address --
365*67e74705SXin Li // CHECK-13-NEXT: 2 | void Test6::C::f()
366*67e74705SXin Li // CHECK-13-NEXT: 3 | offset_to_top (-16)
367*67e74705SXin Li // CHECK-13-NEXT: 4 | Test6::C RTTI
368*67e74705SXin Li // CHECK-13-NEXT: -- (Test6::A2, 16) vtable address --
369*67e74705SXin Li // CHECK-13-NEXT: 5 | void Test6::C::f()
370*67e74705SXin Li // CHECK-13-NEXT: [this adjustment: -16 non-virtual]
371*67e74705SXin Li //
372*67e74705SXin Li // CHECK-13: VTable indices for 'Test6::C' (1 entries).
373*67e74705SXin Li // CHECK-13-NEXT: 0 | void Test6::C::f()
374*67e74705SXin Li struct C : A1, A2 {
375*67e74705SXin Li virtual void f();
376*67e74705SXin Li };
f()377*67e74705SXin Li void C::f() { }
378*67e74705SXin Li
379*67e74705SXin Li }
380*67e74705SXin Li
381*67e74705SXin Li namespace Test7 {
382*67e74705SXin Li
383*67e74705SXin Li // Test that the D::f overrider for A::f have different 'this' pointer
384*67e74705SXin Li // adjustments in the two A base subobjects.
385*67e74705SXin Li
386*67e74705SXin Li struct A {
387*67e74705SXin Li virtual void f();
388*67e74705SXin Li int a;
389*67e74705SXin Li };
390*67e74705SXin Li
391*67e74705SXin Li struct B1 : A { };
392*67e74705SXin Li struct B2 : A { };
393*67e74705SXin Li
394*67e74705SXin Li struct C { virtual void c(); };
395*67e74705SXin Li
396*67e74705SXin Li // CHECK-14: Vtable for 'Test7::D' (10 entries).
397*67e74705SXin Li // CHECK-14-NEXT: 0 | offset_to_top (0)
398*67e74705SXin Li // CHECK-14-NEXT: 1 | Test7::D RTTI
399*67e74705SXin Li // CHECK-14-NEXT: -- (Test7::C, 0) vtable address --
400*67e74705SXin Li // CHECK-14-NEXT: -- (Test7::D, 0) vtable address --
401*67e74705SXin Li // CHECK-14-NEXT: 2 | void Test7::C::c()
402*67e74705SXin Li // CHECK-14-NEXT: 3 | void Test7::D::f()
403*67e74705SXin Li // CHECK-14-NEXT: 4 | offset_to_top (-8)
404*67e74705SXin Li // CHECK-14-NEXT: 5 | Test7::D RTTI
405*67e74705SXin Li // CHECK-14-NEXT: -- (Test7::A, 8) vtable address --
406*67e74705SXin Li // CHECK-14-NEXT: -- (Test7::B1, 8) vtable address --
407*67e74705SXin Li // CHECK-14-NEXT: 6 | void Test7::D::f()
408*67e74705SXin Li // CHECK-14-NEXT: [this adjustment: -8 non-virtual]
409*67e74705SXin Li // CHECK-14-NEXT: 7 | offset_to_top (-24)
410*67e74705SXin Li // CHECK-14-NEXT: 8 | Test7::D RTTI
411*67e74705SXin Li // CHECK-14-NEXT: -- (Test7::A, 24) vtable address --
412*67e74705SXin Li // CHECK-14-NEXT: -- (Test7::B2, 24) vtable address --
413*67e74705SXin Li // CHECK-14-NEXT: 9 | void Test7::D::f()
414*67e74705SXin Li // CHECK-14-NEXT: [this adjustment: -24 non-virtual]
415*67e74705SXin Li //
416*67e74705SXin Li // CHECK-14: VTable indices for 'Test7::D' (1 entries).
417*67e74705SXin Li // CHECK-14-NEXT: 1 | void Test7::D::f()
418*67e74705SXin Li struct D : C, B1, B2 {
419*67e74705SXin Li virtual void f();
420*67e74705SXin Li };
f()421*67e74705SXin Li void D::f() { }
422*67e74705SXin Li
423*67e74705SXin Li }
424*67e74705SXin Li
425*67e74705SXin Li namespace Test8 {
426*67e74705SXin Li
427*67e74705SXin Li // Test that we don't try to layout vtables for classes that don't have
428*67e74705SXin Li // virtual bases or virtual member functions.
429*67e74705SXin Li
430*67e74705SXin Li struct A { };
431*67e74705SXin Li
432*67e74705SXin Li // CHECK-15: Vtable for 'Test8::B' (3 entries).
433*67e74705SXin Li // CHECK-15-NEXT: 0 | offset_to_top (0)
434*67e74705SXin Li // CHECK-15-NEXT: 1 | Test8::B RTTI
435*67e74705SXin Li // CHECK-15-NEXT: -- (Test8::B, 0) vtable address --
436*67e74705SXin Li // CHECK-15-NEXT: 2 | void Test8::B::f()
437*67e74705SXin Li //
438*67e74705SXin Li // CHECK-15: VTable indices for 'Test8::B' (1 entries).
439*67e74705SXin Li // CHECK-15-NEXT: 0 | void Test8::B::f()
440*67e74705SXin Li struct B : A {
441*67e74705SXin Li virtual void f();
442*67e74705SXin Li };
f()443*67e74705SXin Li void B::f() { }
444*67e74705SXin Li
445*67e74705SXin Li }
446*67e74705SXin Li
447*67e74705SXin Li namespace Test9 {
448*67e74705SXin Li
449*67e74705SXin Li // Simple test of vbase offsets.
450*67e74705SXin Li
451*67e74705SXin Li struct A1 { int a1; };
452*67e74705SXin Li struct A2 { int a2; };
453*67e74705SXin Li
454*67e74705SXin Li // CHECK-16: Vtable for 'Test9::B' (5 entries).
455*67e74705SXin Li // CHECK-16-NEXT: 0 | vbase_offset (16)
456*67e74705SXin Li // CHECK-16-NEXT: 1 | vbase_offset (12)
457*67e74705SXin Li // CHECK-16-NEXT: 2 | offset_to_top (0)
458*67e74705SXin Li // CHECK-16-NEXT: 3 | Test9::B RTTI
459*67e74705SXin Li // CHECK-16-NEXT: -- (Test9::B, 0) vtable address --
460*67e74705SXin Li // CHECK-16-NEXT: 4 | void Test9::B::f()
461*67e74705SXin Li //
462*67e74705SXin Li // CHECK-16: VTable indices for 'Test9::B' (1 entries).
463*67e74705SXin Li // CHECK-16-NEXT: 0 | void Test9::B::f()
464*67e74705SXin Li struct B : virtual A1, virtual A2 {
465*67e74705SXin Li int b;
466*67e74705SXin Li
467*67e74705SXin Li virtual void f();
468*67e74705SXin Li };
469*67e74705SXin Li
470*67e74705SXin Li
f()471*67e74705SXin Li void B::f() { }
472*67e74705SXin Li
473*67e74705SXin Li }
474*67e74705SXin Li
475*67e74705SXin Li namespace Test10 {
476*67e74705SXin Li
477*67e74705SXin Li // Test for a bug where we would not emit secondary vtables for bases
478*67e74705SXin Li // of a primary base.
479*67e74705SXin Li struct A1 { virtual void a1(); };
480*67e74705SXin Li struct A2 { virtual void a2(); };
481*67e74705SXin Li
482*67e74705SXin Li // CHECK-17: Vtable for 'Test10::C' (7 entries).
483*67e74705SXin Li // CHECK-17-NEXT: 0 | offset_to_top (0)
484*67e74705SXin Li // CHECK-17-NEXT: 1 | Test10::C RTTI
485*67e74705SXin Li // CHECK-17-NEXT: -- (Test10::A1, 0) vtable address --
486*67e74705SXin Li // CHECK-17-NEXT: -- (Test10::B, 0) vtable address --
487*67e74705SXin Li // CHECK-17-NEXT: -- (Test10::C, 0) vtable address --
488*67e74705SXin Li // CHECK-17-NEXT: 2 | void Test10::A1::a1()
489*67e74705SXin Li // CHECK-17-NEXT: 3 | void Test10::C::f()
490*67e74705SXin Li // CHECK-17-NEXT: 4 | offset_to_top (-8)
491*67e74705SXin Li // CHECK-17-NEXT: 5 | Test10::C RTTI
492*67e74705SXin Li // CHECK-17-NEXT: -- (Test10::A2, 8) vtable address --
493*67e74705SXin Li // CHECK-17-NEXT: 6 | void Test10::A2::a2()
494*67e74705SXin Li //
495*67e74705SXin Li // CHECK-17: VTable indices for 'Test10::C' (1 entries).
496*67e74705SXin Li // CHECK-17-NEXT: 1 | void Test10::C::f()
497*67e74705SXin Li struct B : A1, A2 {
498*67e74705SXin Li int b;
499*67e74705SXin Li };
500*67e74705SXin Li
501*67e74705SXin Li struct C : B {
502*67e74705SXin Li virtual void f();
503*67e74705SXin Li };
f()504*67e74705SXin Li void C::f() { }
505*67e74705SXin Li
506*67e74705SXin Li }
507*67e74705SXin Li
508*67e74705SXin Li namespace Test11 {
509*67e74705SXin Li
510*67e74705SXin Li // Very simple test of vtables for virtual bases.
511*67e74705SXin Li struct A1 { int a; };
512*67e74705SXin Li struct A2 { int b; };
513*67e74705SXin Li
514*67e74705SXin Li struct B : A1, virtual A2 {
515*67e74705SXin Li int b;
516*67e74705SXin Li };
517*67e74705SXin Li
518*67e74705SXin Li // CHECK-18: Vtable for 'Test11::C' (8 entries).
519*67e74705SXin Li // CHECK-18-NEXT: 0 | vbase_offset (24)
520*67e74705SXin Li // CHECK-18-NEXT: 1 | vbase_offset (8)
521*67e74705SXin Li // CHECK-18-NEXT: 2 | offset_to_top (0)
522*67e74705SXin Li // CHECK-18-NEXT: 3 | Test11::C RTTI
523*67e74705SXin Li // CHECK-18-NEXT: -- (Test11::C, 0) vtable address --
524*67e74705SXin Li // CHECK-18-NEXT: 4 | void Test11::C::f()
525*67e74705SXin Li // CHECK-18-NEXT: 5 | vbase_offset (16)
526*67e74705SXin Li // CHECK-18-NEXT: 6 | offset_to_top (-8)
527*67e74705SXin Li // CHECK-18-NEXT: 7 | Test11::C RTTI
528*67e74705SXin Li //
529*67e74705SXin Li // CHECK-18: VTable indices for 'Test11::C' (1 entries).
530*67e74705SXin Li // CHECK-18-NEXT: 0 | void Test11::C::f()
531*67e74705SXin Li struct C : virtual B {
532*67e74705SXin Li virtual void f();
533*67e74705SXin Li };
f()534*67e74705SXin Li void C::f() { }
535*67e74705SXin Li
536*67e74705SXin Li }
537*67e74705SXin Li
538*67e74705SXin Li namespace Test12 {
539*67e74705SXin Li
540*67e74705SXin Li // Test that the right vcall offsets are generated in the right order.
541*67e74705SXin Li
542*67e74705SXin Li // CHECK-19: Vtable for 'Test12::B' (19 entries).
543*67e74705SXin Li // CHECK-19-NEXT: 0 | vbase_offset (8)
544*67e74705SXin Li // CHECK-19-NEXT: 1 | offset_to_top (0)
545*67e74705SXin Li // CHECK-19-NEXT: 2 | Test12::B RTTI
546*67e74705SXin Li // CHECK-19-NEXT: -- (Test12::B, 0) vtable address --
547*67e74705SXin Li // CHECK-19-NEXT: 3 | void Test12::B::f()
548*67e74705SXin Li // CHECK-19-NEXT: 4 | void Test12::B::a()
549*67e74705SXin Li // CHECK-19-NEXT: 5 | vcall_offset (32)
550*67e74705SXin Li // CHECK-19-NEXT: 6 | vcall_offset (16)
551*67e74705SXin Li // CHECK-19-NEXT: 7 | vcall_offset (-8)
552*67e74705SXin Li // CHECK-19-NEXT: 8 | vcall_offset (0)
553*67e74705SXin Li // CHECK-19-NEXT: 9 | offset_to_top (-8)
554*67e74705SXin Li // CHECK-19-NEXT: 10 | Test12::B RTTI
555*67e74705SXin Li // CHECK-19-NEXT: -- (Test12::A, 8) vtable address --
556*67e74705SXin Li // CHECK-19-NEXT: -- (Test12::A1, 8) vtable address --
557*67e74705SXin Li // CHECK-19-NEXT: 11 | void Test12::A1::a1()
558*67e74705SXin Li // CHECK-19-NEXT: 12 | void Test12::B::a()
559*67e74705SXin Li // CHECK-19-NEXT: [this adjustment: 0 non-virtual, -32 vcall offset offset]
560*67e74705SXin Li // CHECK-19-NEXT: 13 | offset_to_top (-24)
561*67e74705SXin Li // CHECK-19-NEXT: 14 | Test12::B RTTI
562*67e74705SXin Li // CHECK-19-NEXT: -- (Test12::A2, 24) vtable address --
563*67e74705SXin Li // CHECK-19-NEXT: 15 | void Test12::A2::a2()
564*67e74705SXin Li // CHECK-19-NEXT: 16 | offset_to_top (-40)
565*67e74705SXin Li // CHECK-19-NEXT: 17 | Test12::B RTTI
566*67e74705SXin Li // CHECK-19-NEXT: -- (Test12::A3, 40) vtable address --
567*67e74705SXin Li // CHECK-19-NEXT: 18 | void Test12::A3::a3()
568*67e74705SXin Li //
569*67e74705SXin Li // CHECK-19: VTable indices for 'Test12::B' (2 entries).
570*67e74705SXin Li // CHECK-19-NEXT: 0 | void Test12::B::f()
571*67e74705SXin Li // CHECK-19-NEXT: 1 | void Test12::B::a()
572*67e74705SXin Li struct A1 {
573*67e74705SXin Li virtual void a1();
574*67e74705SXin Li int a;
575*67e74705SXin Li };
576*67e74705SXin Li
577*67e74705SXin Li struct A2 {
578*67e74705SXin Li virtual void a2();
579*67e74705SXin Li int a;
580*67e74705SXin Li };
581*67e74705SXin Li
582*67e74705SXin Li struct A3 {
583*67e74705SXin Li virtual void a3();
584*67e74705SXin Li int a;
585*67e74705SXin Li };
586*67e74705SXin Li
587*67e74705SXin Li struct A : A1, A2, A3 {
588*67e74705SXin Li virtual void a();
589*67e74705SXin Li int i;
590*67e74705SXin Li };
591*67e74705SXin Li
592*67e74705SXin Li struct B : virtual A {
593*67e74705SXin Li virtual void f();
594*67e74705SXin Li
595*67e74705SXin Li virtual void a();
596*67e74705SXin Li };
f()597*67e74705SXin Li void B::f() { }
598*67e74705SXin Li
599*67e74705SXin Li }
600*67e74705SXin Li
601*67e74705SXin Li namespace Test13 {
602*67e74705SXin Li
603*67e74705SXin Li // Test that we don't try to emit a vtable for 'A' twice.
604*67e74705SXin Li struct A {
605*67e74705SXin Li virtual void f();
606*67e74705SXin Li };
607*67e74705SXin Li
608*67e74705SXin Li struct B : virtual A {
609*67e74705SXin Li virtual void f();
610*67e74705SXin Li };
611*67e74705SXin Li
612*67e74705SXin Li // CHECK-20: Vtable for 'Test13::C' (6 entries).
613*67e74705SXin Li // CHECK-20-NEXT: 0 | vbase_offset (0)
614*67e74705SXin Li // CHECK-20-NEXT: 1 | vbase_offset (0)
615*67e74705SXin Li // CHECK-20-NEXT: 2 | vcall_offset (0)
616*67e74705SXin Li // CHECK-20-NEXT: 3 | offset_to_top (0)
617*67e74705SXin Li // CHECK-20-NEXT: 4 | Test13::C RTTI
618*67e74705SXin Li // CHECK-20-NEXT: -- (Test13::A, 0) vtable address --
619*67e74705SXin Li // CHECK-20-NEXT: -- (Test13::B, 0) vtable address --
620*67e74705SXin Li // CHECK-20-NEXT: -- (Test13::C, 0) vtable address --
621*67e74705SXin Li // CHECK-20-NEXT: 5 | void Test13::C::f()
622*67e74705SXin Li //
623*67e74705SXin Li // CHECK-20: VTable indices for 'Test13::C' (1 entries).
624*67e74705SXin Li // CHECK-20-NEXT: 0 | void Test13::C::f()
625*67e74705SXin Li struct C : virtual B, virtual A {
626*67e74705SXin Li virtual void f();
627*67e74705SXin Li };
f()628*67e74705SXin Li void C::f() { }
629*67e74705SXin Li
630*67e74705SXin Li }
631*67e74705SXin Li
632*67e74705SXin Li namespace Test14 {
633*67e74705SXin Li
634*67e74705SXin Li // Verify that we handle A being a non-virtual base of B, which is a virtual base.
635*67e74705SXin Li
636*67e74705SXin Li struct A {
637*67e74705SXin Li virtual void f();
638*67e74705SXin Li };
639*67e74705SXin Li
640*67e74705SXin Li struct B : A { };
641*67e74705SXin Li
642*67e74705SXin Li struct C : virtual B { };
643*67e74705SXin Li
644*67e74705SXin Li // CHECK-21: Vtable for 'Test14::D' (5 entries).
645*67e74705SXin Li // CHECK-21-NEXT: 0 | vbase_offset (0)
646*67e74705SXin Li // CHECK-21-NEXT: 1 | vcall_offset (0)
647*67e74705SXin Li // CHECK-21-NEXT: 2 | offset_to_top (0)
648*67e74705SXin Li // CHECK-21-NEXT: 3 | Test14::D RTTI
649*67e74705SXin Li // CHECK-21-NEXT: -- (Test14::A, 0) vtable address --
650*67e74705SXin Li // CHECK-21-NEXT: -- (Test14::B, 0) vtable address --
651*67e74705SXin Li // CHECK-21-NEXT: -- (Test14::C, 0) vtable address --
652*67e74705SXin Li // CHECK-21-NEXT: -- (Test14::D, 0) vtable address --
653*67e74705SXin Li // CHECK-21-NEXT: 4 | void Test14::D::f()
654*67e74705SXin Li //
655*67e74705SXin Li // CHECK-21: VTable indices for 'Test14::D' (1 entries).
656*67e74705SXin Li // CHECK-21-NEXT: 0 | void Test14::D::f()
657*67e74705SXin Li struct D : C, virtual B {
658*67e74705SXin Li virtual void f();
659*67e74705SXin Li };
f()660*67e74705SXin Li void D::f() { }
661*67e74705SXin Li
662*67e74705SXin Li }
663*67e74705SXin Li
664*67e74705SXin Li namespace Test15 {
665*67e74705SXin Li
666*67e74705SXin Li // Test that we don't emit an extra vtable for B since it's a primary base of C.
667*67e74705SXin Li struct A { virtual void a(); };
668*67e74705SXin Li struct B { virtual void b(); };
669*67e74705SXin Li
670*67e74705SXin Li struct C : virtual B { };
671*67e74705SXin Li
672*67e74705SXin Li // CHECK-22: Vtable for 'Test15::D' (11 entries).
673*67e74705SXin Li // CHECK-22-NEXT: 0 | vbase_offset (8)
674*67e74705SXin Li // CHECK-22-NEXT: 1 | vbase_offset (8)
675*67e74705SXin Li // CHECK-22-NEXT: 2 | offset_to_top (0)
676*67e74705SXin Li // CHECK-22-NEXT: 3 | Test15::D RTTI
677*67e74705SXin Li // CHECK-22-NEXT: -- (Test15::A, 0) vtable address --
678*67e74705SXin Li // CHECK-22-NEXT: -- (Test15::D, 0) vtable address --
679*67e74705SXin Li // CHECK-22-NEXT: 4 | void Test15::A::a()
680*67e74705SXin Li // CHECK-22-NEXT: 5 | void Test15::D::f()
681*67e74705SXin Li // CHECK-22-NEXT: 6 | vbase_offset (0)
682*67e74705SXin Li // CHECK-22-NEXT: 7 | vcall_offset (0)
683*67e74705SXin Li // CHECK-22-NEXT: 8 | offset_to_top (-8)
684*67e74705SXin Li // CHECK-22-NEXT: 9 | Test15::D RTTI
685*67e74705SXin Li // CHECK-22-NEXT: -- (Test15::B, 8) vtable address --
686*67e74705SXin Li // CHECK-22-NEXT: -- (Test15::C, 8) vtable address --
687*67e74705SXin Li // CHECK-22-NEXT: 10 | void Test15::B::b()
688*67e74705SXin Li //
689*67e74705SXin Li // CHECK-22: VTable indices for 'Test15::D' (1 entries).
690*67e74705SXin Li // CHECK-22-NEXT: 1 | void Test15::D::f()
691*67e74705SXin Li struct D : A, virtual B, virtual C {
692*67e74705SXin Li virtual void f();
693*67e74705SXin Li };
f()694*67e74705SXin Li void D::f() { }
695*67e74705SXin Li
696*67e74705SXin Li }
697*67e74705SXin Li
698*67e74705SXin Li namespace Test16 {
699*67e74705SXin Li
700*67e74705SXin Li // Test that destructors share vcall offsets.
701*67e74705SXin Li
702*67e74705SXin Li struct A { virtual ~A(); };
703*67e74705SXin Li struct B { virtual ~B(); };
704*67e74705SXin Li
705*67e74705SXin Li struct C : A, B { virtual ~C(); };
706*67e74705SXin Li
707*67e74705SXin Li // CHECK-23: Vtable for 'Test16::D' (15 entries).
708*67e74705SXin Li // CHECK-23-NEXT: 0 | vbase_offset (8)
709*67e74705SXin Li // CHECK-23-NEXT: 1 | offset_to_top (0)
710*67e74705SXin Li // CHECK-23-NEXT: 2 | Test16::D RTTI
711*67e74705SXin Li // CHECK-23-NEXT: -- (Test16::D, 0) vtable address --
712*67e74705SXin Li // CHECK-23-NEXT: 3 | void Test16::D::f()
713*67e74705SXin Li // CHECK-23-NEXT: 4 | Test16::D::~D() [complete]
714*67e74705SXin Li // CHECK-23-NEXT: 5 | Test16::D::~D() [deleting]
715*67e74705SXin Li // CHECK-23-NEXT: 6 | vcall_offset (-8)
716*67e74705SXin Li // CHECK-23-NEXT: 7 | offset_to_top (-8)
717*67e74705SXin Li // CHECK-23-NEXT: 8 | Test16::D RTTI
718*67e74705SXin Li // CHECK-23-NEXT: -- (Test16::A, 8) vtable address --
719*67e74705SXin Li // CHECK-23-NEXT: -- (Test16::C, 8) vtable address --
720*67e74705SXin Li // CHECK-23-NEXT: 9 | Test16::D::~D() [complete]
721*67e74705SXin Li // CHECK-23-NEXT: [this adjustment: 0 non-virtual, -24 vcall offset offset]
722*67e74705SXin Li // CHECK-23-NEXT: 10 | Test16::D::~D() [deleting]
723*67e74705SXin Li // CHECK-23-NEXT: [this adjustment: 0 non-virtual, -24 vcall offset offset]
724*67e74705SXin Li // CHECK-23-NEXT: 11 | offset_to_top (-16)
725*67e74705SXin Li // CHECK-23-NEXT: 12 | Test16::D RTTI
726*67e74705SXin Li // CHECK-23-NEXT: -- (Test16::B, 16) vtable address --
727*67e74705SXin Li // CHECK-23-NEXT: 13 | Test16::D::~D() [complete]
728*67e74705SXin Li // CHECK-23-NEXT: [this adjustment: -8 non-virtual, -24 vcall offset offset]
729*67e74705SXin Li // CHECK-23-NEXT: 14 | Test16::D::~D() [deleting]
730*67e74705SXin Li // CHECK-23-NEXT: [this adjustment: -8 non-virtual, -24 vcall offset offset]
731*67e74705SXin Li //
732*67e74705SXin Li // CHECK-23: VTable indices for 'Test16::D' (3 entries).
733*67e74705SXin Li // CHECK-23-NEXT: 0 | void Test16::D::f()
734*67e74705SXin Li // CHECK-23-NEXT: 1 | Test16::D::~D() [complete]
735*67e74705SXin Li // CHECK-23-NEXT: 2 | Test16::D::~D() [deleting]
736*67e74705SXin Li struct D : virtual C {
737*67e74705SXin Li virtual void f();
738*67e74705SXin Li };
f()739*67e74705SXin Li void D::f() { }
740*67e74705SXin Li
741*67e74705SXin Li }
742*67e74705SXin Li
743*67e74705SXin Li namespace Test17 {
744*67e74705SXin Li
745*67e74705SXin Li // Test that we don't mark E::f in the C-in-E vtable as unused.
746*67e74705SXin Li struct A { virtual void f(); };
747*67e74705SXin Li struct B : virtual A { virtual void f(); };
748*67e74705SXin Li struct C : virtual A { virtual void f(); };
749*67e74705SXin Li struct D : virtual B, virtual C { virtual void f(); };
750*67e74705SXin Li
751*67e74705SXin Li // CHECK-24: Vtable for 'Test17::E' (13 entries).
752*67e74705SXin Li // CHECK-24-NEXT: 0 | vbase_offset (0)
753*67e74705SXin Li // CHECK-24-NEXT: 1 | vbase_offset (8)
754*67e74705SXin Li // CHECK-24-NEXT: 2 | vbase_offset (0)
755*67e74705SXin Li // CHECK-24-NEXT: 3 | vbase_offset (0)
756*67e74705SXin Li // CHECK-24-NEXT: 4 | vcall_offset (0)
757*67e74705SXin Li // CHECK-24-NEXT: 5 | offset_to_top (0)
758*67e74705SXin Li // CHECK-24-NEXT: 6 | Test17::E RTTI
759*67e74705SXin Li // CHECK-24-NEXT: -- (Test17::A, 0) vtable address --
760*67e74705SXin Li // CHECK-24-NEXT: -- (Test17::B, 0) vtable address --
761*67e74705SXin Li // CHECK-24-NEXT: -- (Test17::D, 0) vtable address --
762*67e74705SXin Li // CHECK-24-NEXT: -- (Test17::E, 0) vtable address --
763*67e74705SXin Li // CHECK-24-NEXT: 7 | void Test17::E::f()
764*67e74705SXin Li // CHECK-24-NEXT: 8 | vbase_offset (-8)
765*67e74705SXin Li // CHECK-24-NEXT: 9 | vcall_offset (-8)
766*67e74705SXin Li // CHECK-24-NEXT: 10 | offset_to_top (-8)
767*67e74705SXin Li // CHECK-24-NEXT: 11 | Test17::E RTTI
768*67e74705SXin Li // CHECK-24-NEXT: -- (Test17::C, 8) vtable address --
769*67e74705SXin Li // CHECK-24-NEXT: 12 | void Test17::E::f()
770*67e74705SXin Li // CHECK-24-NEXT: [this adjustment: 0 non-virtual, -24 vcall offset offset]
771*67e74705SXin Li //
772*67e74705SXin Li // CHECK-24: VTable indices for 'Test17::E' (1 entries).
773*67e74705SXin Li // CHECK-24-NEXT: 0 | void Test17::E::f()
774*67e74705SXin Li class E : virtual D {
775*67e74705SXin Li virtual void f();
776*67e74705SXin Li };
f()777*67e74705SXin Li void E::f() {}
778*67e74705SXin Li
779*67e74705SXin Li }
780*67e74705SXin Li
781*67e74705SXin Li namespace Test18 {
782*67e74705SXin Li
783*67e74705SXin Li // Test that we compute the right 'this' adjustment offsets.
784*67e74705SXin Li
785*67e74705SXin Li struct A {
786*67e74705SXin Li virtual void f();
787*67e74705SXin Li virtual void g();
788*67e74705SXin Li };
789*67e74705SXin Li
790*67e74705SXin Li struct B : virtual A {
791*67e74705SXin Li virtual void f();
792*67e74705SXin Li };
793*67e74705SXin Li
794*67e74705SXin Li struct C : A, B {
795*67e74705SXin Li virtual void g();
796*67e74705SXin Li };
797*67e74705SXin Li
798*67e74705SXin Li // CHECK-25: Vtable for 'Test18::D' (24 entries).
799*67e74705SXin Li // CHECK-25-NEXT: 0 | vbase_offset (8)
800*67e74705SXin Li // CHECK-25-NEXT: 1 | vbase_offset (0)
801*67e74705SXin Li // CHECK-25-NEXT: 2 | vbase_offset (0)
802*67e74705SXin Li // CHECK-25-NEXT: 3 | vcall_offset (8)
803*67e74705SXin Li // CHECK-25-NEXT: 4 | vcall_offset (0)
804*67e74705SXin Li // CHECK-25-NEXT: 5 | offset_to_top (0)
805*67e74705SXin Li // CHECK-25-NEXT: 6 | Test18::D RTTI
806*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::A, 0) vtable address --
807*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::B, 0) vtable address --
808*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::D, 0) vtable address --
809*67e74705SXin Li // CHECK-25-NEXT: 7 | void Test18::D::f()
810*67e74705SXin Li // CHECK-25-NEXT: 8 | void Test18::C::g()
811*67e74705SXin Li // CHECK-25-NEXT: [this adjustment: 0 non-virtual, -32 vcall offset offset]
812*67e74705SXin Li // CHECK-25-NEXT: 9 | void Test18::D::h()
813*67e74705SXin Li // CHECK-25-NEXT: 10 | vcall_offset (0)
814*67e74705SXin Li // CHECK-25-NEXT: 11 | vcall_offset (-8)
815*67e74705SXin Li // CHECK-25-NEXT: 12 | vbase_offset (-8)
816*67e74705SXin Li // CHECK-25-NEXT: 13 | offset_to_top (-8)
817*67e74705SXin Li // CHECK-25-NEXT: 14 | Test18::D RTTI
818*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::A, 8) vtable address --
819*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::C, 8) vtable address --
820*67e74705SXin Li // CHECK-25-NEXT: 15 | void Test18::D::f()
821*67e74705SXin Li // CHECK-25-NEXT: [this adjustment: 0 non-virtual, -32 vcall offset offset]
822*67e74705SXin Li // CHECK-25-NEXT: 16 | void Test18::C::g()
823*67e74705SXin Li // CHECK-25-NEXT: 17 | vbase_offset (-16)
824*67e74705SXin Li // CHECK-25-NEXT: 18 | vcall_offset (-8)
825*67e74705SXin Li // CHECK-25-NEXT: 19 | vcall_offset (-16)
826*67e74705SXin Li // CHECK-25-NEXT: 20 | offset_to_top (-16)
827*67e74705SXin Li // CHECK-25-NEXT: 21 | Test18::D RTTI
828*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::B, 16) vtable address --
829*67e74705SXin Li // CHECK-25-NEXT: 22 | void Test18::D::f()
830*67e74705SXin Li // CHECK-25-NEXT: [this adjustment: -8 non-virtual, -32 vcall offset offset]
831*67e74705SXin Li // CHECK-25-NEXT: 23 | [unused] void Test18::C::g()
832*67e74705SXin Li //
833*67e74705SXin Li // CHECK-25: VTable indices for 'Test18::D' (2 entries).
834*67e74705SXin Li // CHECK-25-NEXT: 0 | void Test18::D::f()
835*67e74705SXin Li // CHECK-25-NEXT: 2 | void Test18::D::h()
836*67e74705SXin Li
837*67e74705SXin Li // CHECK-25: Construction vtable for ('Test18::B', 0) in 'Test18::D' (7 entries).
838*67e74705SXin Li // CHECK-25-NEXT: 0 | vbase_offset (0)
839*67e74705SXin Li // CHECK-25-NEXT: 1 | vcall_offset (0)
840*67e74705SXin Li // CHECK-25-NEXT: 2 | vcall_offset (0)
841*67e74705SXin Li // CHECK-25-NEXT: 3 | offset_to_top (0)
842*67e74705SXin Li // CHECK-25-NEXT: 4 | Test18::B RTTI
843*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::A, 0) vtable address --
844*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::B, 0) vtable address --
845*67e74705SXin Li // CHECK-25-NEXT: 5 | void Test18::B::f()
846*67e74705SXin Li // CHECK-25-NEXT: 6 | void Test18::A::g()
847*67e74705SXin Li
848*67e74705SXin Li // CHECK-25: Construction vtable for ('Test18::C', 8) in 'Test18::D' (20 entries).
849*67e74705SXin Li // CHECK-25-NEXT: 0 | vcall_offset (0)
850*67e74705SXin Li // CHECK-25-NEXT: 1 | vcall_offset (0)
851*67e74705SXin Li // CHECK-25-NEXT: 2 | vbase_offset (-8)
852*67e74705SXin Li // CHECK-25-NEXT: 3 | offset_to_top (0)
853*67e74705SXin Li // CHECK-25-NEXT: 4 | Test18::C RTTI
854*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::A, 8) vtable address --
855*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::C, 8) vtable address --
856*67e74705SXin Li // CHECK-25-NEXT: 5 | void Test18::A::f()
857*67e74705SXin Li // CHECK-25-NEXT: 6 | void Test18::C::g()
858*67e74705SXin Li // CHECK-25-NEXT: 7 | vbase_offset (-16)
859*67e74705SXin Li // CHECK-25-NEXT: 8 | vcall_offset (-8)
860*67e74705SXin Li // CHECK-25-NEXT: 9 | vcall_offset (0)
861*67e74705SXin Li // CHECK-25-NEXT: 10 | offset_to_top (-8)
862*67e74705SXin Li // CHECK-25-NEXT: 11 | Test18::C RTTI
863*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::B, 16) vtable address --
864*67e74705SXin Li // CHECK-25-NEXT: 12 | void Test18::B::f()
865*67e74705SXin Li // CHECK-25-NEXT: 13 | [unused] void Test18::C::g()
866*67e74705SXin Li // CHECK-25-NEXT: 14 | vcall_offset (8)
867*67e74705SXin Li // CHECK-25-NEXT: 15 | vcall_offset (16)
868*67e74705SXin Li // CHECK-25-NEXT: 16 | offset_to_top (8)
869*67e74705SXin Li // CHECK-25-NEXT: 17 | Test18::C RTTI
870*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::A, 0) vtable address --
871*67e74705SXin Li // CHECK-25-NEXT: 18 | void Test18::B::f()
872*67e74705SXin Li // CHECK-25-NEXT: [this adjustment: 0 non-virtual, -24 vcall offset offset]
873*67e74705SXin Li // CHECK-25-NEXT: 19 | void Test18::C::g()
874*67e74705SXin Li // CHECK-25-NEXT: [this adjustment: 0 non-virtual, -32 vcall offset offset]
875*67e74705SXin Li
876*67e74705SXin Li // CHECK-25: Construction vtable for ('Test18::B', 16) in 'Test18::D' (13 entries).
877*67e74705SXin Li // CHECK-25-NEXT: 0 | vbase_offset (-16)
878*67e74705SXin Li // CHECK-25-NEXT: 1 | vcall_offset (-16)
879*67e74705SXin Li // CHECK-25-NEXT: 2 | vcall_offset (0)
880*67e74705SXin Li // CHECK-25-NEXT: 3 | offset_to_top (0)
881*67e74705SXin Li // CHECK-25-NEXT: 4 | Test18::B RTTI
882*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::B, 16) vtable address --
883*67e74705SXin Li // CHECK-25-NEXT: 5 | void Test18::B::f()
884*67e74705SXin Li // CHECK-25-NEXT: 6 | [unused] void Test18::A::g()
885*67e74705SXin Li // CHECK-25-NEXT: 7 | vcall_offset (0)
886*67e74705SXin Li // CHECK-25-NEXT: 8 | vcall_offset (16)
887*67e74705SXin Li // CHECK-25-NEXT: 9 | offset_to_top (16)
888*67e74705SXin Li // CHECK-25-NEXT: 10 | Test18::B RTTI
889*67e74705SXin Li // CHECK-25-NEXT: -- (Test18::A, 0) vtable address --
890*67e74705SXin Li // CHECK-25-NEXT: 11 | void Test18::B::f()
891*67e74705SXin Li // CHECK-25-NEXT: [this adjustment: 0 non-virtual, -24 vcall offset offset]
892*67e74705SXin Li // CHECK-25-NEXT: 12 | void Test18::A::g()
893*67e74705SXin Li struct D : virtual B, virtual C, virtual A
894*67e74705SXin Li {
895*67e74705SXin Li virtual void f();
896*67e74705SXin Li virtual void h();
897*67e74705SXin Li };
f()898*67e74705SXin Li void D::f() {}
899*67e74705SXin Li
900*67e74705SXin Li }
901*67e74705SXin Li
902*67e74705SXin Li namespace Test19 {
903*67e74705SXin Li
904*67e74705SXin Li // Another 'this' adjustment test.
905*67e74705SXin Li
906*67e74705SXin Li struct A {
907*67e74705SXin Li int a;
908*67e74705SXin Li
909*67e74705SXin Li virtual void f();
910*67e74705SXin Li };
911*67e74705SXin Li
912*67e74705SXin Li struct B : A {
913*67e74705SXin Li int b;
914*67e74705SXin Li
915*67e74705SXin Li virtual void g();
916*67e74705SXin Li };
917*67e74705SXin Li
918*67e74705SXin Li struct C {
919*67e74705SXin Li virtual void c();
920*67e74705SXin Li };
921*67e74705SXin Li
922*67e74705SXin Li // CHECK-26: Vtable for 'Test19::D' (13 entries).
923*67e74705SXin Li // CHECK-26-NEXT: 0 | vbase_offset (24)
924*67e74705SXin Li // CHECK-26-NEXT: 1 | offset_to_top (0)
925*67e74705SXin Li // CHECK-26-NEXT: 2 | Test19::D RTTI
926*67e74705SXin Li // CHECK-26-NEXT: -- (Test19::C, 0) vtable address --
927*67e74705SXin Li // CHECK-26-NEXT: -- (Test19::D, 0) vtable address --
928*67e74705SXin Li // CHECK-26-NEXT: 3 | void Test19::C::c()
929*67e74705SXin Li // CHECK-26-NEXT: 4 | void Test19::D::f()
930*67e74705SXin Li // CHECK-26-NEXT: 5 | offset_to_top (-8)
931*67e74705SXin Li // CHECK-26-NEXT: 6 | Test19::D RTTI
932*67e74705SXin Li // CHECK-26-NEXT: -- (Test19::A, 8) vtable address --
933*67e74705SXin Li // CHECK-26-NEXT: -- (Test19::B, 8) vtable address --
934*67e74705SXin Li // CHECK-26-NEXT: 7 | void Test19::D::f()
935*67e74705SXin Li // CHECK-26-NEXT: [this adjustment: -8 non-virtual]
936*67e74705SXin Li // CHECK-26-NEXT: 8 | void Test19::B::g()
937*67e74705SXin Li // CHECK-26-NEXT: 9 | vcall_offset (-24)
938*67e74705SXin Li // CHECK-26-NEXT: 10 | offset_to_top (-24)
939*67e74705SXin Li // CHECK-26-NEXT: 11 | Test19::D RTTI
940*67e74705SXin Li // CHECK-26-NEXT: -- (Test19::A, 24) vtable address --
941*67e74705SXin Li // CHECK-26-NEXT: 12 | void Test19::D::f()
942*67e74705SXin Li // CHECK-26-NEXT: [this adjustment: 0 non-virtual, -24 vcall offset offset]
943*67e74705SXin Li //
944*67e74705SXin Li // CHECK-26: VTable indices for 'Test19::D' (1 entries).
945*67e74705SXin Li // CHECK-26-NEXT: 1 | void Test19::D::f()
946*67e74705SXin Li struct D : C, B, virtual A {
947*67e74705SXin Li virtual void f();
948*67e74705SXin Li };
f()949*67e74705SXin Li void D::f() { }
950*67e74705SXin Li
951*67e74705SXin Li }
952*67e74705SXin Li
953*67e74705SXin Li namespace Test20 {
954*67e74705SXin Li
955*67e74705SXin Li // pure virtual member functions should never have 'this' adjustments.
956*67e74705SXin Li
957*67e74705SXin Li struct A {
958*67e74705SXin Li virtual void f() = 0;
959*67e74705SXin Li virtual void g();
960*67e74705SXin Li };
961*67e74705SXin Li
962*67e74705SXin Li struct B : A { };
963*67e74705SXin Li
964*67e74705SXin Li // CHECK-27: Vtable for 'Test20::C' (9 entries).
965*67e74705SXin Li // CHECK-27-NEXT: 0 | offset_to_top (0)
966*67e74705SXin Li // CHECK-27-NEXT: 1 | Test20::C RTTI
967*67e74705SXin Li // CHECK-27-NEXT: -- (Test20::A, 0) vtable address --
968*67e74705SXin Li // CHECK-27-NEXT: -- (Test20::C, 0) vtable address --
969*67e74705SXin Li // CHECK-27-NEXT: 2 | void Test20::C::f() [pure]
970*67e74705SXin Li // CHECK-27-NEXT: 3 | void Test20::A::g()
971*67e74705SXin Li // CHECK-27-NEXT: 4 | void Test20::C::h()
972*67e74705SXin Li // CHECK-27-NEXT: 5 | offset_to_top (-8)
973*67e74705SXin Li // CHECK-27-NEXT: 6 | Test20::C RTTI
974*67e74705SXin Li // CHECK-27-NEXT: -- (Test20::A, 8) vtable address --
975*67e74705SXin Li // CHECK-27-NEXT: -- (Test20::B, 8) vtable address --
976*67e74705SXin Li // CHECK-27-NEXT: 7 | void Test20::C::f() [pure]
977*67e74705SXin Li // CHECK-27-NEXT: 8 | void Test20::A::g()
978*67e74705SXin Li //
979*67e74705SXin Li // CHECK-27: VTable indices for 'Test20::C' (2 entries).
980*67e74705SXin Li // CHECK-27-NEXT: 0 | void Test20::C::f()
981*67e74705SXin Li // CHECK-27-NEXT: 2 | void Test20::C::h()
982*67e74705SXin Li struct C : A, B {
983*67e74705SXin Li virtual void f() = 0;
984*67e74705SXin Li virtual void h();
985*67e74705SXin Li };
h()986*67e74705SXin Li void C::h() { }
987*67e74705SXin Li
988*67e74705SXin Li }
989*67e74705SXin Li
990*67e74705SXin Li namespace Test21 {
991*67e74705SXin Li
992*67e74705SXin Li // Test that we get vbase offsets right in secondary vtables.
993*67e74705SXin Li struct A {
994*67e74705SXin Li virtual void f();
995*67e74705SXin Li };
996*67e74705SXin Li
997*67e74705SXin Li struct B : virtual A { };
998*67e74705SXin Li class C : virtual B { };
999*67e74705SXin Li class D : virtual C { };
1000*67e74705SXin Li
1001*67e74705SXin Li class E : virtual C { };
1002*67e74705SXin Li
1003*67e74705SXin Li // CHECK-28: Vtable for 'Test21::F' (16 entries).
1004*67e74705SXin Li // CHECK-28-NEXT: 0 | vbase_offset (8)
1005*67e74705SXin Li // CHECK-28-NEXT: 1 | vbase_offset (0)
1006*67e74705SXin Li // CHECK-28-NEXT: 2 | vbase_offset (0)
1007*67e74705SXin Li // CHECK-28-NEXT: 3 | vbase_offset (0)
1008*67e74705SXin Li // CHECK-28-NEXT: 4 | vbase_offset (0)
1009*67e74705SXin Li // CHECK-28-NEXT: 5 | vcall_offset (0)
1010*67e74705SXin Li // CHECK-28-NEXT: 6 | offset_to_top (0)
1011*67e74705SXin Li // CHECK-28-NEXT: 7 | Test21::F RTTI
1012*67e74705SXin Li // CHECK-28-NEXT: -- (Test21::A, 0) vtable address --
1013*67e74705SXin Li // CHECK-28-NEXT: -- (Test21::B, 0) vtable address --
1014*67e74705SXin Li // CHECK-28-NEXT: -- (Test21::C, 0) vtable address --
1015*67e74705SXin Li // CHECK-28-NEXT: -- (Test21::D, 0) vtable address --
1016*67e74705SXin Li // CHECK-28-NEXT: -- (Test21::F, 0) vtable address --
1017*67e74705SXin Li // CHECK-28-NEXT: 8 | void Test21::F::f()
1018*67e74705SXin Li // CHECK-28-NEXT: 9 | vbase_offset (-8)
1019*67e74705SXin Li // CHECK-28-NEXT: 10 | vbase_offset (-8)
1020*67e74705SXin Li // CHECK-28-NEXT: 11 | vbase_offset (-8)
1021*67e74705SXin Li // CHECK-28-NEXT: 12 | vcall_offset (-8)
1022*67e74705SXin Li // CHECK-28-NEXT: 13 | offset_to_top (-8)
1023*67e74705SXin Li // CHECK-28-NEXT: 14 | Test21::F RTTI
1024*67e74705SXin Li // CHECK-28-NEXT: -- (Test21::E, 8) vtable address --
1025*67e74705SXin Li // CHECK-28-NEXT: 15 | [unused] void Test21::F::f()
1026*67e74705SXin Li //
1027*67e74705SXin Li // CHECK-28: Virtual base offset offsets for 'Test21::F' (5 entries).
1028*67e74705SXin Li // CHECK-28-NEXT: Test21::A | -32
1029*67e74705SXin Li // CHECK-28-NEXT: Test21::B | -40
1030*67e74705SXin Li // CHECK-28-NEXT: Test21::C | -48
1031*67e74705SXin Li // CHECK-28-NEXT: Test21::D | -56
1032*67e74705SXin Li // CHECK-28-NEXT: Test21::E | -64
1033*67e74705SXin Li //
1034*67e74705SXin Li // CHECK-28: VTable indices for 'Test21::F' (1 entries).
1035*67e74705SXin Li // CHECK-28-NEXT: 0 | void Test21::F::f()
1036*67e74705SXin Li class F : virtual D, virtual E {
1037*67e74705SXin Li virtual void f();
1038*67e74705SXin Li };
f()1039*67e74705SXin Li void F::f() { }
1040*67e74705SXin Li
1041*67e74705SXin Li }
1042*67e74705SXin Li
1043*67e74705SXin Li namespace Test22 {
1044*67e74705SXin Li
1045*67e74705SXin Li // Very simple construction vtable test.
1046*67e74705SXin Li struct V1 {
1047*67e74705SXin Li int v1;
1048*67e74705SXin Li };
1049*67e74705SXin Li
1050*67e74705SXin Li struct V2 : virtual V1 {
1051*67e74705SXin Li int v2;
1052*67e74705SXin Li };
1053*67e74705SXin Li
1054*67e74705SXin Li // CHECK-29: Vtable for 'Test22::C' (8 entries).
1055*67e74705SXin Li // CHECK-29-NEXT: 0 | vbase_offset (16)
1056*67e74705SXin Li // CHECK-29-NEXT: 1 | vbase_offset (12)
1057*67e74705SXin Li // CHECK-29-NEXT: 2 | offset_to_top (0)
1058*67e74705SXin Li // CHECK-29-NEXT: 3 | Test22::C RTTI
1059*67e74705SXin Li // CHECK-29-NEXT: -- (Test22::C, 0) vtable address --
1060*67e74705SXin Li // CHECK-29-NEXT: 4 | void Test22::C::f()
1061*67e74705SXin Li // CHECK-29-NEXT: 5 | vbase_offset (-4)
1062*67e74705SXin Li // CHECK-29-NEXT: 6 | offset_to_top (-16)
1063*67e74705SXin Li // CHECK-29-NEXT: 7 | Test22::C RTTI
1064*67e74705SXin Li // CHECK-29-NEXT: -- (Test22::V2, 16) vtable address --
1065*67e74705SXin Li //
1066*67e74705SXin Li // CHECK-29: VTable indices for 'Test22::C' (1 entries).
1067*67e74705SXin Li // CHECK-29-NEXT: 0 | void Test22::C::f()
1068*67e74705SXin Li
1069*67e74705SXin Li // CHECK-29: Construction vtable for ('Test22::V2', 16) in 'Test22::C' (3 entries).
1070*67e74705SXin Li // CHECK-29-NEXT: 0 | vbase_offset (-4)
1071*67e74705SXin Li // CHECK-29-NEXT: 1 | offset_to_top (0)
1072*67e74705SXin Li // CHECK-29-NEXT: 2 | Test22::V2 RTTI
1073*67e74705SXin Li
1074*67e74705SXin Li struct C : virtual V1, virtual V2 {
1075*67e74705SXin Li int c;
1076*67e74705SXin Li virtual void f();
1077*67e74705SXin Li };
f()1078*67e74705SXin Li void C::f() { }
1079*67e74705SXin Li
1080*67e74705SXin Li }
1081*67e74705SXin Li
1082*67e74705SXin Li namespace Test23 {
1083*67e74705SXin Li
1084*67e74705SXin Li struct A {
1085*67e74705SXin Li int a;
1086*67e74705SXin Li };
1087*67e74705SXin Li
1088*67e74705SXin Li struct B : virtual A {
1089*67e74705SXin Li int b;
1090*67e74705SXin Li };
1091*67e74705SXin Li
1092*67e74705SXin Li struct C : A, virtual B {
1093*67e74705SXin Li int c;
1094*67e74705SXin Li };
1095*67e74705SXin Li
1096*67e74705SXin Li // CHECK-30: Vtable for 'Test23::D' (7 entries).
1097*67e74705SXin Li // CHECK-30-NEXT: 0 | vbase_offset (20)
1098*67e74705SXin Li // CHECK-30-NEXT: 1 | vbase_offset (24)
1099*67e74705SXin Li // CHECK-30-NEXT: 2 | offset_to_top (0)
1100*67e74705SXin Li // CHECK-30-NEXT: 3 | Test23::D RTTI
1101*67e74705SXin Li // CHECK-30-NEXT: -- (Test23::C, 0) vtable address --
1102*67e74705SXin Li // CHECK-30-NEXT: -- (Test23::D, 0) vtable address --
1103*67e74705SXin Li // CHECK-30-NEXT: 4 | vbase_offset (-4)
1104*67e74705SXin Li // CHECK-30-NEXT: 5 | offset_to_top (-24)
1105*67e74705SXin Li // CHECK-30-NEXT: 6 | Test23::D RTTI
1106*67e74705SXin Li // CHECK-30-NEXT: -- (Test23::B, 24) vtable address --
1107*67e74705SXin Li
1108*67e74705SXin Li // CHECK-30: Construction vtable for ('Test23::C', 0) in 'Test23::D' (7 entries).
1109*67e74705SXin Li // CHECK-30-NEXT: 0 | vbase_offset (20)
1110*67e74705SXin Li // CHECK-30-NEXT: 1 | vbase_offset (24)
1111*67e74705SXin Li // CHECK-30-NEXT: 2 | offset_to_top (0)
1112*67e74705SXin Li // CHECK-30-NEXT: 3 | Test23::C RTTI
1113*67e74705SXin Li // CHECK-30-NEXT: -- (Test23::C, 0) vtable address --
1114*67e74705SXin Li // CHECK-30-NEXT: 4 | vbase_offset (-4)
1115*67e74705SXin Li // CHECK-30-NEXT: 5 | offset_to_top (-24)
1116*67e74705SXin Li // CHECK-30-NEXT: 6 | Test23::C RTTI
1117*67e74705SXin Li // CHECK-30-NEXT: -- (Test23::B, 24) vtable address --
1118*67e74705SXin Li
1119*67e74705SXin Li // CHECK-30: Construction vtable for ('Test23::B', 24) in 'Test23::D' (3 entries).
1120*67e74705SXin Li // CHECK-30-NEXT: 0 | vbase_offset (-4)
1121*67e74705SXin Li // CHECK-30-NEXT: 1 | offset_to_top (0)
1122*67e74705SXin Li // CHECK-30-NEXT: 2 | Test23::B RTTI
1123*67e74705SXin Li // CHECK-30-NEXT: -- (Test23::B, 24) vtable address --
1124*67e74705SXin Li
1125*67e74705SXin Li struct D : virtual A, virtual B, C {
1126*67e74705SXin Li int d;
1127*67e74705SXin Li
1128*67e74705SXin Li void f();
1129*67e74705SXin Li };
f()1130*67e74705SXin Li void D::f() { }
1131*67e74705SXin Li D d;
1132*67e74705SXin Li }
1133*67e74705SXin Li
1134*67e74705SXin Li namespace Test24 {
1135*67e74705SXin Li
1136*67e74705SXin Li // Another construction vtable test.
1137*67e74705SXin Li
1138*67e74705SXin Li struct A {
1139*67e74705SXin Li virtual void f();
1140*67e74705SXin Li };
1141*67e74705SXin Li
1142*67e74705SXin Li struct B : virtual A { };
1143*67e74705SXin Li struct C : virtual A { };
1144*67e74705SXin Li
1145*67e74705SXin Li // CHECK-31: Vtable for 'Test24::D' (10 entries).
1146*67e74705SXin Li // CHECK-31-NEXT: 0 | vbase_offset (0)
1147*67e74705SXin Li // CHECK-31-NEXT: 1 | vcall_offset (0)
1148*67e74705SXin Li // CHECK-31-NEXT: 2 | offset_to_top (0)
1149*67e74705SXin Li // CHECK-31-NEXT: 3 | Test24::D RTTI
1150*67e74705SXin Li // CHECK-31-NEXT: -- (Test24::A, 0) vtable address --
1151*67e74705SXin Li // CHECK-31-NEXT: -- (Test24::B, 0) vtable address --
1152*67e74705SXin Li // CHECK-31-NEXT: -- (Test24::D, 0) vtable address --
1153*67e74705SXin Li // CHECK-31-NEXT: 4 | void Test24::D::f()
1154*67e74705SXin Li // CHECK-31-NEXT: 5 | vbase_offset (-8)
1155*67e74705SXin Li // CHECK-31-NEXT: 6 | vcall_offset (-8)
1156*67e74705SXin Li // CHECK-31-NEXT: 7 | offset_to_top (-8)
1157*67e74705SXin Li // CHECK-31-NEXT: 8 | Test24::D RTTI
1158*67e74705SXin Li // CHECK-31-NEXT: -- (Test24::C, 8) vtable address --
1159*67e74705SXin Li // CHECK-31-NEXT: 9 | [unused] void Test24::D::f()
1160*67e74705SXin Li //
1161*67e74705SXin Li // CHECK-31: VTable indices for 'Test24::D' (1 entries).
1162*67e74705SXin Li // CHECK-31-NEXT: 0 | void Test24::D::f()
1163*67e74705SXin Li
1164*67e74705SXin Li // CHECK-31: Construction vtable for ('Test24::B', 0) in 'Test24::D' (5 entries).
1165*67e74705SXin Li // CHECK-31-NEXT: 0 | vbase_offset (0)
1166*67e74705SXin Li // CHECK-31-NEXT: 1 | vcall_offset (0)
1167*67e74705SXin Li // CHECK-31-NEXT: 2 | offset_to_top (0)
1168*67e74705SXin Li // CHECK-31-NEXT: 3 | Test24::B RTTI
1169*67e74705SXin Li // CHECK-31-NEXT: -- (Test24::A, 0) vtable address --
1170*67e74705SXin Li // CHECK-31-NEXT: -- (Test24::B, 0) vtable address --
1171*67e74705SXin Li // CHECK-31-NEXT: 4 | void Test24::A::f()
1172*67e74705SXin Li
1173*67e74705SXin Li // CHECK-31: Construction vtable for ('Test24::C', 8) in 'Test24::D' (9 entries).
1174*67e74705SXin Li // CHECK-31-NEXT: 0 | vbase_offset (-8)
1175*67e74705SXin Li // CHECK-31-NEXT: 1 | vcall_offset (-8)
1176*67e74705SXin Li // CHECK-31-NEXT: 2 | offset_to_top (0)
1177*67e74705SXin Li // CHECK-31-NEXT: 3 | Test24::C RTTI
1178*67e74705SXin Li // CHECK-31-NEXT: -- (Test24::C, 8) vtable address --
1179*67e74705SXin Li // CHECK-31-NEXT: 4 | [unused] void Test24::A::f()
1180*67e74705SXin Li // CHECK-31-NEXT: 5 | vcall_offset (0)
1181*67e74705SXin Li // CHECK-31-NEXT: 6 | offset_to_top (8)
1182*67e74705SXin Li // CHECK-31-NEXT: 7 | Test24::C RTTI
1183*67e74705SXin Li // CHECK-31-NEXT: -- (Test24::A, 0) vtable address --
1184*67e74705SXin Li // CHECK-31-NEXT: 8 | void Test24::A::f()
1185*67e74705SXin Li struct D : B, C {
1186*67e74705SXin Li virtual void f();
1187*67e74705SXin Li };
f()1188*67e74705SXin Li void D::f() { }
1189*67e74705SXin Li
1190*67e74705SXin Li }
1191*67e74705SXin Li
1192*67e74705SXin Li namespace Test25 {
1193*67e74705SXin Li
1194*67e74705SXin Li // This mainly tests that we don't assert on this class hierarchy.
1195*67e74705SXin Li
1196*67e74705SXin Li struct V {
1197*67e74705SXin Li virtual void f();
1198*67e74705SXin Li };
1199*67e74705SXin Li
1200*67e74705SXin Li struct A : virtual V { };
1201*67e74705SXin Li struct B : virtual V { };
1202*67e74705SXin Li
1203*67e74705SXin Li // CHECK-32: Vtable for 'Test25::C' (11 entries).
1204*67e74705SXin Li // CHECK-32-NEXT: 0 | vbase_offset (0)
1205*67e74705SXin Li // CHECK-32-NEXT: 1 | vcall_offset (0)
1206*67e74705SXin Li // CHECK-32-NEXT: 2 | offset_to_top (0)
1207*67e74705SXin Li // CHECK-32-NEXT: 3 | Test25::C RTTI
1208*67e74705SXin Li // CHECK-32-NEXT: -- (Test25::A, 0) vtable address --
1209*67e74705SXin Li // CHECK-32-NEXT: -- (Test25::C, 0) vtable address --
1210*67e74705SXin Li // CHECK-32-NEXT: -- (Test25::V, 0) vtable address --
1211*67e74705SXin Li // CHECK-32-NEXT: 4 | void Test25::V::f()
1212*67e74705SXin Li // CHECK-32-NEXT: 5 | void Test25::C::g()
1213*67e74705SXin Li // CHECK-32-NEXT: 6 | vbase_offset (-8)
1214*67e74705SXin Li // CHECK-32-NEXT: 7 | vcall_offset (-8)
1215*67e74705SXin Li // CHECK-32-NEXT: 8 | offset_to_top (-8)
1216*67e74705SXin Li // CHECK-32-NEXT: 9 | Test25::C RTTI
1217*67e74705SXin Li // CHECK-32-NEXT: -- (Test25::B, 8) vtable address --
1218*67e74705SXin Li // CHECK-32-NEXT: 10 | [unused] void Test25::V::f()
1219*67e74705SXin Li //
1220*67e74705SXin Li // CHECK-32: VTable indices for 'Test25::C' (1 entries).
1221*67e74705SXin Li // CHECK-32-NEXT: 1 | void Test25::C::g()
1222*67e74705SXin Li
1223*67e74705SXin Li // CHECK-32: Construction vtable for ('Test25::A', 0) in 'Test25::C' (5 entries).
1224*67e74705SXin Li // CHECK-32-NEXT: 0 | vbase_offset (0)
1225*67e74705SXin Li // CHECK-32-NEXT: 1 | vcall_offset (0)
1226*67e74705SXin Li // CHECK-32-NEXT: 2 | offset_to_top (0)
1227*67e74705SXin Li // CHECK-32-NEXT: 3 | Test25::A RTTI
1228*67e74705SXin Li // CHECK-32-NEXT: -- (Test25::A, 0) vtable address --
1229*67e74705SXin Li // CHECK-32-NEXT: -- (Test25::V, 0) vtable address --
1230*67e74705SXin Li // CHECK-32-NEXT: 4 | void Test25::V::f()
1231*67e74705SXin Li
1232*67e74705SXin Li // CHECK-32: Construction vtable for ('Test25::B', 8) in 'Test25::C' (9 entries).
1233*67e74705SXin Li // CHECK-32-NEXT: 0 | vbase_offset (-8)
1234*67e74705SXin Li // CHECK-32-NEXT: 1 | vcall_offset (-8)
1235*67e74705SXin Li // CHECK-32-NEXT: 2 | offset_to_top (0)
1236*67e74705SXin Li // CHECK-32-NEXT: 3 | Test25::B RTTI
1237*67e74705SXin Li // CHECK-32-NEXT: -- (Test25::B, 8) vtable address --
1238*67e74705SXin Li // CHECK-32-NEXT: 4 | [unused] void Test25::V::f()
1239*67e74705SXin Li // CHECK-32-NEXT: 5 | vcall_offset (0)
1240*67e74705SXin Li // CHECK-32-NEXT: 6 | offset_to_top (8)
1241*67e74705SXin Li // CHECK-32-NEXT: 7 | Test25::B RTTI
1242*67e74705SXin Li // CHECK-32-NEXT: -- (Test25::V, 0) vtable address --
1243*67e74705SXin Li // CHECK-32-NEXT: 8 | void Test25::V::f()
1244*67e74705SXin Li struct C : A, virtual V, B {
1245*67e74705SXin Li virtual void g();
1246*67e74705SXin Li };
g()1247*67e74705SXin Li void C::g() { }
1248*67e74705SXin Li
1249*67e74705SXin Li }
1250*67e74705SXin Li
1251*67e74705SXin Li namespace Test26 {
1252*67e74705SXin Li
1253*67e74705SXin Li // Test that we generate the right number of entries in the C-in-D construction vtable, and that
1254*67e74705SXin Li // we don't mark A::a as unused.
1255*67e74705SXin Li
1256*67e74705SXin Li struct A {
1257*67e74705SXin Li virtual void a();
1258*67e74705SXin Li };
1259*67e74705SXin Li
1260*67e74705SXin Li struct B {
1261*67e74705SXin Li virtual void c();
1262*67e74705SXin Li };
1263*67e74705SXin Li
1264*67e74705SXin Li struct C : virtual A {
1265*67e74705SXin Li virtual void b();
1266*67e74705SXin Li };
1267*67e74705SXin Li
1268*67e74705SXin Li // CHECK-33: Vtable for 'Test26::D' (15 entries).
1269*67e74705SXin Li // CHECK-33-NEXT: 0 | vbase_offset (8)
1270*67e74705SXin Li // CHECK-33-NEXT: 1 | vbase_offset (8)
1271*67e74705SXin Li // CHECK-33-NEXT: 2 | vbase_offset (0)
1272*67e74705SXin Li // CHECK-33-NEXT: 3 | vcall_offset (0)
1273*67e74705SXin Li // CHECK-33-NEXT: 4 | offset_to_top (0)
1274*67e74705SXin Li // CHECK-33-NEXT: 5 | Test26::D RTTI
1275*67e74705SXin Li // CHECK-33-NEXT: -- (Test26::B, 0) vtable address --
1276*67e74705SXin Li // CHECK-33-NEXT: -- (Test26::D, 0) vtable address --
1277*67e74705SXin Li // CHECK-33-NEXT: 6 | void Test26::B::c()
1278*67e74705SXin Li // CHECK-33-NEXT: 7 | void Test26::D::d()
1279*67e74705SXin Li // CHECK-33-NEXT: 8 | vcall_offset (0)
1280*67e74705SXin Li // CHECK-33-NEXT: 9 | vbase_offset (0)
1281*67e74705SXin Li // CHECK-33-NEXT: 10 | vcall_offset (0)
1282*67e74705SXin Li // CHECK-33-NEXT: 11 | offset_to_top (-8)
1283*67e74705SXin Li // CHECK-33-NEXT: 12 | Test26::D RTTI
1284*67e74705SXin Li // CHECK-33-NEXT: -- (Test26::A, 8) vtable address --
1285*67e74705SXin Li // CHECK-33-NEXT: -- (Test26::C, 8) vtable address --
1286*67e74705SXin Li // CHECK-33-NEXT: 13 | void Test26::A::a()
1287*67e74705SXin Li // CHECK-33-NEXT: 14 | void Test26::C::b()
1288*67e74705SXin Li //
1289*67e74705SXin Li // CHECK-33: VTable indices for 'Test26::D' (1 entries).
1290*67e74705SXin Li // CHECK-33-NEXT: 1 | void Test26::D::d()
1291*67e74705SXin Li
1292*67e74705SXin Li // CHECK-33: Construction vtable for ('Test26::C', 8) in 'Test26::D' (7 entries).
1293*67e74705SXin Li // CHECK-33-NEXT: 0 | vcall_offset (0)
1294*67e74705SXin Li // CHECK-33-NEXT: 1 | vbase_offset (0)
1295*67e74705SXin Li // CHECK-33-NEXT: 2 | vcall_offset (0)
1296*67e74705SXin Li // CHECK-33-NEXT: 3 | offset_to_top (0)
1297*67e74705SXin Li // CHECK-33-NEXT: 4 | Test26::C RTTI
1298*67e74705SXin Li // CHECK-33-NEXT: -- (Test26::A, 8) vtable address --
1299*67e74705SXin Li // CHECK-33-NEXT: -- (Test26::C, 8) vtable address --
1300*67e74705SXin Li // CHECK-33-NEXT: 5 | void Test26::A::a()
1301*67e74705SXin Li // CHECK-33-NEXT: 6 | void Test26::C::b()
1302*67e74705SXin Li class D : virtual B, virtual C {
1303*67e74705SXin Li virtual void d();
1304*67e74705SXin Li };
d()1305*67e74705SXin Li void D::d() { }
1306*67e74705SXin Li
1307*67e74705SXin Li }
1308*67e74705SXin Li
1309*67e74705SXin Li namespace Test27 {
1310*67e74705SXin Li
1311*67e74705SXin Li // Test that we don't generate a secondary vtable for C in the D-in-E vtable, since
1312*67e74705SXin Li // C doesn't have any virtual bases.
1313*67e74705SXin Li
1314*67e74705SXin Li struct A {
1315*67e74705SXin Li virtual void a();
1316*67e74705SXin Li };
1317*67e74705SXin Li
1318*67e74705SXin Li struct B {
1319*67e74705SXin Li virtual void b();
1320*67e74705SXin Li };
1321*67e74705SXin Li
1322*67e74705SXin Li struct C {
1323*67e74705SXin Li virtual void c();
1324*67e74705SXin Li };
1325*67e74705SXin Li
1326*67e74705SXin Li struct D : A, virtual B, C {
1327*67e74705SXin Li virtual void d();
1328*67e74705SXin Li };
1329*67e74705SXin Li
1330*67e74705SXin Li // CHECK-34: Vtable for 'Test27::E' (13 entries).
1331*67e74705SXin Li // CHECK-34-NEXT: 0 | vbase_offset (16)
1332*67e74705SXin Li // CHECK-34-NEXT: 1 | offset_to_top (0)
1333*67e74705SXin Li // CHECK-34-NEXT: 2 | Test27::E RTTI
1334*67e74705SXin Li // CHECK-34-NEXT: -- (Test27::A, 0) vtable address --
1335*67e74705SXin Li // CHECK-34-NEXT: -- (Test27::D, 0) vtable address --
1336*67e74705SXin Li // CHECK-34-NEXT: -- (Test27::E, 0) vtable address --
1337*67e74705SXin Li // CHECK-34-NEXT: 3 | void Test27::A::a()
1338*67e74705SXin Li // CHECK-34-NEXT: 4 | void Test27::D::d()
1339*67e74705SXin Li // CHECK-34-NEXT: 5 | void Test27::E::e()
1340*67e74705SXin Li // CHECK-34-NEXT: 6 | offset_to_top (-8)
1341*67e74705SXin Li // CHECK-34-NEXT: 7 | Test27::E RTTI
1342*67e74705SXin Li // CHECK-34-NEXT: -- (Test27::C, 8) vtable address --
1343*67e74705SXin Li // CHECK-34-NEXT: 8 | void Test27::C::c()
1344*67e74705SXin Li // CHECK-34-NEXT: 9 | vcall_offset (0)
1345*67e74705SXin Li // CHECK-34-NEXT: 10 | offset_to_top (-16)
1346*67e74705SXin Li // CHECK-34-NEXT: 11 | Test27::E RTTI
1347*67e74705SXin Li // CHECK-34-NEXT: -- (Test27::B, 16) vtable address --
1348*67e74705SXin Li // CHECK-34-NEXT: 12 | void Test27::B::b()
1349*67e74705SXin Li //
1350*67e74705SXin Li // CHECK-34: VTable indices for 'Test27::E' (1 entries).
1351*67e74705SXin Li // CHECK-34-NEXT: 2 | void Test27::E::e()
1352*67e74705SXin Li
1353*67e74705SXin Li // CHECK-34: Construction vtable for ('Test27::D', 0) in 'Test27::E' (9 entries).
1354*67e74705SXin Li // CHECK-34-NEXT: 0 | vbase_offset (16)
1355*67e74705SXin Li // CHECK-34-NEXT: 1 | offset_to_top (0)
1356*67e74705SXin Li // CHECK-34-NEXT: 2 | Test27::D RTTI
1357*67e74705SXin Li // CHECK-34-NEXT: -- (Test27::A, 0) vtable address --
1358*67e74705SXin Li // CHECK-34-NEXT: -- (Test27::D, 0) vtable address --
1359*67e74705SXin Li // CHECK-34-NEXT: 3 | void Test27::A::a()
1360*67e74705SXin Li // CHECK-34-NEXT: 4 | void Test27::D::d()
1361*67e74705SXin Li // CHECK-34-NEXT: 5 | vcall_offset (0)
1362*67e74705SXin Li // CHECK-34-NEXT: 6 | offset_to_top (-16)
1363*67e74705SXin Li // CHECK-34-NEXT: 7 | Test27::D RTTI
1364*67e74705SXin Li // CHECK-34-NEXT: -- (Test27::B, 16) vtable address --
1365*67e74705SXin Li // CHECK-34-NEXT: 8 | void Test27::B::b()
1366*67e74705SXin Li struct E : D {
1367*67e74705SXin Li virtual void e();
1368*67e74705SXin Li };
e()1369*67e74705SXin Li void E::e() { }
1370*67e74705SXin Li
1371*67e74705SXin Li }
1372*67e74705SXin Li
1373*67e74705SXin Li namespace Test28 {
1374*67e74705SXin Li
1375*67e74705SXin Li // Check that we do include the vtable for B in the D-in-E construction vtable, since
1376*67e74705SXin Li // B is a base class of a virtual base (C).
1377*67e74705SXin Li
1378*67e74705SXin Li struct A {
1379*67e74705SXin Li virtual void a();
1380*67e74705SXin Li };
1381*67e74705SXin Li
1382*67e74705SXin Li struct B {
1383*67e74705SXin Li virtual void b();
1384*67e74705SXin Li };
1385*67e74705SXin Li
1386*67e74705SXin Li struct C : A, B {
1387*67e74705SXin Li virtual void c();
1388*67e74705SXin Li };
1389*67e74705SXin Li
1390*67e74705SXin Li struct D : virtual C {
1391*67e74705SXin Li };
1392*67e74705SXin Li
1393*67e74705SXin Li // CHECK-35: Vtable for 'Test28::E' (14 entries).
1394*67e74705SXin Li // CHECK-35-NEXT: 0 | vbase_offset (8)
1395*67e74705SXin Li // CHECK-35-NEXT: 1 | offset_to_top (0)
1396*67e74705SXin Li // CHECK-35-NEXT: 2 | Test28::E RTTI
1397*67e74705SXin Li // CHECK-35-NEXT: -- (Test28::D, 0) vtable address --
1398*67e74705SXin Li // CHECK-35-NEXT: -- (Test28::E, 0) vtable address --
1399*67e74705SXin Li // CHECK-35-NEXT: 3 | void Test28::E::e()
1400*67e74705SXin Li // CHECK-35-NEXT: 4 | vcall_offset (8)
1401*67e74705SXin Li // CHECK-35-NEXT: 5 | vcall_offset (0)
1402*67e74705SXin Li // CHECK-35-NEXT: 6 | vcall_offset (0)
1403*67e74705SXin Li // CHECK-35-NEXT: 7 | offset_to_top (-8)
1404*67e74705SXin Li // CHECK-35-NEXT: 8 | Test28::E RTTI
1405*67e74705SXin Li // CHECK-35-NEXT: -- (Test28::A, 8) vtable address --
1406*67e74705SXin Li // CHECK-35-NEXT: -- (Test28::C, 8) vtable address --
1407*67e74705SXin Li // CHECK-35-NEXT: 9 | void Test28::A::a()
1408*67e74705SXin Li // CHECK-35-NEXT: 10 | void Test28::C::c()
1409*67e74705SXin Li // CHECK-35-NEXT: 11 | offset_to_top (-16)
1410*67e74705SXin Li // CHECK-35-NEXT: 12 | Test28::E RTTI
1411*67e74705SXin Li // CHECK-35-NEXT: -- (Test28::B, 16) vtable address --
1412*67e74705SXin Li // CHECK-35-NEXT: 13 | void Test28::B::b()
1413*67e74705SXin Li //
1414*67e74705SXin Li // CHECK-35: VTable indices for 'Test28::E' (1 entries).
1415*67e74705SXin Li // CHECK-35-NEXT : 0 | void Test28::E::e()
1416*67e74705SXin Li
1417*67e74705SXin Li // CHECK-35: Construction vtable for ('Test28::D', 0) in 'Test28::E' (13 entries).
1418*67e74705SXin Li // CHECK-35-NEXT: 0 | vbase_offset (8)
1419*67e74705SXin Li // CHECK-35-NEXT: 1 | offset_to_top (0)
1420*67e74705SXin Li // CHECK-35-NEXT: 2 | Test28::D RTTI
1421*67e74705SXin Li // CHECK-35-NEXT: -- (Test28::D, 0) vtable address --
1422*67e74705SXin Li // CHECK-35-NEXT: 3 | vcall_offset (8)
1423*67e74705SXin Li // CHECK-35-NEXT: 4 | vcall_offset (0)
1424*67e74705SXin Li // CHECK-35-NEXT: 5 | vcall_offset (0)
1425*67e74705SXin Li // CHECK-35-NEXT: 6 | offset_to_top (-8)
1426*67e74705SXin Li // CHECK-35-NEXT: 7 | Test28::D RTTI
1427*67e74705SXin Li // CHECK-35-NEXT: -- (Test28::A, 8) vtable address --
1428*67e74705SXin Li // CHECK-35-NEXT: -- (Test28::C, 8) vtable address --
1429*67e74705SXin Li // CHECK-35-NEXT: 8 | void Test28::A::a()
1430*67e74705SXin Li // CHECK-35-NEXT: 9 | void Test28::C::c()
1431*67e74705SXin Li // CHECK-35-NEXT: 10 | offset_to_top (-16)
1432*67e74705SXin Li // CHECK-35-NEXT: 11 | Test28::D RTTI
1433*67e74705SXin Li // CHECK-35-NEXT: -- (Test28::B, 16) vtable address --
1434*67e74705SXin Li // CHECK-35-NEXT: 12 | void Test28::B::b()
1435*67e74705SXin Li struct E : D {
1436*67e74705SXin Li virtual void e();
1437*67e74705SXin Li };
e()1438*67e74705SXin Li void E::e() { }
1439*67e74705SXin Li
1440*67e74705SXin Li }
1441*67e74705SXin Li
1442*67e74705SXin Li namespace Test29 {
1443*67e74705SXin Li
1444*67e74705SXin Li // Test that the covariant return thunk for B::f will have a virtual 'this' adjustment,
1445*67e74705SXin Li // matching gcc.
1446*67e74705SXin Li
1447*67e74705SXin Li struct V1 { };
1448*67e74705SXin Li struct V2 : virtual V1 { };
1449*67e74705SXin Li
1450*67e74705SXin Li struct A {
1451*67e74705SXin Li virtual V1 *f();
1452*67e74705SXin Li };
1453*67e74705SXin Li
1454*67e74705SXin Li // CHECK-36: Vtable for 'Test29::B' (6 entries).
1455*67e74705SXin Li // CHECK-36-NEXT: 0 | vbase_offset (0)
1456*67e74705SXin Li // CHECK-36-NEXT: 1 | vcall_offset (0)
1457*67e74705SXin Li // CHECK-36-NEXT: 2 | offset_to_top (0)
1458*67e74705SXin Li // CHECK-36-NEXT: 3 | Test29::B RTTI
1459*67e74705SXin Li // CHECK-36-NEXT: -- (Test29::A, 0) vtable address --
1460*67e74705SXin Li // CHECK-36-NEXT: -- (Test29::B, 0) vtable address --
1461*67e74705SXin Li // CHECK-36-NEXT: 4 | Test29::V2 *Test29::B::f()
1462*67e74705SXin Li // CHECK-36-NEXT: [return adjustment: 0 non-virtual, -24 vbase offset offset]
1463*67e74705SXin Li // CHECK-36-NEXT: [this adjustment: 0 non-virtual, -24 vcall offset offset]
1464*67e74705SXin Li // CHECK-36-NEXT: 5 | Test29::V2 *Test29::B::f()
1465*67e74705SXin Li //
1466*67e74705SXin Li // CHECK-36: VTable indices for 'Test29::B' (1 entries).
1467*67e74705SXin Li // CHECK-36-NEXT: 1 | Test29::V2 *Test29::B::f()
1468*67e74705SXin Li struct B : virtual A {
1469*67e74705SXin Li virtual V2 *f();
1470*67e74705SXin Li };
f()1471*67e74705SXin Li V2 *B::f() { return 0; }
1472*67e74705SXin Li
1473*67e74705SXin Li }
1474*67e74705SXin Li
1475*67e74705SXin Li namespace Test30 {
1476*67e74705SXin Li
1477*67e74705SXin Li // Test that we don't assert when generating a vtable for F.
1478*67e74705SXin Li struct A { };
1479*67e74705SXin Li
1480*67e74705SXin Li struct B : virtual A {
1481*67e74705SXin Li int i;
1482*67e74705SXin Li };
1483*67e74705SXin Li
1484*67e74705SXin Li struct C {
1485*67e74705SXin Li virtual void f();
1486*67e74705SXin Li };
1487*67e74705SXin Li
1488*67e74705SXin Li struct D : virtual C, B { };
1489*67e74705SXin Li struct E : virtual D { };
1490*67e74705SXin Li
1491*67e74705SXin Li struct F : E {
1492*67e74705SXin Li virtual void f();
1493*67e74705SXin Li };
f()1494*67e74705SXin Li void F::f() { }
1495*67e74705SXin Li
1496*67e74705SXin Li }
1497*67e74705SXin Li
1498*67e74705SXin Li namespace Test31 {
1499*67e74705SXin Li
1500*67e74705SXin Li // Test that we don't add D::f twice to the primary vtable.
1501*67e74705SXin Li struct A {
1502*67e74705SXin Li int a;
1503*67e74705SXin Li };
1504*67e74705SXin Li
1505*67e74705SXin Li struct B {
1506*67e74705SXin Li virtual void f();
1507*67e74705SXin Li };
1508*67e74705SXin Li
1509*67e74705SXin Li struct C : A, virtual B {
1510*67e74705SXin Li virtual void f();
1511*67e74705SXin Li };
1512*67e74705SXin Li
1513*67e74705SXin Li // CHECK-37: Vtable for 'Test31::D' (11 entries).
1514*67e74705SXin Li // CHECK-37-NEXT: 0 | vbase_offset (0)
1515*67e74705SXin Li // CHECK-37-NEXT: 1 | vbase_offset (8)
1516*67e74705SXin Li // CHECK-37-NEXT: 2 | vcall_offset (0)
1517*67e74705SXin Li // CHECK-37-NEXT: 3 | offset_to_top (0)
1518*67e74705SXin Li // CHECK-37-NEXT: 4 | Test31::D RTTI
1519*67e74705SXin Li // CHECK-37-NEXT: -- (Test31::B, 0) vtable address --
1520*67e74705SXin Li // CHECK-37-NEXT: -- (Test31::D, 0) vtable address --
1521*67e74705SXin Li // CHECK-37-NEXT: 5 | void Test31::D::f()
1522*67e74705SXin Li // CHECK-37-NEXT: 6 | vbase_offset (-8)
1523*67e74705SXin Li // CHECK-37-NEXT: 7 | vcall_offset (-8)
1524*67e74705SXin Li // CHECK-37-NEXT: 8 | offset_to_top (-8)
1525*67e74705SXin Li // CHECK-37-NEXT: 9 | Test31::D RTTI
1526*67e74705SXin Li // CHECK-37-NEXT: -- (Test31::C, 8) vtable address --
1527*67e74705SXin Li // CHECK-37-NEXT: 10 | void Test31::D::f()
1528*67e74705SXin Li // CHECK-37-NEXT: [this adjustment: 0 non-virtual, -24 vcall offset offset]
1529*67e74705SXin Li //
1530*67e74705SXin Li // CHECK-37: VTable indices for 'Test31::D' (1 entries).
1531*67e74705SXin Li // CHECK-37-NEXT: 0 | void Test31::D::f()
1532*67e74705SXin Li struct D : virtual C {
1533*67e74705SXin Li virtual void f();
1534*67e74705SXin Li };
f()1535*67e74705SXin Li void D::f() { }
1536*67e74705SXin Li
1537*67e74705SXin Li }
1538*67e74705SXin Li
1539*67e74705SXin Li namespace Test32 {
1540*67e74705SXin Li
1541*67e74705SXin Li // Check that we correctly lay out the virtual bases of 'Test32::D'.
1542*67e74705SXin Li
1543*67e74705SXin Li struct A {
1544*67e74705SXin Li virtual void f();
1545*67e74705SXin Li };
1546*67e74705SXin Li
1547*67e74705SXin Li struct B : virtual A { };
1548*67e74705SXin Li struct C : A, virtual B { };
1549*67e74705SXin Li struct D : virtual B { };
1550*67e74705SXin Li
1551*67e74705SXin Li // CHECK-38: Virtual base offset offsets for 'Test32::E' (3 entries).
1552*67e74705SXin Li // CHECK-38-NEXT: Test32::A | -32
1553*67e74705SXin Li // CHECK-38-NEXT: Test32::B | -24
1554*67e74705SXin Li // CHECK-38-NEXT: Test32::D | -40
1555*67e74705SXin Li struct E : C, virtual D {
1556*67e74705SXin Li virtual void f();
1557*67e74705SXin Li };
f()1558*67e74705SXin Li void E::f() { }
1559*67e74705SXin Li
1560*67e74705SXin Li }
1561*67e74705SXin Li
1562*67e74705SXin Li namespace Test33 {
1563*67e74705SXin Li
1564*67e74705SXin Li // Test that we don't emit too many vcall offsets in 'Test32::F'.
1565*67e74705SXin Li
1566*67e74705SXin Li struct A {
1567*67e74705SXin Li virtual void a();
1568*67e74705SXin Li };
1569*67e74705SXin Li
1570*67e74705SXin Li struct B {
1571*67e74705SXin Li virtual void b();
1572*67e74705SXin Li };
1573*67e74705SXin Li
1574*67e74705SXin Li struct C : virtual A, virtual B {
1575*67e74705SXin Li virtual void c();
1576*67e74705SXin Li };
1577*67e74705SXin Li
1578*67e74705SXin Li struct D : virtual C { };
1579*67e74705SXin Li
1580*67e74705SXin Li struct E : A, D {
1581*67e74705SXin Li virtual void e();
1582*67e74705SXin Li };
1583*67e74705SXin Li
1584*67e74705SXin Li // CHECK-39: Vtable for 'Test33::F' (30 entries).
1585*67e74705SXin Li // CHECK-39-NEXT: 0 | vbase_offset (24)
1586*67e74705SXin Li // CHECK-39-NEXT: 1 | vbase_offset (16)
1587*67e74705SXin Li // CHECK-39-NEXT: 2 | vbase_offset (16)
1588*67e74705SXin Li // CHECK-39-NEXT: 3 | vbase_offset (8)
1589*67e74705SXin Li // CHECK-39-NEXT: 4 | offset_to_top (0)
1590*67e74705SXin Li // CHECK-39-NEXT: 5 | Test33::F RTTI
1591*67e74705SXin Li // CHECK-39-NEXT: -- (Test33::A, 0) vtable address --
1592*67e74705SXin Li // CHECK-39-NEXT: -- (Test33::F, 0) vtable address --
1593*67e74705SXin Li // CHECK-39-NEXT: 6 | void Test33::A::a()
1594*67e74705SXin Li // CHECK-39-NEXT: 7 | void Test33::F::f()
1595*67e74705SXin Li // CHECK-39-NEXT: 8 | vcall_offset (0)
1596*67e74705SXin Li // CHECK-39-NEXT: 9 | vcall_offset (0)
1597*67e74705SXin Li // CHECK-39-NEXT: 10 | vbase_offset (16)
1598*67e74705SXin Li // CHECK-39-NEXT: 11 | vbase_offset (8)
1599*67e74705SXin Li // CHECK-39-NEXT: 12 | vbase_offset (8)
1600*67e74705SXin Li // CHECK-39-NEXT: 13 | offset_to_top (-8)
1601*67e74705SXin Li // CHECK-39-NEXT: 14 | Test33::F RTTI
1602*67e74705SXin Li // CHECK-39-NEXT: -- (Test33::A, 8) vtable address --
1603*67e74705SXin Li // CHECK-39-NEXT: -- (Test33::E, 8) vtable address --
1604*67e74705SXin Li // CHECK-39-NEXT: 15 | void Test33::A::a()
1605*67e74705SXin Li // CHECK-39-NEXT: 16 | void Test33::E::e()
1606*67e74705SXin Li // CHECK-39-NEXT: 17 | vbase_offset (0)
1607*67e74705SXin Li // CHECK-39-NEXT: 18 | vcall_offset (0)
1608*67e74705SXin Li // CHECK-39-NEXT: 19 | vbase_offset (8)
1609*67e74705SXin Li // CHECK-39-NEXT: 20 | vbase_offset (0)
1610*67e74705SXin Li // CHECK-39-NEXT: 21 | vcall_offset (0)
1611*67e74705SXin Li // CHECK-39-NEXT: 22 | offset_to_top (-16)
1612*67e74705SXin Li // CHECK-39-NEXT: 23 | Test33::F RTTI
1613*67e74705SXin Li // CHECK-39-NEXT: -- (Test33::A, 16) vtable address --
1614*67e74705SXin Li // CHECK-39-NEXT: -- (Test33::C, 16) vtable address --
1615*67e74705SXin Li // CHECK-39-NEXT: -- (Test33::D, 16) vtable address --
1616*67e74705SXin Li // CHECK-39-NEXT: 24 | void Test33::A::a()
1617*67e74705SXin Li // CHECK-39-NEXT: 25 | void Test33::C::c()
1618*67e74705SXin Li // CHECK-39-NEXT: 26 | vcall_offset (0)
1619*67e74705SXin Li // CHECK-39-NEXT: 27 | offset_to_top (-24)
1620*67e74705SXin Li // CHECK-39-NEXT: 28 | Test33::F RTTI
1621*67e74705SXin Li // CHECK-39-NEXT: -- (Test33::B, 24) vtable address --
1622*67e74705SXin Li // CHECK-39-NEXT: 29 | void Test33::B::b()
1623*67e74705SXin Li //
1624*67e74705SXin Li // CHECK-39: VTable indices for 'Test33::F' (1 entries).
1625*67e74705SXin Li // CHECK-39-NEXT: 1 | void Test33::F::f()
1626*67e74705SXin Li struct F : virtual E, A {
1627*67e74705SXin Li virtual void f();
1628*67e74705SXin Li };
f()1629*67e74705SXin Li void F::f() { }
1630*67e74705SXin Li
1631*67e74705SXin Li }
1632*67e74705SXin Li
1633*67e74705SXin Li namespace Test34 {
1634*67e74705SXin Li
1635*67e74705SXin Li // Test that we lay out the construction vtable for 'Test34::E' in 'Test34::F' correctly.
1636*67e74705SXin Li
1637*67e74705SXin Li struct A {
1638*67e74705SXin Li virtual void a();
1639*67e74705SXin Li };
1640*67e74705SXin Li struct B : virtual A { };
1641*67e74705SXin Li
1642*67e74705SXin Li struct C : B, A {
1643*67e74705SXin Li virtual void c();
1644*67e74705SXin Li };
1645*67e74705SXin Li
1646*67e74705SXin Li struct D : A, C { };
1647*67e74705SXin Li
1648*67e74705SXin Li struct E : virtual D {
1649*67e74705SXin Li virtual void e();
1650*67e74705SXin Li };
1651*67e74705SXin Li
1652*67e74705SXin Li // CHECK-40: Construction vtable for ('Test34::E', 0) in 'Test34::F' (22 entries).
1653*67e74705SXin Li // CHECK-40-NEXT: 0 | vbase_offset (0)
1654*67e74705SXin Li // CHECK-40-NEXT: 1 | vbase_offset (8)
1655*67e74705SXin Li // CHECK-40-NEXT: 2 | vcall_offset (0)
1656*67e74705SXin Li // CHECK-40-NEXT: 3 | offset_to_top (0)
1657*67e74705SXin Li // CHECK-40-NEXT: 4 | Test34::E RTTI
1658*67e74705SXin Li // CHECK-40-NEXT: -- (Test34::A, 0) vtable address --
1659*67e74705SXin Li // CHECK-40-NEXT: -- (Test34::E, 0) vtable address --
1660*67e74705SXin Li // CHECK-40-NEXT: 5 | void Test34::A::a()
1661*67e74705SXin Li // CHECK-40-NEXT: 6 | void Test34::E::e()
1662*67e74705SXin Li // CHECK-40-NEXT: 7 | vcall_offset (8)
1663*67e74705SXin Li // CHECK-40-NEXT: 8 | vcall_offset (0)
1664*67e74705SXin Li // CHECK-40-NEXT: 9 | vbase_offset (-8)
1665*67e74705SXin Li // CHECK-40-NEXT: 10 | offset_to_top (-8)
1666*67e74705SXin Li // CHECK-40-NEXT: 11 | Test34::E RTTI
1667*67e74705SXin Li // CHECK-40-NEXT: -- (Test34::A, 8) vtable address --
1668*67e74705SXin Li // CHECK-40-NEXT: -- (Test34::D, 8) vtable address --
1669*67e74705SXin Li // CHECK-40-NEXT: 12 | void Test34::A::a()
1670*67e74705SXin Li // CHECK-40-NEXT: 13 | vbase_offset (-16)
1671*67e74705SXin Li // CHECK-40-NEXT: 14 | vcall_offset (-16)
1672*67e74705SXin Li // CHECK-40-NEXT: 15 | offset_to_top (-16)
1673*67e74705SXin Li // CHECK-40-NEXT: 16 | Test34::E RTTI
1674*67e74705SXin Li // CHECK-40-NEXT: -- (Test34::B, 16) vtable address --
1675*67e74705SXin Li // CHECK-40-NEXT: -- (Test34::C, 16) vtable address --
1676*67e74705SXin Li // CHECK-40-NEXT: 17 | [unused] void Test34::A::a()
1677*67e74705SXin Li // CHECK-40-NEXT: 18 | void Test34::C::c()
1678*67e74705SXin Li // CHECK-40-NEXT: 19 | offset_to_top (-24)
1679*67e74705SXin Li // CHECK-40-NEXT: 20 | Test34::E RTTI
1680*67e74705SXin Li // CHECK-40-NEXT: -- (Test34::A, 24) vtable address --
1681*67e74705SXin Li // CHECK-40-NEXT: 21 | void Test34::A::a()
1682*67e74705SXin Li struct F : E {
1683*67e74705SXin Li virtual void f();
1684*67e74705SXin Li };
f()1685*67e74705SXin Li void F::f() { }
1686*67e74705SXin Li
1687*67e74705SXin Li }
1688*67e74705SXin Li
1689*67e74705SXin Li namespace Test35 {
1690*67e74705SXin Li
1691*67e74705SXin Li // Test that we lay out the virtual bases of 'Test35::H' in the correct order.
1692*67e74705SXin Li
1693*67e74705SXin Li struct A {
1694*67e74705SXin Li virtual void a();
1695*67e74705SXin Li
1696*67e74705SXin Li int i;
1697*67e74705SXin Li };
1698*67e74705SXin Li
1699*67e74705SXin Li struct B : virtual A {
1700*67e74705SXin Li virtual void b();
1701*67e74705SXin Li };
1702*67e74705SXin Li
1703*67e74705SXin Li struct C {
1704*67e74705SXin Li virtual void c();
1705*67e74705SXin Li };
1706*67e74705SXin Li
1707*67e74705SXin Li struct D : C, virtual B {
1708*67e74705SXin Li virtual void d();
1709*67e74705SXin Li };
1710*67e74705SXin Li
1711*67e74705SXin Li struct E : D {
1712*67e74705SXin Li virtual void e();
1713*67e74705SXin Li
1714*67e74705SXin Li bool b;
1715*67e74705SXin Li };
1716*67e74705SXin Li
1717*67e74705SXin Li struct F : virtual D { };
1718*67e74705SXin Li struct G : virtual E { };
1719*67e74705SXin Li
1720*67e74705SXin Li // CHECK-41: Vtable for 'Test35::H' (32 entries).
1721*67e74705SXin Li // CHECK-41-NEXT: 0 | vbase_offset (32)
1722*67e74705SXin Li // CHECK-41-NEXT: 1 | vbase_offset (0)
1723*67e74705SXin Li // CHECK-41-NEXT: 2 | vcall_offset (0)
1724*67e74705SXin Li // CHECK-41-NEXT: 3 | vcall_offset (0)
1725*67e74705SXin Li // CHECK-41-NEXT: 4 | vbase_offset (16)
1726*67e74705SXin Li // CHECK-41-NEXT: 5 | vbase_offset (8)
1727*67e74705SXin Li // CHECK-41-NEXT: 6 | offset_to_top (0)
1728*67e74705SXin Li // CHECK-41-NEXT: 7 | Test35::H RTTI
1729*67e74705SXin Li // CHECK-41-NEXT: -- (Test35::C, 0) vtable address --
1730*67e74705SXin Li // CHECK-41-NEXT: -- (Test35::D, 0) vtable address --
1731*67e74705SXin Li // CHECK-41-NEXT: -- (Test35::F, 0) vtable address --
1732*67e74705SXin Li // CHECK-41-NEXT: -- (Test35::H, 0) vtable address --
1733*67e74705SXin Li // CHECK-41-NEXT: 8 | void Test35::C::c()
1734*67e74705SXin Li // CHECK-41-NEXT: 9 | void Test35::D::d()
1735*67e74705SXin Li // CHECK-41-NEXT: 10 | void Test35::H::h()
1736*67e74705SXin Li // CHECK-41-NEXT: 11 | vbase_offset (0)
1737*67e74705SXin Li // CHECK-41-NEXT: 12 | vbase_offset (24)
1738*67e74705SXin Li // CHECK-41-NEXT: 13 | vcall_offset (0)
1739*67e74705SXin Li // CHECK-41-NEXT: 14 | vbase_offset (8)
1740*67e74705SXin Li // CHECK-41-NEXT: 15 | offset_to_top (-8)
1741*67e74705SXin Li // CHECK-41-NEXT: 16 | Test35::H RTTI
1742*67e74705SXin Li // CHECK-41-NEXT: -- (Test35::B, 8) vtable address --
1743*67e74705SXin Li // CHECK-41-NEXT: -- (Test35::G, 8) vtable address --
1744*67e74705SXin Li // CHECK-41-NEXT: 17 | void Test35::B::b()
1745*67e74705SXin Li // CHECK-41-NEXT: 18 | vcall_offset (0)
1746*67e74705SXin Li // CHECK-41-NEXT: 19 | offset_to_top (-16)
1747*67e74705SXin Li // CHECK-41-NEXT: 20 | Test35::H RTTI
1748*67e74705SXin Li // CHECK-41-NEXT: -- (Test35::A, 16) vtable address --
1749*67e74705SXin Li // CHECK-41-NEXT: 21 | void Test35::A::a()
1750*67e74705SXin Li // CHECK-41-NEXT: 22 | vcall_offset (0)
1751*67e74705SXin Li // CHECK-41-NEXT: 23 | vcall_offset (0)
1752*67e74705SXin Li // CHECK-41-NEXT: 24 | vcall_offset (0)
1753*67e74705SXin Li // CHECK-41-NEXT: 25 | vbase_offset (-16)
1754*67e74705SXin Li // CHECK-41-NEXT: 26 | vbase_offset (-24)
1755*67e74705SXin Li // CHECK-41-NEXT: 27 | offset_to_top (-32)
1756*67e74705SXin Li // CHECK-41-NEXT: 28 | Test35::H RTTI
1757*67e74705SXin Li // CHECK-41-NEXT: -- (Test35::C, 32) vtable address --
1758*67e74705SXin Li // CHECK-41-NEXT: -- (Test35::D, 32) vtable address --
1759*67e74705SXin Li // CHECK-41-NEXT: -- (Test35::E, 32) vtable address --
1760*67e74705SXin Li // CHECK-41-NEXT: 29 | void Test35::C::c()
1761*67e74705SXin Li // CHECK-41-NEXT: 30 | void Test35::D::d()
1762*67e74705SXin Li // CHECK-41-NEXT: 31 | void Test35::E::e()
1763*67e74705SXin Li //
1764*67e74705SXin Li // CHECK-41: Virtual base offset offsets for 'Test35::H' (4 entries).
1765*67e74705SXin Li // CHECK-41-NEXT: Test35::A | -32
1766*67e74705SXin Li // CHECK-41-NEXT: Test35::B | -24
1767*67e74705SXin Li // CHECK-41-NEXT: Test35::D | -56
1768*67e74705SXin Li // CHECK-41-NEXT: Test35::E | -64
1769*67e74705SXin Li //
1770*67e74705SXin Li // CHECK-41: VTable indices for 'Test35::H' (1 entries).
1771*67e74705SXin Li // CHECK-41-NEXT: 2 | void Test35::H::h()
1772*67e74705SXin Li struct H : F, G {
1773*67e74705SXin Li virtual void h();
1774*67e74705SXin Li };
h()1775*67e74705SXin Li void H::h() { }
1776*67e74705SXin Li
1777*67e74705SXin Li }
1778*67e74705SXin Li
1779*67e74705SXin Li namespace Test36 {
1780*67e74705SXin Li
1781*67e74705SXin Li // Test that we don't mark B::f as unused in the vtable for D.
1782*67e74705SXin Li
1783*67e74705SXin Li struct A {
1784*67e74705SXin Li virtual void f();
1785*67e74705SXin Li };
1786*67e74705SXin Li
1787*67e74705SXin Li struct B : virtual A { };
1788*67e74705SXin Li
1789*67e74705SXin Li struct C : virtual A {
1790*67e74705SXin Li virtual void f();
1791*67e74705SXin Li };
1792*67e74705SXin Li
1793*67e74705SXin Li // CHECK-42: Vtable for 'Test36::D' (12 entries).
1794*67e74705SXin Li // CHECK-42-NEXT: 0 | vbase_offset (8)
1795*67e74705SXin Li // CHECK-42-NEXT: 1 | vbase_offset (8)
1796*67e74705SXin Li // CHECK-42-NEXT: 2 | vcall_offset (0)
1797*67e74705SXin Li // CHECK-42-NEXT: 3 | offset_to_top (0)
1798*67e74705SXin Li // CHECK-42-NEXT: 4 | Test36::D RTTI
1799*67e74705SXin Li // CHECK-42-NEXT: -- (Test36::C, 0) vtable address --
1800*67e74705SXin Li // CHECK-42-NEXT: -- (Test36::D, 0) vtable address --
1801*67e74705SXin Li // CHECK-42-NEXT: 5 | void Test36::C::f()
1802*67e74705SXin Li // CHECK-42-NEXT: 6 | void Test36::D::g()
1803*67e74705SXin Li // CHECK-42-NEXT: 7 | vbase_offset (0)
1804*67e74705SXin Li // CHECK-42-NEXT: 8 | vcall_offset (-8)
1805*67e74705SXin Li // CHECK-42-NEXT: 9 | offset_to_top (-8)
1806*67e74705SXin Li // CHECK-42-NEXT: 10 | Test36::D RTTI
1807*67e74705SXin Li // CHECK-42-NEXT: -- (Test36::A, 8) vtable address --
1808*67e74705SXin Li // CHECK-42-NEXT: -- (Test36::B, 8) vtable address --
1809*67e74705SXin Li // CHECK-42-NEXT: 11 | void Test36::C::f()
1810*67e74705SXin Li // CHECK-42-NEXT: [this adjustment: 0 non-virtual, -24 vcall offset offset]
1811*67e74705SXin Li //
1812*67e74705SXin Li // CHECK-42: VTable indices for 'Test36::D' (1 entries).
1813*67e74705SXin Li // CHECK-42-NEXT: 1 | void Test36::D::g()
1814*67e74705SXin Li struct D : virtual B, C {
1815*67e74705SXin Li virtual void g();
1816*67e74705SXin Li };
g()1817*67e74705SXin Li void D::g() { }
1818*67e74705SXin Li
1819*67e74705SXin Li }
1820*67e74705SXin Li
1821*67e74705SXin Li namespace Test37 {
1822*67e74705SXin Li
1823*67e74705SXin Li // Test that we give C::f the right vtable index. (PR9660).
1824*67e74705SXin Li struct A {
1825*67e74705SXin Li virtual A* f() = 0;
1826*67e74705SXin Li };
1827*67e74705SXin Li
1828*67e74705SXin Li struct B : virtual A {
1829*67e74705SXin Li virtual B* f();
1830*67e74705SXin Li };
1831*67e74705SXin Li
1832*67e74705SXin Li // CHECK-43: VTable indices for 'Test37::C' (1 entries).
1833*67e74705SXin Li // CHECK-43-NEXT: 1 | Test37::C *Test37::C::f()
1834*67e74705SXin Li struct C : B {
1835*67e74705SXin Li virtual C* f();
1836*67e74705SXin Li };
1837*67e74705SXin Li
f()1838*67e74705SXin Li C* C::f() { return 0; }
1839*67e74705SXin Li
1840*67e74705SXin Li }
1841*67e74705SXin Li
1842*67e74705SXin Li // rdar://problem/10959710
1843*67e74705SXin Li namespace Test38 {
1844*67e74705SXin Li struct A {
1845*67e74705SXin Li virtual void *foo();
1846*67e74705SXin Li virtual const void *foo() const;
1847*67e74705SXin Li };
1848*67e74705SXin Li
1849*67e74705SXin Li // CHECK-44: Vtable for 'Test38::B' (7 entries).
1850*67e74705SXin Li // CHECK-44-NEXT: 0 | vbase_offset (0)
1851*67e74705SXin Li // CHECK-44-NEXT: 1 | vcall_offset (0)
1852*67e74705SXin Li // CHECK-44-NEXT: 2 | vcall_offset (0)
1853*67e74705SXin Li // CHECK-44-NEXT: 3 | offset_to_top (0)
1854*67e74705SXin Li // CHECK-44-NEXT: 4 | Test38::B RTTI
1855*67e74705SXin Li // CHECK-44-NEXT: -- (Test38::A, 0) vtable address --
1856*67e74705SXin Li // CHECK-44-NEXT: -- (Test38::B, 0) vtable address --
1857*67e74705SXin Li // CHECK-44-NEXT: 5 | void *Test38::B::foo()
1858*67e74705SXin Li // CHECK-44-NEXT: 6 | const void *Test38::B::foo() const
1859*67e74705SXin Li //
1860*67e74705SXin Li // CHECK-44: VTable indices for 'Test38::B' (2 entries).
1861*67e74705SXin Li // CHECK-44-NEXT: 0 | void *Test38::B::foo()
1862*67e74705SXin Li // CHECK-44-NEXT: 1 | const void *Test38::B::foo() const
1863*67e74705SXin Li class B : virtual public A {
1864*67e74705SXin Li void *foo();
1865*67e74705SXin Li const void *foo() const;
1866*67e74705SXin Li };
1867*67e74705SXin Li
foo()1868*67e74705SXin Li void *B::foo() { return 0; }
1869*67e74705SXin Li }
1870*67e74705SXin Li
1871*67e74705SXin Li namespace Test39 {
1872*67e74705SXin Li struct A {
1873*67e74705SXin Li virtual void foo() = delete;
1874*67e74705SXin Li };
1875*67e74705SXin Li
1876*67e74705SXin Li // CHECK-45: Vtable for 'Test39::B' (4 entries).
1877*67e74705SXin Li // CHECK-45-NEXT: 0 | offset_to_top (0)
1878*67e74705SXin Li // CHECK-45-NEXT: 1 | Test39::B RTTI
1879*67e74705SXin Li // CHECK-45-NEXT: -- (Test39::A, 0) vtable address --
1880*67e74705SXin Li // CHECK-45-NEXT: -- (Test39::B, 0) vtable address --
1881*67e74705SXin Li // CHECK-45-NEXT: 2 | void Test39::A::foo() [deleted]
1882*67e74705SXin Li // CHECK-45-NEXT: 3 | void Test39::B::foo2()
1883*67e74705SXin Li //
1884*67e74705SXin Li // CHECK-45: VTable indices for 'Test39::B' (1 entries).
1885*67e74705SXin Li // CHECK-45-NEXT: 1 | void Test39::B::foo2()
1886*67e74705SXin Li struct B: A {
1887*67e74705SXin Li virtual void foo2();
1888*67e74705SXin Li };
1889*67e74705SXin Li
foo2()1890*67e74705SXin Li void B::foo2() {
1891*67e74705SXin Li }
1892*67e74705SXin Li }
1893*67e74705SXin Li
1894*67e74705SXin Li namespace Test40 {
1895*67e74705SXin Li struct A {
1896*67e74705SXin Li virtual void foo() = 0;
1897*67e74705SXin Li };
1898*67e74705SXin Li
1899*67e74705SXin Li struct B : public A {
1900*67e74705SXin Li virtual void foo();
1901*67e74705SXin Li };
1902*67e74705SXin Li
1903*67e74705SXin Li struct C: public B {
1904*67e74705SXin Li // CHECK-46: VTable indices for 'Test40::C' (8 entries).
1905*67e74705SXin Li // CHECK-46-NEXT: 1 | int Test40::C::f(int)
1906*67e74705SXin Li // CHECK-46-NEXT: 2 | int Test40::C::f()
1907*67e74705SXin Li // CHECK-46-NEXT: 3 | int Test40::C::g(int)
1908*67e74705SXin Li // CHECK-46-NEXT: 4 | int Test40::C::g()
1909*67e74705SXin Li // CHECK-46-NEXT: 5 | int Test40::C::h(int)
1910*67e74705SXin Li // CHECK-46-NEXT: 6 | int Test40::C::h()
1911*67e74705SXin Li // CHECK-46-NEXT: 7 | int Test40::C::i(int)
1912*67e74705SXin Li // CHECK-46-NEXT: 8 | int Test40::C::i()
1913*67e74705SXin Li virtual int f(int);
1914*67e74705SXin Li virtual int f();
1915*67e74705SXin Li virtual int g(int);
1916*67e74705SXin Li virtual int g();
1917*67e74705SXin Li virtual int h(int);
1918*67e74705SXin Li virtual int h();
1919*67e74705SXin Li virtual int i(int);
1920*67e74705SXin Li virtual int i();
1921*67e74705SXin Li };
1922*67e74705SXin Li
1923*67e74705SXin Li class D : C {};
1924*67e74705SXin Li
1925*67e74705SXin Li D d;
1926*67e74705SXin Li }
1927