1 //===- StringMap.h - String Hash table map interface ------------*- C++ -*-===//
3 // 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
7 //===----------------------------------------------------------------------===//
9 // This file defines the StringMap class.
11 //===----------------------------------------------------------------------===//
13 #ifndef LLVM_ADT_STRINGMAP_H
14 #define LLVM_ADT_STRINGMAP_H
16 #include "llvm/ADT/StringRef.h"
17 #include "llvm/ADT/iterator.h"
18 #include "llvm/ADT/iterator_range.h"
19 #include "llvm/Support/Allocator.h"
20 #include "llvm/Support/PointerLikeTypeTraits.h"
21 #include "llvm/Support/ErrorHandling.h"
27 #include <initializer_list>
33 template<typename ValueTy
> class StringMapConstIterator
;
34 template<typename ValueTy
> class StringMapIterator
;
35 template<typename ValueTy
> class StringMapKeyIterator
;
37 /// StringMapEntryBase - Shared base class of StringMapEntry instances.
38 class StringMapEntryBase
{
42 explicit StringMapEntryBase(size_t Len
) : StrLen(Len
) {}
44 size_t getKeyLength() const { return StrLen
; }
47 /// StringMapImpl - This is the base class of StringMap that is shared among
48 /// all of its instantiations.
51 // Array of NumBuckets pointers to entries, null pointers are holes.
52 // TheTable[NumBuckets] contains a sentinel value for easy iteration. Followed
53 // by an array of the actual hash values as unsigned integers.
54 StringMapEntryBase
**TheTable
= nullptr;
55 unsigned NumBuckets
= 0;
56 unsigned NumItems
= 0;
57 unsigned NumTombstones
= 0;
61 explicit StringMapImpl(unsigned itemSize
)
62 : ItemSize(itemSize
) {}
63 StringMapImpl(StringMapImpl
&&RHS
)
64 : TheTable(RHS
.TheTable
), NumBuckets(RHS
.NumBuckets
),
65 NumItems(RHS
.NumItems
), NumTombstones(RHS
.NumTombstones
),
66 ItemSize(RHS
.ItemSize
) {
67 RHS
.TheTable
= nullptr;
70 RHS
.NumTombstones
= 0;
73 StringMapImpl(unsigned InitSize
, unsigned ItemSize
);
74 unsigned RehashTable(unsigned BucketNo
= 0);
76 /// LookupBucketFor - Look up the bucket that the specified string should end
77 /// up in. If it already exists as a key in the map, the Item pointer for the
78 /// specified bucket will be non-null. Otherwise, it will be null. In either
79 /// case, the FullHashValue field of the bucket will be set to the hash value
81 unsigned LookupBucketFor(StringRef Key
);
83 /// FindKey - Look up the bucket that contains the specified key. If it exists
84 /// in the map, return the bucket number of the key. Otherwise return -1.
85 /// This does not modify the map.
86 int FindKey(StringRef Key
) const;
88 /// RemoveKey - Remove the specified StringMapEntry from the table, but do not
89 /// delete it. This aborts if the value isn't in the table.
90 void RemoveKey(StringMapEntryBase
*V
);
92 /// RemoveKey - Remove the StringMapEntry for the specified key from the
93 /// table, returning it. If the key is not in the table, this returns null.
94 StringMapEntryBase
*RemoveKey(StringRef Key
);
96 /// Allocate the table with the specified number of buckets and otherwise
97 /// setup the map as empty.
98 void init(unsigned Size
);
101 static StringMapEntryBase
*getTombstoneVal() {
102 uintptr_t Val
= static_cast<uintptr_t>(-1);
103 Val
<<= PointerLikeTypeTraits
<StringMapEntryBase
*>::NumLowBitsAvailable
;
104 return reinterpret_cast<StringMapEntryBase
*>(Val
);
107 unsigned getNumBuckets() const { return NumBuckets
; }
108 unsigned getNumItems() const { return NumItems
; }
110 bool empty() const { return NumItems
== 0; }
111 unsigned size() const { return NumItems
; }
113 void swap(StringMapImpl
&Other
) {
114 std::swap(TheTable
, Other
.TheTable
);
115 std::swap(NumBuckets
, Other
.NumBuckets
);
116 std::swap(NumItems
, Other
.NumItems
);
117 std::swap(NumTombstones
, Other
.NumTombstones
);
121 /// StringMapEntry - This is used to represent one value that is inserted into
122 /// a StringMap. It contains the Value itself and the key: the string length
124 template<typename ValueTy
>
125 class StringMapEntry
: public StringMapEntryBase
{
129 explicit StringMapEntry(size_t strLen
)
130 : StringMapEntryBase(strLen
), second() {}
131 template <typename
... InitTy
>
132 StringMapEntry(size_t strLen
, InitTy
&&... InitVals
)
133 : StringMapEntryBase(strLen
), second(std::forward
<InitTy
>(InitVals
)...) {}
134 StringMapEntry(StringMapEntry
&E
) = delete;
136 StringRef
getKey() const {
137 return StringRef(getKeyData(), getKeyLength());
140 const ValueTy
&getValue() const { return second
; }
141 ValueTy
&getValue() { return second
; }
143 void setValue(const ValueTy
&V
) { second
= V
; }
145 /// getKeyData - Return the start of the string data that is the key for this
146 /// value. The string data is always stored immediately after the
147 /// StringMapEntry object.
148 const char *getKeyData() const {return reinterpret_cast<const char*>(this+1);}
150 StringRef
first() const { return StringRef(getKeyData(), getKeyLength()); }
152 /// Create a StringMapEntry for the specified key construct the value using
154 template <typename AllocatorTy
, typename
... InitTy
>
155 static StringMapEntry
*Create(StringRef Key
, AllocatorTy
&Allocator
,
156 InitTy
&&... InitVals
) {
157 size_t KeyLength
= Key
.size();
159 // Allocate a new item with space for the string at the end and a null
161 size_t AllocSize
= sizeof(StringMapEntry
) + KeyLength
+ 1;
162 size_t Alignment
= alignof(StringMapEntry
);
164 StringMapEntry
*NewItem
=
165 static_cast<StringMapEntry
*>(Allocator
.Allocate(AllocSize
,Alignment
));
166 assert(NewItem
&& "Unhandled out-of-memory");
168 // Construct the value.
169 new (NewItem
) StringMapEntry(KeyLength
, std::forward
<InitTy
>(InitVals
)...);
171 // Copy the string information.
172 char *StrBuffer
= const_cast<char*>(NewItem
->getKeyData());
174 memcpy(StrBuffer
, Key
.data(), KeyLength
);
175 StrBuffer
[KeyLength
] = 0; // Null terminate for convenience of clients.
179 /// Create - Create a StringMapEntry with normal malloc/free.
180 template <typename
... InitType
>
181 static StringMapEntry
*Create(StringRef Key
, InitType
&&... InitVal
) {
183 return Create(Key
, A
, std::forward
<InitType
>(InitVal
)...);
186 static StringMapEntry
*Create(StringRef Key
) {
187 return Create(Key
, ValueTy());
190 /// GetStringMapEntryFromKeyData - Given key data that is known to be embedded
191 /// into a StringMapEntry, return the StringMapEntry itself.
192 static StringMapEntry
&GetStringMapEntryFromKeyData(const char *KeyData
) {
193 char *Ptr
= const_cast<char*>(KeyData
) - sizeof(StringMapEntry
<ValueTy
>);
194 return *reinterpret_cast<StringMapEntry
*>(Ptr
);
197 /// Destroy - Destroy this StringMapEntry, releasing memory back to the
198 /// specified allocator.
199 template<typename AllocatorTy
>
200 void Destroy(AllocatorTy
&Allocator
) {
201 // Free memory referenced by the item.
202 size_t AllocSize
= sizeof(StringMapEntry
) + getKeyLength() + 1;
203 this->~StringMapEntry();
204 Allocator
.Deallocate(static_cast<void *>(this), AllocSize
);
207 /// Destroy this object, releasing memory back to the malloc allocator.
214 /// StringMap - This is an unconventional map that is specialized for handling
215 /// keys that are "strings", which are basically ranges of bytes. This does some
216 /// funky memory allocation and hashing things to make it extremely efficient,
217 /// storing the string data *after* the value in the map.
218 template<typename ValueTy
, typename AllocatorTy
= MallocAllocator
>
219 class StringMap
: public StringMapImpl
{
220 AllocatorTy Allocator
;
223 using MapEntryTy
= StringMapEntry
<ValueTy
>;
225 StringMap() : StringMapImpl(static_cast<unsigned>(sizeof(MapEntryTy
))) {}
227 explicit StringMap(unsigned InitialSize
)
228 : StringMapImpl(InitialSize
, static_cast<unsigned>(sizeof(MapEntryTy
))) {}
230 explicit StringMap(AllocatorTy A
)
231 : StringMapImpl(static_cast<unsigned>(sizeof(MapEntryTy
))), Allocator(A
) {}
233 StringMap(unsigned InitialSize
, AllocatorTy A
)
234 : StringMapImpl(InitialSize
, static_cast<unsigned>(sizeof(MapEntryTy
))),
237 StringMap(std::initializer_list
<std::pair
<StringRef
, ValueTy
>> List
)
238 : StringMapImpl(List
.size(), static_cast<unsigned>(sizeof(MapEntryTy
))) {
239 for (const auto &P
: List
) {
244 StringMap(StringMap
&&RHS
)
245 : StringMapImpl(std::move(RHS
)), Allocator(std::move(RHS
.Allocator
)) {}
247 StringMap(const StringMap
&RHS
) :
248 StringMapImpl(static_cast<unsigned>(sizeof(MapEntryTy
))),
249 Allocator(RHS
.Allocator
) {
253 // Allocate TheTable of the same size as RHS's TheTable, and set the
254 // sentinel appropriately (and NumBuckets).
255 init(RHS
.NumBuckets
);
256 unsigned *HashTable
= (unsigned *)(TheTable
+ NumBuckets
+ 1),
257 *RHSHashTable
= (unsigned *)(RHS
.TheTable
+ NumBuckets
+ 1);
259 NumItems
= RHS
.NumItems
;
260 NumTombstones
= RHS
.NumTombstones
;
261 for (unsigned I
= 0, E
= NumBuckets
; I
!= E
; ++I
) {
262 StringMapEntryBase
*Bucket
= RHS
.TheTable
[I
];
263 if (!Bucket
|| Bucket
== getTombstoneVal()) {
264 TheTable
[I
] = Bucket
;
268 TheTable
[I
] = MapEntryTy::Create(
269 static_cast<MapEntryTy
*>(Bucket
)->getKey(), Allocator
,
270 static_cast<MapEntryTy
*>(Bucket
)->getValue());
271 HashTable
[I
] = RHSHashTable
[I
];
274 // Note that here we've copied everything from the RHS into this object,
275 // tombstones included. We could, instead, have re-probed for each key to
276 // instantiate this new object without any tombstone buckets. The
277 // assumption here is that items are rarely deleted from most StringMaps,
278 // and so tombstones are rare, so the cost of re-probing for all inputs is
282 StringMap
&operator=(StringMap RHS
) {
283 StringMapImpl::swap(RHS
);
284 std::swap(Allocator
, RHS
.Allocator
);
289 // Delete all the elements in the map, but don't reset the elements
290 // to default values. This is a copy of clear(), but avoids unnecessary
291 // work not required in the destructor.
293 for (unsigned I
= 0, E
= NumBuckets
; I
!= E
; ++I
) {
294 StringMapEntryBase
*Bucket
= TheTable
[I
];
295 if (Bucket
&& Bucket
!= getTombstoneVal()) {
296 static_cast<MapEntryTy
*>(Bucket
)->Destroy(Allocator
);
303 AllocatorTy
&getAllocator() { return Allocator
; }
304 const AllocatorTy
&getAllocator() const { return Allocator
; }
306 using key_type
= const char*;
307 using mapped_type
= ValueTy
;
308 using value_type
= StringMapEntry
<ValueTy
>;
309 using size_type
= size_t;
311 using const_iterator
= StringMapConstIterator
<ValueTy
>;
312 using iterator
= StringMapIterator
<ValueTy
>;
315 return iterator(TheTable
, NumBuckets
== 0);
318 return iterator(TheTable
+NumBuckets
, true);
320 const_iterator
begin() const {
321 return const_iterator(TheTable
, NumBuckets
== 0);
323 const_iterator
end() const {
324 return const_iterator(TheTable
+NumBuckets
, true);
327 iterator_range
<StringMapKeyIterator
<ValueTy
>> keys() const {
328 return make_range(StringMapKeyIterator
<ValueTy
>(begin()),
329 StringMapKeyIterator
<ValueTy
>(end()));
332 iterator
find(StringRef Key
) {
333 int Bucket
= FindKey(Key
);
334 if (Bucket
== -1) return end();
335 return iterator(TheTable
+Bucket
, true);
338 const_iterator
find(StringRef Key
) const {
339 int Bucket
= FindKey(Key
);
340 if (Bucket
== -1) return end();
341 return const_iterator(TheTable
+Bucket
, true);
344 /// lookup - Return the entry for the specified key, or a default
345 /// constructed value if no such entry exists.
346 ValueTy
lookup(StringRef Key
) const {
347 const_iterator it
= find(Key
);
353 /// Lookup the ValueTy for the \p Key, or create a default constructed value
354 /// if the key is not in the map.
355 ValueTy
&operator[](StringRef Key
) { return try_emplace(Key
).first
->second
; }
357 /// count - Return 1 if the element is in the map, 0 otherwise.
358 size_type
count(StringRef Key
) const {
359 return find(Key
) == end() ? 0 : 1;
362 /// insert - Insert the specified key/value pair into the map. If the key
363 /// already exists in the map, return false and ignore the request, otherwise
364 /// insert it and return true.
365 bool insert(MapEntryTy
*KeyValue
) {
366 unsigned BucketNo
= LookupBucketFor(KeyValue
->getKey());
367 StringMapEntryBase
*&Bucket
= TheTable
[BucketNo
];
368 if (Bucket
&& Bucket
!= getTombstoneVal())
369 return false; // Already exists in map.
371 if (Bucket
== getTombstoneVal())
375 assert(NumItems
+ NumTombstones
<= NumBuckets
);
381 /// insert - Inserts the specified key/value pair into the map if the key
382 /// isn't already in the map. The bool component of the returned pair is true
383 /// if and only if the insertion takes place, and the iterator component of
384 /// the pair points to the element with key equivalent to the key of the pair.
385 std::pair
<iterator
, bool> insert(std::pair
<StringRef
, ValueTy
> KV
) {
386 return try_emplace(KV
.first
, std::move(KV
.second
));
389 /// Emplace a new element for the specified key into the map if the key isn't
390 /// already in the map. The bool component of the returned pair is true
391 /// if and only if the insertion takes place, and the iterator component of
392 /// the pair points to the element with key equivalent to the key of the pair.
393 template <typename
... ArgsTy
>
394 std::pair
<iterator
, bool> try_emplace(StringRef Key
, ArgsTy
&&... Args
) {
395 unsigned BucketNo
= LookupBucketFor(Key
);
396 StringMapEntryBase
*&Bucket
= TheTable
[BucketNo
];
397 if (Bucket
&& Bucket
!= getTombstoneVal())
398 return std::make_pair(iterator(TheTable
+ BucketNo
, false),
399 false); // Already exists in map.
401 if (Bucket
== getTombstoneVal())
403 Bucket
= MapEntryTy::Create(Key
, Allocator
, std::forward
<ArgsTy
>(Args
)...);
405 assert(NumItems
+ NumTombstones
<= NumBuckets
);
407 BucketNo
= RehashTable(BucketNo
);
408 return std::make_pair(iterator(TheTable
+ BucketNo
, false), true);
411 // clear - Empties out the StringMap
415 // Zap all values, resetting the keys back to non-present (not tombstone),
416 // which is safe because we're removing all elements.
417 for (unsigned I
= 0, E
= NumBuckets
; I
!= E
; ++I
) {
418 StringMapEntryBase
*&Bucket
= TheTable
[I
];
419 if (Bucket
&& Bucket
!= getTombstoneVal()) {
420 static_cast<MapEntryTy
*>(Bucket
)->Destroy(Allocator
);
429 /// remove - Remove the specified key/value pair from the map, but do not
430 /// erase it. This aborts if the key is not in the map.
431 void remove(MapEntryTy
*KeyValue
) {
435 void erase(iterator I
) {
438 V
.Destroy(Allocator
);
441 bool erase(StringRef Key
) {
442 iterator I
= find(Key
);
443 if (I
== end()) return false;
449 template <typename DerivedTy
, typename ValueTy
>
450 class StringMapIterBase
451 : public iterator_facade_base
<DerivedTy
, std::forward_iterator_tag
,
454 StringMapEntryBase
**Ptr
= nullptr;
457 StringMapIterBase() = default;
459 explicit StringMapIterBase(StringMapEntryBase
**Bucket
,
460 bool NoAdvance
= false)
462 if (!NoAdvance
) AdvancePastEmptyBuckets();
465 DerivedTy
&operator=(const DerivedTy
&Other
) {
467 return static_cast<DerivedTy
&>(*this);
470 bool operator==(const DerivedTy
&RHS
) const { return Ptr
== RHS
.Ptr
; }
472 DerivedTy
&operator++() { // Preincrement
474 AdvancePastEmptyBuckets();
475 return static_cast<DerivedTy
&>(*this);
478 DerivedTy
operator++(int) { // Post-increment
485 void AdvancePastEmptyBuckets() {
486 while (*Ptr
== nullptr || *Ptr
== StringMapImpl::getTombstoneVal())
491 template <typename ValueTy
>
492 class StringMapConstIterator
493 : public StringMapIterBase
<StringMapConstIterator
<ValueTy
>,
494 const StringMapEntry
<ValueTy
>> {
495 using base
= StringMapIterBase
<StringMapConstIterator
<ValueTy
>,
496 const StringMapEntry
<ValueTy
>>;
499 StringMapConstIterator() = default;
500 explicit StringMapConstIterator(StringMapEntryBase
**Bucket
,
501 bool NoAdvance
= false)
502 : base(Bucket
, NoAdvance
) {}
504 const StringMapEntry
<ValueTy
> &operator*() const {
505 return *static_cast<const StringMapEntry
<ValueTy
> *>(*this->Ptr
);
509 template <typename ValueTy
>
510 class StringMapIterator
: public StringMapIterBase
<StringMapIterator
<ValueTy
>,
511 StringMapEntry
<ValueTy
>> {
513 StringMapIterBase
<StringMapIterator
<ValueTy
>, StringMapEntry
<ValueTy
>>;
516 StringMapIterator() = default;
517 explicit StringMapIterator(StringMapEntryBase
**Bucket
,
518 bool NoAdvance
= false)
519 : base(Bucket
, NoAdvance
) {}
521 StringMapEntry
<ValueTy
> &operator*() const {
522 return *static_cast<StringMapEntry
<ValueTy
> *>(*this->Ptr
);
525 operator StringMapConstIterator
<ValueTy
>() const {
526 return StringMapConstIterator
<ValueTy
>(this->Ptr
, true);
530 template <typename ValueTy
>
531 class StringMapKeyIterator
532 : public iterator_adaptor_base
<StringMapKeyIterator
<ValueTy
>,
533 StringMapConstIterator
<ValueTy
>,
534 std::forward_iterator_tag
, StringRef
> {
535 using base
= iterator_adaptor_base
<StringMapKeyIterator
<ValueTy
>,
536 StringMapConstIterator
<ValueTy
>,
537 std::forward_iterator_tag
, StringRef
>;
540 StringMapKeyIterator() = default;
541 explicit StringMapKeyIterator(StringMapConstIterator
<ValueTy
> Iter
)
542 : base(std::move(Iter
)) {}
544 StringRef
&operator*() {
545 Key
= this->wrapped()->getKey();
553 } // end namespace llvm
555 #endif // LLVM_ADT_STRINGMAP_H