blob: 9dcb4fa7fba391b6992932dabfba96618b3d168f [file] [log] [blame]
Jeffrey Yasskin4546d312009-10-22 22:11:22 +00001//===- llvm/unittest/ADT/ValueMapTest.cpp - ValueMap unit tests -*- C++ -*-===//
2//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Jeffrey Yasskin4546d312009-10-22 22:11:22 +00006//
7//===----------------------------------------------------------------------===//
8
Chandler Carrutha4ea2692014-03-04 11:26:31 +00009#include "llvm/IR/ValueMap.h"
Chandler Carruth130cec22012-12-04 10:23:08 +000010#include "llvm/Config/llvm-config.h"
Chandler Carruth9fb823b2013-01-02 11:36:10 +000011#include "llvm/IR/Constants.h"
12#include "llvm/IR/Instructions.h"
13#include "llvm/IR/LLVMContext.h"
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000014#include "gtest/gtest.h"
15
16using namespace llvm;
17
18namespace {
19
20// Test fixture
21template<typename T>
22class ValueMapTest : public testing::Test {
23protected:
Mehdi Amini03b42e42016-04-14 21:59:01 +000024 LLVMContext Context;
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000025 Constant *ConstantV;
Ahmed Charles56440fd2014-03-06 05:51:42 +000026 std::unique_ptr<BitCastInst> BitcastV;
27 std::unique_ptr<BinaryOperator> AddV;
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000028
Mehdi Amini03b42e42016-04-14 21:59:01 +000029 ValueMapTest()
30 : ConstantV(ConstantInt::get(Type::getInt32Ty(Context), 0)),
31 BitcastV(new BitCastInst(ConstantV, Type::getInt32Ty(Context))),
32 AddV(BinaryOperator::CreateAdd(ConstantV, ConstantV)) {}
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000033};
34
35// Run everything on Value*, a subtype to make sure that casting works as
36// expected, and a const subtype to make sure we cast const correctly.
37typedef ::testing::Types<Value, Instruction, const Instruction> KeyTypes;
38TYPED_TEST_CASE(ValueMapTest, KeyTypes);
39
40TYPED_TEST(ValueMapTest, Null) {
41 ValueMap<TypeParam*, int> VM1;
Craig Topper66f09ad2014-06-08 22:29:17 +000042 VM1[nullptr] = 7;
43 EXPECT_EQ(7, VM1.lookup(nullptr));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000044}
45
46TYPED_TEST(ValueMapTest, FollowsValue) {
47 ValueMap<TypeParam*, int> VM;
48 VM[this->BitcastV.get()] = 7;
49 EXPECT_EQ(7, VM.lookup(this->BitcastV.get()));
David Blaikie7c8d1392014-06-20 19:54:13 +000050 EXPECT_EQ(0u, VM.count(this->AddV.get()));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000051 this->BitcastV->replaceAllUsesWith(this->AddV.get());
52 EXPECT_EQ(7, VM.lookup(this->AddV.get()));
David Blaikie7c8d1392014-06-20 19:54:13 +000053 EXPECT_EQ(0u, VM.count(this->BitcastV.get()));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000054 this->AddV.reset();
David Blaikie7c8d1392014-06-20 19:54:13 +000055 EXPECT_EQ(0u, VM.count(this->AddV.get()));
56 EXPECT_EQ(0u, VM.count(this->BitcastV.get()));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000057 EXPECT_EQ(0U, VM.size());
58}
59
60TYPED_TEST(ValueMapTest, OperationsWork) {
61 ValueMap<TypeParam*, int> VM;
Chris Lattnerf58e8c02010-04-18 03:28:20 +000062 ValueMap<TypeParam*, int> VM2(16); (void)VM2;
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000063 typename ValueMapConfig<TypeParam*>::ExtraData Data;
Chris Lattnerf58e8c02010-04-18 03:28:20 +000064 ValueMap<TypeParam*, int> VM3(Data, 16); (void)VM3;
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000065 EXPECT_TRUE(VM.empty());
66
67 VM[this->BitcastV.get()] = 7;
68
69 // Find:
70 typename ValueMap<TypeParam*, int>::iterator I =
71 VM.find(this->BitcastV.get());
72 ASSERT_TRUE(I != VM.end());
Jim Grosbach96ef39a2010-11-18 01:38:07 +000073 EXPECT_EQ(this->BitcastV.get(), I->first);
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000074 EXPECT_EQ(7, I->second);
Jim Grosbach96ef39a2010-11-18 01:38:07 +000075 EXPECT_TRUE(VM.find(this->AddV.get()) == VM.end());
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000076
77 // Const find:
78 const ValueMap<TypeParam*, int> &CVM = VM;
79 typename ValueMap<TypeParam*, int>::const_iterator CI =
80 CVM.find(this->BitcastV.get());
81 ASSERT_TRUE(CI != CVM.end());
Jim Grosbach96ef39a2010-11-18 01:38:07 +000082 EXPECT_EQ(this->BitcastV.get(), CI->first);
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000083 EXPECT_EQ(7, CI->second);
84 EXPECT_TRUE(CVM.find(this->AddV.get()) == CVM.end());
85
86 // Insert:
87 std::pair<typename ValueMap<TypeParam*, int>::iterator, bool> InsertResult1 =
88 VM.insert(std::make_pair(this->AddV.get(), 3));
Jim Grosbach96ef39a2010-11-18 01:38:07 +000089 EXPECT_EQ(this->AddV.get(), InsertResult1.first->first);
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000090 EXPECT_EQ(3, InsertResult1.first->second);
91 EXPECT_TRUE(InsertResult1.second);
David Blaikie7c8d1392014-06-20 19:54:13 +000092 EXPECT_EQ(1u, VM.count(this->AddV.get()));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000093 std::pair<typename ValueMap<TypeParam*, int>::iterator, bool> InsertResult2 =
94 VM.insert(std::make_pair(this->AddV.get(), 5));
Jim Grosbach96ef39a2010-11-18 01:38:07 +000095 EXPECT_EQ(this->AddV.get(), InsertResult2.first->first);
Jeffrey Yasskin4546d312009-10-22 22:11:22 +000096 EXPECT_EQ(3, InsertResult2.first->second);
97 EXPECT_FALSE(InsertResult2.second);
98
99 // Erase:
100 VM.erase(InsertResult2.first);
Bill Wendling9a2652b2010-07-10 18:56:35 +0000101 EXPECT_EQ(0U, VM.count(this->AddV.get()));
102 EXPECT_EQ(1U, VM.count(this->BitcastV.get()));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000103 VM.erase(this->BitcastV.get());
Bill Wendling9a2652b2010-07-10 18:56:35 +0000104 EXPECT_EQ(0U, VM.count(this->BitcastV.get()));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000105 EXPECT_EQ(0U, VM.size());
106
107 // Range insert:
108 SmallVector<std::pair<Instruction*, int>, 2> Elems;
109 Elems.push_back(std::make_pair(this->AddV.get(), 1));
110 Elems.push_back(std::make_pair(this->BitcastV.get(), 2));
111 VM.insert(Elems.begin(), Elems.end());
112 EXPECT_EQ(1, VM.lookup(this->AddV.get()));
113 EXPECT_EQ(2, VM.lookup(this->BitcastV.get()));
114}
115
116template<typename ExpectedType, typename VarType>
117void CompileAssertHasType(VarType) {
Benjamin Kramerbcc77b02014-03-07 15:54:23 +0000118 static_assert(std::is_same<ExpectedType, VarType>::value,
119 "Not the same type");
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000120}
121
122TYPED_TEST(ValueMapTest, Iteration) {
123 ValueMap<TypeParam*, int> VM;
124 VM[this->BitcastV.get()] = 2;
125 VM[this->AddV.get()] = 3;
126 size_t size = 0;
127 for (typename ValueMap<TypeParam*, int>::iterator I = VM.begin(), E = VM.end();
128 I != E; ++I) {
129 ++size;
Chris Lattnerf58e8c02010-04-18 03:28:20 +0000130 std::pair<TypeParam*, int> value = *I; (void)value;
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000131 CompileAssertHasType<TypeParam*>(I->first);
132 if (I->second == 2) {
Jim Grosbach96ef39a2010-11-18 01:38:07 +0000133 EXPECT_EQ(this->BitcastV.get(), I->first);
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000134 I->second = 5;
135 } else if (I->second == 3) {
Jim Grosbach96ef39a2010-11-18 01:38:07 +0000136 EXPECT_EQ(this->AddV.get(), I->first);
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000137 I->second = 6;
138 } else {
139 ADD_FAILURE() << "Iterated through an extra value.";
140 }
141 }
142 EXPECT_EQ(2U, size);
143 EXPECT_EQ(5, VM[this->BitcastV.get()]);
144 EXPECT_EQ(6, VM[this->AddV.get()]);
145
146 size = 0;
147 // Cast to const ValueMap to avoid a bug in DenseMap's iterators.
148 const ValueMap<TypeParam*, int>& CVM = VM;
149 for (typename ValueMap<TypeParam*, int>::const_iterator I = CVM.begin(),
150 E = CVM.end(); I != E; ++I) {
151 ++size;
Chris Lattnerf58e8c02010-04-18 03:28:20 +0000152 std::pair<TypeParam*, int> value = *I; (void)value;
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000153 CompileAssertHasType<TypeParam*>(I->first);
154 if (I->second == 5) {
Jim Grosbach96ef39a2010-11-18 01:38:07 +0000155 EXPECT_EQ(this->BitcastV.get(), I->first);
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000156 } else if (I->second == 6) {
Jim Grosbach96ef39a2010-11-18 01:38:07 +0000157 EXPECT_EQ(this->AddV.get(), I->first);
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000158 } else {
159 ADD_FAILURE() << "Iterated through an extra value.";
160 }
161 }
162 EXPECT_EQ(2U, size);
163}
164
165TYPED_TEST(ValueMapTest, DefaultCollisionBehavior) {
166 // By default, we overwrite the old value with the replaced value.
167 ValueMap<TypeParam*, int> VM;
168 VM[this->BitcastV.get()] = 7;
169 VM[this->AddV.get()] = 9;
170 this->BitcastV->replaceAllUsesWith(this->AddV.get());
David Blaikie7c8d1392014-06-20 19:54:13 +0000171 EXPECT_EQ(0u, VM.count(this->BitcastV.get()));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000172 EXPECT_EQ(9, VM.lookup(this->AddV.get()));
173}
174
175TYPED_TEST(ValueMapTest, ConfiguredCollisionBehavior) {
176 // TODO: Implement this when someone needs it.
177}
178
Zachary Turner814a4932014-06-17 00:17:38 +0000179template<typename KeyT, typename MutexT>
180struct LockMutex : ValueMapConfig<KeyT, MutexT> {
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000181 struct ExtraData {
Zachary Turnerb07f1e12014-06-17 00:38:40 +0000182 MutexT *M;
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000183 bool *CalledRAUW;
184 bool *CalledDeleted;
185 };
186 static void onRAUW(const ExtraData &Data, KeyT Old, KeyT New) {
187 *Data.CalledRAUW = true;
Dylan Noblesmith13044d1c2014-08-23 22:49:22 +0000188 EXPECT_FALSE(Data.M->try_lock()) << "Mutex should already be locked.";
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000189 }
Jeffrey Yasskin61ade252009-10-23 20:54:00 +0000190 static void onDelete(const ExtraData &Data, KeyT Old) {
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000191 *Data.CalledDeleted = true;
Dylan Noblesmith13044d1c2014-08-23 22:49:22 +0000192 EXPECT_FALSE(Data.M->try_lock()) << "Mutex should already be locked.";
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000193 }
Zachary Turnerb07f1e12014-06-17 00:38:40 +0000194 static MutexT *getMutex(const ExtraData &Data) { return Data.M; }
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000195};
Reid Klecknerad4cee12015-04-06 21:49:55 +0000196// FIXME: These tests started failing on Windows.
Nico Weber712e8d22018-04-29 00:45:03 +0000197#if LLVM_ENABLE_THREADS && !defined(_WIN32)
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000198TYPED_TEST(ValueMapTest, LocksMutex) {
Zachary Turnerc04b8922014-06-20 21:07:14 +0000199 sys::Mutex M(false); // Not recursive.
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000200 bool CalledRAUW = false, CalledDeleted = false;
Zachary Turnerc04b8922014-06-20 21:07:14 +0000201 typedef LockMutex<TypeParam*, sys::Mutex> ConfigType;
Zachary Turner814a4932014-06-17 00:17:38 +0000202 typename ConfigType::ExtraData Data = {&M, &CalledRAUW, &CalledDeleted};
203 ValueMap<TypeParam*, int, ConfigType> VM(Data);
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000204 VM[this->BitcastV.get()] = 7;
205 this->BitcastV->replaceAllUsesWith(this->AddV.get());
206 this->AddV.reset();
207 EXPECT_TRUE(CalledRAUW);
208 EXPECT_TRUE(CalledDeleted);
209}
Duncan Sandsc8cee4f2009-11-19 20:48:14 +0000210#endif
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000211
212template<typename KeyT>
213struct NoFollow : ValueMapConfig<KeyT> {
214 enum { FollowRAUW = false };
215};
216
217TYPED_TEST(ValueMapTest, NoFollowRAUW) {
218 ValueMap<TypeParam*, int, NoFollow<TypeParam*> > VM;
219 VM[this->BitcastV.get()] = 7;
220 EXPECT_EQ(7, VM.lookup(this->BitcastV.get()));
David Blaikie7c8d1392014-06-20 19:54:13 +0000221 EXPECT_EQ(0u, VM.count(this->AddV.get()));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000222 this->BitcastV->replaceAllUsesWith(this->AddV.get());
223 EXPECT_EQ(7, VM.lookup(this->BitcastV.get()));
224 EXPECT_EQ(0, VM.lookup(this->AddV.get()));
225 this->AddV.reset();
226 EXPECT_EQ(7, VM.lookup(this->BitcastV.get()));
227 EXPECT_EQ(0, VM.lookup(this->AddV.get()));
228 this->BitcastV.reset();
229 EXPECT_EQ(0, VM.lookup(this->BitcastV.get()));
230 EXPECT_EQ(0, VM.lookup(this->AddV.get()));
231 EXPECT_EQ(0U, VM.size());
232}
233
234template<typename KeyT>
235struct CountOps : ValueMapConfig<KeyT> {
236 struct ExtraData {
237 int *Deletions;
238 int *RAUWs;
239 };
240
241 static void onRAUW(const ExtraData &Data, KeyT Old, KeyT New) {
242 ++*Data.RAUWs;
243 }
Jeffrey Yasskin61ade252009-10-23 20:54:00 +0000244 static void onDelete(const ExtraData &Data, KeyT Old) {
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000245 ++*Data.Deletions;
246 }
247};
248
249TYPED_TEST(ValueMapTest, CallsConfig) {
250 int Deletions = 0, RAUWs = 0;
251 typename CountOps<TypeParam*>::ExtraData Data = {&Deletions, &RAUWs};
252 ValueMap<TypeParam*, int, CountOps<TypeParam*> > VM(Data);
253 VM[this->BitcastV.get()] = 7;
254 this->BitcastV->replaceAllUsesWith(this->AddV.get());
255 EXPECT_EQ(0, Deletions);
256 EXPECT_EQ(1, RAUWs);
257 this->AddV.reset();
258 EXPECT_EQ(1, Deletions);
259 EXPECT_EQ(1, RAUWs);
260 this->BitcastV.reset();
261 EXPECT_EQ(1, Deletions);
262 EXPECT_EQ(1, RAUWs);
263}
264
265template<typename KeyT>
266struct ModifyingConfig : ValueMapConfig<KeyT> {
267 // We'll put a pointer here back to the ValueMap this key is in, so
268 // that we can modify it (and clobber *this) before the ValueMap
269 // tries to do the same modification. In previous versions of
270 // ValueMap, that exploded.
271 typedef ValueMap<KeyT, int, ModifyingConfig<KeyT> > **ExtraData;
272
273 static void onRAUW(ExtraData Map, KeyT Old, KeyT New) {
274 (*Map)->erase(Old);
275 }
Jeffrey Yasskin61ade252009-10-23 20:54:00 +0000276 static void onDelete(ExtraData Map, KeyT Old) {
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000277 (*Map)->erase(Old);
278 }
279};
280TYPED_TEST(ValueMapTest, SurvivesModificationByConfig) {
281 ValueMap<TypeParam*, int, ModifyingConfig<TypeParam*> > *MapAddress;
282 ValueMap<TypeParam*, int, ModifyingConfig<TypeParam*> > VM(&MapAddress);
283 MapAddress = &VM;
284 // Now the ModifyingConfig can modify the Map inside a callback.
285 VM[this->BitcastV.get()] = 7;
286 this->BitcastV->replaceAllUsesWith(this->AddV.get());
David Blaikie7c8d1392014-06-20 19:54:13 +0000287 EXPECT_EQ(0u, VM.count(this->BitcastV.get()));
288 EXPECT_EQ(0u, VM.count(this->AddV.get()));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000289 VM[this->AddV.get()] = 7;
290 this->AddV.reset();
David Blaikie7c8d1392014-06-20 19:54:13 +0000291 EXPECT_EQ(0u, VM.count(this->AddV.get()));
Jeffrey Yasskin4546d312009-10-22 22:11:22 +0000292}
293
Duncan P. N. Exon Smithddbb1cd2016-04-02 16:45:51 +0000294} // end namespace