458 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			458 lines
		
	
	
		
			16 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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK-X64-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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]
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| }
 | |
| 
 | |
| 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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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]
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| }
 | |
| 
 | |
| 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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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]
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| }
 | |
| 
 | |
| 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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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]
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| }
 | |
| 
 | |
| 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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK: *** Dumping AST Record Layout
 | |
| // CHECK-NEXT:    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: *** Dumping AST Record Layout
 | |
| // CHECK-X64: *** Dumping AST Record Layout
 | |
| // CHECK-X64-NEXT:    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];
 |