ms-x86-vfvb-alignment.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388
  1. // RUN: %clang_cc1 -fno-rtti -emit-llvm-only -triple i686-pc-win32 -fms-extensions -fdump-record-layouts -fsyntax-only %s 2>&1 \
  2. // RUN: | FileCheck %s
  3. // RUN: %clang_cc1 -fno-rtti -emit-llvm-only -triple x86_64-pc-win32 -fms-extensions -fdump-record-layouts -fsyntax-only %s 2>/dev/null \
  4. // RUN: | FileCheck %s -check-prefix CHECK-X64
  5. extern "C" int printf(const char *fmt, ...);
  6. struct B0 { int a; B0() : a(0xf00000B0) {} };
  7. struct B1 { char a; B1() : a(0xB1) {} };
  8. struct B2 : virtual B1 { int a; B2() : a(0xf00000B2) {} };
  9. struct B3 { __declspec(align(16)) int a; B3() : a(0xf00000B3) {} };
  10. struct B4 : virtual B3 { int a; B4() : a(0xf00000B4) {} };
  11. struct B5 { __declspec(align(32)) int a; B5() : a(0xf00000B5) {} };
  12. struct B6 { int a; B6() : a(0xf00000B6) {} virtual void f() { printf("B6"); } };
  13. struct A : B0, virtual B1 { __declspec(align(16)) int a; A() : a(0xf000000A) {} virtual void f() { printf("A"); } };
  14. // CHECK: *** Dumping AST Record Layout
  15. // CHECK: *** Dumping AST Record Layout
  16. // CHECK: *** Dumping AST Record Layout
  17. // CHECK-NEXT: 0 | struct A
  18. // CHECK-NEXT: 0 | (A vftable pointer)
  19. // CHECK-NEXT: 16 | struct B0 (base)
  20. // CHECK-NEXT: 16 | int a
  21. // CHECK-NEXT: 20 | (A vbtable pointer)
  22. // CHECK-NEXT: 48 | int a
  23. // CHECK-NEXT: 64 | struct B1 (virtual base)
  24. // CHECK-NEXT: 64 | char a
  25. // CHECK-NEXT: | [sizeof=80, align=16
  26. // CHECK-NEXT: | nvsize=64, nvalign=16]
  27. // CHECK-X64: *** Dumping AST Record Layout
  28. // CHECK-X64: *** Dumping AST Record Layout
  29. // CHECK-X64: *** Dumping AST Record Layout
  30. // CHECK-X64-NEXT: 0 | struct A
  31. // CHECK-X64-NEXT: 0 | (A vftable pointer)
  32. // CHECK-X64-NEXT: 16 | struct B0 (base)
  33. // CHECK-X64-NEXT: 16 | int a
  34. // CHECK-X64-NEXT: 24 | (A vbtable pointer)
  35. // CHECK-X64-NEXT: 48 | int a
  36. // CHECK-X64-NEXT: 64 | struct B1 (virtual base)
  37. // CHECK-X64-NEXT: 64 | char a
  38. // CHECK-X64-NEXT: | [sizeof=80, align=16
  39. // CHECK-X64-NEXT: | nvsize=64, nvalign=16]
  40. struct B : A, B2 { int a; B() : a(0xf000000B) {} virtual void f() { printf("B"); } };
  41. // CHECK: *** Dumping AST Record Layout
  42. // CHECK: *** Dumping AST Record Layout
  43. // CHECK-NEXT: 0 | struct B
  44. // CHECK-NEXT: 0 | struct A (primary base)
  45. // CHECK-NEXT: 0 | (A vftable pointer)
  46. // CHECK-NEXT: 16 | struct B0 (base)
  47. // CHECK-NEXT: 16 | int a
  48. // CHECK-NEXT: 20 | (A vbtable pointer)
  49. // CHECK-NEXT: 48 | int a
  50. // CHECK-NEXT: 64 | struct B2 (base)
  51. // CHECK-NEXT: 64 | (B2 vbtable pointer)
  52. // CHECK-NEXT: 68 | int a
  53. // CHECK-NEXT: 72 | int a
  54. // CHECK-NEXT: 80 | struct B1 (virtual base)
  55. // CHECK-NEXT: 80 | char a
  56. // CHECK-NEXT: | [sizeof=96, align=16
  57. // CHECK-NEXT: | nvsize=80, nvalign=16]
  58. // CHECK-X64: *** Dumping AST Record Layout
  59. // CHECK-X64: *** Dumping AST Record Layout
  60. // CHECK-X64-NEXT: 0 | struct B
  61. // CHECK-X64-NEXT: 0 | struct A (primary base)
  62. // CHECK-X64-NEXT: 0 | (A vftable pointer)
  63. // CHECK-X64-NEXT: 16 | struct B0 (base)
  64. // CHECK-X64-NEXT: 16 | int a
  65. // CHECK-X64-NEXT: 24 | (A vbtable pointer)
  66. // CHECK-X64-NEXT: 48 | int a
  67. // CHECK-X64-NEXT: 64 | struct B2 (base)
  68. // CHECK-X64-NEXT: 64 | (B2 vbtable pointer)
  69. // CHECK-X64-NEXT: 72 | int a
  70. // CHECK-X64-NEXT: 80 | int a
  71. // CHECK-X64-NEXT: 96 | struct B1 (virtual base)
  72. // CHECK-X64-NEXT: 96 | char a
  73. // CHECK-X64-NEXT: | [sizeof=112, align=16
  74. // CHECK-X64-NEXT: | nvsize=96, nvalign=16]
  75. struct C : B4 { int a; C() : a(0xf000000C) {} virtual void f() { printf("C"); } };
  76. // CHECK: *** Dumping AST Record Layout
  77. // CHECK: *** Dumping AST Record Layout
  78. // CHECK: *** Dumping AST Record Layout
  79. // CHECK-NEXT: 0 | struct C
  80. // CHECK-NEXT: 0 | (C vftable pointer)
  81. // CHECK-NEXT: 16 | struct B4 (base)
  82. // CHECK-NEXT: 16 | (B4 vbtable pointer)
  83. // CHECK-NEXT: 20 | int a
  84. // CHECK-NEXT: 24 | int a
  85. // CHECK-NEXT: 32 | struct B3 (virtual base)
  86. // CHECK-NEXT: 32 | int a
  87. // CHECK-NEXT: | [sizeof=48, align=16
  88. // CHECK-NEXT: | nvsize=32, nvalign=16]
  89. // CHECK-X64: *** Dumping AST Record Layout
  90. // CHECK-X64: *** Dumping AST Record Layout
  91. // CHECK-X64: *** Dumping AST Record Layout
  92. // CHECK-X64-NEXT: 0 | struct C
  93. // CHECK-X64-NEXT: 0 | (C vftable pointer)
  94. // CHECK-X64-NEXT: 16 | struct B4 (base)
  95. // CHECK-X64-NEXT: 16 | (B4 vbtable pointer)
  96. // CHECK-X64-NEXT: 24 | int a
  97. // CHECK-X64-NEXT: 32 | int a
  98. // CHECK-X64-NEXT: 48 | struct B3 (virtual base)
  99. // CHECK-X64-NEXT: 48 | int a
  100. // CHECK-X64-NEXT: | [sizeof=64, align=16
  101. // CHECK-X64-NEXT: | nvsize=48, nvalign=16]
  102. struct D : C { int a; D() : a(0xf000000D) {} virtual void f() { printf("D"); } };
  103. // CHECK: *** Dumping AST Record Layout
  104. // CHECK-NEXT: 0 | struct D
  105. // CHECK-NEXT: 0 | struct C (primary base)
  106. // CHECK-NEXT: 0 | (C vftable pointer)
  107. // CHECK-NEXT: 16 | struct B4 (base)
  108. // CHECK-NEXT: 16 | (B4 vbtable pointer)
  109. // CHECK-NEXT: 20 | int a
  110. // CHECK-NEXT: 24 | int a
  111. // CHECK-NEXT: 32 | int a
  112. // CHECK-NEXT: 48 | struct B3 (virtual base)
  113. // CHECK-NEXT: 48 | int a
  114. // CHECK-NEXT: | [sizeof=64, align=16
  115. // CHECK-NEXT: | nvsize=48, nvalign=16]
  116. // CHECK-X64: *** Dumping AST Record Layout
  117. // CHECK-X64-NEXT: 0 | struct D
  118. // CHECK-X64-NEXT: 0 | struct C (primary base)
  119. // CHECK-X64-NEXT: 0 | (C vftable pointer)
  120. // CHECK-X64-NEXT: 16 | struct B4 (base)
  121. // CHECK-X64-NEXT: 16 | (B4 vbtable pointer)
  122. // CHECK-X64-NEXT: 24 | int a
  123. // CHECK-X64-NEXT: 32 | int a
  124. // CHECK-X64-NEXT: 48 | int a
  125. // CHECK-X64-NEXT: 64 | struct B3 (virtual base)
  126. // CHECK-X64-NEXT: 64 | int a
  127. // CHECK-X64-NEXT: | [sizeof=80, align=16
  128. // CHECK-X64-NEXT: | nvsize=64, nvalign=16]
  129. struct E : virtual C { int a; E() : a(0xf000000E) {} virtual void f() { printf("E"); } };
  130. // CHECK: *** Dumping AST Record Layout
  131. // CHECK-NEXT: 0 | struct E
  132. // CHECK-NEXT: 0 | (E vbtable pointer)
  133. // CHECK-NEXT: 4 | int a
  134. // CHECK-NEXT: 16 | struct B3 (virtual base)
  135. // CHECK-NEXT: 16 | int a
  136. // CHECK-NEXT: 44 | (vtordisp for vbase C)
  137. // CHECK-NEXT: 48 | struct C (virtual base)
  138. // CHECK-NEXT: 48 | (C vftable pointer)
  139. // CHECK-NEXT: 64 | struct B4 (base)
  140. // CHECK-NEXT: 64 | (B4 vbtable pointer)
  141. // CHECK-NEXT: 68 | int a
  142. // CHECK-NEXT: 72 | int a
  143. // CHECK-NEXT: | [sizeof=80, align=16
  144. // CHECK-NEXT: | nvsize=8, nvalign=16]
  145. // CHECK-X64: *** Dumping AST Record Layout
  146. // CHECK-X64-NEXT: 0 | struct E
  147. // CHECK-X64-NEXT: 0 | (E vbtable pointer)
  148. // CHECK-X64-NEXT: 8 | int a
  149. // CHECK-X64-NEXT: 16 | struct B3 (virtual base)
  150. // CHECK-X64-NEXT: 16 | int a
  151. // CHECK-X64-NEXT: 44 | (vtordisp for vbase C)
  152. // CHECK-X64-NEXT: 48 | struct C (virtual base)
  153. // CHECK-X64-NEXT: 48 | (C vftable pointer)
  154. // CHECK-X64-NEXT: 64 | struct B4 (base)
  155. // CHECK-X64-NEXT: 64 | (B4 vbtable pointer)
  156. // CHECK-X64-NEXT: 72 | int a
  157. // CHECK-X64-NEXT: 80 | int a
  158. // CHECK-X64-NEXT: | [sizeof=96, align=16
  159. // CHECK-X64-NEXT: | nvsize=16, nvalign=16]
  160. struct F : B3, virtual B0 { int a; F() : a(0xf000000F) {} virtual void f() { printf("F"); } };
  161. // CHECK: *** Dumping AST Record Layout
  162. // CHECK-NEXT: 0 | struct F
  163. // CHECK-NEXT: 0 | (F vftable pointer)
  164. // CHECK-NEXT: 16 | struct B3 (base)
  165. // CHECK-NEXT: 16 | int a
  166. // CHECK-NEXT: 32 | (F vbtable pointer)
  167. // CHECK-NEXT: 48 | int a
  168. // CHECK-NEXT: 64 | struct B0 (virtual base)
  169. // CHECK-NEXT: 64 | int a
  170. // CHECK-NEXT: | [sizeof=80, align=16
  171. // CHECK-NEXT: | nvsize=64, nvalign=16]
  172. // CHECK-X64: *** Dumping AST Record Layout
  173. // CHECK-X64-NEXT: 0 | struct F
  174. // CHECK-X64-NEXT: 0 | (F vftable pointer)
  175. // CHECK-X64-NEXT: 16 | struct B3 (base)
  176. // CHECK-X64-NEXT: 16 | int a
  177. // CHECK-X64-NEXT: 32 | (F vbtable pointer)
  178. // CHECK-X64-NEXT: 48 | int a
  179. // CHECK-X64-NEXT: 64 | struct B0 (virtual base)
  180. // CHECK-X64-NEXT: 64 | int a
  181. // CHECK-X64-NEXT: | [sizeof=80, align=16
  182. // CHECK-X64-NEXT: | nvsize=64, nvalign=16]
  183. struct G : B2, B6, virtual B1 { int a; G() : a(0xf0000010) {} };
  184. // CHECK: *** Dumping AST Record Layout
  185. // CHECK: *** Dumping AST Record Layout
  186. // CHECK-NEXT: 0 | struct G
  187. // CHECK-NEXT: 0 | struct B6 (primary base)
  188. // CHECK-NEXT: 0 | (B6 vftable pointer)
  189. // CHECK-NEXT: 4 | int a
  190. // CHECK-NEXT: 8 | struct B2 (base)
  191. // CHECK-NEXT: 8 | (B2 vbtable pointer)
  192. // CHECK-NEXT: 12 | int a
  193. // CHECK-NEXT: 16 | int a
  194. // CHECK-NEXT: 20 | struct B1 (virtual base)
  195. // CHECK-NEXT: 20 | char a
  196. // CHECK-NEXT: | [sizeof=21, align=4
  197. // CHECK-NEXT: | nvsize=20, nvalign=4]
  198. // CHECK-X64: *** Dumping AST Record Layout
  199. // CHECK-X64: *** Dumping AST Record Layout
  200. // CHECK-X64-NEXT: 0 | struct G
  201. // CHECK-X64-NEXT: 0 | struct B6 (primary base)
  202. // CHECK-X64-NEXT: 0 | (B6 vftable pointer)
  203. // CHECK-X64-NEXT: 8 | int a
  204. // CHECK-X64-NEXT: 16 | struct B2 (base)
  205. // CHECK-X64-NEXT: 16 | (B2 vbtable pointer)
  206. // CHECK-X64-NEXT: 24 | int a
  207. // CHECK-X64-NEXT: 32 | int a
  208. // CHECK-X64-NEXT: 40 | struct B1 (virtual base)
  209. // CHECK-X64-NEXT: 40 | char a
  210. // CHECK-X64-NEXT: | [sizeof=48, align=8
  211. // CHECK-X64-NEXT: | nvsize=40, nvalign=8]
  212. struct H : B6, B2, virtual B1 { int a; H() : a(0xf0000011) {} };
  213. // CHECK: *** Dumping AST Record Layout
  214. // CHECK-NEXT: 0 | struct H
  215. // CHECK-NEXT: 0 | struct B6 (primary base)
  216. // CHECK-NEXT: 0 | (B6 vftable pointer)
  217. // CHECK-NEXT: 4 | int a
  218. // CHECK-NEXT: 8 | struct B2 (base)
  219. // CHECK-NEXT: 8 | (B2 vbtable pointer)
  220. // CHECK-NEXT: 12 | int a
  221. // CHECK-NEXT: 16 | int a
  222. // CHECK-NEXT: 20 | struct B1 (virtual base)
  223. // CHECK-NEXT: 20 | char a
  224. // CHECK-NEXT: | [sizeof=21, align=4
  225. // CHECK-NEXT: | nvsize=20, nvalign=4]
  226. // CHECK-X64: *** Dumping AST Record Layout
  227. // CHECK-X64-NEXT: 0 | struct H
  228. // CHECK-X64-NEXT: 0 | struct B6 (primary base)
  229. // CHECK-X64-NEXT: 0 | (B6 vftable pointer)
  230. // CHECK-X64-NEXT: 8 | int a
  231. // CHECK-X64-NEXT: 16 | struct B2 (base)
  232. // CHECK-X64-NEXT: 16 | (B2 vbtable pointer)
  233. // CHECK-X64-NEXT: 24 | int a
  234. // CHECK-X64-NEXT: 32 | int a
  235. // CHECK-X64-NEXT: 40 | struct B1 (virtual base)
  236. // CHECK-X64-NEXT: 40 | char a
  237. // CHECK-X64-NEXT: | [sizeof=48, align=8
  238. // CHECK-X64-NEXT: | nvsize=40, nvalign=8]
  239. struct I : B0, virtual B1 { int a; int a1; __declspec(align(16)) int a2; I() : a(0xf0000011), a1(0xf0000011), a2(0xf0000011) {} };
  240. // CHECK: *** Dumping AST Record Layout
  241. // CHECK-NEXT: 0 | struct I
  242. // CHECK-NEXT: 0 | struct B0 (base)
  243. // CHECK-NEXT: 0 | int a
  244. // CHECK-NEXT: 4 | (I vbtable pointer)
  245. // CHECK-NEXT: 20 | int a
  246. // CHECK-NEXT: 24 | int a1
  247. // CHECK-NEXT: 32 | int a2
  248. // CHECK-NEXT: 48 | struct B1 (virtual base)
  249. // CHECK-NEXT: 48 | char a
  250. // CHECK-NEXT: | [sizeof=64, align=16
  251. // CHECK-NEXT: | nvsize=48, nvalign=16]
  252. // CHECK-X64: *** Dumping AST Record Layout
  253. // CHECK-X64-NEXT: 0 | struct I
  254. // CHECK-X64-NEXT: 0 | struct B0 (base)
  255. // CHECK-X64-NEXT: 0 | int a
  256. // CHECK-X64-NEXT: 8 | (I vbtable pointer)
  257. // CHECK-X64-NEXT: 20 | int a
  258. // CHECK-X64-NEXT: 24 | int a1
  259. // CHECK-X64-NEXT: 32 | int a2
  260. // CHECK-X64-NEXT: 48 | struct B1 (virtual base)
  261. // CHECK-X64-NEXT: 48 | char a
  262. // CHECK-X64-NEXT: | [sizeof=64, align=16
  263. // CHECK-X64-NEXT: | nvsize=48, nvalign=16]
  264. struct J : B0, B3, virtual B1 { int a; int a1; J() : a(0xf0000012), a1(0xf0000012) {} };
  265. // CHECK: *** Dumping AST Record Layout
  266. // CHECK-NEXT: 0 | struct J
  267. // CHECK-NEXT: 0 | struct B0 (base)
  268. // CHECK-NEXT: 0 | int a
  269. // CHECK-NEXT: 16 | struct B3 (base)
  270. // CHECK-NEXT: 16 | int a
  271. // CHECK-NEXT: 32 | (J vbtable pointer)
  272. // CHECK-NEXT: 48 | int a
  273. // CHECK-NEXT: 52 | int a1
  274. // CHECK-NEXT: 64 | struct B1 (virtual base)
  275. // CHECK-NEXT: 64 | char a
  276. // CHECK-NEXT: | [sizeof=80, align=16
  277. // CHECK-NEXT: | nvsize=64, nvalign=16]
  278. // CHECK-X64: *** Dumping AST Record Layout
  279. // CHECK-X64-NEXT: 0 | struct J
  280. // CHECK-X64-NEXT: 0 | struct B0 (base)
  281. // CHECK-X64-NEXT: 0 | int a
  282. // CHECK-X64-NEXT: 16 | struct B3 (base)
  283. // CHECK-X64-NEXT: 16 | int a
  284. // CHECK-X64-NEXT: 32 | (J vbtable pointer)
  285. // CHECK-X64-NEXT: 48 | int a
  286. // CHECK-X64-NEXT: 52 | int a1
  287. // CHECK-X64-NEXT: 64 | struct B1 (virtual base)
  288. // CHECK-X64-NEXT: 64 | char a
  289. // CHECK-X64-NEXT: | [sizeof=80, align=16
  290. // CHECK-X64-NEXT: | nvsize=64, nvalign=16]
  291. struct K { int a; K() : a(0xf0000013) {} virtual void f() { printf("K"); } };
  292. // CHECK: *** Dumping AST Record Layout
  293. // CHECK-NEXT: 0 | struct K
  294. // CHECK-NEXT: 0 | (K vftable pointer)
  295. // CHECK-NEXT: 4 | int a
  296. // CHECK-NEXT: | [sizeof=8, align=4
  297. // CHECK-NEXT: | nvsize=8, nvalign=4]
  298. // CHECK-X64: *** Dumping AST Record Layout
  299. // CHECK-X64-NEXT: 0 | struct K
  300. // CHECK-X64-NEXT: 0 | (K vftable pointer)
  301. // CHECK-X64-NEXT: 8 | int a
  302. // CHECK-X64-NEXT: | [sizeof=16, align=8
  303. // CHECK-X64-NEXT: | nvsize=16, nvalign=8]
  304. struct L : virtual K { int a; L() : a(0xf0000014) {} virtual void g() { printf("L"); } };
  305. // CHECK: *** Dumping AST Record Layout
  306. // CHECK-NEXT: 0 | struct L
  307. // CHECK-NEXT: 0 | (L vftable pointer)
  308. // CHECK-NEXT: 4 | (L vbtable pointer)
  309. // CHECK-NEXT: 8 | int a
  310. // CHECK-NEXT: 12 | struct K (virtual base)
  311. // CHECK-NEXT: 12 | (K vftable pointer)
  312. // CHECK-NEXT: 16 | int a
  313. // CHECK-NEXT: | [sizeof=20, align=4
  314. // CHECK-NEXT: | nvsize=12, nvalign=4]
  315. // CHECK-X64: *** Dumping AST Record Layout
  316. // CHECK-X64-NEXT: 0 | struct L
  317. // CHECK-X64-NEXT: 0 | (L vftable pointer)
  318. // CHECK-X64-NEXT: 8 | (L vbtable pointer)
  319. // CHECK-X64-NEXT: 16 | int a
  320. // CHECK-X64-NEXT: 24 | struct K (virtual base)
  321. // CHECK-X64-NEXT: 24 | (K vftable pointer)
  322. // CHECK-X64-NEXT: 32 | int a
  323. // CHECK-X64-NEXT: | [sizeof=40, align=8
  324. // CHECK-X64-NEXT: | nvsize=24, nvalign=8]
  325. struct M : virtual K { int a; M() : a(0xf0000015) {} virtual void f() { printf("M"); } };
  326. // CHECK: *** Dumping AST Record Layout
  327. // CHECK-NEXT: 0 | struct M
  328. // CHECK-NEXT: 0 | (M vbtable pointer)
  329. // CHECK-NEXT: 4 | int a
  330. // CHECK-NEXT: 8 | (vtordisp for vbase K)
  331. // CHECK-NEXT: 12 | struct K (virtual base)
  332. // CHECK-NEXT: 12 | (K vftable pointer)
  333. // CHECK-NEXT: 16 | int a
  334. // CHECK-NEXT: | [sizeof=20, align=4
  335. // CHECK-NEXT: | nvsize=8, nvalign=4]
  336. // CHECK-X64: *** Dumping AST Record Layout
  337. // CHECK-X64-NEXT: 0 | struct M
  338. // CHECK-X64-NEXT: 0 | (M vbtable pointer)
  339. // CHECK-X64-NEXT: 8 | int a
  340. // CHECK-X64-NEXT: 20 | (vtordisp for vbase K)
  341. // CHECK-X64-NEXT: 24 | struct K (virtual base)
  342. // CHECK-X64-NEXT: 24 | (K vftable pointer)
  343. // CHECK-X64-NEXT: 32 | int a
  344. // CHECK-X64-NEXT: | [sizeof=40, align=8
  345. // CHECK-X64-NEXT: | nvsize=16, nvalign=8]
  346. int a[
  347. sizeof(A)+
  348. sizeof(B)+
  349. sizeof(C)+
  350. sizeof(D)+
  351. sizeof(E)+
  352. sizeof(F)+
  353. sizeof(G)+
  354. sizeof(H)+
  355. sizeof(I)+
  356. sizeof(J)+
  357. sizeof(K)+
  358. sizeof(L)+
  359. sizeof(M)];