1 //===- llvm/unittest/ADT/ValueMapTest.cpp - ValueMap unit tests -*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "llvm/ADT/ValueMap.h"
11 #include "llvm/Constants.h"
12 #include "llvm/Instructions.h"
13 #include "llvm/LLVMContext.h"
14 #include "llvm/ADT/OwningPtr.h"
15 #include "llvm/Config/config.h"
16 
17 #include "gtest/gtest.h"
18 
19 using namespace llvm;
20 
21 namespace {
22 
23 // Test fixture
24 template<typename T>
25 class ValueMapTest : public testing::Test {
26 protected:
27   Constant *ConstantV;
28   OwningPtr<BitCastInst> BitcastV;
29   OwningPtr<BinaryOperator> AddV;
30 
ValueMapTest()31   ValueMapTest() :
32     ConstantV(ConstantInt::get(Type::getInt32Ty(getGlobalContext()), 0)),
33     BitcastV(new BitCastInst(ConstantV, Type::getInt32Ty(getGlobalContext()))),
34     AddV(BinaryOperator::CreateAdd(ConstantV, ConstantV)) {
35   }
36 };
37 
38 // Run everything on Value*, a subtype to make sure that casting works as
39 // expected, and a const subtype to make sure we cast const correctly.
40 typedef ::testing::Types<Value, Instruction, const Instruction> KeyTypes;
41 TYPED_TEST_CASE(ValueMapTest, KeyTypes);
42 
TYPED_TEST(ValueMapTest,Null)43 TYPED_TEST(ValueMapTest, Null) {
44   ValueMap<TypeParam*, int> VM1;
45   VM1[NULL] = 7;
46   EXPECT_EQ(7, VM1.lookup(NULL));
47 }
48 
TYPED_TEST(ValueMapTest,FollowsValue)49 TYPED_TEST(ValueMapTest, FollowsValue) {
50   ValueMap<TypeParam*, int> VM;
51   VM[this->BitcastV.get()] = 7;
52   EXPECT_EQ(7, VM.lookup(this->BitcastV.get()));
53   EXPECT_EQ(0, VM.count(this->AddV.get()));
54   this->BitcastV->replaceAllUsesWith(this->AddV.get());
55   EXPECT_EQ(7, VM.lookup(this->AddV.get()));
56   EXPECT_EQ(0, VM.count(this->BitcastV.get()));
57   this->AddV.reset();
58   EXPECT_EQ(0, VM.count(this->AddV.get()));
59   EXPECT_EQ(0, VM.count(this->BitcastV.get()));
60   EXPECT_EQ(0U, VM.size());
61 }
62 
TYPED_TEST(ValueMapTest,OperationsWork)63 TYPED_TEST(ValueMapTest, OperationsWork) {
64   ValueMap<TypeParam*, int> VM;
65   ValueMap<TypeParam*, int> VM2(16);  (void)VM2;
66   typename ValueMapConfig<TypeParam*>::ExtraData Data;
67   ValueMap<TypeParam*, int> VM3(Data, 16);  (void)VM3;
68   EXPECT_TRUE(VM.empty());
69 
70   VM[this->BitcastV.get()] = 7;
71 
72   // Find:
73   typename ValueMap<TypeParam*, int>::iterator I =
74     VM.find(this->BitcastV.get());
75   ASSERT_TRUE(I != VM.end());
76   EXPECT_EQ(this->BitcastV.get(), I->first);
77   EXPECT_EQ(7, I->second);
78   EXPECT_TRUE(VM.find(this->AddV.get()) == VM.end());
79 
80   // Const find:
81   const ValueMap<TypeParam*, int> &CVM = VM;
82   typename ValueMap<TypeParam*, int>::const_iterator CI =
83     CVM.find(this->BitcastV.get());
84   ASSERT_TRUE(CI != CVM.end());
85   EXPECT_EQ(this->BitcastV.get(), CI->first);
86   EXPECT_EQ(7, CI->second);
87   EXPECT_TRUE(CVM.find(this->AddV.get()) == CVM.end());
88 
89   // Insert:
90   std::pair<typename ValueMap<TypeParam*, int>::iterator, bool> InsertResult1 =
91     VM.insert(std::make_pair(this->AddV.get(), 3));
92   EXPECT_EQ(this->AddV.get(), InsertResult1.first->first);
93   EXPECT_EQ(3, InsertResult1.first->second);
94   EXPECT_TRUE(InsertResult1.second);
95   EXPECT_EQ(true, VM.count(this->AddV.get()));
96   std::pair<typename ValueMap<TypeParam*, int>::iterator, bool> InsertResult2 =
97     VM.insert(std::make_pair(this->AddV.get(), 5));
98   EXPECT_EQ(this->AddV.get(), InsertResult2.first->first);
99   EXPECT_EQ(3, InsertResult2.first->second);
100   EXPECT_FALSE(InsertResult2.second);
101 
102   // Erase:
103   VM.erase(InsertResult2.first);
104   EXPECT_EQ(0U, VM.count(this->AddV.get()));
105   EXPECT_EQ(1U, VM.count(this->BitcastV.get()));
106   VM.erase(this->BitcastV.get());
107   EXPECT_EQ(0U, VM.count(this->BitcastV.get()));
108   EXPECT_EQ(0U, VM.size());
109 
110   // Range insert:
111   SmallVector<std::pair<Instruction*, int>, 2> Elems;
112   Elems.push_back(std::make_pair(this->AddV.get(), 1));
113   Elems.push_back(std::make_pair(this->BitcastV.get(), 2));
114   VM.insert(Elems.begin(), Elems.end());
115   EXPECT_EQ(1, VM.lookup(this->AddV.get()));
116   EXPECT_EQ(2, VM.lookup(this->BitcastV.get()));
117 }
118 
119 template<typename ExpectedType, typename VarType>
CompileAssertHasType(VarType)120 void CompileAssertHasType(VarType) {
121   typedef char assert[is_same<ExpectedType, VarType>::value ? 1 : -1];
122 }
123 
TYPED_TEST(ValueMapTest,Iteration)124 TYPED_TEST(ValueMapTest, Iteration) {
125   ValueMap<TypeParam*, int> VM;
126   VM[this->BitcastV.get()] = 2;
127   VM[this->AddV.get()] = 3;
128   size_t size = 0;
129   for (typename ValueMap<TypeParam*, int>::iterator I = VM.begin(), E = VM.end();
130        I != E; ++I) {
131     ++size;
132     std::pair<TypeParam*, int> value = *I; (void)value;
133     CompileAssertHasType<TypeParam*>(I->first);
134     if (I->second == 2) {
135       EXPECT_EQ(this->BitcastV.get(), I->first);
136       I->second = 5;
137     } else if (I->second == 3) {
138       EXPECT_EQ(this->AddV.get(), I->first);
139       I->second = 6;
140     } else {
141       ADD_FAILURE() << "Iterated through an extra value.";
142     }
143   }
144   EXPECT_EQ(2U, size);
145   EXPECT_EQ(5, VM[this->BitcastV.get()]);
146   EXPECT_EQ(6, VM[this->AddV.get()]);
147 
148   size = 0;
149   // Cast to const ValueMap to avoid a bug in DenseMap's iterators.
150   const ValueMap<TypeParam*, int>& CVM = VM;
151   for (typename ValueMap<TypeParam*, int>::const_iterator I = CVM.begin(),
152          E = CVM.end(); I != E; ++I) {
153     ++size;
154     std::pair<TypeParam*, int> value = *I;  (void)value;
155     CompileAssertHasType<TypeParam*>(I->first);
156     if (I->second == 5) {
157       EXPECT_EQ(this->BitcastV.get(), I->first);
158     } else if (I->second == 6) {
159       EXPECT_EQ(this->AddV.get(), I->first);
160     } else {
161       ADD_FAILURE() << "Iterated through an extra value.";
162     }
163   }
164   EXPECT_EQ(2U, size);
165 }
166 
TYPED_TEST(ValueMapTest,DefaultCollisionBehavior)167 TYPED_TEST(ValueMapTest, DefaultCollisionBehavior) {
168   // By default, we overwrite the old value with the replaced value.
169   ValueMap<TypeParam*, int> VM;
170   VM[this->BitcastV.get()] = 7;
171   VM[this->AddV.get()] = 9;
172   this->BitcastV->replaceAllUsesWith(this->AddV.get());
173   EXPECT_EQ(0, VM.count(this->BitcastV.get()));
174   EXPECT_EQ(9, VM.lookup(this->AddV.get()));
175 }
176 
TYPED_TEST(ValueMapTest,ConfiguredCollisionBehavior)177 TYPED_TEST(ValueMapTest, ConfiguredCollisionBehavior) {
178   // TODO: Implement this when someone needs it.
179 }
180 
181 template<typename KeyT>
182 struct LockMutex : ValueMapConfig<KeyT> {
183   struct ExtraData {
184     sys::Mutex *M;
185     bool *CalledRAUW;
186     bool *CalledDeleted;
187   };
onRAUW__anon3f058fb00111::LockMutex188   static void onRAUW(const ExtraData &Data, KeyT Old, KeyT New) {
189     *Data.CalledRAUW = true;
190     EXPECT_FALSE(Data.M->tryacquire()) << "Mutex should already be locked.";
191   }
onDelete__anon3f058fb00111::LockMutex192   static void onDelete(const ExtraData &Data, KeyT Old) {
193     *Data.CalledDeleted = true;
194     EXPECT_FALSE(Data.M->tryacquire()) << "Mutex should already be locked.";
195   }
getMutex__anon3f058fb00111::LockMutex196   static sys::Mutex *getMutex(const ExtraData &Data) { return Data.M; }
197 };
198 #if ENABLE_THREADS
TYPED_TEST(ValueMapTest,LocksMutex)199 TYPED_TEST(ValueMapTest, LocksMutex) {
200   sys::Mutex M(false);  // Not recursive.
201   bool CalledRAUW = false, CalledDeleted = false;
202   typename LockMutex<TypeParam*>::ExtraData Data =
203     {&M, &CalledRAUW, &CalledDeleted};
204   ValueMap<TypeParam*, int, LockMutex<TypeParam*> > VM(Data);
205   VM[this->BitcastV.get()] = 7;
206   this->BitcastV->replaceAllUsesWith(this->AddV.get());
207   this->AddV.reset();
208   EXPECT_TRUE(CalledRAUW);
209   EXPECT_TRUE(CalledDeleted);
210 }
211 #endif
212 
213 template<typename KeyT>
214 struct NoFollow : ValueMapConfig<KeyT> {
215   enum { FollowRAUW = false };
216 };
217 
TYPED_TEST(ValueMapTest,NoFollowRAUW)218 TYPED_TEST(ValueMapTest, NoFollowRAUW) {
219   ValueMap<TypeParam*, int, NoFollow<TypeParam*> > VM;
220   VM[this->BitcastV.get()] = 7;
221   EXPECT_EQ(7, VM.lookup(this->BitcastV.get()));
222   EXPECT_EQ(0, VM.count(this->AddV.get()));
223   this->BitcastV->replaceAllUsesWith(this->AddV.get());
224   EXPECT_EQ(7, VM.lookup(this->BitcastV.get()));
225   EXPECT_EQ(0, VM.lookup(this->AddV.get()));
226   this->AddV.reset();
227   EXPECT_EQ(7, VM.lookup(this->BitcastV.get()));
228   EXPECT_EQ(0, VM.lookup(this->AddV.get()));
229   this->BitcastV.reset();
230   EXPECT_EQ(0, VM.lookup(this->BitcastV.get()));
231   EXPECT_EQ(0, VM.lookup(this->AddV.get()));
232   EXPECT_EQ(0U, VM.size());
233 }
234 
235 template<typename KeyT>
236 struct CountOps : ValueMapConfig<KeyT> {
237   struct ExtraData {
238     int *Deletions;
239     int *RAUWs;
240   };
241 
onRAUW__anon3f058fb00111::CountOps242   static void onRAUW(const ExtraData &Data, KeyT Old, KeyT New) {
243     ++*Data.RAUWs;
244   }
onDelete__anon3f058fb00111::CountOps245   static void onDelete(const ExtraData &Data, KeyT Old) {
246     ++*Data.Deletions;
247   }
248 };
249 
TYPED_TEST(ValueMapTest,CallsConfig)250 TYPED_TEST(ValueMapTest, CallsConfig) {
251   int Deletions = 0, RAUWs = 0;
252   typename CountOps<TypeParam*>::ExtraData Data = {&Deletions, &RAUWs};
253   ValueMap<TypeParam*, int, CountOps<TypeParam*> > VM(Data);
254   VM[this->BitcastV.get()] = 7;
255   this->BitcastV->replaceAllUsesWith(this->AddV.get());
256   EXPECT_EQ(0, Deletions);
257   EXPECT_EQ(1, RAUWs);
258   this->AddV.reset();
259   EXPECT_EQ(1, Deletions);
260   EXPECT_EQ(1, RAUWs);
261   this->BitcastV.reset();
262   EXPECT_EQ(1, Deletions);
263   EXPECT_EQ(1, RAUWs);
264 }
265 
266 template<typename KeyT>
267 struct ModifyingConfig : ValueMapConfig<KeyT> {
268   // We'll put a pointer here back to the ValueMap this key is in, so
269   // that we can modify it (and clobber *this) before the ValueMap
270   // tries to do the same modification.  In previous versions of
271   // ValueMap, that exploded.
272   typedef ValueMap<KeyT, int, ModifyingConfig<KeyT> > **ExtraData;
273 
onRAUW__anon3f058fb00111::ModifyingConfig274   static void onRAUW(ExtraData Map, KeyT Old, KeyT New) {
275     (*Map)->erase(Old);
276   }
onDelete__anon3f058fb00111::ModifyingConfig277   static void onDelete(ExtraData Map, KeyT Old) {
278     (*Map)->erase(Old);
279   }
280 };
TYPED_TEST(ValueMapTest,SurvivesModificationByConfig)281 TYPED_TEST(ValueMapTest, SurvivesModificationByConfig) {
282   ValueMap<TypeParam*, int, ModifyingConfig<TypeParam*> > *MapAddress;
283   ValueMap<TypeParam*, int, ModifyingConfig<TypeParam*> > VM(&MapAddress);
284   MapAddress = &VM;
285   // Now the ModifyingConfig can modify the Map inside a callback.
286   VM[this->BitcastV.get()] = 7;
287   this->BitcastV->replaceAllUsesWith(this->AddV.get());
288   EXPECT_FALSE(VM.count(this->BitcastV.get()));
289   EXPECT_FALSE(VM.count(this->AddV.get()));
290   VM[this->AddV.get()] = 7;
291   this->AddV.reset();
292   EXPECT_FALSE(VM.count(this->AddV.get()));
293 }
294 
295 }
296