1
2 /*
3 * Copyright (C) 2016 The Android Open Source Project
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 */
17
18 #include "gtest/gtest.h"
19
20 #include "chre/util/non_copyable.h"
21 #include "chre/util/optional.h"
22
23 using chre::Optional;
24
25 class DestructorTestingObject {
26 public:
~DestructorTestingObject()27 ~DestructorTestingObject() {
28 if (valueToFlipWhenDestruct != nullptr) {
29 *valueToFlipWhenDestruct = !*valueToFlipWhenDestruct;
30 }
31 }
32
setValueToFlipWhenDestruct(bool * value)33 void setValueToFlipWhenDestruct(bool *value) {
34 valueToFlipWhenDestruct = value;
35 }
36
37 private:
38 bool *valueToFlipWhenDestruct = nullptr;
39 };
40
TEST(Optional,ShouldDestructContainedObject)41 TEST(Optional, ShouldDestructContainedObject) {
42 bool destructed = false;
43 {
44 Optional<DestructorTestingObject> object(DestructorTestingObject{});
45 object.value().setValueToFlipWhenDestruct(&destructed);
46 }
47 EXPECT_TRUE(destructed);
48 }
49
TEST(Optional,NoValueByDefault)50 TEST(Optional, NoValueByDefault) {
51 Optional<int> myInt;
52 EXPECT_FALSE(myInt.has_value());
53 }
54
TEST(Optional,NonDefaultValueByDefault)55 TEST(Optional, NonDefaultValueByDefault) {
56 Optional<int> myInt(0x1337);
57 EXPECT_TRUE(myInt.has_value());
58 EXPECT_EQ(*myInt, 0x1337);
59 }
60
TEST(Optional,NonDefaultMovedValueByDefault)61 TEST(Optional, NonDefaultMovedValueByDefault) {
62 Optional<int> myInt(std::move(0x1337));
63 EXPECT_TRUE(myInt.has_value());
64 EXPECT_EQ(*myInt, 0x1337);
65 }
66
TEST(Optional,CopyConstruct)67 TEST(Optional, CopyConstruct) {
68 Optional<int> myInt(0x1337);
69 Optional<int> myNewInt(myInt);
70 EXPECT_TRUE(myNewInt.has_value());
71 EXPECT_EQ(*myNewInt, 0x1337);
72 }
73
TEST(Optional,CopyConstructConst)74 TEST(Optional, CopyConstructConst) {
75 const Optional<int> myInt(0x1337);
76 Optional<int> myNewInt(myInt);
77 EXPECT_TRUE(myNewInt.has_value());
78 EXPECT_EQ(*myNewInt, 0x1337);
79 }
80
TEST(Optional,CopyAssignAndRead)81 TEST(Optional, CopyAssignAndRead) {
82 Optional<int> myInt;
83 EXPECT_FALSE(myInt.has_value());
84 myInt = 0x1337;
85 EXPECT_EQ(*myInt, 0x1337);
86 EXPECT_TRUE(myInt.has_value());
87 myInt.reset();
88 EXPECT_FALSE(myInt.has_value());
89 }
90
TEST(Optional,MoveAssignAndRead)91 TEST(Optional, MoveAssignAndRead) {
92 Optional<int> myInt;
93 EXPECT_FALSE(myInt.has_value());
94 myInt = std::move(0xcafe);
95 EXPECT_TRUE(myInt.has_value());
96 EXPECT_EQ(*myInt, 0xcafe);
97 }
98
TEST(Optional,OptionalMoveAssignAndRead)99 TEST(Optional, OptionalMoveAssignAndRead) {
100 Optional<int> myInt(0x1337);
101 Optional<int> myMovedInt;
102 EXPECT_FALSE(myMovedInt.has_value());
103 myMovedInt = std::move(myInt);
104 EXPECT_TRUE(myInt.has_value());
105 EXPECT_TRUE(myMovedInt.has_value());
106 EXPECT_EQ(*myMovedInt, 0x1337);
107 }
108
TEST(Optional,OptionalCopyAssignAndRead)109 TEST(Optional, OptionalCopyAssignAndRead) {
110 Optional<int> myInt(0x1337);
111 Optional<int> myCopiedInt;
112 EXPECT_FALSE(myCopiedInt.has_value());
113 myCopiedInt = myInt;
114 EXPECT_TRUE(myInt.has_value());
115 EXPECT_TRUE(myCopiedInt.has_value());
116 EXPECT_EQ(*myInt, 0x1337);
117 EXPECT_EQ(*myCopiedInt, 0x1337);
118 }
119
120 static constexpr int kInvalidValue = -1;
121
122 class MovableButNonCopyable : public chre::NonCopyable {
123 public:
124 MovableButNonCopyable() = default;
MovableButNonCopyable(int value)125 MovableButNonCopyable(int value) : mValue(value) {}
MovableButNonCopyable(MovableButNonCopyable && other)126 MovableButNonCopyable(MovableButNonCopyable &&other) {
127 mValue = other.mValue;
128 other.mValue = kInvalidValue;
129 }
130
operator =(MovableButNonCopyable && other)131 MovableButNonCopyable &operator=(MovableButNonCopyable &&other) {
132 assert(mMagic == kConstructedMagic);
133 mValue = other.mValue;
134 other.mValue = kInvalidValue;
135 return *this;
136 }
137
~MovableButNonCopyable()138 ~MovableButNonCopyable() {
139 mMagic = kUninitializedMagic;
140 mValue = kUninitializedMagic;
141 }
142
getValue() const143 int getValue() const {
144 return mValue;
145 }
146
147 private:
148 static constexpr int kConstructedMagic = 0xfeedc0fe;
149 static constexpr int kUninitializedMagic = 0xdeadbeef;
150
151 int mMagic = kConstructedMagic;
152 int mValue = kInvalidValue;
153 };
154
TEST(Optional,UninitializedAssignment)155 TEST(Optional, UninitializedAssignment) {
156 constexpr int kValue1 = 0xd00d;
157 constexpr int kValue2 = 0xcafe;
158 MovableButNonCopyable transferee1(kValue1);
159 MovableButNonCopyable transferee2(kValue2);
160
161 Optional<MovableButNonCopyable> container;
162 EXPECT_FALSE(container.has_value());
163
164 container = std::move(transferee1);
165 EXPECT_TRUE(container.has_value());
166 EXPECT_EQ(container->getValue(), kValue1);
167 EXPECT_EQ(transferee1.getValue(), kInvalidValue);
168
169 container.reset();
170 EXPECT_FALSE(container.has_value());
171
172 container = std::move(transferee2);
173 EXPECT_TRUE(container.has_value());
174 EXPECT_EQ(container->getValue(), kValue2);
175 EXPECT_EQ(transferee2.getValue(), kInvalidValue);
176 }
177
178 // TODO: should add some tests to cover the possible assignment outcomes between
179 // two Optional instances (e.g. assign one w/o value to one w/value, etc)
180