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