xref: /aosp_15_r20/system/chre/util/tests/unique_ptr_test.cc (revision 84e339476a462649f82315436d70fd732297a399)
1*84e33947SAndroid Build Coastguard Worker /*
2*84e33947SAndroid Build Coastguard Worker  * Copyright (C) 2017 The Android Open Source Project
3*84e33947SAndroid Build Coastguard Worker  *
4*84e33947SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*84e33947SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*84e33947SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*84e33947SAndroid Build Coastguard Worker  *
8*84e33947SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*84e33947SAndroid Build Coastguard Worker  *
10*84e33947SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*84e33947SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*84e33947SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*84e33947SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*84e33947SAndroid Build Coastguard Worker  * limitations under the License.
15*84e33947SAndroid Build Coastguard Worker  */
16*84e33947SAndroid Build Coastguard Worker 
17*84e33947SAndroid Build Coastguard Worker #include <cstring>
18*84e33947SAndroid Build Coastguard Worker 
19*84e33947SAndroid Build Coastguard Worker #include "gtest/gtest.h"
20*84e33947SAndroid Build Coastguard Worker 
21*84e33947SAndroid Build Coastguard Worker #include "chre/util/unique_ptr.h"
22*84e33947SAndroid Build Coastguard Worker 
23*84e33947SAndroid Build Coastguard Worker using chre::MakeUnique;
24*84e33947SAndroid Build Coastguard Worker using chre::MakeUniqueArray;
25*84e33947SAndroid Build Coastguard Worker using chre::MakeUniqueZeroFill;
26*84e33947SAndroid Build Coastguard Worker using chre::UniquePtr;
27*84e33947SAndroid Build Coastguard Worker using chre::util::internal::is_unbounded_array_v;
28*84e33947SAndroid Build Coastguard Worker 
29*84e33947SAndroid Build Coastguard Worker struct Value {
ValueValue30*84e33947SAndroid Build Coastguard Worker   Value(int value) : value(value) {
31*84e33947SAndroid Build Coastguard Worker     constructionCounter++;
32*84e33947SAndroid Build Coastguard Worker   }
33*84e33947SAndroid Build Coastguard Worker 
~ValueValue34*84e33947SAndroid Build Coastguard Worker   ~Value() {
35*84e33947SAndroid Build Coastguard Worker     constructionCounter--;
36*84e33947SAndroid Build Coastguard Worker   }
37*84e33947SAndroid Build Coastguard Worker 
operator =Value38*84e33947SAndroid Build Coastguard Worker   Value &operator=(Value &&other) {
39*84e33947SAndroid Build Coastguard Worker     value = other.value;
40*84e33947SAndroid Build Coastguard Worker     return *this;
41*84e33947SAndroid Build Coastguard Worker   }
42*84e33947SAndroid Build Coastguard Worker 
43*84e33947SAndroid Build Coastguard Worker   int value;
44*84e33947SAndroid Build Coastguard Worker   static int constructionCounter;
45*84e33947SAndroid Build Coastguard Worker };
46*84e33947SAndroid Build Coastguard Worker 
47*84e33947SAndroid Build Coastguard Worker int Value::constructionCounter = 0;
48*84e33947SAndroid Build Coastguard Worker 
TEST(UniquePtr,NullInit)49*84e33947SAndroid Build Coastguard Worker TEST(UniquePtr, NullInit) {
50*84e33947SAndroid Build Coastguard Worker   // Put something on the stack to help catch uninitialized memory errors
51*84e33947SAndroid Build Coastguard Worker   {
52*84e33947SAndroid Build Coastguard Worker     UniquePtr<int> p1 = MakeUnique<int>();
53*84e33947SAndroid Build Coastguard Worker     // Verify that the typical null checks are implemented correctly
54*84e33947SAndroid Build Coastguard Worker     ASSERT_FALSE(p1.isNull());
55*84e33947SAndroid Build Coastguard Worker     EXPECT_TRUE(p1);
56*84e33947SAndroid Build Coastguard Worker     EXPECT_NE(p1, nullptr);
57*84e33947SAndroid Build Coastguard Worker   }
58*84e33947SAndroid Build Coastguard Worker   {
59*84e33947SAndroid Build Coastguard Worker     UniquePtr<int> p1;
60*84e33947SAndroid Build Coastguard Worker     EXPECT_FALSE(p1);
61*84e33947SAndroid Build Coastguard Worker     EXPECT_TRUE(p1.isNull());
62*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(p1, nullptr);
63*84e33947SAndroid Build Coastguard Worker   }
64*84e33947SAndroid Build Coastguard Worker   UniquePtr<int> p2(nullptr);
65*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(p2.isNull());
66*84e33947SAndroid Build Coastguard Worker }
67*84e33947SAndroid Build Coastguard Worker 
TEST(UniquePtr,Construct)68*84e33947SAndroid Build Coastguard Worker TEST(UniquePtr, Construct) {
69*84e33947SAndroid Build Coastguard Worker   UniquePtr<Value> myInt = MakeUnique<Value>(0xcafe);
70*84e33947SAndroid Build Coastguard Worker   ASSERT_TRUE(myInt);
71*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(myInt.get()->value, 0xcafe);
72*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(myInt->value, 0xcafe);
73*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ((*myInt).value, 0xcafe);
74*84e33947SAndroid Build Coastguard Worker 
75*84e33947SAndroid Build Coastguard Worker   int *realInt = chre::memoryAlloc<int>();
76*84e33947SAndroid Build Coastguard Worker   ASSERT_NE(realInt, nullptr);
77*84e33947SAndroid Build Coastguard Worker   UniquePtr<int> wrapped(realInt);
78*84e33947SAndroid Build Coastguard Worker   ASSERT_TRUE(realInt);
79*84e33947SAndroid Build Coastguard Worker }
80*84e33947SAndroid Build Coastguard Worker 
81*84e33947SAndroid Build Coastguard Worker struct BigArray {
82*84e33947SAndroid Build Coastguard Worker   int x[2048];
83*84e33947SAndroid Build Coastguard Worker };
84*84e33947SAndroid Build Coastguard Worker 
85*84e33947SAndroid Build Coastguard Worker // Check the is_unbounded_array_v backport used in memoryAllocArray to help
86*84e33947SAndroid Build Coastguard Worker // constrain usage of MakeUniqueArray to only the proper type category
87*84e33947SAndroid Build Coastguard Worker static_assert(is_unbounded_array_v<int> == false &&
88*84e33947SAndroid Build Coastguard Worker                   is_unbounded_array_v<char[2]> == false &&
89*84e33947SAndroid Build Coastguard Worker                   is_unbounded_array_v<char[]> == true,
90*84e33947SAndroid Build Coastguard Worker               "is_unbounded_array_v implemented incorrectly");
91*84e33947SAndroid Build Coastguard Worker 
TEST(UniquePtr,MakeUniqueArray)92*84e33947SAndroid Build Coastguard Worker TEST(UniquePtr, MakeUniqueArray) {
93*84e33947SAndroid Build Coastguard Worker   // For these tests, we are just allocating and writing to the array - the main
94*84e33947SAndroid Build Coastguard Worker   // thing we are looking for is that the allocation is of an appropriate size,
95*84e33947SAndroid Build Coastguard Worker   // which should be checked when running this test with sanitizers enabled
96*84e33947SAndroid Build Coastguard Worker   {
97*84e33947SAndroid Build Coastguard Worker     constexpr size_t kSize = 32;
98*84e33947SAndroid Build Coastguard Worker     UniquePtr<char[]> ptr = MakeUniqueArray<char[]>(kSize);
99*84e33947SAndroid Build Coastguard Worker     ASSERT_FALSE(ptr.isNull());
100*84e33947SAndroid Build Coastguard Worker     std::memset(ptr.get(), 0x98, kSize);
101*84e33947SAndroid Build Coastguard Worker     ptr[0] = 0x11;
102*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(*ptr.get(), 0x11);
103*84e33947SAndroid Build Coastguard Worker   }
104*84e33947SAndroid Build Coastguard Worker   {
105*84e33947SAndroid Build Coastguard Worker     constexpr size_t kSize = 4;
106*84e33947SAndroid Build Coastguard Worker     auto ptr = MakeUniqueArray<BigArray[]>(kSize);
107*84e33947SAndroid Build Coastguard Worker     ASSERT_FALSE(ptr.isNull());
108*84e33947SAndroid Build Coastguard Worker     std::memset(ptr.get(), 0x37, sizeof(BigArray) * kSize);
109*84e33947SAndroid Build Coastguard Worker   }
110*84e33947SAndroid Build Coastguard Worker }
111*84e33947SAndroid Build Coastguard Worker 
TEST(UniquePtr,MakeUniqueZeroFill)112*84e33947SAndroid Build Coastguard Worker TEST(UniquePtr, MakeUniqueZeroFill) {
113*84e33947SAndroid Build Coastguard Worker   BigArray baseline = {};
114*84e33947SAndroid Build Coastguard Worker   auto myArray = MakeUniqueZeroFill<BigArray>();
115*84e33947SAndroid Build Coastguard Worker   ASSERT_FALSE(myArray.isNull());
116*84e33947SAndroid Build Coastguard Worker   // Note that this doesn't actually test things properly, because we don't
117*84e33947SAndroid Build Coastguard Worker   // guarantee that malloc is not already giving us zeroed out memory. To
118*84e33947SAndroid Build Coastguard Worker   // properly do it, we could inject the allocator, but this function is simple
119*84e33947SAndroid Build Coastguard Worker   // enough that it's not really worth the effort.
120*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(std::memcmp(&baseline, myArray.get(), sizeof(baseline)), 0);
121*84e33947SAndroid Build Coastguard Worker }
122*84e33947SAndroid Build Coastguard Worker 
TEST(UniquePtr,MoveConstruct)123*84e33947SAndroid Build Coastguard Worker TEST(UniquePtr, MoveConstruct) {
124*84e33947SAndroid Build Coastguard Worker   UniquePtr<Value> myInt = MakeUnique<Value>(0xcafe);
125*84e33947SAndroid Build Coastguard Worker   ASSERT_FALSE(myInt.isNull());
126*84e33947SAndroid Build Coastguard Worker   Value *value = myInt.get();
127*84e33947SAndroid Build Coastguard Worker 
128*84e33947SAndroid Build Coastguard Worker   UniquePtr<Value> moved(std::move(myInt));
129*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(moved.get(), value);
130*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(myInt.get(), nullptr);
131*84e33947SAndroid Build Coastguard Worker }
132*84e33947SAndroid Build Coastguard Worker 
TEST(UniquePtr,Move)133*84e33947SAndroid Build Coastguard Worker TEST(UniquePtr, Move) {
134*84e33947SAndroid Build Coastguard Worker   Value::constructionCounter = 0;
135*84e33947SAndroid Build Coastguard Worker 
136*84e33947SAndroid Build Coastguard Worker   {
137*84e33947SAndroid Build Coastguard Worker     UniquePtr<Value> myInt = MakeUnique<Value>(0xcafe);
138*84e33947SAndroid Build Coastguard Worker     ASSERT_FALSE(myInt.isNull());
139*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(Value::constructionCounter, 1);
140*84e33947SAndroid Build Coastguard Worker 
141*84e33947SAndroid Build Coastguard Worker     UniquePtr<Value> myMovedInt = MakeUnique<Value>(0);
142*84e33947SAndroid Build Coastguard Worker     ASSERT_FALSE(myMovedInt.isNull());
143*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(Value::constructionCounter, 2);
144*84e33947SAndroid Build Coastguard Worker     myMovedInt = std::move(myInt);
145*84e33947SAndroid Build Coastguard Worker     ASSERT_FALSE(myMovedInt.isNull());
146*84e33947SAndroid Build Coastguard Worker     ASSERT_TRUE(myInt.isNull());
147*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(myMovedInt.get()->value, 0xcafe);
148*84e33947SAndroid Build Coastguard Worker   }
149*84e33947SAndroid Build Coastguard Worker 
150*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(Value::constructionCounter, 0);
151*84e33947SAndroid Build Coastguard Worker }
152*84e33947SAndroid Build Coastguard Worker 
TEST(UniquePtr,Release)153*84e33947SAndroid Build Coastguard Worker TEST(UniquePtr, Release) {
154*84e33947SAndroid Build Coastguard Worker   Value::constructionCounter = 0;
155*84e33947SAndroid Build Coastguard Worker 
156*84e33947SAndroid Build Coastguard Worker   Value *value1, *value2;
157*84e33947SAndroid Build Coastguard Worker   {
158*84e33947SAndroid Build Coastguard Worker     UniquePtr<Value> myInt = MakeUnique<Value>(0xcafe);
159*84e33947SAndroid Build Coastguard Worker     ASSERT_FALSE(myInt.isNull());
160*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(Value::constructionCounter, 1);
161*84e33947SAndroid Build Coastguard Worker     value1 = myInt.get();
162*84e33947SAndroid Build Coastguard Worker     EXPECT_NE(value1, nullptr);
163*84e33947SAndroid Build Coastguard Worker     value2 = myInt.release();
164*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(value1, value2);
165*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(myInt.get(), nullptr);
166*84e33947SAndroid Build Coastguard Worker     EXPECT_TRUE(myInt.isNull());
167*84e33947SAndroid Build Coastguard Worker   }
168*84e33947SAndroid Build Coastguard Worker 
169*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(Value::constructionCounter, 1);
170*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(value2->value, 0xcafe);
171*84e33947SAndroid Build Coastguard Worker   value2->~Value();
172*84e33947SAndroid Build Coastguard Worker   chre::memoryFree(value2);
173*84e33947SAndroid Build Coastguard Worker }
174*84e33947SAndroid Build Coastguard Worker 
TEST(UniquePtr,Reset)175*84e33947SAndroid Build Coastguard Worker TEST(UniquePtr, Reset) {
176*84e33947SAndroid Build Coastguard Worker   Value::constructionCounter = 0;
177*84e33947SAndroid Build Coastguard Worker 
178*84e33947SAndroid Build Coastguard Worker   {
179*84e33947SAndroid Build Coastguard Worker     UniquePtr<Value> myInt = MakeUnique<Value>(0xcafe);
180*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(myInt.get()->value, 0xcafe);
181*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(Value::constructionCounter, 1);
182*84e33947SAndroid Build Coastguard Worker     myInt.reset(nullptr);
183*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(myInt.get(), nullptr);
184*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(Value::constructionCounter, 0);
185*84e33947SAndroid Build Coastguard Worker 
186*84e33947SAndroid Build Coastguard Worker     myInt = MakeUnique<Value>(0xcafe);
187*84e33947SAndroid Build Coastguard Worker     UniquePtr<Value> myInt2 = MakeUnique<Value>(0xface);
188*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(Value::constructionCounter, 2);
189*84e33947SAndroid Build Coastguard Worker     myInt.reset(myInt2.release());
190*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(Value::constructionCounter, 1);
191*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(myInt.get()->value, 0xface);
192*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(myInt2.get(), nullptr);
193*84e33947SAndroid Build Coastguard Worker 
194*84e33947SAndroid Build Coastguard Worker     myInt.reset();
195*84e33947SAndroid Build Coastguard Worker     EXPECT_EQ(myInt.get(), nullptr);
196*84e33947SAndroid Build Coastguard Worker   }
197*84e33947SAndroid Build Coastguard Worker 
198*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(Value::constructionCounter, 0);
199*84e33947SAndroid Build Coastguard Worker }
200*84e33947SAndroid Build Coastguard Worker 
TEST(UniquePtr,EqualityOperator)201*84e33947SAndroid Build Coastguard Worker TEST(UniquePtr, EqualityOperator) {
202*84e33947SAndroid Build Coastguard Worker   Value::constructionCounter = 0;
203*84e33947SAndroid Build Coastguard Worker 
204*84e33947SAndroid Build Coastguard Worker   {
205*84e33947SAndroid Build Coastguard Worker     UniquePtr<Value> myInt = MakeUnique<Value>(0xcafe);
206*84e33947SAndroid Build Coastguard Worker     EXPECT_TRUE(myInt != nullptr);
207*84e33947SAndroid Build Coastguard Worker 
208*84e33947SAndroid Build Coastguard Worker     myInt.reset();
209*84e33947SAndroid Build Coastguard Worker     EXPECT_TRUE(myInt == nullptr);
210*84e33947SAndroid Build Coastguard Worker   }
211*84e33947SAndroid Build Coastguard Worker 
212*84e33947SAndroid Build Coastguard Worker   EXPECT_EQ(Value::constructionCounter, 0);
213*84e33947SAndroid Build Coastguard Worker }
214*84e33947SAndroid Build Coastguard Worker 
TEST(UniquePtr,OverAlignedTest)215*84e33947SAndroid Build Coastguard Worker TEST(UniquePtr, OverAlignedTest) {
216*84e33947SAndroid Build Coastguard Worker   // Explicitly overaligned structure larger than std::max_align_t.
217*84e33947SAndroid Build Coastguard Worker   struct alignas(32) OverAlignedStruct {
218*84e33947SAndroid Build Coastguard Worker     uint32_t x[32];
219*84e33947SAndroid Build Coastguard Worker   };
220*84e33947SAndroid Build Coastguard Worker   static_assert(alignof(OverAlignedStruct) > alignof(std::max_align_t));
221*84e33947SAndroid Build Coastguard Worker 
222*84e33947SAndroid Build Coastguard Worker   UniquePtr<OverAlignedStruct> ptr = MakeUnique<OverAlignedStruct>();
223*84e33947SAndroid Build Coastguard Worker   ASSERT_EQ(reinterpret_cast<uintptr_t>(ptr.get()) % alignof(OverAlignedStruct),
224*84e33947SAndroid Build Coastguard Worker             0);
225*84e33947SAndroid Build Coastguard Worker }
226