TrailingObjectsTest.cpp 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. //=== - llvm/unittest/Support/TrailingObjectsTest.cpp ---------------------===//
  2. //
  3. // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
  4. // See https://llvm.org/LICENSE.txt for license information.
  5. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  6. //
  7. //===----------------------------------------------------------------------===//
  8. #include "llvm/Support/TrailingObjects.h"
  9. #include "gtest/gtest.h"
  10. using namespace llvm;
  11. namespace {
  12. // This class, beyond being used by the test case, a nice
  13. // demonstration of the intended usage of TrailingObjects, with a
  14. // single trailing array.
  15. class Class1 final : protected TrailingObjects<Class1, short> {
  16. friend TrailingObjects;
  17. unsigned NumShorts;
  18. protected:
  19. size_t numTrailingObjects(OverloadToken<short>) const { return NumShorts; }
  20. Class1(int *ShortArray, unsigned NumShorts) : NumShorts(NumShorts) {
  21. std::uninitialized_copy(ShortArray, ShortArray + NumShorts,
  22. getTrailingObjects<short>());
  23. }
  24. public:
  25. static Class1 *create(int *ShortArray, unsigned NumShorts) {
  26. void *Mem = ::operator new(totalSizeToAlloc<short>(NumShorts));
  27. return new (Mem) Class1(ShortArray, NumShorts);
  28. }
  29. void operator delete(void *p) { ::operator delete(p); }
  30. short get(unsigned Num) const { return getTrailingObjects<short>()[Num]; }
  31. unsigned numShorts() const { return NumShorts; }
  32. // Pull some protected members in as public, for testability.
  33. template <typename... Ty>
  34. using FixedSizeStorage = TrailingObjects::FixedSizeStorage<Ty...>;
  35. using TrailingObjects::totalSizeToAlloc;
  36. using TrailingObjects::additionalSizeToAlloc;
  37. using TrailingObjects::getTrailingObjects;
  38. };
  39. // Here, there are two singular optional object types appended. Note
  40. // that the alignment of Class2 is automatically increased to account
  41. // for the alignment requirements of the trailing objects.
  42. class Class2 final : protected TrailingObjects<Class2, double, short> {
  43. friend TrailingObjects;
  44. bool HasShort, HasDouble;
  45. protected:
  46. size_t numTrailingObjects(OverloadToken<short>) const {
  47. return HasShort ? 1 : 0;
  48. }
  49. size_t numTrailingObjects(OverloadToken<double>) const {
  50. return HasDouble ? 1 : 0;
  51. }
  52. Class2(bool HasShort, bool HasDouble)
  53. : HasShort(HasShort), HasDouble(HasDouble) {}
  54. public:
  55. static Class2 *create(short S = 0, double D = 0.0) {
  56. bool HasShort = S != 0;
  57. bool HasDouble = D != 0.0;
  58. void *Mem =
  59. ::operator new(totalSizeToAlloc<double, short>(HasDouble, HasShort));
  60. Class2 *C = new (Mem) Class2(HasShort, HasDouble);
  61. if (HasShort)
  62. *C->getTrailingObjects<short>() = S;
  63. if (HasDouble)
  64. *C->getTrailingObjects<double>() = D;
  65. return C;
  66. }
  67. void operator delete(void *p) { ::operator delete(p); }
  68. short getShort() const {
  69. if (!HasShort)
  70. return 0;
  71. return *getTrailingObjects<short>();
  72. }
  73. double getDouble() const {
  74. if (!HasDouble)
  75. return 0.0;
  76. return *getTrailingObjects<double>();
  77. }
  78. // Pull some protected members in as public, for testability.
  79. template <typename... Ty>
  80. using FixedSizeStorage = TrailingObjects::FixedSizeStorage<Ty...>;
  81. using TrailingObjects::totalSizeToAlloc;
  82. using TrailingObjects::additionalSizeToAlloc;
  83. using TrailingObjects::getTrailingObjects;
  84. };
  85. TEST(TrailingObjects, OneArg) {
  86. int arr[] = {1, 2, 3};
  87. Class1 *C = Class1::create(arr, 3);
  88. EXPECT_EQ(sizeof(Class1), sizeof(unsigned));
  89. EXPECT_EQ(Class1::additionalSizeToAlloc<short>(1), sizeof(short));
  90. EXPECT_EQ(Class1::additionalSizeToAlloc<short>(3), sizeof(short) * 3);
  91. EXPECT_EQ(alignof(Class1),
  92. alignof(Class1::FixedSizeStorage<short>::with_counts<1>::type));
  93. EXPECT_EQ(sizeof(Class1::FixedSizeStorage<short>::with_counts<1>::type),
  94. llvm::alignTo(Class1::totalSizeToAlloc<short>(1), alignof(Class1)));
  95. EXPECT_EQ(Class1::totalSizeToAlloc<short>(1), sizeof(Class1) + sizeof(short));
  96. EXPECT_EQ(alignof(Class1),
  97. alignof(Class1::FixedSizeStorage<short>::with_counts<3>::type));
  98. EXPECT_EQ(sizeof(Class1::FixedSizeStorage<short>::with_counts<3>::type),
  99. llvm::alignTo(Class1::totalSizeToAlloc<short>(3), alignof(Class1)));
  100. EXPECT_EQ(Class1::totalSizeToAlloc<short>(3),
  101. sizeof(Class1) + sizeof(short) * 3);
  102. EXPECT_EQ(C->getTrailingObjects<short>(), reinterpret_cast<short *>(C + 1));
  103. EXPECT_EQ(C->get(0), 1);
  104. EXPECT_EQ(C->get(2), 3);
  105. delete C;
  106. }
  107. TEST(TrailingObjects, TwoArg) {
  108. Class2 *C1 = Class2::create(4);
  109. Class2 *C2 = Class2::create(0, 4.2);
  110. EXPECT_EQ(sizeof(Class2), llvm::alignTo(sizeof(bool) * 2, alignof(double)));
  111. EXPECT_EQ(alignof(Class2), alignof(double));
  112. EXPECT_EQ((Class2::additionalSizeToAlloc<double, short>(1, 0)),
  113. sizeof(double));
  114. EXPECT_EQ((Class2::additionalSizeToAlloc<double, short>(0, 1)),
  115. sizeof(short));
  116. EXPECT_EQ((Class2::additionalSizeToAlloc<double, short>(3, 1)),
  117. sizeof(double) * 3 + sizeof(short));
  118. EXPECT_EQ(
  119. alignof(Class2),
  120. (alignof(
  121. Class2::FixedSizeStorage<double, short>::with_counts<1, 1>::type)));
  122. EXPECT_EQ(
  123. sizeof(Class2::FixedSizeStorage<double, short>::with_counts<1, 1>::type),
  124. llvm::alignTo(Class2::totalSizeToAlloc<double, short>(1, 1),
  125. alignof(Class2)));
  126. EXPECT_EQ((Class2::totalSizeToAlloc<double, short>(1, 1)),
  127. sizeof(Class2) + sizeof(double) + sizeof(short));
  128. EXPECT_EQ(C1->getDouble(), 0);
  129. EXPECT_EQ(C1->getShort(), 4);
  130. EXPECT_EQ(C1->getTrailingObjects<double>(),
  131. reinterpret_cast<double *>(C1 + 1));
  132. EXPECT_EQ(C1->getTrailingObjects<short>(), reinterpret_cast<short *>(C1 + 1));
  133. EXPECT_EQ(C2->getDouble(), 4.2);
  134. EXPECT_EQ(C2->getShort(), 0);
  135. EXPECT_EQ(C2->getTrailingObjects<double>(),
  136. reinterpret_cast<double *>(C2 + 1));
  137. EXPECT_EQ(C2->getTrailingObjects<short>(),
  138. reinterpret_cast<short *>(reinterpret_cast<double *>(C2 + 1) + 1));
  139. delete C1;
  140. delete C2;
  141. }
  142. // This test class is not trying to be a usage demo, just asserting
  143. // that three args does actually work too (it's the same code as
  144. // handles the second arg, so it's basically covered by the above, but
  145. // just in case..)
  146. class Class3 final : public TrailingObjects<Class3, double, short, bool> {
  147. friend TrailingObjects;
  148. size_t numTrailingObjects(OverloadToken<double>) const { return 1; }
  149. size_t numTrailingObjects(OverloadToken<short>) const { return 1; }
  150. };
  151. TEST(TrailingObjects, ThreeArg) {
  152. EXPECT_EQ((Class3::additionalSizeToAlloc<double, short, bool>(1, 1, 3)),
  153. sizeof(double) + sizeof(short) + 3 * sizeof(bool));
  154. EXPECT_EQ(sizeof(Class3), llvm::alignTo(1, alignof(double)));
  155. EXPECT_EQ(
  156. alignof(Class3),
  157. (alignof(Class3::FixedSizeStorage<double, short,
  158. bool>::with_counts<1, 1, 3>::type)));
  159. EXPECT_EQ(
  160. sizeof(Class3::FixedSizeStorage<double, short,
  161. bool>::with_counts<1, 1, 3>::type),
  162. llvm::alignTo(Class3::totalSizeToAlloc<double, short, bool>(1, 1, 3),
  163. alignof(Class3)));
  164. std::unique_ptr<char[]> P(new char[1000]);
  165. Class3 *C = reinterpret_cast<Class3 *>(P.get());
  166. EXPECT_EQ(C->getTrailingObjects<double>(), reinterpret_cast<double *>(C + 1));
  167. EXPECT_EQ(C->getTrailingObjects<short>(),
  168. reinterpret_cast<short *>(reinterpret_cast<double *>(C + 1) + 1));
  169. EXPECT_EQ(
  170. C->getTrailingObjects<bool>(),
  171. reinterpret_cast<bool *>(
  172. reinterpret_cast<short *>(reinterpret_cast<double *>(C + 1) + 1) +
  173. 1));
  174. }
  175. class Class4 final : public TrailingObjects<Class4, char, long> {
  176. friend TrailingObjects;
  177. size_t numTrailingObjects(OverloadToken<char>) const { return 1; }
  178. };
  179. TEST(TrailingObjects, Realignment) {
  180. EXPECT_EQ((Class4::additionalSizeToAlloc<char, long>(1, 1)),
  181. llvm::alignTo(sizeof(long) + 1, alignof(long)));
  182. EXPECT_EQ(sizeof(Class4), llvm::alignTo(1, alignof(long)));
  183. EXPECT_EQ(
  184. alignof(Class4),
  185. (alignof(Class4::FixedSizeStorage<char, long>::with_counts<1, 1>::type)));
  186. EXPECT_EQ(
  187. sizeof(Class4::FixedSizeStorage<char, long>::with_counts<1, 1>::type),
  188. llvm::alignTo(Class4::totalSizeToAlloc<char, long>(1, 1),
  189. alignof(Class4)));
  190. std::unique_ptr<char[]> P(new char[1000]);
  191. Class4 *C = reinterpret_cast<Class4 *>(P.get());
  192. EXPECT_EQ(C->getTrailingObjects<char>(), reinterpret_cast<char *>(C + 1));
  193. EXPECT_EQ(C->getTrailingObjects<long>(),
  194. reinterpret_cast<long *>(llvm::alignAddr(
  195. reinterpret_cast<char *>(C + 1) + 1, Align::Of<long>())));
  196. }
  197. }
  198. // Test the use of TrailingObjects with a template class. This
  199. // previously failed to compile due to a bug in MSVC's member access
  200. // control/lookup handling for OverloadToken.
  201. template <typename Derived>
  202. class Class5Tmpl : private llvm::TrailingObjects<Derived, float, int> {
  203. using TrailingObjects = typename llvm::TrailingObjects<Derived, float>;
  204. friend TrailingObjects;
  205. size_t numTrailingObjects(
  206. typename TrailingObjects::template OverloadToken<float>) const {
  207. return 1;
  208. }
  209. size_t numTrailingObjects(
  210. typename TrailingObjects::template OverloadToken<int>) const {
  211. return 2;
  212. }
  213. };
  214. class Class5 : public Class5Tmpl<Class5> {};