|
@@ -0,0 +1,217 @@
|
|
|
+// 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;
|
|
|
+};
|