218 lines
7.5 KiB
C++
218 lines
7.5 KiB
C++
|
// RUN: %clang_cc1 -fms-extensions -fexceptions -fcxx-exceptions -emit-llvm-only -triple x86_64-pc-win32 -fdump-record-layouts -fsyntax-only %s 2>&1 | FileCheck %s
|
||
|
|
||
|
struct Base {
|
||
|
virtual ~Base() {}
|
||
|
virtual void BaseFunc() {}
|
||
|
};
|
||
|
|
||
|
#pragma vtordisp(0)
|
||
|
|
||
|
struct Container {
|
||
|
static void f() try {
|
||
|
#pragma vtordisp(2)
|
||
|
struct HasVtorDisp : virtual Base {
|
||
|
virtual ~HasVtorDisp() {}
|
||
|
virtual void Func() {}
|
||
|
};
|
||
|
|
||
|
int x[sizeof(HasVtorDisp)];
|
||
|
|
||
|
// HasVtorDisp: vtordisp because of pragma right before it.
|
||
|
//
|
||
|
// CHECK: *** Dumping AST Record Layout
|
||
|
// CHECK: *** Dumping AST Record Layout
|
||
|
// CHECK-NEXT: 0 | struct HasVtorDisp
|
||
|
// CHECK-NEXT: 0 | (HasVtorDisp vftable pointer)
|
||
|
// CHECK-NEXT: 8 | (HasVtorDisp vbtable pointer)
|
||
|
// CHECK-NEXT: 20 | (vtordisp for vbase Base)
|
||
|
// CHECK-NEXT: 24 | struct Base (virtual base)
|
||
|
// CHECK-NEXT: 24 | (Base vftable pointer)
|
||
|
// CHECK-NEXT: | [sizeof=32, align=8,
|
||
|
// CHECK-NEXT: | nvsize=16, nvalign=8]
|
||
|
} catch (...) {
|
||
|
}
|
||
|
};
|
||
|
|
||
|
struct NoVtorDisp1 : virtual Base {
|
||
|
virtual ~NoVtorDisp1() {}
|
||
|
virtual void Func() {}
|
||
|
};
|
||
|
|
||
|
int x1[sizeof(NoVtorDisp1)];
|
||
|
|
||
|
// NoVtroDisp1: no vtordisp because of pragma disabling it.
|
||
|
//
|
||
|
// CHECK: *** Dumping AST Record Layout
|
||
|
// CHECK-NEXT: 0 | struct NoVtorDisp1
|
||
|
// CHECK-NEXT: 0 | (NoVtorDisp1 vftable pointer)
|
||
|
// CHECK-NEXT: 8 | (NoVtorDisp1 vbtable pointer)
|
||
|
// CHECK-NEXT: 16 | struct Base (virtual base)
|
||
|
// CHECK-NEXT: 16 | (Base vftable pointer)
|
||
|
// CHECK-NEXT: | [sizeof=24, align=8,
|
||
|
// CHECK-NEXT: | nvsize=16, nvalign=8]
|
||
|
|
||
|
struct Container2 {
|
||
|
static void f1() {
|
||
|
// Local pragma #1 - must be disabled on exit from f1().
|
||
|
#pragma vtordisp(push, 2)
|
||
|
struct HasVtorDisp1 : virtual Base {
|
||
|
virtual ~HasVtorDisp1() {}
|
||
|
virtual void Func() {}
|
||
|
};
|
||
|
|
||
|
int x2[sizeof(HasVtorDisp1)];
|
||
|
|
||
|
// HasVtorDisp1: vtordisp because of pragma right before it.
|
||
|
//
|
||
|
// CHECK: *** Dumping AST Record Layout
|
||
|
// CHECK-NEXT: 0 | struct HasVtorDisp1
|
||
|
// CHECK-NEXT: 0 | (HasVtorDisp1 vftable pointer)
|
||
|
// CHECK-NEXT: 8 | (HasVtorDisp1 vbtable pointer)
|
||
|
// CHECK-NEXT: 20 | (vtordisp for vbase Base)
|
||
|
// CHECK-NEXT: 24 | struct Base (virtual base)
|
||
|
// CHECK-NEXT: 24 | (Base vftable pointer)
|
||
|
// CHECK-NEXT: | [sizeof=32, align=8,
|
||
|
// CHECK-NEXT: | nvsize=16, nvalign=8]
|
||
|
|
||
|
struct InnerContainer {
|
||
|
static void g1() {
|
||
|
struct HasVtorDisp2 : virtual Base {
|
||
|
virtual ~HasVtorDisp2() {}
|
||
|
virtual void Func() {}
|
||
|
};
|
||
|
|
||
|
int x3[sizeof(HasVtorDisp2)];
|
||
|
|
||
|
// HasVtorDisp2: vtordisp because of vtordisp(2) in f1().
|
||
|
//
|
||
|
// CHECK: *** Dumping AST Record Layout
|
||
|
// CHECK-NEXT: 0 | struct HasVtorDisp2
|
||
|
// CHECK-NEXT: 0 | (HasVtorDisp2 vftable pointer)
|
||
|
// CHECK-NEXT: 8 | (HasVtorDisp2 vbtable pointer)
|
||
|
// CHECK-NEXT: 20 | (vtordisp for vbase Base)
|
||
|
// CHECK-NEXT: 24 | struct Base (virtual base)
|
||
|
// CHECK-NEXT: 24 | (Base vftable pointer)
|
||
|
// CHECK-NEXT: | [sizeof=32, align=8,
|
||
|
// CHECK-NEXT: | nvsize=16, nvalign=8]
|
||
|
|
||
|
// Local pragma #2 - must be disabled on exit from g1().
|
||
|
#pragma vtordisp(push, 0)
|
||
|
struct NoVtorDisp2 : virtual Base {
|
||
|
virtual ~NoVtorDisp2() {}
|
||
|
virtual void Func() {}
|
||
|
};
|
||
|
|
||
|
int x4[sizeof(NoVtorDisp2)];
|
||
|
|
||
|
// NoVtroDisp2: no vtordisp because of vtordisp(0) in g1().
|
||
|
//
|
||
|
// CHECK: *** Dumping AST Record Layout
|
||
|
// CHECK-NEXT: 0 | struct NoVtorDisp2
|
||
|
// CHECK-NEXT: 0 | (NoVtorDisp2 vftable pointer)
|
||
|
// CHECK-NEXT: 8 | (NoVtorDisp2 vbtable pointer)
|
||
|
// CHECK-NEXT: 16 | struct Base (virtual base)
|
||
|
// CHECK-NEXT: 16 | (Base vftable pointer)
|
||
|
// CHECK-NEXT: | [sizeof=24, align=8,
|
||
|
// CHECK-NEXT: | nvsize=16, nvalign=8]
|
||
|
}
|
||
|
|
||
|
static void g2() {
|
||
|
struct HasVtorDisp3 : virtual Base {
|
||
|
virtual ~HasVtorDisp3() {}
|
||
|
virtual void Func() {}
|
||
|
};
|
||
|
|
||
|
int x5[sizeof(HasVtorDisp3)];
|
||
|
|
||
|
// HasVtorDisp3: vtordisp because of vtordisp(2) in f1(),
|
||
|
// local vtordisp(0) in g1() is disabled.
|
||
|
//
|
||
|
// CHECK: *** Dumping AST Record Layout
|
||
|
// CHECK-NEXT: 0 | struct HasVtorDisp3
|
||
|
// CHECK-NEXT: 0 | (HasVtorDisp3 vftable pointer)
|
||
|
// CHECK-NEXT: 8 | (HasVtorDisp3 vbtable pointer)
|
||
|
// CHECK-NEXT: 20 | (vtordisp for vbase Base)
|
||
|
// CHECK-NEXT: 24 | struct Base (virtual base)
|
||
|
// CHECK-NEXT: 24 | (Base vftable pointer)
|
||
|
// CHECK-NEXT: | [sizeof=32, align=8,
|
||
|
// CHECK-NEXT: | nvsize=16, nvalign=8]
|
||
|
}
|
||
|
};
|
||
|
|
||
|
struct HasVtorDisp4 : virtual Base {
|
||
|
virtual ~HasVtorDisp4() {}
|
||
|
virtual void Func() {}
|
||
|
};
|
||
|
|
||
|
int x6[sizeof(HasVtorDisp4)];
|
||
|
|
||
|
// HasVtorDisp4: vtordisp because of vtordisp(2) in f1(),
|
||
|
// local vtordisp(0) in g1() is disabled,
|
||
|
// g2() has no pragmas - stack is not affected.
|
||
|
//
|
||
|
// CHECK: *** Dumping AST Record Layout
|
||
|
// CHECK-NEXT: 0 | struct HasVtorDisp4
|
||
|
// CHECK-NEXT: 0 | (HasVtorDisp4 vftable pointer)
|
||
|
// CHECK-NEXT: 8 | (HasVtorDisp4 vbtable pointer)
|
||
|
// CHECK-NEXT: 20 | (vtordisp for vbase Base)
|
||
|
// CHECK-NEXT: 24 | struct Base (virtual base)
|
||
|
// CHECK-NEXT: 24 | (Base vftable pointer)
|
||
|
// CHECK-NEXT: | [sizeof=32, align=8,
|
||
|
// CHECK-NEXT: | nvsize=16, nvalign=8]
|
||
|
|
||
|
InnerContainer::g1();
|
||
|
InnerContainer::g2();
|
||
|
}
|
||
|
|
||
|
static void f2() {
|
||
|
struct NoVtorDisp3 : virtual Base {
|
||
|
virtual ~NoVtorDisp3() {}
|
||
|
virtual void Func() {}
|
||
|
};
|
||
|
|
||
|
int x7[sizeof(NoVtorDisp3)];
|
||
|
|
||
|
// NoVtroDisp3: no vtordisp because of global pragma (0),
|
||
|
// local vtordisp(2) is disabled on exit from f1().
|
||
|
//
|
||
|
// CHECK: *** Dumping AST Record Layout
|
||
|
// CHECK-NEXT: 0 | struct NoVtorDisp3
|
||
|
// CHECK-NEXT: 0 | (NoVtorDisp3 vftable pointer)
|
||
|
// CHECK-NEXT: 8 | (NoVtorDisp3 vbtable pointer)
|
||
|
// CHECK-NEXT: 16 | struct Base (virtual base)
|
||
|
// CHECK-NEXT: 16 | (Base vftable pointer)
|
||
|
// CHECK-NEXT: | [sizeof=24, align=8,
|
||
|
// CHECK-NEXT: | nvsize=16, nvalign=8]
|
||
|
}
|
||
|
};
|
||
|
|
||
|
struct Container3 {
|
||
|
#pragma vtordisp(2)
|
||
|
struct HasVtorDisp5 : virtual Base {
|
||
|
virtual ~HasVtorDisp5() {}
|
||
|
virtual void Func() {}
|
||
|
};
|
||
|
|
||
|
int x8[sizeof(HasVtorDisp5)];
|
||
|
|
||
|
// HasVtorDisp5: vtordisp because of pragma right before it.
|
||
|
//
|
||
|
// CHECK: *** Dumping AST Record Layout
|
||
|
// CHECK-NEXT: 0 | struct Container3::HasVtorDisp5
|
||
|
// CHECK-NEXT: 0 | (HasVtorDisp5 vftable pointer)
|
||
|
// CHECK-NEXT: 8 | (HasVtorDisp5 vbtable pointer)
|
||
|
// CHECK-NEXT: 20 | (vtordisp for vbase Base)
|
||
|
// CHECK-NEXT: 24 | struct Base (virtual base)
|
||
|
// CHECK-NEXT: 24 | (Base vftable pointer)
|
||
|
// CHECK-NEXT: | [sizeof=32, align=8,
|
||
|
// CHECK-NEXT: | nvsize=16, nvalign=8]
|
||
|
};
|
||
|
|
||
|
int main() {
|
||
|
Container::f();
|
||
|
Container2::f1();
|
||
|
Container2::f2();
|
||
|
Container3 cont3;
|
||
|
return 0;
|
||
|
};
|