ms-vtordisp-local.cpp 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. // 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
  2. struct Base {
  3. virtual ~Base() {}
  4. virtual void BaseFunc() {}
  5. };
  6. #pragma vtordisp(0)
  7. struct Container {
  8. static void f() try {
  9. #pragma vtordisp(2)
  10. struct HasVtorDisp : virtual Base {
  11. virtual ~HasVtorDisp() {}
  12. virtual void Func() {}
  13. };
  14. int x[sizeof(HasVtorDisp)];
  15. // HasVtorDisp: vtordisp because of pragma right before it.
  16. //
  17. // CHECK: *** Dumping AST Record Layout
  18. // CHECK: *** Dumping AST Record Layout
  19. // CHECK-NEXT: 0 | struct HasVtorDisp
  20. // CHECK-NEXT: 0 | (HasVtorDisp vftable pointer)
  21. // CHECK-NEXT: 8 | (HasVtorDisp vbtable pointer)
  22. // CHECK-NEXT: 20 | (vtordisp for vbase Base)
  23. // CHECK-NEXT: 24 | struct Base (virtual base)
  24. // CHECK-NEXT: 24 | (Base vftable pointer)
  25. // CHECK-NEXT: | [sizeof=32, align=8,
  26. // CHECK-NEXT: | nvsize=16, nvalign=8]
  27. } catch (...) {
  28. }
  29. };
  30. struct NoVtorDisp1 : virtual Base {
  31. virtual ~NoVtorDisp1() {}
  32. virtual void Func() {}
  33. };
  34. int x1[sizeof(NoVtorDisp1)];
  35. // NoVtroDisp1: no vtordisp because of pragma disabling it.
  36. //
  37. // CHECK: *** Dumping AST Record Layout
  38. // CHECK-NEXT: 0 | struct NoVtorDisp1
  39. // CHECK-NEXT: 0 | (NoVtorDisp1 vftable pointer)
  40. // CHECK-NEXT: 8 | (NoVtorDisp1 vbtable pointer)
  41. // CHECK-NEXT: 16 | struct Base (virtual base)
  42. // CHECK-NEXT: 16 | (Base vftable pointer)
  43. // CHECK-NEXT: | [sizeof=24, align=8,
  44. // CHECK-NEXT: | nvsize=16, nvalign=8]
  45. struct Container2 {
  46. static void f1() {
  47. // Local pragma #1 - must be disabled on exit from f1().
  48. #pragma vtordisp(push, 2)
  49. struct HasVtorDisp1 : virtual Base {
  50. virtual ~HasVtorDisp1() {}
  51. virtual void Func() {}
  52. };
  53. int x2[sizeof(HasVtorDisp1)];
  54. // HasVtorDisp1: vtordisp because of pragma right before it.
  55. //
  56. // CHECK: *** Dumping AST Record Layout
  57. // CHECK-NEXT: 0 | struct HasVtorDisp1
  58. // CHECK-NEXT: 0 | (HasVtorDisp1 vftable pointer)
  59. // CHECK-NEXT: 8 | (HasVtorDisp1 vbtable pointer)
  60. // CHECK-NEXT: 20 | (vtordisp for vbase Base)
  61. // CHECK-NEXT: 24 | struct Base (virtual base)
  62. // CHECK-NEXT: 24 | (Base vftable pointer)
  63. // CHECK-NEXT: | [sizeof=32, align=8,
  64. // CHECK-NEXT: | nvsize=16, nvalign=8]
  65. struct InnerContainer {
  66. static void g1() {
  67. struct HasVtorDisp2 : virtual Base {
  68. virtual ~HasVtorDisp2() {}
  69. virtual void Func() {}
  70. };
  71. int x3[sizeof(HasVtorDisp2)];
  72. // HasVtorDisp2: vtordisp because of vtordisp(2) in f1().
  73. //
  74. // CHECK: *** Dumping AST Record Layout
  75. // CHECK-NEXT: 0 | struct HasVtorDisp2
  76. // CHECK-NEXT: 0 | (HasVtorDisp2 vftable pointer)
  77. // CHECK-NEXT: 8 | (HasVtorDisp2 vbtable pointer)
  78. // CHECK-NEXT: 20 | (vtordisp for vbase Base)
  79. // CHECK-NEXT: 24 | struct Base (virtual base)
  80. // CHECK-NEXT: 24 | (Base vftable pointer)
  81. // CHECK-NEXT: | [sizeof=32, align=8,
  82. // CHECK-NEXT: | nvsize=16, nvalign=8]
  83. // Local pragma #2 - must be disabled on exit from g1().
  84. #pragma vtordisp(push, 0)
  85. struct NoVtorDisp2 : virtual Base {
  86. virtual ~NoVtorDisp2() {}
  87. virtual void Func() {}
  88. };
  89. int x4[sizeof(NoVtorDisp2)];
  90. // NoVtroDisp2: no vtordisp because of vtordisp(0) in g1().
  91. //
  92. // CHECK: *** Dumping AST Record Layout
  93. // CHECK-NEXT: 0 | struct NoVtorDisp2
  94. // CHECK-NEXT: 0 | (NoVtorDisp2 vftable pointer)
  95. // CHECK-NEXT: 8 | (NoVtorDisp2 vbtable pointer)
  96. // CHECK-NEXT: 16 | struct Base (virtual base)
  97. // CHECK-NEXT: 16 | (Base vftable pointer)
  98. // CHECK-NEXT: | [sizeof=24, align=8,
  99. // CHECK-NEXT: | nvsize=16, nvalign=8]
  100. }
  101. static void g2() {
  102. struct HasVtorDisp3 : virtual Base {
  103. virtual ~HasVtorDisp3() {}
  104. virtual void Func() {}
  105. };
  106. int x5[sizeof(HasVtorDisp3)];
  107. // HasVtorDisp3: vtordisp because of vtordisp(2) in f1(),
  108. // local vtordisp(0) in g1() is disabled.
  109. //
  110. // CHECK: *** Dumping AST Record Layout
  111. // CHECK-NEXT: 0 | struct HasVtorDisp3
  112. // CHECK-NEXT: 0 | (HasVtorDisp3 vftable pointer)
  113. // CHECK-NEXT: 8 | (HasVtorDisp3 vbtable pointer)
  114. // CHECK-NEXT: 20 | (vtordisp for vbase Base)
  115. // CHECK-NEXT: 24 | struct Base (virtual base)
  116. // CHECK-NEXT: 24 | (Base vftable pointer)
  117. // CHECK-NEXT: | [sizeof=32, align=8,
  118. // CHECK-NEXT: | nvsize=16, nvalign=8]
  119. }
  120. };
  121. struct HasVtorDisp4 : virtual Base {
  122. virtual ~HasVtorDisp4() {}
  123. virtual void Func() {}
  124. };
  125. int x6[sizeof(HasVtorDisp4)];
  126. // HasVtorDisp4: vtordisp because of vtordisp(2) in f1(),
  127. // local vtordisp(0) in g1() is disabled,
  128. // g2() has no pragmas - stack is not affected.
  129. //
  130. // CHECK: *** Dumping AST Record Layout
  131. // CHECK-NEXT: 0 | struct HasVtorDisp4
  132. // CHECK-NEXT: 0 | (HasVtorDisp4 vftable pointer)
  133. // CHECK-NEXT: 8 | (HasVtorDisp4 vbtable pointer)
  134. // CHECK-NEXT: 20 | (vtordisp for vbase Base)
  135. // CHECK-NEXT: 24 | struct Base (virtual base)
  136. // CHECK-NEXT: 24 | (Base vftable pointer)
  137. // CHECK-NEXT: | [sizeof=32, align=8,
  138. // CHECK-NEXT: | nvsize=16, nvalign=8]
  139. InnerContainer::g1();
  140. InnerContainer::g2();
  141. }
  142. static void f2() {
  143. struct NoVtorDisp3 : virtual Base {
  144. virtual ~NoVtorDisp3() {}
  145. virtual void Func() {}
  146. };
  147. int x7[sizeof(NoVtorDisp3)];
  148. // NoVtroDisp3: no vtordisp because of global pragma (0),
  149. // local vtordisp(2) is disabled on exit from f1().
  150. //
  151. // CHECK: *** Dumping AST Record Layout
  152. // CHECK-NEXT: 0 | struct NoVtorDisp3
  153. // CHECK-NEXT: 0 | (NoVtorDisp3 vftable pointer)
  154. // CHECK-NEXT: 8 | (NoVtorDisp3 vbtable pointer)
  155. // CHECK-NEXT: 16 | struct Base (virtual base)
  156. // CHECK-NEXT: 16 | (Base vftable pointer)
  157. // CHECK-NEXT: | [sizeof=24, align=8,
  158. // CHECK-NEXT: | nvsize=16, nvalign=8]
  159. }
  160. };
  161. struct Container3 {
  162. #pragma vtordisp(2)
  163. struct HasVtorDisp5 : virtual Base {
  164. virtual ~HasVtorDisp5() {}
  165. virtual void Func() {}
  166. };
  167. int x8[sizeof(HasVtorDisp5)];
  168. // HasVtorDisp5: vtordisp because of pragma right before it.
  169. //
  170. // CHECK: *** Dumping AST Record Layout
  171. // CHECK-NEXT: 0 | struct Container3::HasVtorDisp5
  172. // CHECK-NEXT: 0 | (HasVtorDisp5 vftable pointer)
  173. // CHECK-NEXT: 8 | (HasVtorDisp5 vbtable pointer)
  174. // CHECK-NEXT: 20 | (vtordisp for vbase Base)
  175. // CHECK-NEXT: 24 | struct Base (virtual base)
  176. // CHECK-NEXT: 24 | (Base vftable pointer)
  177. // CHECK-NEXT: | [sizeof=32, align=8,
  178. // CHECK-NEXT: | nvsize=16, nvalign=8]
  179. };
  180. int main() {
  181. Container::f();
  182. Container2::f1();
  183. Container2::f2();
  184. Container3 cont3;
  185. return 0;
  186. };