|  | //=== - llvm/unittest/Support/TrailingObjectsTest.cpp ---------------------===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "llvm/Support/TrailingObjects.h" | 
|  | #include "gtest/gtest.h" | 
|  |  | 
|  | using namespace llvm; | 
|  |  | 
|  | namespace { | 
|  | // This class, beyond being used by the test case, a nice | 
|  | // demonstration of the intended usage of TrailingObjects, with a | 
|  | // single trailing array. | 
|  | class Class1 final : protected TrailingObjects<Class1, short> { | 
|  | friend TrailingObjects; | 
|  |  | 
|  | unsigned NumShorts; | 
|  |  | 
|  | protected: | 
|  | size_t numTrailingObjects(OverloadToken<short>) const { return NumShorts; } | 
|  |  | 
|  | Class1(int *ShortArray, unsigned NumShorts) : NumShorts(NumShorts) { | 
|  | std::uninitialized_copy(ShortArray, ShortArray + NumShorts, | 
|  | getTrailingObjects<short>()); | 
|  | } | 
|  |  | 
|  | public: | 
|  | static Class1 *create(int *ShortArray, unsigned NumShorts) { | 
|  | void *Mem = ::operator new(totalSizeToAlloc<short>(NumShorts)); | 
|  | return new (Mem) Class1(ShortArray, NumShorts); | 
|  | } | 
|  | void operator delete(void *p) { ::operator delete(p); } | 
|  |  | 
|  | short get(unsigned Num) const { return getTrailingObjects<short>()[Num]; } | 
|  |  | 
|  | unsigned numShorts() const { return NumShorts; } | 
|  |  | 
|  | // Pull some protected members in as public, for testability. | 
|  | template <typename... Ty> | 
|  | using FixedSizeStorage = TrailingObjects::FixedSizeStorage<Ty...>; | 
|  |  | 
|  | using TrailingObjects::totalSizeToAlloc; | 
|  | using TrailingObjects::additionalSizeToAlloc; | 
|  | using TrailingObjects::getTrailingObjects; | 
|  | }; | 
|  |  | 
|  | // Here, there are two singular optional object types appended.  Note | 
|  | // that the alignment of Class2 is automatically increased to account | 
|  | // for the alignment requirements of the trailing objects. | 
|  | class Class2 final : protected TrailingObjects<Class2, double, short> { | 
|  | friend TrailingObjects; | 
|  |  | 
|  | bool HasShort, HasDouble; | 
|  |  | 
|  | protected: | 
|  | size_t numTrailingObjects(OverloadToken<short>) const { | 
|  | return HasShort ? 1 : 0; | 
|  | } | 
|  | size_t numTrailingObjects(OverloadToken<double>) const { | 
|  | return HasDouble ? 1 : 0; | 
|  | } | 
|  |  | 
|  | Class2(bool HasShort, bool HasDouble) | 
|  | : HasShort(HasShort), HasDouble(HasDouble) {} | 
|  |  | 
|  | public: | 
|  | static Class2 *create(short S = 0, double D = 0.0) { | 
|  | bool HasShort = S != 0; | 
|  | bool HasDouble = D != 0.0; | 
|  |  | 
|  | void *Mem = | 
|  | ::operator new(totalSizeToAlloc<double, short>(HasDouble, HasShort)); | 
|  | Class2 *C = new (Mem) Class2(HasShort, HasDouble); | 
|  | if (HasShort) | 
|  | *C->getTrailingObjects<short>() = S; | 
|  | if (HasDouble) | 
|  | *C->getTrailingObjects<double>() = D; | 
|  | return C; | 
|  | } | 
|  | void operator delete(void *p) { ::operator delete(p); } | 
|  |  | 
|  | short getShort() const { | 
|  | if (!HasShort) | 
|  | return 0; | 
|  | return *getTrailingObjects<short>(); | 
|  | } | 
|  |  | 
|  | double getDouble() const { | 
|  | if (!HasDouble) | 
|  | return 0.0; | 
|  | return *getTrailingObjects<double>(); | 
|  | } | 
|  |  | 
|  | // Pull some protected members in as public, for testability. | 
|  | template <typename... Ty> | 
|  | using FixedSizeStorage = TrailingObjects::FixedSizeStorage<Ty...>; | 
|  |  | 
|  | using TrailingObjects::totalSizeToAlloc; | 
|  | using TrailingObjects::additionalSizeToAlloc; | 
|  | using TrailingObjects::getTrailingObjects; | 
|  | }; | 
|  |  | 
|  | TEST(TrailingObjects, OneArg) { | 
|  | int arr[] = {1, 2, 3}; | 
|  | Class1 *C = Class1::create(arr, 3); | 
|  | EXPECT_EQ(sizeof(Class1), sizeof(unsigned)); | 
|  | EXPECT_EQ(Class1::additionalSizeToAlloc<short>(1), sizeof(short)); | 
|  | EXPECT_EQ(Class1::additionalSizeToAlloc<short>(3), sizeof(short) * 3); | 
|  |  | 
|  | EXPECT_EQ(alignof(Class1), | 
|  | alignof(Class1::FixedSizeStorage<short>::with_counts<1>::type)); | 
|  | EXPECT_EQ(sizeof(Class1::FixedSizeStorage<short>::with_counts<1>::type), | 
|  | llvm::alignTo(Class1::totalSizeToAlloc<short>(1), alignof(Class1))); | 
|  | EXPECT_EQ(Class1::totalSizeToAlloc<short>(1), sizeof(Class1) + sizeof(short)); | 
|  |  | 
|  | EXPECT_EQ(alignof(Class1), | 
|  | alignof(Class1::FixedSizeStorage<short>::with_counts<3>::type)); | 
|  | EXPECT_EQ(sizeof(Class1::FixedSizeStorage<short>::with_counts<3>::type), | 
|  | llvm::alignTo(Class1::totalSizeToAlloc<short>(3), alignof(Class1))); | 
|  | EXPECT_EQ(Class1::totalSizeToAlloc<short>(3), | 
|  | sizeof(Class1) + sizeof(short) * 3); | 
|  |  | 
|  | EXPECT_EQ(C->getTrailingObjects<short>(), reinterpret_cast<short *>(C + 1)); | 
|  | EXPECT_EQ(C->get(0), 1); | 
|  | EXPECT_EQ(C->get(2), 3); | 
|  | delete C; | 
|  | } | 
|  |  | 
|  | TEST(TrailingObjects, TwoArg) { | 
|  | Class2 *C1 = Class2::create(4); | 
|  | Class2 *C2 = Class2::create(0, 4.2); | 
|  |  | 
|  | EXPECT_EQ(sizeof(Class2), llvm::alignTo(sizeof(bool) * 2, alignof(double))); | 
|  | EXPECT_EQ(alignof(Class2), alignof(double)); | 
|  |  | 
|  | EXPECT_EQ((Class2::additionalSizeToAlloc<double, short>(1, 0)), | 
|  | sizeof(double)); | 
|  | EXPECT_EQ((Class2::additionalSizeToAlloc<double, short>(0, 1)), | 
|  | sizeof(short)); | 
|  | EXPECT_EQ((Class2::additionalSizeToAlloc<double, short>(3, 1)), | 
|  | sizeof(double) * 3 + sizeof(short)); | 
|  |  | 
|  | EXPECT_EQ( | 
|  | alignof(Class2), | 
|  | (alignof( | 
|  | Class2::FixedSizeStorage<double, short>::with_counts<1, 1>::type))); | 
|  | EXPECT_EQ( | 
|  | sizeof(Class2::FixedSizeStorage<double, short>::with_counts<1, 1>::type), | 
|  | llvm::alignTo(Class2::totalSizeToAlloc<double, short>(1, 1), | 
|  | alignof(Class2))); | 
|  | EXPECT_EQ((Class2::totalSizeToAlloc<double, short>(1, 1)), | 
|  | sizeof(Class2) + sizeof(double) + sizeof(short)); | 
|  |  | 
|  | EXPECT_EQ(C1->getDouble(), 0); | 
|  | EXPECT_EQ(C1->getShort(), 4); | 
|  | EXPECT_EQ(C1->getTrailingObjects<double>(), | 
|  | reinterpret_cast<double *>(C1 + 1)); | 
|  | EXPECT_EQ(C1->getTrailingObjects<short>(), reinterpret_cast<short *>(C1 + 1)); | 
|  |  | 
|  | EXPECT_EQ(C2->getDouble(), 4.2); | 
|  | EXPECT_EQ(C2->getShort(), 0); | 
|  | EXPECT_EQ(C2->getTrailingObjects<double>(), | 
|  | reinterpret_cast<double *>(C2 + 1)); | 
|  | EXPECT_EQ(C2->getTrailingObjects<short>(), | 
|  | reinterpret_cast<short *>(reinterpret_cast<double *>(C2 + 1) + 1)); | 
|  | delete C1; | 
|  | delete C2; | 
|  | } | 
|  |  | 
|  | // This test class is not trying to be a usage demo, just asserting | 
|  | // that three args does actually work too (it's the same code as | 
|  | // handles the second arg, so it's basically covered by the above, but | 
|  | // just in case..) | 
|  | class Class3 final : public TrailingObjects<Class3, double, short, bool> { | 
|  | friend TrailingObjects; | 
|  |  | 
|  | size_t numTrailingObjects(OverloadToken<double>) const { return 1; } | 
|  | size_t numTrailingObjects(OverloadToken<short>) const { return 1; } | 
|  | }; | 
|  |  | 
|  | TEST(TrailingObjects, ThreeArg) { | 
|  | EXPECT_EQ((Class3::additionalSizeToAlloc<double, short, bool>(1, 1, 3)), | 
|  | sizeof(double) + sizeof(short) + 3 * sizeof(bool)); | 
|  | EXPECT_EQ(sizeof(Class3), llvm::alignTo(1, alignof(double))); | 
|  |  | 
|  | EXPECT_EQ( | 
|  | alignof(Class3), | 
|  | (alignof(Class3::FixedSizeStorage<double, short, | 
|  | bool>::with_counts<1, 1, 3>::type))); | 
|  | EXPECT_EQ( | 
|  | sizeof(Class3::FixedSizeStorage<double, short, | 
|  | bool>::with_counts<1, 1, 3>::type), | 
|  | llvm::alignTo(Class3::totalSizeToAlloc<double, short, bool>(1, 1, 3), | 
|  | alignof(Class3))); | 
|  |  | 
|  | std::unique_ptr<char[]> P(new char[1000]); | 
|  | Class3 *C = reinterpret_cast<Class3 *>(P.get()); | 
|  | EXPECT_EQ(C->getTrailingObjects<double>(), reinterpret_cast<double *>(C + 1)); | 
|  | EXPECT_EQ(C->getTrailingObjects<short>(), | 
|  | reinterpret_cast<short *>(reinterpret_cast<double *>(C + 1) + 1)); | 
|  | EXPECT_EQ( | 
|  | C->getTrailingObjects<bool>(), | 
|  | reinterpret_cast<bool *>( | 
|  | reinterpret_cast<short *>(reinterpret_cast<double *>(C + 1) + 1) + | 
|  | 1)); | 
|  | } | 
|  |  | 
|  | class Class4 final : public TrailingObjects<Class4, char, long> { | 
|  | friend TrailingObjects; | 
|  | size_t numTrailingObjects(OverloadToken<char>) const { return 1; } | 
|  | }; | 
|  |  | 
|  | TEST(TrailingObjects, Realignment) { | 
|  | EXPECT_EQ((Class4::additionalSizeToAlloc<char, long>(1, 1)), | 
|  | llvm::alignTo(sizeof(long) + 1, alignof(long))); | 
|  | EXPECT_EQ(sizeof(Class4), llvm::alignTo(1, alignof(long))); | 
|  |  | 
|  | EXPECT_EQ( | 
|  | alignof(Class4), | 
|  | (alignof(Class4::FixedSizeStorage<char, long>::with_counts<1, 1>::type))); | 
|  | EXPECT_EQ( | 
|  | sizeof(Class4::FixedSizeStorage<char, long>::with_counts<1, 1>::type), | 
|  | llvm::alignTo(Class4::totalSizeToAlloc<char, long>(1, 1), | 
|  | alignof(Class4))); | 
|  |  | 
|  | std::unique_ptr<char[]> P(new char[1000]); | 
|  | Class4 *C = reinterpret_cast<Class4 *>(P.get()); | 
|  | EXPECT_EQ(C->getTrailingObjects<char>(), reinterpret_cast<char *>(C + 1)); | 
|  | EXPECT_EQ(C->getTrailingObjects<long>(), | 
|  | reinterpret_cast<long *>(llvm::alignAddr( | 
|  | reinterpret_cast<char *>(C + 1) + 1, alignof(long)))); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Test the use of TrailingObjects with a template class. This | 
|  | // previously failed to compile due to a bug in MSVC's member access | 
|  | // control/lookup handling for OverloadToken. | 
|  | template <typename Derived> | 
|  | class Class5Tmpl : private llvm::TrailingObjects<Derived, float, int> { | 
|  | using TrailingObjects = typename llvm::TrailingObjects<Derived, float>; | 
|  | friend TrailingObjects; | 
|  |  | 
|  | size_t numTrailingObjects( | 
|  | typename TrailingObjects::template OverloadToken<float>) const { | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | size_t numTrailingObjects( | 
|  | typename TrailingObjects::template OverloadToken<int>) const { | 
|  | return 2; | 
|  | } | 
|  | }; | 
|  |  | 
|  | class Class5 : public Class5Tmpl<Class5> {}; |