1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "variant_map.h"
18 #include "gtest/gtest.h"
19 
20 #define EXPECT_NULL(expected) EXPECT_EQ(reinterpret_cast<const void*>(expected), \
21                                         static_cast<void*>(nullptr));
22 
23 namespace art {
24 
25 namespace {
26 template <typename TValue>
27 struct FruitMapKey : VariantMapKey<TValue> {
FruitMapKeyart::__anon08c9e08c0111::FruitMapKey28   FruitMapKey() {}
29 };
30 
31 struct FruitMap : VariantMap<FruitMap, FruitMapKey> {
32   // This 'using' line is necessary to inherit the variadic constructor.
33   using VariantMap<FruitMap, FruitMapKey>::VariantMap;
34 
35   // Make the next '4' usages of Key slightly shorter to type.
36   template <typename TValue>
37   using Key = FruitMapKey<TValue>;
38 
39   static const Key<int> Apple;
40   static const Key<double> Orange;
41   static const Key<std::string> Label;
42 };
43 
44 const FruitMap::Key<int> FruitMap::Apple;
45 const FruitMap::Key<double> FruitMap::Orange;
46 const FruitMap::Key<std::string> FruitMap::Label;
47 }  // namespace
48 
TEST(VariantMaps,BasicReadWrite)49 TEST(VariantMaps, BasicReadWrite) {
50   FruitMap fm;
51 
52   EXPECT_NULL(fm.Get(FruitMap::Apple));
53   EXPECT_FALSE(fm.Exists(FruitMap::Apple));
54   EXPECT_NULL(fm.Get(FruitMap::Orange));
55   EXPECT_FALSE(fm.Exists(FruitMap::Orange));
56 
57   fm.Set(FruitMap::Apple, 1);
58   EXPECT_NULL(fm.Get(FruitMap::Orange));
59   EXPECT_EQ(1, *fm.Get(FruitMap::Apple));
60   EXPECT_TRUE(fm.Exists(FruitMap::Apple));
61 
62   fm.Set(FruitMap::Apple, 5);
63   EXPECT_NULL(fm.Get(FruitMap::Orange));
64   EXPECT_EQ(5, *fm.Get(FruitMap::Apple));
65   EXPECT_TRUE(fm.Exists(FruitMap::Apple));
66 
67   fm.Set(FruitMap::Orange, 555.0);
68   EXPECT_EQ(5, *fm.Get(FruitMap::Apple));
69   EXPECT_DOUBLE_EQ(555.0, *fm.Get(FruitMap::Orange));
70   EXPECT_EQ(size_t(2), fm.Size());
71 
72   // Simple remove
73   fm.Remove(FruitMap::Apple);
74   EXPECT_FALSE(fm.Exists(FruitMap::Apple));
75 
76   fm.Clear();
77   EXPECT_EQ(size_t(0), fm.Size());
78   EXPECT_FALSE(fm.Exists(FruitMap::Orange));
79 }
80 
TEST(VariantMaps,SetPreviousValue)81 TEST(VariantMaps, SetPreviousValue) {
82   FruitMap fm;
83 
84   // Indirect remove by setting yourself again
85   fm.Set(FruitMap::Label, std::string("hello_world"));
86   auto* ptr = fm.Get(FruitMap::Label);
87   ASSERT_TRUE(ptr != nullptr);
88   *ptr = "foobar";
89 
90   // Set the value to the same exact pointer which we got out of the map.
91   // This should cleanly 'just work' and not try to delete the value too early.
92   fm.Set(FruitMap::Label, *ptr);
93 
94   auto* new_ptr = fm.Get(FruitMap::Label);
95   ASSERT_TRUE(ptr != nullptr);
96   EXPECT_EQ(std::string("foobar"), *new_ptr);
97 }
98 
TEST(VariantMaps,RuleOfFive)99 TEST(VariantMaps, RuleOfFive) {
100   // Test empty constructor
101   FruitMap fmEmpty;
102   EXPECT_EQ(size_t(0), fmEmpty.Size());
103 
104   // Test empty constructor
105   FruitMap fmFilled;
106   fmFilled.Set(FruitMap::Apple, 1);
107   fmFilled.Set(FruitMap::Orange, 555.0);
108   EXPECT_EQ(size_t(2), fmFilled.Size());
109 
110   // Test copy constructor
111   FruitMap fmEmptyCopy(fmEmpty);  // NOLINT
112   EXPECT_EQ(size_t(0), fmEmptyCopy.Size());
113 
114   // Test copy constructor
115   FruitMap fmFilledCopy(fmFilled);
116   EXPECT_EQ(size_t(2), fmFilledCopy.Size());
117   EXPECT_EQ(*fmFilled.Get(FruitMap::Apple), *fmFilledCopy.Get(FruitMap::Apple));
118   EXPECT_DOUBLE_EQ(*fmFilled.Get(FruitMap::Orange), *fmFilledCopy.Get(FruitMap::Orange));
119 
120   // Test operator=
121   FruitMap fmFilledCopy2;
122   fmFilledCopy2 = fmFilled;
123   EXPECT_EQ(size_t(2), fmFilledCopy2.Size());
124   EXPECT_EQ(*fmFilled.Get(FruitMap::Apple), *fmFilledCopy2.Get(FruitMap::Apple));
125   EXPECT_DOUBLE_EQ(*fmFilled.Get(FruitMap::Orange), *fmFilledCopy2.Get(FruitMap::Orange));
126 
127   // Test move constructor
128   FruitMap fmMoved(std::move(fmFilledCopy));
129   // NOLINTNEXTLINE - checking underlying storage has been freed
130   EXPECT_EQ(size_t(0), fmFilledCopy.Size());
131   EXPECT_EQ(size_t(2), fmMoved.Size());
132   EXPECT_EQ(*fmFilled.Get(FruitMap::Apple), *fmMoved.Get(FruitMap::Apple));
133   EXPECT_DOUBLE_EQ(*fmFilled.Get(FruitMap::Orange), *fmMoved.Get(FruitMap::Orange));
134 
135   // Test operator= move
136   FruitMap fmMoved2;
137   fmMoved2.Set(FruitMap::Apple, 12345);  // This value will be clobbered after the move
138 
139   fmMoved2 = std::move(fmFilledCopy2);
140   // NOLINTNEXTLINE - checking underlying storage has been freed
141   EXPECT_EQ(size_t(0), fmFilledCopy2.Size());
142   EXPECT_EQ(size_t(2), fmMoved2.Size());
143   EXPECT_EQ(*fmFilled.Get(FruitMap::Apple), *fmMoved2.Get(FruitMap::Apple));
144   EXPECT_DOUBLE_EQ(*fmFilled.Get(FruitMap::Orange), *fmMoved2.Get(FruitMap::Orange));
145 }
146 
TEST(VariantMaps,VariadicConstructors)147 TEST(VariantMaps, VariadicConstructors) {
148   // Variadic constructor, 1 kv/pair
149   FruitMap fmApple(FruitMap::Apple, 12345);
150   EXPECT_EQ(size_t(1), fmApple.Size());
151   EXPECT_EQ(12345, *fmApple.Get(FruitMap::Apple));
152 
153   // Variadic constructor, 2 kv/pair
154   FruitMap fmAppleAndOrange(FruitMap::Apple,   12345,
155                             FruitMap::Orange,  100.0);
156   EXPECT_EQ(size_t(2), fmAppleAndOrange.Size());
157   EXPECT_EQ(12345, *fmAppleAndOrange.Get(FruitMap::Apple));
158   EXPECT_DOUBLE_EQ(100.0, *fmAppleAndOrange.Get(FruitMap::Orange));
159 }
160 
TEST(VariantMaps,ReleaseOrDefault)161 TEST(VariantMaps, ReleaseOrDefault) {
162   FruitMap fmAppleAndOrange(FruitMap::Apple,   12345,
163                             FruitMap::Orange,  100.0);
164 
165   int apple = fmAppleAndOrange.ReleaseOrDefault(FruitMap::Apple);
166   EXPECT_EQ(12345, apple);
167 
168   // Releasing will also remove the Apple key.
169   EXPECT_EQ(size_t(1), fmAppleAndOrange.Size());
170 
171   // Releasing again yields a default value.
172   int apple2 = fmAppleAndOrange.ReleaseOrDefault(FruitMap::Apple);
173   EXPECT_EQ(0, apple2);
174 }
175 
TEST(VariantMaps,GetOrDefault)176 TEST(VariantMaps, GetOrDefault) {
177   FruitMap fm(FruitMap::Apple,   12345);
178 
179   // Apple gives the expected value we set.
180   int apple = fm.GetOrDefault(FruitMap::Apple);
181   EXPECT_EQ(12345, apple);
182 
183   // Map is still 1.
184   EXPECT_EQ(size_t(1), fm.Size());
185 
186   // Orange gives back a default value, since it's not in the map.
187   double orange = fm.GetOrDefault(FruitMap::Orange);
188   EXPECT_DOUBLE_EQ(0.0, orange);
189 }
190 
191 }  // namespace art
192