2013-01-21 21:02:41 +08:00
|
|
|
// RUN: %clang_cc1 %s -fno-rtti -cxx-abi microsoft -triple=i386-pc-win32 -emit-llvm -fdump-vtable-layouts -o - > %t 2>&1
|
|
|
|
// RUN: FileCheck --check-prefix=EMITS-VTABLE %s < %t
|
2013-05-30 00:18:30 +08:00
|
|
|
// RUN: FileCheck --check-prefix=NO-VTABLE %s < %t
|
2013-01-21 21:02:41 +08:00
|
|
|
// RUN: FileCheck --check-prefix=CHECK-A %s < %t
|
|
|
|
// RUN: FileCheck --check-prefix=CHECK-B %s < %t
|
|
|
|
// RUN: FileCheck --check-prefix=CHECK-C %s < %t
|
|
|
|
// RUN: FileCheck --check-prefix=CHECK-D %s < %t
|
|
|
|
// RUN: FileCheck --check-prefix=CHECK-E %s < %t
|
|
|
|
// RUN: FileCheck --check-prefix=CHECK-F %s < %t
|
|
|
|
// RUN: FileCheck --check-prefix=CHECK-G %s < %t
|
2013-07-30 17:46:19 +08:00
|
|
|
// RUN: FileCheck --check-prefix=CHECK-I %s < %t
|
|
|
|
|
|
|
|
// FIXME: Currently, we only test VFTableContext in the AST, but still use
|
|
|
|
// VTableContext for CodeGen. We should remove the "Vtable" checks below when we
|
|
|
|
// completely switch from VTableContext to VFTableContext.
|
|
|
|
// Currently, the order of Vtable vs VFTable output depends on whether the
|
|
|
|
// v*table info was required by a constructor or a method definition.
|
2013-01-21 21:02:41 +08:00
|
|
|
|
|
|
|
struct A {
|
|
|
|
// CHECK-A: Vtable for 'A' (3 entries)
|
|
|
|
// CHECK-A-NEXT: 0 | void A::f()
|
|
|
|
// CHECK-A-NEXT: 1 | void A::g()
|
|
|
|
// CHECK-A-NEXT: 2 | void A::h()
|
2013-07-30 17:46:19 +08:00
|
|
|
|
|
|
|
// CHECK-A: VFTable for 'A' (3 entries)
|
|
|
|
// CHECK-A-NEXT: 0 | void A::f()
|
|
|
|
// CHECK-A-NEXT: 1 | void A::g()
|
|
|
|
// CHECK-A-NEXT: 2 | void A::h()
|
|
|
|
|
2013-01-21 21:02:41 +08:00
|
|
|
virtual void f();
|
|
|
|
virtual void g();
|
|
|
|
virtual void h();
|
|
|
|
int ia;
|
|
|
|
};
|
2013-05-30 00:18:30 +08:00
|
|
|
A a;
|
|
|
|
// EMITS-VTABLE-DAG: @"\01??_7A@@6B@" = linkonce_odr unnamed_addr constant [3 x i8*]
|
2013-01-21 21:02:41 +08:00
|
|
|
|
|
|
|
struct B : A {
|
|
|
|
// CHECK-B: Vtable for 'B' (5 entries)
|
|
|
|
// CHECK-B-NEXT: 0 | void B::f()
|
|
|
|
// CHECK-B-NEXT: 1 | void A::g()
|
|
|
|
// CHECK-B-NEXT: 2 | void A::h()
|
|
|
|
// CHECK-B-NEXT: 3 | void B::i()
|
|
|
|
// CHECK-B-NEXT: 4 | void B::j()
|
2013-07-30 17:46:19 +08:00
|
|
|
|
|
|
|
// CHECK-B: VFTable for 'A' in 'B' (5 entries)
|
|
|
|
// CHECK-B-NEXT: 0 | void B::f()
|
|
|
|
// CHECK-B-NEXT: 1 | void A::g()
|
|
|
|
// CHECK-B-NEXT: 2 | void A::h()
|
|
|
|
// CHECK-B-NEXT: 3 | void B::i()
|
|
|
|
// CHECK-B-NEXT: 4 | void B::j()
|
|
|
|
|
2013-01-21 21:02:41 +08:00
|
|
|
virtual void f(); // overrides A::f()
|
|
|
|
virtual void i();
|
|
|
|
virtual void j();
|
|
|
|
};
|
2013-05-30 00:18:30 +08:00
|
|
|
B b;
|
|
|
|
// EMITS-VTABLE-DAG: @"\01??_7B@@6B@" = linkonce_odr unnamed_addr constant [5 x i8*]
|
2013-01-21 21:02:41 +08:00
|
|
|
|
|
|
|
struct C {
|
2013-07-30 17:46:19 +08:00
|
|
|
// CHECK-C: VFTable for 'C' (2 entries)
|
|
|
|
// CHECK-C-NEXT: 0 | C::~C() [scalar deleting]
|
|
|
|
// CHECK-C-NEXT: 1 | void C::f()
|
|
|
|
// CHECK-C: VFTable indices for 'C' (2 entries).
|
|
|
|
// CHECK-C-NEXT: 0 | C::~C() [scalar deleting]
|
|
|
|
// CHECK-C-NEXT: 1 | void C::f()
|
|
|
|
|
2013-01-21 21:02:41 +08:00
|
|
|
// CHECK-C: Vtable for 'C' (2 entries)
|
2013-02-13 16:37:51 +08:00
|
|
|
// CHECK-C-NEXT: 0 | C::~C() [scalar deleting]
|
2013-01-21 21:02:41 +08:00
|
|
|
// CHECK-C-NEXT: 1 | void C::f()
|
|
|
|
// CHECK-C: VTable indices for 'C' (2 entries).
|
2013-02-13 16:37:51 +08:00
|
|
|
// CHECK-C-NEXT: 0 | C::~C() [scalar deleting]
|
2013-01-21 21:02:41 +08:00
|
|
|
// CHECK-C-NEXT: 1 | void C::f()
|
|
|
|
|
2013-07-30 17:46:19 +08:00
|
|
|
virtual ~C();
|
2013-01-21 21:02:41 +08:00
|
|
|
virtual void f();
|
|
|
|
};
|
|
|
|
void C::f() {}
|
2013-05-30 00:18:30 +08:00
|
|
|
// NO-VTABLE-NOT: @"\01??_7C@@6B@"
|
2013-01-21 21:02:41 +08:00
|
|
|
|
|
|
|
struct D {
|
|
|
|
// CHECK-D: Vtable for 'D' (2 entries)
|
|
|
|
// CHECK-D-NEXT: 0 | void D::f()
|
2013-02-13 16:37:51 +08:00
|
|
|
// CHECK-D-NEXT: 1 | D::~D() [scalar deleting]
|
2013-01-21 21:02:41 +08:00
|
|
|
|
2013-07-30 17:46:19 +08:00
|
|
|
// CHECK-D: VFTable for 'D' (2 entries)
|
|
|
|
// CHECK-D-NEXT: 0 | void D::f()
|
|
|
|
// CHECK-D-NEXT: 1 | D::~D() [scalar deleting]
|
|
|
|
|
|
|
|
virtual void f();
|
2013-01-21 21:02:41 +08:00
|
|
|
virtual ~D();
|
|
|
|
};
|
2013-05-30 00:18:30 +08:00
|
|
|
D d;
|
|
|
|
// EMITS-VTABLE-DAG: @"\01??_7D@@6B@" = linkonce_odr unnamed_addr constant [2 x i8*]
|
2013-01-21 21:02:41 +08:00
|
|
|
|
|
|
|
struct E : A {
|
2013-07-30 17:46:19 +08:00
|
|
|
// CHECK-E: VFTable for 'A' in 'E' (5 entries)
|
|
|
|
// CHECK-E-NEXT: 0 | void A::f()
|
|
|
|
// CHECK-E-NEXT: 1 | void A::g()
|
|
|
|
// CHECK-E-NEXT: 2 | void A::h()
|
|
|
|
// CHECK-E-NEXT: 3 | E::~E() [scalar deleting]
|
|
|
|
// CHECK-E-NEXT: 4 | void E::i()
|
|
|
|
// CHECK-E: VFTable indices for 'E' (2 entries).
|
|
|
|
// CHECK-E-NEXT: 3 | E::~E() [scalar deleting]
|
|
|
|
// CHECK-E-NEXT: 4 | void E::i()
|
|
|
|
|
2013-01-21 21:02:41 +08:00
|
|
|
// CHECK-E: Vtable for 'E' (5 entries)
|
|
|
|
// CHECK-E-NEXT: 0 | void A::f()
|
|
|
|
// CHECK-E-NEXT: 1 | void A::g()
|
|
|
|
// CHECK-E-NEXT: 2 | void A::h()
|
2013-02-13 16:37:51 +08:00
|
|
|
// CHECK-E-NEXT: 3 | E::~E() [scalar deleting]
|
2013-01-21 21:02:41 +08:00
|
|
|
// CHECK-E-NEXT: 4 | void E::i()
|
|
|
|
// CHECK-E: VTable indices for 'E' (2 entries).
|
2013-02-13 16:37:51 +08:00
|
|
|
// CHECK-E-NEXT: 3 | E::~E() [scalar deleting]
|
2013-01-21 21:02:41 +08:00
|
|
|
// CHECK-E-NEXT: 4 | void E::i()
|
|
|
|
|
2013-05-30 00:18:30 +08:00
|
|
|
// ~E would be the key method, but it isn't used, and MS ABI has no key
|
|
|
|
// methods.
|
2013-01-21 21:02:41 +08:00
|
|
|
virtual ~E();
|
|
|
|
virtual void i();
|
|
|
|
};
|
|
|
|
void E::i() {}
|
2013-05-30 00:18:30 +08:00
|
|
|
// NO-VTABLE-NOT: @"\01??_7E@@6B@"
|
2013-01-21 21:02:41 +08:00
|
|
|
|
|
|
|
struct F : A {
|
|
|
|
// CHECK-F: Vtable for 'F' (5 entries)
|
|
|
|
// CHECK-F-NEXT: 0 | void A::f()
|
|
|
|
// CHECK-F-NEXT: 1 | void A::g()
|
|
|
|
// CHECK-F-NEXT: 2 | void A::h()
|
|
|
|
// CHECK-F-NEXT: 3 | void F::i()
|
2013-02-13 16:37:51 +08:00
|
|
|
// CHECK-F-NEXT: 4 | F::~F() [scalar deleting]
|
2013-01-21 21:02:41 +08:00
|
|
|
// CHECK-F: VTable indices for 'F' (2 entries).
|
|
|
|
// CHECK-F-NEXT: 3 | void F::i()
|
2013-02-13 16:37:51 +08:00
|
|
|
// CHECK-F-NEXT: 4 | F::~F() [scalar deleting]
|
2013-07-30 17:46:19 +08:00
|
|
|
|
|
|
|
// CHECK-F: VFTable for 'A' in 'F' (5 entries)
|
|
|
|
// CHECK-F-NEXT: 0 | void A::f()
|
|
|
|
// CHECK-F-NEXT: 1 | void A::g()
|
|
|
|
// CHECK-F-NEXT: 2 | void A::h()
|
|
|
|
// CHECK-F-NEXT: 3 | void F::i()
|
|
|
|
// CHECK-F-NEXT: 4 | F::~F() [scalar deleting]
|
|
|
|
// CHECK-F: VFTable indices for 'F' (2 entries).
|
|
|
|
// CHECK-F-NEXT: 3 | void F::i()
|
|
|
|
// CHECK-F-NEXT: 4 | F::~F() [scalar deleting]
|
|
|
|
|
2013-01-21 21:02:41 +08:00
|
|
|
virtual void i();
|
|
|
|
virtual ~F();
|
|
|
|
};
|
2013-05-30 00:18:30 +08:00
|
|
|
F f;
|
|
|
|
// EMITS-VTABLE-DAG: @"\01??_7F@@6B@" = linkonce_odr unnamed_addr constant [5 x i8*]
|
2013-01-21 21:02:41 +08:00
|
|
|
|
|
|
|
struct G : E {
|
2013-07-30 17:46:19 +08:00
|
|
|
// CHECK-G: VFTable for 'A' in 'E' in 'G' (6 entries)
|
|
|
|
// CHECK-G-NEXT: 0 | void G::f()
|
|
|
|
// CHECK-G-NEXT: 1 | void A::g()
|
|
|
|
// CHECK-G-NEXT: 2 | void A::h()
|
|
|
|
// CHECK-G-NEXT: 3 | G::~G() [scalar deleting]
|
|
|
|
// CHECK-G-NEXT: 4 | void E::i()
|
|
|
|
// CHECK-G-NEXT: 5 | void G::j()
|
|
|
|
// CHECK-G: VFTable indices for 'G' (3 entries).
|
|
|
|
// CHECK-G-NEXT: 0 | void G::f()
|
|
|
|
// CHECK-G-NEXT: 3 | G::~G() [scalar deleting]
|
|
|
|
// CHECK-G-NEXT: 5 | void G::j()
|
|
|
|
|
2013-01-21 21:02:41 +08:00
|
|
|
// CHECK-G: Vtable for 'G' (6 entries)
|
|
|
|
// CHECK-G-NEXT: 0 | void G::f()
|
|
|
|
// CHECK-G-NEXT: 1 | void A::g()
|
|
|
|
// CHECK-G-NEXT: 2 | void A::h()
|
2013-02-13 16:37:51 +08:00
|
|
|
// CHECK-G-NEXT: 3 | G::~G() [scalar deleting]
|
2013-01-21 21:02:41 +08:00
|
|
|
// CHECK-G-NEXT: 4 | void E::i()
|
|
|
|
// CHECK-G-NEXT: 5 | void G::j()
|
|
|
|
// CHECK-G: VTable indices for 'G' (3 entries).
|
|
|
|
// CHECK-G-NEXT: 0 | void G::f()
|
2013-02-13 16:37:51 +08:00
|
|
|
// CHECK-G-NEXT: 3 | G::~G() [scalar deleting]
|
2013-01-21 21:02:41 +08:00
|
|
|
// CHECK-G-NEXT: 5 | void G::j()
|
2013-07-30 17:46:19 +08:00
|
|
|
|
2013-01-21 21:02:41 +08:00
|
|
|
virtual void f(); // overrides A::f()
|
|
|
|
virtual ~G();
|
|
|
|
virtual void j();
|
|
|
|
};
|
|
|
|
void G::j() {}
|
2013-05-30 00:18:30 +08:00
|
|
|
// NO-VTABLE-NOT: @"\01??_7G@@6B@"
|
2013-05-30 00:51:17 +08:00
|
|
|
|
|
|
|
// Test that the usual Itanium-style key method does not emit a vtable.
|
|
|
|
struct H {
|
|
|
|
virtual void f();
|
|
|
|
};
|
|
|
|
void H::f() {}
|
|
|
|
// NO-VTABLE-NOT: @"\01??_7H@@6B@"
|
2013-07-30 17:46:19 +08:00
|
|
|
|
|
|
|
struct Empty { };
|
|
|
|
|
|
|
|
struct I : Empty {
|
|
|
|
// CHECK-I: VFTable for 'I' (2 entries)
|
|
|
|
// CHECK-I-NEXT: 0 | void I::f()
|
|
|
|
// CHECK-I-NEXT: 1 | void I::g()
|
|
|
|
virtual void f();
|
|
|
|
virtual void g();
|
|
|
|
};
|
|
|
|
|
|
|
|
I i;
|