forked from OSchip/llvm-project
392 lines
14 KiB
C++
392 lines
14 KiB
C++
// RUN: %clang_cc1 -fno-rtti -fms-extensions -emit-llvm-only -triple i686-pc-win32 -fdump-record-layouts -fsyntax-only %s 2>&1 \
|
|
// RUN: | FileCheck %s
|
|
// RUN: %clang_cc1 -fno-rtti -fms-extensions -emit-llvm-only -triple x86_64-pc-win32 -fdump-record-layouts -fsyntax-only %s 2>/dev/null \
|
|
// RUN: | FileCheck %s -check-prefix CHECK-X64
|
|
|
|
extern "C" int printf(const char *fmt, ...);
|
|
|
|
struct B0 {
|
|
int a;
|
|
B0() : a(0xf00000B0) {}
|
|
virtual void f() { printf("B0"); }
|
|
};
|
|
|
|
struct __declspec(align(16)) B1 {
|
|
int a;
|
|
B1() : a(0xf00000B1) {}
|
|
virtual void f() { printf("B1"); }
|
|
};
|
|
|
|
struct __declspec(align(16)) Align16 {};
|
|
struct __declspec(align(32)) Align32 {};
|
|
struct VAlign16 : virtual Align16 {};
|
|
struct VAlign32 : virtual Align32 {};
|
|
|
|
struct A : virtual B0, virtual B1 {
|
|
int a;
|
|
A() : a(0xf000000A) {}
|
|
virtual void f() { printf("A"); }
|
|
virtual void g() { printf("A"); }
|
|
};
|
|
|
|
// CHECK-LABEL: 0 | struct A{{$}}
|
|
// CHECK-NEXT: 0 | (A vftable pointer)
|
|
// CHECK-NEXT: 4 | (A vbtable pointer)
|
|
// CHECK-NEXT: 8 | int a
|
|
// CHECK-NEXT: 16 | (vtordisp for vbase B0)
|
|
// CHECK-NEXT: 20 | struct B0 (virtual base)
|
|
// CHECK-NEXT: 20 | (B0 vftable pointer)
|
|
// CHECK-NEXT: 24 | int a
|
|
// CHECK-NEXT: 44 | (vtordisp for vbase B1)
|
|
// CHECK-NEXT: 48 | struct B1 (virtual base)
|
|
// CHECK-NEXT: 48 | (B1 vftable pointer)
|
|
// CHECK-NEXT: 52 | int a
|
|
// CHECK-NEXT: | [sizeof=64, align=16
|
|
// CHECK-NEXT: | nvsize=12, nvalign=16]
|
|
// CHECK-X64-LABEL: 0 | struct A{{$}}
|
|
// CHECK-X64-NEXT: 0 | (A vftable pointer)
|
|
// CHECK-X64-NEXT: 8 | (A vbtable pointer)
|
|
// CHECK-X64-NEXT: 16 | int a
|
|
// CHECK-X64-NEXT: 36 | (vtordisp for vbase B0)
|
|
// CHECK-X64-NEXT: 40 | struct B0 (virtual base)
|
|
// CHECK-X64-NEXT: 40 | (B0 vftable pointer)
|
|
// CHECK-X64-NEXT: 48 | int a
|
|
// CHECK-X64-NEXT: 76 | (vtordisp for vbase B1)
|
|
// CHECK-X64-NEXT: 80 | struct B1 (virtual base)
|
|
// CHECK-X64-NEXT: 80 | (B1 vftable pointer)
|
|
// CHECK-X64-NEXT: 88 | int a
|
|
// CHECK-X64-NEXT: | [sizeof=96, align=16
|
|
// CHECK-X64-NEXT: | nvsize=24, nvalign=16]
|
|
|
|
struct C : virtual B0, virtual B1, VAlign32 {
|
|
int a;
|
|
C() : a(0xf000000C) {}
|
|
virtual void f() { printf("C"); }
|
|
virtual void g() { printf("C"); }
|
|
};
|
|
|
|
// CHECK-LABEL: 0 | struct C{{$}}
|
|
// CHECK-NEXT: 0 | (C vftable pointer)
|
|
// CHECK-NEXT: 32 | struct VAlign32 (base)
|
|
// CHECK-NEXT: 32 | (VAlign32 vbtable pointer)
|
|
// CHECK-NEXT: 36 | int a
|
|
// CHECK-NEXT: 64 | (vtordisp for vbase B0)
|
|
// CHECK-NEXT: 68 | struct B0 (virtual base)
|
|
// CHECK-NEXT: 68 | (B0 vftable pointer)
|
|
// CHECK-NEXT: 72 | int a
|
|
// CHECK-NEXT: 108 | (vtordisp for vbase B1)
|
|
// CHECK-NEXT: 112 | struct B1 (virtual base)
|
|
// CHECK-NEXT: 112 | (B1 vftable pointer)
|
|
// CHECK-NEXT: 116 | int a
|
|
// CHECK-NEXT: 128 | struct Align32 (virtual base) (empty)
|
|
// CHECK-NEXT: | [sizeof=128, align=32
|
|
// CHECK-NEXT: | nvsize=64, nvalign=32]
|
|
// CHECK-X64-LABEL: 0 | struct C{{$}}
|
|
// CHECK-X64-NEXT: 0 | (C vftable pointer)
|
|
// CHECK-X64-NEXT: 32 | struct VAlign32 (base)
|
|
// CHECK-X64-NEXT: 32 | (VAlign32 vbtable pointer)
|
|
// CHECK-X64-NEXT: 40 | int a
|
|
// CHECK-X64-NEXT: 68 | (vtordisp for vbase B0)
|
|
// CHECK-X64-NEXT: 72 | struct B0 (virtual base)
|
|
// CHECK-X64-NEXT: 72 | (B0 vftable pointer)
|
|
// CHECK-X64-NEXT: 80 | int a
|
|
// CHECK-X64-NEXT: 108 | (vtordisp for vbase B1)
|
|
// CHECK-X64-NEXT: 112 | struct B1 (virtual base)
|
|
// CHECK-X64-NEXT: 112 | (B1 vftable pointer)
|
|
// CHECK-X64-NEXT: 120 | int a
|
|
// CHECK-X64-NEXT: 128 | struct Align32 (virtual base) (empty)
|
|
// CHECK-X64-NEXT: | [sizeof=128, align=32
|
|
// CHECK-X64-NEXT: | nvsize=64, nvalign=32]
|
|
|
|
struct __declspec(align(32)) D : virtual B0, virtual B1 {
|
|
int a;
|
|
D() : a(0xf000000D) {}
|
|
virtual void f() { printf("D"); }
|
|
virtual void g() { printf("D"); }
|
|
};
|
|
|
|
// CHECK-LABEL: 0 | struct D{{$}}
|
|
// CHECK-NEXT: 0 | (D vftable pointer)
|
|
// CHECK-NEXT: 4 | (D vbtable pointer)
|
|
// CHECK-NEXT: 8 | int a
|
|
// CHECK-NEXT: 32 | (vtordisp for vbase B0)
|
|
// CHECK-NEXT: 36 | struct B0 (virtual base)
|
|
// CHECK-NEXT: 36 | (B0 vftable pointer)
|
|
// CHECK-NEXT: 40 | int a
|
|
// CHECK-NEXT: 76 | (vtordisp for vbase B1)
|
|
// CHECK-NEXT: 80 | struct B1 (virtual base)
|
|
// CHECK-NEXT: 80 | (B1 vftable pointer)
|
|
// CHECK-NEXT: 84 | int a
|
|
// CHECK-NEXT: | [sizeof=96, align=32
|
|
// CHECK-NEXT: | nvsize=12, nvalign=32]
|
|
// CHECK-X64-LABEL: 0 | struct D{{$}}
|
|
// CHECK-X64-NEXT: 0 | (D vftable pointer)
|
|
// CHECK-X64-NEXT: 8 | (D vbtable pointer)
|
|
// CHECK-X64-NEXT: 16 | int a
|
|
// CHECK-X64-NEXT: 36 | (vtordisp for vbase B0)
|
|
// CHECK-X64-NEXT: 40 | struct B0 (virtual base)
|
|
// CHECK-X64-NEXT: 40 | (B0 vftable pointer)
|
|
// CHECK-X64-NEXT: 48 | int a
|
|
// CHECK-X64-NEXT: 76 | (vtordisp for vbase B1)
|
|
// CHECK-X64-NEXT: 80 | struct B1 (virtual base)
|
|
// CHECK-X64-NEXT: 80 | (B1 vftable pointer)
|
|
// CHECK-X64-NEXT: 88 | int a
|
|
// CHECK-X64-NEXT: | [sizeof=96, align=32
|
|
// CHECK-X64-NEXT: | nvsize=24, nvalign=32]
|
|
|
|
struct AT {
|
|
virtual ~AT(){}
|
|
};
|
|
struct CT : virtual AT {
|
|
virtual ~CT();
|
|
};
|
|
CT::~CT(){}
|
|
|
|
// CHECK-LABEL: 0 | struct CT{{$}}
|
|
// CHECK-NEXT: 0 | (CT vbtable pointer)
|
|
// CHECK-NEXT: 4 | struct AT (virtual base)
|
|
// CHECK-NEXT: 4 | (AT vftable pointer)
|
|
// CHECK-NEXT: | [sizeof=8, align=4
|
|
// CHECK-NEXT: | nvsize=4, nvalign=4]
|
|
// CHECK-X64-LABEL: 0 | struct CT{{$}}
|
|
// CHECK-X64-NEXT: 0 | (CT vbtable pointer)
|
|
// CHECK-X64-NEXT: 8 | struct AT (virtual base)
|
|
// CHECK-X64-NEXT: 8 | (AT vftable pointer)
|
|
// CHECK-X64-NEXT: | [sizeof=16, align=8
|
|
// CHECK-X64-NEXT: | nvsize=8, nvalign=8]
|
|
|
|
struct XA {
|
|
XA() { printf("XA"); }
|
|
long long ll;
|
|
};
|
|
struct XB : XA {
|
|
XB() { printf("XB"); }
|
|
virtual void foo() {}
|
|
int b;
|
|
};
|
|
struct XC : virtual XB {
|
|
XC() { printf("XC"); }
|
|
virtual void foo() {}
|
|
};
|
|
|
|
// CHECK-LABEL: 0 | struct XC{{$}}
|
|
// CHECK-NEXT: 0 | (XC vbtable pointer)
|
|
// CHECK-NEXT: 4 | (vtordisp for vbase XB)
|
|
// CHECK-NEXT: 8 | struct XB (virtual base)
|
|
// CHECK-NEXT: 8 | (XB vftable pointer)
|
|
// CHECK-NEXT: 16 | struct XA (base)
|
|
// CHECK-NEXT: 16 | long long ll
|
|
// CHECK-NEXT: 24 | int b
|
|
// CHECK-NEXT: | [sizeof=32, align=8
|
|
// CHECK-NEXT: | nvsize=4, nvalign=8]
|
|
// CHECK-X64-LABEL: 0 | struct XC{{$}}
|
|
// CHECK-X64-NEXT: 0 | (XC vbtable pointer)
|
|
// CHECK-X64-NEXT: 12 | (vtordisp for vbase XB)
|
|
// CHECK-X64-NEXT: 16 | struct XB (virtual base)
|
|
// CHECK-X64-NEXT: 16 | (XB vftable pointer)
|
|
// CHECK-X64-NEXT: 24 | struct XA (base)
|
|
// CHECK-X64-NEXT: 24 | long long ll
|
|
// CHECK-X64-NEXT: 32 | int b
|
|
// CHECK-X64-NEXT: | [sizeof=40, align=8
|
|
// CHECK-X64-NEXT: | nvsize=8, nvalign=8]
|
|
|
|
namespace pragma_test1 {
|
|
// No overrides means no vtordisps by default.
|
|
struct A { virtual ~A(); virtual void foo(); int a; };
|
|
struct B : virtual A { virtual ~B(); virtual void bar(); int b; };
|
|
struct C : virtual B { int c; };
|
|
// CHECK-LABEL: 0 | struct pragma_test1::C{{$}}
|
|
// CHECK-NEXT: 0 | (C vbtable pointer)
|
|
// CHECK-NEXT: 4 | int c
|
|
// CHECK-NEXT: 8 | struct pragma_test1::A (virtual base)
|
|
// CHECK-NEXT: 8 | (A vftable pointer)
|
|
// CHECK-NEXT: 12 | int a
|
|
// CHECK-NEXT: 16 | struct pragma_test1::B (virtual base)
|
|
// CHECK-NEXT: 16 | (B vftable pointer)
|
|
// CHECK-NEXT: 20 | (B vbtable pointer)
|
|
// CHECK-NEXT: 24 | int b
|
|
// CHECK-NEXT: | [sizeof=28, align=4
|
|
// CHECK-NEXT: | nvsize=8, nvalign=4]
|
|
}
|
|
|
|
namespace pragma_test2 {
|
|
struct A { virtual ~A(); virtual void foo(); int a; };
|
|
#pragma vtordisp(push,2)
|
|
struct B : virtual A { virtual ~B(); virtual void bar(); int b; };
|
|
struct C : virtual B { int c; };
|
|
#pragma vtordisp(pop)
|
|
// CHECK-LABEL: 0 | struct pragma_test2::C{{$}}
|
|
// CHECK-NEXT: 0 | (C vbtable pointer)
|
|
// CHECK-NEXT: 4 | int c
|
|
// CHECK-NEXT: 8 | (vtordisp for vbase A)
|
|
// CHECK-NEXT: 12 | struct pragma_test2::A (virtual base)
|
|
// CHECK-NEXT: 12 | (A vftable pointer)
|
|
// CHECK-NEXT: 16 | int a
|
|
// By adding a virtual method and vftable to B, now we need a vtordisp.
|
|
// CHECK-NEXT: 20 | (vtordisp for vbase B)
|
|
// CHECK-NEXT: 24 | struct pragma_test2::B (virtual base)
|
|
// CHECK-NEXT: 24 | (B vftable pointer)
|
|
// CHECK-NEXT: 28 | (B vbtable pointer)
|
|
// CHECK-NEXT: 32 | int b
|
|
// CHECK-NEXT: | [sizeof=36, align=4
|
|
// CHECK-NEXT: | nvsize=8, nvalign=4]
|
|
}
|
|
|
|
namespace pragma_test3 {
|
|
struct A { virtual ~A(); virtual void foo(); int a; };
|
|
#pragma vtordisp(push,2)
|
|
struct B : virtual A { virtual ~B(); virtual void foo(); int b; };
|
|
struct C : virtual B { int c; };
|
|
#pragma vtordisp(pop)
|
|
// CHECK-LABEL: 0 | struct pragma_test3::C{{$}}
|
|
// CHECK-NEXT: 0 | (C vbtable pointer)
|
|
// CHECK-NEXT: 4 | int c
|
|
// CHECK-NEXT: 8 | (vtordisp for vbase A)
|
|
// CHECK-NEXT: 12 | struct pragma_test3::A (virtual base)
|
|
// CHECK-NEXT: 12 | (A vftable pointer)
|
|
// CHECK-NEXT: 16 | int a
|
|
// No vtordisp before B! It doesn't have its own vftable.
|
|
// CHECK-NEXT: 20 | struct pragma_test3::B (virtual base)
|
|
// CHECK-NEXT: 20 | (B vbtable pointer)
|
|
// CHECK-NEXT: 24 | int b
|
|
// CHECK-NEXT: | [sizeof=28, align=4
|
|
// CHECK-NEXT: | nvsize=8, nvalign=4]
|
|
}
|
|
|
|
namespace pragma_test4 {
|
|
struct A {
|
|
A();
|
|
virtual void foo();
|
|
int a;
|
|
};
|
|
|
|
// Make sure the pragma applies to class template decls before they've been
|
|
// instantiated.
|
|
#pragma vtordisp(push,2)
|
|
template <typename T>
|
|
struct B : virtual A {
|
|
B();
|
|
virtual ~B();
|
|
virtual void bar();
|
|
T b;
|
|
};
|
|
#pragma vtordisp(pop)
|
|
|
|
struct C : virtual B<int> { int c; };
|
|
// CHECK-LABEL: 0 | struct pragma_test4::C{{$}}
|
|
// CHECK-NEXT: 0 | (C vbtable pointer)
|
|
// CHECK-NEXT: 4 | int c
|
|
// Pragma applies to B, which has vbase A.
|
|
// CHECK-NEXT: 8 | (vtordisp for vbase A)
|
|
// CHECK-NEXT: 12 | struct pragma_test4::A (virtual base)
|
|
// CHECK-NEXT: 12 | (A vftable pointer)
|
|
// CHECK-NEXT: 16 | int a
|
|
// Pragma does not apply to C, and B doesn't usually need a vtordisp in C.
|
|
// CHECK-NEXT: 20 | struct pragma_test4::B<int> (virtual base)
|
|
// CHECK-NEXT: 20 | (B vftable pointer)
|
|
// CHECK-NEXT: 24 | (B vbtable pointer)
|
|
// CHECK-NEXT: 28 | int b
|
|
// CHECK-NEXT: | [sizeof=32, align=4
|
|
// CHECK-NEXT: | nvsize=8, nvalign=4]
|
|
}
|
|
|
|
struct GA {
|
|
virtual void fun() {}
|
|
};
|
|
struct GB: public GA {};
|
|
struct GC: public virtual GA {
|
|
virtual void fun() {}
|
|
GC() {}
|
|
};
|
|
struct GD: public virtual GC, public virtual GB {};
|
|
|
|
// CHECK-LABEL: 0 | struct GD{{$}}
|
|
// CHECK-NEXT: 0 | (GD vbtable pointer)
|
|
// CHECK-NEXT: 4 | (vtordisp for vbase GA)
|
|
// CHECK-NEXT: 8 | struct GA (virtual base)
|
|
// CHECK-NEXT: 8 | (GA vftable pointer)
|
|
// CHECK-NEXT: 12 | struct GC (virtual base)
|
|
// CHECK-NEXT: 12 | (GC vbtable pointer)
|
|
// CHECK-NEXT: 16 | struct GB (virtual base)
|
|
// CHECK-NEXT: 16 | struct GA (primary base)
|
|
// CHECK-NEXT: 16 | (GA vftable pointer)
|
|
// CHECK-NEXT: | [sizeof=20, align=4
|
|
// CHECK-NEXT: | nvsize=4, nvalign=4]
|
|
// CHECK-X64-LABEL: 0 | struct GD{{$}}
|
|
// CHECK-X64-NEXT: 0 | (GD vbtable pointer)
|
|
// CHECK-X64-NEXT: 12 | (vtordisp for vbase GA)
|
|
// CHECK-X64-NEXT: 16 | struct GA (virtual base)
|
|
// CHECK-X64-NEXT: 16 | (GA vftable pointer)
|
|
// CHECK-X64-NEXT: 24 | struct GC (virtual base)
|
|
// CHECK-X64-NEXT: 24 | (GC vbtable pointer)
|
|
// CHECK-X64-NEXT: 32 | struct GB (virtual base)
|
|
// CHECK-X64-NEXT: 32 | struct GA (primary base)
|
|
// CHECK-X64-NEXT: 32 | (GA vftable pointer)
|
|
// CHECK-X64-NEXT: | [sizeof=40, align=8
|
|
// CHECK-X64-NEXT: | nvsize=8, nvalign=8]
|
|
|
|
struct HA {
|
|
virtual void fun() {}
|
|
};
|
|
#pragma vtordisp(push, 2)
|
|
struct HB : virtual HA {};
|
|
#pragma vtordisp(pop)
|
|
#pragma vtordisp(push, 0)
|
|
struct HC : virtual HB {};
|
|
#pragma vtordisp(pop)
|
|
|
|
// CHECK-LABEL: 0 | struct HC{{$}}
|
|
// CHECK-NEXT: 0 | (HC vbtable pointer)
|
|
// CHECK-NEXT: 4 | (vtordisp for vbase HA)
|
|
// CHECK-NEXT: 8 | struct HA (virtual base)
|
|
// CHECK-NEXT: 8 | (HA vftable pointer)
|
|
// CHECK-NEXT: 12 | struct HB (virtual base)
|
|
// CHECK-NEXT: 12 | (HB vbtable pointer)
|
|
// CHECK-NEXT: | [sizeof=16, align=4
|
|
// CHECK-NEXT: | nvsize=4, nvalign=4]
|
|
// CHECK-X64-LABEL: 0 | struct HC{{$}}
|
|
// CHECK-X64-NEXT: 0 | (HC vbtable pointer)
|
|
// CHECK-X64-NEXT: 12 | (vtordisp for vbase HA)
|
|
// CHECK-X64-NEXT: 16 | struct HA (virtual base)
|
|
// CHECK-X64-NEXT: 16 | (HA vftable pointer)
|
|
// CHECK-X64-NEXT: 24 | struct HB (virtual base)
|
|
// CHECK-X64-NEXT: 24 | (HB vbtable pointer)
|
|
// CHECK-X64-NEXT: | [sizeof=32, align=8
|
|
// CHECK-X64-NEXT: | nvsize=8, nvalign=8]
|
|
|
|
struct IA {
|
|
virtual void f();
|
|
};
|
|
struct __declspec(dllexport) IB : virtual IA {
|
|
virtual void f() = 0;
|
|
IB() {}
|
|
};
|
|
|
|
// CHECK-LABEL: 0 | struct IB{{$}}
|
|
// CHECK-NEXT: 0 | (IB vbtable pointer)
|
|
// CHECK-NEXT: 4 | struct IA (virtual base)
|
|
// CHECK-NEXT: 4 | (IA vftable pointer)
|
|
// CHECK-NEXT: | [sizeof=8, align=4
|
|
// CHECK-NEXT: | nvsize=4, nvalign=4]
|
|
// CHECK-X64-LABEL: 0 | struct IB{{$}}
|
|
// CHECK-X64-NEXT: 0 | (IB vbtable pointer)
|
|
// CHECK-X64-NEXT: 8 | struct IA (virtual base)
|
|
// CHECK-X64-NEXT: 8 | (IA vftable pointer)
|
|
// CHECK-X64-NEXT: | [sizeof=16, align=8
|
|
// CHECK-X64-NEXT: | nvsize=8, nvalign=8]
|
|
|
|
int a[
|
|
sizeof(A)+
|
|
sizeof(C)+
|
|
sizeof(D)+
|
|
sizeof(CT)+
|
|
sizeof(XC)+
|
|
sizeof(pragma_test1::C)+
|
|
sizeof(pragma_test2::C)+
|
|
sizeof(pragma_test3::C)+
|
|
sizeof(pragma_test4::C)+
|
|
sizeof(GD)+
|
|
sizeof(HC)+
|
|
sizeof(IB)+
|
|
0];
|