blob: cbcd6f6e53f4a1d9d2c1fb59043f80fb0cfd8bb2 [file] [log] [blame]
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +00001//===- 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
Chandler Carruth7225e272014-03-04 11:26:31 +000010#include "llvm/IR/ValueMap.h"
Chandler Carruth5a88dda2012-12-04 10:23:08 +000011#include "llvm/Config/llvm-config.h"
Chandler Carruth0b8c9a82013-01-02 11:36:10 +000012#include "llvm/IR/Constants.h"
13#include "llvm/IR/Instructions.h"
14#include "llvm/IR/LLVMContext.h"
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000015#include "gtest/gtest.h"
16
17using namespace llvm;
18
19namespace {
20
21// Test fixture
22template<typename T>
23class ValueMapTest : public testing::Test {
24protected:
Mehdi Amini8be77072016-04-14 21:59:01 +000025 LLVMContext Context;
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000026 Constant *ConstantV;
Ahmed Charlesf4ccd112014-03-06 05:51:42 +000027 std::unique_ptr<BitCastInst> BitcastV;
28 std::unique_ptr<BinaryOperator> AddV;
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000029
Mehdi Amini8be77072016-04-14 21:59:01 +000030 ValueMapTest()
31 : ConstantV(ConstantInt::get(Type::getInt32Ty(Context), 0)),
32 BitcastV(new BitCastInst(ConstantV, Type::getInt32Ty(Context))),
33 AddV(BinaryOperator::CreateAdd(ConstantV, ConstantV)) {}
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000034};
35
36// Run everything on Value*, a subtype to make sure that casting works as
37// expected, and a const subtype to make sure we cast const correctly.
38typedef ::testing::Types<Value, Instruction, const Instruction> KeyTypes;
39TYPED_TEST_CASE(ValueMapTest, KeyTypes);
40
41TYPED_TEST(ValueMapTest, Null) {
42 ValueMap<TypeParam*, int> VM1;
Craig Topperb1770412014-06-08 22:29:17 +000043 VM1[nullptr] = 7;
44 EXPECT_EQ(7, VM1.lookup(nullptr));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000045}
46
47TYPED_TEST(ValueMapTest, FollowsValue) {
48 ValueMap<TypeParam*, int> VM;
49 VM[this->BitcastV.get()] = 7;
50 EXPECT_EQ(7, VM.lookup(this->BitcastV.get()));
David Blaikie20dead82014-06-20 19:54:13 +000051 EXPECT_EQ(0u, VM.count(this->AddV.get()));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000052 this->BitcastV->replaceAllUsesWith(this->AddV.get());
53 EXPECT_EQ(7, VM.lookup(this->AddV.get()));
David Blaikie20dead82014-06-20 19:54:13 +000054 EXPECT_EQ(0u, VM.count(this->BitcastV.get()));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000055 this->AddV.reset();
David Blaikie20dead82014-06-20 19:54:13 +000056 EXPECT_EQ(0u, VM.count(this->AddV.get()));
57 EXPECT_EQ(0u, VM.count(this->BitcastV.get()));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000058 EXPECT_EQ(0U, VM.size());
59}
60
61TYPED_TEST(ValueMapTest, OperationsWork) {
62 ValueMap<TypeParam*, int> VM;
Chris Lattnera07cd902010-04-18 03:28:20 +000063 ValueMap<TypeParam*, int> VM2(16); (void)VM2;
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000064 typename ValueMapConfig<TypeParam*>::ExtraData Data;
Chris Lattnera07cd902010-04-18 03:28:20 +000065 ValueMap<TypeParam*, int> VM3(Data, 16); (void)VM3;
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000066 EXPECT_TRUE(VM.empty());
67
68 VM[this->BitcastV.get()] = 7;
69
70 // Find:
71 typename ValueMap<TypeParam*, int>::iterator I =
72 VM.find(this->BitcastV.get());
73 ASSERT_TRUE(I != VM.end());
Jim Grosbache5415282010-11-18 01:38:07 +000074 EXPECT_EQ(this->BitcastV.get(), I->first);
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000075 EXPECT_EQ(7, I->second);
Jim Grosbache5415282010-11-18 01:38:07 +000076 EXPECT_TRUE(VM.find(this->AddV.get()) == VM.end());
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000077
78 // Const find:
79 const ValueMap<TypeParam*, int> &CVM = VM;
80 typename ValueMap<TypeParam*, int>::const_iterator CI =
81 CVM.find(this->BitcastV.get());
82 ASSERT_TRUE(CI != CVM.end());
Jim Grosbache5415282010-11-18 01:38:07 +000083 EXPECT_EQ(this->BitcastV.get(), CI->first);
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000084 EXPECT_EQ(7, CI->second);
85 EXPECT_TRUE(CVM.find(this->AddV.get()) == CVM.end());
86
87 // Insert:
88 std::pair<typename ValueMap<TypeParam*, int>::iterator, bool> InsertResult1 =
89 VM.insert(std::make_pair(this->AddV.get(), 3));
Jim Grosbache5415282010-11-18 01:38:07 +000090 EXPECT_EQ(this->AddV.get(), InsertResult1.first->first);
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000091 EXPECT_EQ(3, InsertResult1.first->second);
92 EXPECT_TRUE(InsertResult1.second);
David Blaikie20dead82014-06-20 19:54:13 +000093 EXPECT_EQ(1u, VM.count(this->AddV.get()));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000094 std::pair<typename ValueMap<TypeParam*, int>::iterator, bool> InsertResult2 =
95 VM.insert(std::make_pair(this->AddV.get(), 5));
Jim Grosbache5415282010-11-18 01:38:07 +000096 EXPECT_EQ(this->AddV.get(), InsertResult2.first->first);
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +000097 EXPECT_EQ(3, InsertResult2.first->second);
98 EXPECT_FALSE(InsertResult2.second);
99
100 // Erase:
101 VM.erase(InsertResult2.first);
Bill Wendling37323962010-07-10 18:56:35 +0000102 EXPECT_EQ(0U, VM.count(this->AddV.get()));
103 EXPECT_EQ(1U, VM.count(this->BitcastV.get()));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000104 VM.erase(this->BitcastV.get());
Bill Wendling37323962010-07-10 18:56:35 +0000105 EXPECT_EQ(0U, VM.count(this->BitcastV.get()));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000106 EXPECT_EQ(0U, VM.size());
107
108 // Range insert:
109 SmallVector<std::pair<Instruction*, int>, 2> Elems;
110 Elems.push_back(std::make_pair(this->AddV.get(), 1));
111 Elems.push_back(std::make_pair(this->BitcastV.get(), 2));
112 VM.insert(Elems.begin(), Elems.end());
113 EXPECT_EQ(1, VM.lookup(this->AddV.get()));
114 EXPECT_EQ(2, VM.lookup(this->BitcastV.get()));
115}
116
117template<typename ExpectedType, typename VarType>
118void CompileAssertHasType(VarType) {
Benjamin Kramer1dd31112014-03-07 15:54:23 +0000119 static_assert(std::is_same<ExpectedType, VarType>::value,
120 "Not the same type");
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000121}
122
123TYPED_TEST(ValueMapTest, Iteration) {
124 ValueMap<TypeParam*, int> VM;
125 VM[this->BitcastV.get()] = 2;
126 VM[this->AddV.get()] = 3;
127 size_t size = 0;
128 for (typename ValueMap<TypeParam*, int>::iterator I = VM.begin(), E = VM.end();
129 I != E; ++I) {
130 ++size;
Chris Lattnera07cd902010-04-18 03:28:20 +0000131 std::pair<TypeParam*, int> value = *I; (void)value;
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000132 CompileAssertHasType<TypeParam*>(I->first);
133 if (I->second == 2) {
Jim Grosbache5415282010-11-18 01:38:07 +0000134 EXPECT_EQ(this->BitcastV.get(), I->first);
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000135 I->second = 5;
136 } else if (I->second == 3) {
Jim Grosbache5415282010-11-18 01:38:07 +0000137 EXPECT_EQ(this->AddV.get(), I->first);
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000138 I->second = 6;
139 } else {
140 ADD_FAILURE() << "Iterated through an extra value.";
141 }
142 }
143 EXPECT_EQ(2U, size);
144 EXPECT_EQ(5, VM[this->BitcastV.get()]);
145 EXPECT_EQ(6, VM[this->AddV.get()]);
146
147 size = 0;
148 // Cast to const ValueMap to avoid a bug in DenseMap's iterators.
149 const ValueMap<TypeParam*, int>& CVM = VM;
150 for (typename ValueMap<TypeParam*, int>::const_iterator I = CVM.begin(),
151 E = CVM.end(); I != E; ++I) {
152 ++size;
Chris Lattnera07cd902010-04-18 03:28:20 +0000153 std::pair<TypeParam*, int> value = *I; (void)value;
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000154 CompileAssertHasType<TypeParam*>(I->first);
155 if (I->second == 5) {
Jim Grosbache5415282010-11-18 01:38:07 +0000156 EXPECT_EQ(this->BitcastV.get(), I->first);
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000157 } else if (I->second == 6) {
Jim Grosbache5415282010-11-18 01:38:07 +0000158 EXPECT_EQ(this->AddV.get(), I->first);
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000159 } else {
160 ADD_FAILURE() << "Iterated through an extra value.";
161 }
162 }
163 EXPECT_EQ(2U, size);
164}
165
166TYPED_TEST(ValueMapTest, DefaultCollisionBehavior) {
167 // By default, we overwrite the old value with the replaced value.
168 ValueMap<TypeParam*, int> VM;
169 VM[this->BitcastV.get()] = 7;
170 VM[this->AddV.get()] = 9;
171 this->BitcastV->replaceAllUsesWith(this->AddV.get());
David Blaikie20dead82014-06-20 19:54:13 +0000172 EXPECT_EQ(0u, VM.count(this->BitcastV.get()));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000173 EXPECT_EQ(9, VM.lookup(this->AddV.get()));
174}
175
176TYPED_TEST(ValueMapTest, ConfiguredCollisionBehavior) {
177 // TODO: Implement this when someone needs it.
178}
179
Zachary Turner84fea772014-06-17 00:17:38 +0000180template<typename KeyT, typename MutexT>
181struct LockMutex : ValueMapConfig<KeyT, MutexT> {
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000182 struct ExtraData {
Zachary Turner4072aa32014-06-17 00:38:40 +0000183 MutexT *M;
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000184 bool *CalledRAUW;
185 bool *CalledDeleted;
186 };
187 static void onRAUW(const ExtraData &Data, KeyT Old, KeyT New) {
188 *Data.CalledRAUW = true;
Dylan Noblesmithd930a302014-08-23 22:49:22 +0000189 EXPECT_FALSE(Data.M->try_lock()) << "Mutex should already be locked.";
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000190 }
Jeffrey Yasskin4ab74cd2009-10-23 20:54:00 +0000191 static void onDelete(const ExtraData &Data, KeyT Old) {
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000192 *Data.CalledDeleted = true;
Dylan Noblesmithd930a302014-08-23 22:49:22 +0000193 EXPECT_FALSE(Data.M->try_lock()) << "Mutex should already be locked.";
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000194 }
Zachary Turner4072aa32014-06-17 00:38:40 +0000195 static MutexT *getMutex(const ExtraData &Data) { return Data.M; }
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000196};
Reid Kleckner0123b6e2015-04-06 21:49:55 +0000197// FIXME: These tests started failing on Windows.
Nico Weber63033d32018-04-29 00:45:03 +0000198#if LLVM_ENABLE_THREADS && !defined(_WIN32)
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000199TYPED_TEST(ValueMapTest, LocksMutex) {
Zachary Turner91e18f72014-06-20 21:07:14 +0000200 sys::Mutex M(false); // Not recursive.
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000201 bool CalledRAUW = false, CalledDeleted = false;
Zachary Turner91e18f72014-06-20 21:07:14 +0000202 typedef LockMutex<TypeParam*, sys::Mutex> ConfigType;
Zachary Turner84fea772014-06-17 00:17:38 +0000203 typename ConfigType::ExtraData Data = {&M, &CalledRAUW, &CalledDeleted};
204 ValueMap<TypeParam*, int, ConfigType> VM(Data);
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000205 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}
Duncan Sandsb83012a2009-11-19 20:48:14 +0000211#endif
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000212
213template<typename KeyT>
214struct NoFollow : ValueMapConfig<KeyT> {
215 enum { FollowRAUW = false };
216};
217
218TYPED_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()));
David Blaikie20dead82014-06-20 19:54:13 +0000222 EXPECT_EQ(0u, VM.count(this->AddV.get()));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000223 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
235template<typename KeyT>
236struct CountOps : ValueMapConfig<KeyT> {
237 struct ExtraData {
238 int *Deletions;
239 int *RAUWs;
240 };
241
242 static void onRAUW(const ExtraData &Data, KeyT Old, KeyT New) {
243 ++*Data.RAUWs;
244 }
Jeffrey Yasskin4ab74cd2009-10-23 20:54:00 +0000245 static void onDelete(const ExtraData &Data, KeyT Old) {
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000246 ++*Data.Deletions;
247 }
248};
249
250TYPED_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
266template<typename KeyT>
267struct 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
274 static void onRAUW(ExtraData Map, KeyT Old, KeyT New) {
275 (*Map)->erase(Old);
276 }
Jeffrey Yasskin4ab74cd2009-10-23 20:54:00 +0000277 static void onDelete(ExtraData Map, KeyT Old) {
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000278 (*Map)->erase(Old);
279 }
280};
281TYPED_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());
David Blaikie20dead82014-06-20 19:54:13 +0000288 EXPECT_EQ(0u, VM.count(this->BitcastV.get()));
289 EXPECT_EQ(0u, VM.count(this->AddV.get()));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000290 VM[this->AddV.get()] = 7;
291 this->AddV.reset();
David Blaikie20dead82014-06-20 19:54:13 +0000292 EXPECT_EQ(0u, VM.count(this->AddV.get()));
Jeffrey Yasskin71a5c222009-10-22 22:11:22 +0000293}
294
Duncan P. N. Exon Smithec78cb02016-04-02 16:45:51 +0000295} // end namespace