diff options
author | Fangrui Song <maskray@google.com> | 2019-09-25 04:58:02 +0000 |
---|---|---|
committer | Fangrui Song <maskray@google.com> | 2019-09-25 04:58:02 +0000 |
commit | 2bc478844aeda510ccd5539276cbb509cc47aafa (patch) | |
tree | 46d5f9f1a12f16e2633bc79122d8a67d96dd2262 | |
parent | f674d7dab1b17c7e8b1f37ce06345760f94d2ede (diff) | |
download | bcm5719-llvm-2bc478844aeda510ccd5539276cbb509cc47aafa.tar.gz bcm5719-llvm-2bc478844aeda510ccd5539276cbb509cc47aafa.zip |
[ADT] Add StringMap::insert_or_assign
Summary: Similar to std::unordered_map::insert_or_assign
Reviewers: alexshap, bkramer, dblaikie, lhames
Subscribers: jkorous, dexonsmith, kristina, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D67668
llvm-svn: 372813
-rw-r--r-- | llvm/include/llvm/ADT/StringMap.h | 10 | ||||
-rw-r--r-- | llvm/unittests/ADT/StringMapTest.cpp | 31 |
2 files changed, 41 insertions, 0 deletions
diff --git a/llvm/include/llvm/ADT/StringMap.h b/llvm/include/llvm/ADT/StringMap.h index 8a586fc2670..51fa4844f30 100644 --- a/llvm/include/llvm/ADT/StringMap.h +++ b/llvm/include/llvm/ADT/StringMap.h @@ -391,6 +391,16 @@ public: return try_emplace(KV.first, std::move(KV.second)); } + /// Inserts an element or assigns to the current element if the key already + /// exists. The return type is the same as try_emplace. + template <typename V> + std::pair<iterator, bool> insert_or_assign(StringRef Key, V &&Val) { + auto Ret = try_emplace(Key, std::forward<V>(Val)); + if (!Ret.second) + Ret.first->second = std::forward<V>(Val); + return Ret; + } + /// Emplace a new element for the specified key into the map if the key isn't /// already in the map. The bool component of the returned pair is true /// if and only if the insertion takes place, and the iterator component of diff --git a/llvm/unittests/ADT/StringMapTest.cpp b/llvm/unittests/ADT/StringMapTest.cpp index 4038d4d3d2f..5f0c83d0de6 100644 --- a/llvm/unittests/ADT/StringMapTest.cpp +++ b/llvm/unittests/ADT/StringMapTest.cpp @@ -75,6 +75,16 @@ const char* StringMapTest::testKeyFirst = testKey; size_t StringMapTest::testKeyLength = sizeof(testKey) - 1; const std::string StringMapTest::testKeyStr(testKey); +struct CountCopyAndMove { + CountCopyAndMove() = default; + CountCopyAndMove(const CountCopyAndMove &) { copy = 1; } + CountCopyAndMove(CountCopyAndMove &&) { move = 1; } + void operator=(const CountCopyAndMove &) { ++copy; } + void operator=(CountCopyAndMove &&) { ++move; } + int copy = 0; + int move = 0; +}; + // Empty map tests. TEST_F(StringMapTest, EmptyMapTest) { assertEmptyMap(); @@ -269,6 +279,27 @@ TEST_F(StringMapTest, InsertRehashingPairTest) { EXPECT_EQ(42u, It->second); } +TEST_F(StringMapTest, InsertOrAssignTest) { + struct A : CountCopyAndMove { + A(int v) : v(v) {} + int v; + }; + StringMap<A> t(0); + + auto try1 = t.insert_or_assign("A", A(1)); + EXPECT_TRUE(try1.second); + EXPECT_EQ(1, try1.first->second.v); + EXPECT_EQ(1, try1.first->second.move); + + auto try2 = t.insert_or_assign("A", A(2)); + EXPECT_FALSE(try2.second); + EXPECT_EQ(2, try2.first->second.v); + EXPECT_EQ(2, try1.first->second.move); + + EXPECT_EQ(try1.first, try2.first); + EXPECT_EQ(0, try1.first->second.copy); +} + TEST_F(StringMapTest, IterMapKeys) { StringMap<int> Map; Map["A"] = 1; |