xref: /aosp_15_r20/external/llvm/unittests/Support/TrailingObjectsTest.cpp (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker //=== - llvm/unittest/Support/TrailingObjectsTest.cpp ---------------------===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker //                     The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker 
10*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/TrailingObjects.h"
11*9880d681SAndroid Build Coastguard Worker #include "gtest/gtest.h"
12*9880d681SAndroid Build Coastguard Worker 
13*9880d681SAndroid Build Coastguard Worker using namespace llvm;
14*9880d681SAndroid Build Coastguard Worker 
15*9880d681SAndroid Build Coastguard Worker namespace {
16*9880d681SAndroid Build Coastguard Worker // This class, beyond being used by the test case, a nice
17*9880d681SAndroid Build Coastguard Worker // demonstration of the intended usage of TrailingObjects, with a
18*9880d681SAndroid Build Coastguard Worker // single trailing array.
19*9880d681SAndroid Build Coastguard Worker class Class1 final : protected TrailingObjects<Class1, short> {
20*9880d681SAndroid Build Coastguard Worker   friend TrailingObjects;
21*9880d681SAndroid Build Coastguard Worker 
22*9880d681SAndroid Build Coastguard Worker   unsigned NumShorts;
23*9880d681SAndroid Build Coastguard Worker 
24*9880d681SAndroid Build Coastguard Worker protected:
numTrailingObjects(OverloadToken<short>) const25*9880d681SAndroid Build Coastguard Worker   size_t numTrailingObjects(OverloadToken<short>) const { return NumShorts; }
26*9880d681SAndroid Build Coastguard Worker 
Class1(int * ShortArray,unsigned NumShorts)27*9880d681SAndroid Build Coastguard Worker   Class1(int *ShortArray, unsigned NumShorts) : NumShorts(NumShorts) {
28*9880d681SAndroid Build Coastguard Worker     std::uninitialized_copy(ShortArray, ShortArray + NumShorts,
29*9880d681SAndroid Build Coastguard Worker                             getTrailingObjects<short>());
30*9880d681SAndroid Build Coastguard Worker   }
31*9880d681SAndroid Build Coastguard Worker 
32*9880d681SAndroid Build Coastguard Worker public:
create(int * ShortArray,unsigned NumShorts)33*9880d681SAndroid Build Coastguard Worker   static Class1 *create(int *ShortArray, unsigned NumShorts) {
34*9880d681SAndroid Build Coastguard Worker     void *Mem = ::operator new(totalSizeToAlloc<short>(NumShorts));
35*9880d681SAndroid Build Coastguard Worker     return new (Mem) Class1(ShortArray, NumShorts);
36*9880d681SAndroid Build Coastguard Worker   }
operator delete(void * p)37*9880d681SAndroid Build Coastguard Worker   void operator delete(void *p) { ::operator delete(p); }
38*9880d681SAndroid Build Coastguard Worker 
get(unsigned Num) const39*9880d681SAndroid Build Coastguard Worker   short get(unsigned Num) const { return getTrailingObjects<short>()[Num]; }
40*9880d681SAndroid Build Coastguard Worker 
numShorts() const41*9880d681SAndroid Build Coastguard Worker   unsigned numShorts() const { return NumShorts; }
42*9880d681SAndroid Build Coastguard Worker 
43*9880d681SAndroid Build Coastguard Worker   // Pull some protected members in as public, for testability.
44*9880d681SAndroid Build Coastguard Worker   using TrailingObjects::totalSizeToAlloc;
45*9880d681SAndroid Build Coastguard Worker   using TrailingObjects::additionalSizeToAlloc;
46*9880d681SAndroid Build Coastguard Worker   using TrailingObjects::getTrailingObjects;
47*9880d681SAndroid Build Coastguard Worker };
48*9880d681SAndroid Build Coastguard Worker 
49*9880d681SAndroid Build Coastguard Worker // Here, there are two singular optional object types appended.  Note
50*9880d681SAndroid Build Coastguard Worker // that the alignment of Class2 is automatically increased to account
51*9880d681SAndroid Build Coastguard Worker // for the alignment requirements of the trailing objects.
52*9880d681SAndroid Build Coastguard Worker class Class2 final : protected TrailingObjects<Class2, double, short> {
53*9880d681SAndroid Build Coastguard Worker   friend TrailingObjects;
54*9880d681SAndroid Build Coastguard Worker 
55*9880d681SAndroid Build Coastguard Worker   bool HasShort, HasDouble;
56*9880d681SAndroid Build Coastguard Worker 
57*9880d681SAndroid Build Coastguard Worker protected:
numTrailingObjects(OverloadToken<short>) const58*9880d681SAndroid Build Coastguard Worker   size_t numTrailingObjects(OverloadToken<short>) const {
59*9880d681SAndroid Build Coastguard Worker     return HasShort ? 1 : 0;
60*9880d681SAndroid Build Coastguard Worker   }
numTrailingObjects(OverloadToken<double>) const61*9880d681SAndroid Build Coastguard Worker   size_t numTrailingObjects(OverloadToken<double>) const {
62*9880d681SAndroid Build Coastguard Worker     return HasDouble ? 1 : 0;
63*9880d681SAndroid Build Coastguard Worker   }
64*9880d681SAndroid Build Coastguard Worker 
Class2(bool HasShort,bool HasDouble)65*9880d681SAndroid Build Coastguard Worker   Class2(bool HasShort, bool HasDouble)
66*9880d681SAndroid Build Coastguard Worker       : HasShort(HasShort), HasDouble(HasDouble) {}
67*9880d681SAndroid Build Coastguard Worker 
68*9880d681SAndroid Build Coastguard Worker public:
create(short S=0,double D=0.0)69*9880d681SAndroid Build Coastguard Worker   static Class2 *create(short S = 0, double D = 0.0) {
70*9880d681SAndroid Build Coastguard Worker     bool HasShort = S != 0;
71*9880d681SAndroid Build Coastguard Worker     bool HasDouble = D != 0.0;
72*9880d681SAndroid Build Coastguard Worker 
73*9880d681SAndroid Build Coastguard Worker     void *Mem =
74*9880d681SAndroid Build Coastguard Worker         ::operator new(totalSizeToAlloc<double, short>(HasDouble, HasShort));
75*9880d681SAndroid Build Coastguard Worker     Class2 *C = new (Mem) Class2(HasShort, HasDouble);
76*9880d681SAndroid Build Coastguard Worker     if (HasShort)
77*9880d681SAndroid Build Coastguard Worker       *C->getTrailingObjects<short>() = S;
78*9880d681SAndroid Build Coastguard Worker     if (HasDouble)
79*9880d681SAndroid Build Coastguard Worker       *C->getTrailingObjects<double>() = D;
80*9880d681SAndroid Build Coastguard Worker     return C;
81*9880d681SAndroid Build Coastguard Worker   }
operator delete(void * p)82*9880d681SAndroid Build Coastguard Worker   void operator delete(void *p) { ::operator delete(p); }
83*9880d681SAndroid Build Coastguard Worker 
getShort() const84*9880d681SAndroid Build Coastguard Worker   short getShort() const {
85*9880d681SAndroid Build Coastguard Worker     if (!HasShort)
86*9880d681SAndroid Build Coastguard Worker       return 0;
87*9880d681SAndroid Build Coastguard Worker     return *getTrailingObjects<short>();
88*9880d681SAndroid Build Coastguard Worker   }
89*9880d681SAndroid Build Coastguard Worker 
getDouble() const90*9880d681SAndroid Build Coastguard Worker   double getDouble() const {
91*9880d681SAndroid Build Coastguard Worker     if (!HasDouble)
92*9880d681SAndroid Build Coastguard Worker       return 0.0;
93*9880d681SAndroid Build Coastguard Worker     return *getTrailingObjects<double>();
94*9880d681SAndroid Build Coastguard Worker   }
95*9880d681SAndroid Build Coastguard Worker 
96*9880d681SAndroid Build Coastguard Worker   // Pull some protected members in as public, for testability.
97*9880d681SAndroid Build Coastguard Worker   using TrailingObjects::totalSizeToAlloc;
98*9880d681SAndroid Build Coastguard Worker   using TrailingObjects::additionalSizeToAlloc;
99*9880d681SAndroid Build Coastguard Worker   using TrailingObjects::getTrailingObjects;
100*9880d681SAndroid Build Coastguard Worker };
101*9880d681SAndroid Build Coastguard Worker 
TEST(TrailingObjects,OneArg)102*9880d681SAndroid Build Coastguard Worker TEST(TrailingObjects, OneArg) {
103*9880d681SAndroid Build Coastguard Worker   int arr[] = {1, 2, 3};
104*9880d681SAndroid Build Coastguard Worker   Class1 *C = Class1::create(arr, 3);
105*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(sizeof(Class1), sizeof(unsigned));
106*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(Class1::additionalSizeToAlloc<short>(1), sizeof(short));
107*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(Class1::additionalSizeToAlloc<short>(3), sizeof(short) * 3);
108*9880d681SAndroid Build Coastguard Worker 
109*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(Class1::totalSizeToAlloc<short>(1), sizeof(Class1) + sizeof(short));
110*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(Class1::totalSizeToAlloc<short>(3),
111*9880d681SAndroid Build Coastguard Worker             sizeof(Class1) + sizeof(short) * 3);
112*9880d681SAndroid Build Coastguard Worker 
113*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C->getTrailingObjects<short>(), reinterpret_cast<short *>(C + 1));
114*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C->get(0), 1);
115*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C->get(2), 3);
116*9880d681SAndroid Build Coastguard Worker   delete C;
117*9880d681SAndroid Build Coastguard Worker }
118*9880d681SAndroid Build Coastguard Worker 
TEST(TrailingObjects,TwoArg)119*9880d681SAndroid Build Coastguard Worker TEST(TrailingObjects, TwoArg) {
120*9880d681SAndroid Build Coastguard Worker   Class2 *C1 = Class2::create(4);
121*9880d681SAndroid Build Coastguard Worker   Class2 *C2 = Class2::create(0, 4.2);
122*9880d681SAndroid Build Coastguard Worker 
123*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(sizeof(Class2),
124*9880d681SAndroid Build Coastguard Worker             llvm::alignTo(sizeof(bool) * 2, llvm::alignOf<double>()));
125*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(llvm::alignOf<Class2>(), llvm::alignOf<double>());
126*9880d681SAndroid Build Coastguard Worker 
127*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ((Class2::additionalSizeToAlloc<double, short>(1, 0)),
128*9880d681SAndroid Build Coastguard Worker             sizeof(double));
129*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ((Class2::additionalSizeToAlloc<double, short>(0, 1)),
130*9880d681SAndroid Build Coastguard Worker             sizeof(short));
131*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ((Class2::additionalSizeToAlloc<double, short>(3, 1)),
132*9880d681SAndroid Build Coastguard Worker             sizeof(double) * 3 + sizeof(short));
133*9880d681SAndroid Build Coastguard Worker 
134*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ((Class2::totalSizeToAlloc<double, short>(1, 1)),
135*9880d681SAndroid Build Coastguard Worker             sizeof(Class2) + sizeof(double) + sizeof(short));
136*9880d681SAndroid Build Coastguard Worker 
137*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C1->getDouble(), 0);
138*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C1->getShort(), 4);
139*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C1->getTrailingObjects<double>(),
140*9880d681SAndroid Build Coastguard Worker             reinterpret_cast<double *>(C1 + 1));
141*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C1->getTrailingObjects<short>(), reinterpret_cast<short *>(C1 + 1));
142*9880d681SAndroid Build Coastguard Worker 
143*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C2->getDouble(), 4.2);
144*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C2->getShort(), 0);
145*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C2->getTrailingObjects<double>(),
146*9880d681SAndroid Build Coastguard Worker             reinterpret_cast<double *>(C2 + 1));
147*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C2->getTrailingObjects<short>(),
148*9880d681SAndroid Build Coastguard Worker             reinterpret_cast<short *>(reinterpret_cast<double *>(C2 + 1) + 1));
149*9880d681SAndroid Build Coastguard Worker   delete C1;
150*9880d681SAndroid Build Coastguard Worker   delete C2;
151*9880d681SAndroid Build Coastguard Worker }
152*9880d681SAndroid Build Coastguard Worker 
153*9880d681SAndroid Build Coastguard Worker // This test class is not trying to be a usage demo, just asserting
154*9880d681SAndroid Build Coastguard Worker // that three args does actually work too (it's the same code as
155*9880d681SAndroid Build Coastguard Worker // handles the second arg, so it's basically covered by the above, but
156*9880d681SAndroid Build Coastguard Worker // just in case..)
157*9880d681SAndroid Build Coastguard Worker class Class3 final : public TrailingObjects<Class3, double, short, bool> {
158*9880d681SAndroid Build Coastguard Worker   friend TrailingObjects;
159*9880d681SAndroid Build Coastguard Worker 
numTrailingObjects(OverloadToken<double>) const160*9880d681SAndroid Build Coastguard Worker   size_t numTrailingObjects(OverloadToken<double>) const { return 1; }
numTrailingObjects(OverloadToken<short>) const161*9880d681SAndroid Build Coastguard Worker   size_t numTrailingObjects(OverloadToken<short>) const { return 1; }
162*9880d681SAndroid Build Coastguard Worker };
163*9880d681SAndroid Build Coastguard Worker 
TEST(TrailingObjects,ThreeArg)164*9880d681SAndroid Build Coastguard Worker TEST(TrailingObjects, ThreeArg) {
165*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ((Class3::additionalSizeToAlloc<double, short, bool>(1, 1, 3)),
166*9880d681SAndroid Build Coastguard Worker             sizeof(double) + sizeof(short) + 3 * sizeof(bool));
167*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(sizeof(Class3), llvm::alignTo(1, llvm::alignOf<double>()));
168*9880d681SAndroid Build Coastguard Worker   std::unique_ptr<char[]> P(new char[1000]);
169*9880d681SAndroid Build Coastguard Worker   Class3 *C = reinterpret_cast<Class3 *>(P.get());
170*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C->getTrailingObjects<double>(), reinterpret_cast<double *>(C + 1));
171*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C->getTrailingObjects<short>(),
172*9880d681SAndroid Build Coastguard Worker             reinterpret_cast<short *>(reinterpret_cast<double *>(C + 1) + 1));
173*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(
174*9880d681SAndroid Build Coastguard Worker       C->getTrailingObjects<bool>(),
175*9880d681SAndroid Build Coastguard Worker       reinterpret_cast<bool *>(
176*9880d681SAndroid Build Coastguard Worker           reinterpret_cast<short *>(reinterpret_cast<double *>(C + 1) + 1) +
177*9880d681SAndroid Build Coastguard Worker           1));
178*9880d681SAndroid Build Coastguard Worker }
179*9880d681SAndroid Build Coastguard Worker 
180*9880d681SAndroid Build Coastguard Worker class Class4 final : public TrailingObjects<Class4, char, long> {
181*9880d681SAndroid Build Coastguard Worker   friend TrailingObjects;
numTrailingObjects(OverloadToken<char>) const182*9880d681SAndroid Build Coastguard Worker   size_t numTrailingObjects(OverloadToken<char>) const { return 1; }
183*9880d681SAndroid Build Coastguard Worker };
184*9880d681SAndroid Build Coastguard Worker 
TEST(TrailingObjects,Realignment)185*9880d681SAndroid Build Coastguard Worker TEST(TrailingObjects, Realignment) {
186*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ((Class4::additionalSizeToAlloc<char, long>(1, 1)),
187*9880d681SAndroid Build Coastguard Worker             llvm::alignTo(sizeof(long) + 1, llvm::alignOf<long>()));
188*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(sizeof(Class4), llvm::alignTo(1, llvm::alignOf<long>()));
189*9880d681SAndroid Build Coastguard Worker   std::unique_ptr<char[]> P(new char[1000]);
190*9880d681SAndroid Build Coastguard Worker   Class4 *C = reinterpret_cast<Class4 *>(P.get());
191*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C->getTrailingObjects<char>(), reinterpret_cast<char *>(C + 1));
192*9880d681SAndroid Build Coastguard Worker   EXPECT_EQ(C->getTrailingObjects<long>(),
193*9880d681SAndroid Build Coastguard Worker             reinterpret_cast<long *>(llvm::alignAddr(
194*9880d681SAndroid Build Coastguard Worker                 reinterpret_cast<char *>(C + 1) + 1, llvm::alignOf<long>())));
195*9880d681SAndroid Build Coastguard Worker }
196*9880d681SAndroid Build Coastguard Worker }
197