Skip to content

Commit 9780916

Browse files
committed
[esan] Add iterator to esan's generic hashtable
Summary: Adds simple iterator support to the esan hashtable. Reviewers: aizatsky Subscribers: vitalybuka, zhaoqin, kcc, eugenis, llvm-commits, kubabrecka Differential Revision: https://reviews.llvm.org/D22682 git-svn-id: https://llvm.org/svn/llvm-project/compiler-rt/trunk@278027 91177308-0d34-0410-b5e6-96231b3b80d8
1 parent 8e4aa20 commit 9780916

File tree

2 files changed

+179
-3
lines changed

2 files changed

+179
-3
lines changed

lib/esan/esan_hashtable.h

+134-3
Original file line numberDiff line numberDiff line change
@@ -66,19 +66,58 @@ class HashTable {
6666
// If the table is internally-synchronized, this lock must not be held
6767
// while a hashtable function is called as it will deadlock: the lock
6868
// is not recursive. This is meant for use with externally-synchronized
69-
// tables.
69+
// tables or with an iterator.
7070
void lock();
7171
void unlock();
7272

7373
private:
74-
void resize();
75-
7674
struct HashEntry {
7775
KeyTy Key;
7876
DataTy Payload;
7977
HashEntry *Next;
8078
};
8179

80+
public:
81+
struct HashPair {
82+
HashPair(KeyTy Key, DataTy Data) : Key(Key), Data(Data) {}
83+
KeyTy Key;
84+
DataTy Data;
85+
};
86+
87+
// This iterator does not perform any synchronization.
88+
// It expects the caller to lock the table across the whole iteration.
89+
// Calling HashTable functions while using the iterator is not supported.
90+
// The iterator returns copies of the keys and data.
91+
class iterator {
92+
public:
93+
iterator(
94+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy> *Table);
95+
iterator(const iterator &Src) = default;
96+
iterator &operator=(const iterator &Src) = default;
97+
HashPair operator*();
98+
iterator &operator++();
99+
iterator &operator++(int);
100+
bool operator==(const iterator &Cmp) const;
101+
bool operator!=(const iterator &Cmp) const;
102+
103+
private:
104+
iterator(
105+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy> *Table,
106+
int Idx);
107+
friend HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>;
108+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy> *Table;
109+
int Idx;
110+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::HashEntry
111+
*Entry;
112+
};
113+
114+
// No erase or insert iterator supported
115+
iterator begin();
116+
iterator end();
117+
118+
private:
119+
void resize();
120+
82121
HashEntry **Table;
83122
u32 Capacity;
84123
u32 Entries;
@@ -247,4 +286,96 @@ void HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::unlock() {
247286
Mutex.Unlock();
248287
}
249288

289+
//===----------------------------------------------------------------------===//
290+
// Iterator implementation
291+
//===----------------------------------------------------------------------===//
292+
293+
template <typename KeyTy, typename DataTy, bool ExternalLock,
294+
typename HashFuncTy, typename EqualFuncTy>
295+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::iterator::
296+
iterator(
297+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy> *Table)
298+
: Table(Table), Idx(-1), Entry(nullptr) {
299+
operator++();
300+
}
301+
302+
template <typename KeyTy, typename DataTy, bool ExternalLock,
303+
typename HashFuncTy, typename EqualFuncTy>
304+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::iterator::
305+
iterator(
306+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy> *Table,
307+
int Idx)
308+
: Table(Table), Idx(Idx), Entry(nullptr) {
309+
CHECK(Idx >= (int)Table->Capacity); // Only used to create end().
310+
}
311+
312+
template <typename KeyTy, typename DataTy, bool ExternalLock,
313+
typename HashFuncTy, typename EqualFuncTy>
314+
typename HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy,
315+
EqualFuncTy>::HashPair
316+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::iterator::
317+
operator*() {
318+
CHECK(Idx >= 0 && Idx < (int)Table->Capacity);
319+
CHECK(Entry != nullptr);
320+
return HashPair(Entry->Key, Entry->Payload);
321+
}
322+
323+
template <typename KeyTy, typename DataTy, bool ExternalLock,
324+
typename HashFuncTy, typename EqualFuncTy>
325+
typename HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy,
326+
EqualFuncTy>::iterator &
327+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::iterator::
328+
operator++() {
329+
if (Entry != nullptr)
330+
Entry = Entry->Next;
331+
while (Entry == nullptr) {
332+
++Idx;
333+
if (Idx >= (int)Table->Capacity)
334+
break; // At end().
335+
Entry = Table->Table[Idx];
336+
}
337+
return *this;
338+
}
339+
340+
template <typename KeyTy, typename DataTy, bool ExternalLock,
341+
typename HashFuncTy, typename EqualFuncTy>
342+
typename HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy,
343+
EqualFuncTy>::iterator &
344+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::iterator::
345+
operator++(int) {
346+
iterator Temp(*this);
347+
operator++();
348+
return Temp;
349+
}
350+
351+
template <typename KeyTy, typename DataTy, bool ExternalLock,
352+
typename HashFuncTy, typename EqualFuncTy>
353+
bool HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::iterator::
354+
operator==(const iterator &Cmp) const {
355+
return Cmp.Table == Table && Cmp.Idx == Idx && Cmp.Entry == Entry;
356+
}
357+
358+
template <typename KeyTy, typename DataTy, bool ExternalLock,
359+
typename HashFuncTy, typename EqualFuncTy>
360+
bool HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::iterator::
361+
operator!=(const iterator &Cmp) const {
362+
return Cmp.Table != Table || Cmp.Idx != Idx || Cmp.Entry != Entry;
363+
}
364+
365+
template <typename KeyTy, typename DataTy, bool ExternalLock,
366+
typename HashFuncTy, typename EqualFuncTy>
367+
typename HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy,
368+
EqualFuncTy>::iterator
369+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::begin() {
370+
return iterator(this);
371+
}
372+
373+
template <typename KeyTy, typename DataTy, bool ExternalLock,
374+
typename HashFuncTy, typename EqualFuncTy>
375+
typename HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy,
376+
EqualFuncTy>::iterator
377+
HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::end() {
378+
return iterator(this, Capacity);
379+
}
380+
250381
} // namespace __esan

test/esan/Unit/hashtable.cpp

+45
Original file line numberDiff line numberDiff line change
@@ -56,16 +56,37 @@ int main()
5656
{
5757
__esan::HashTable<int, int> IntTable;
5858
assert(IntTable.size() == 0);
59+
60+
// Test iteration on an empty table.
61+
int Count = 0;
62+
for (auto Iter = IntTable.begin(); Iter != IntTable.end();
63+
++Iter, ++Count) {
64+
// Empty.
65+
}
66+
assert(Count == 0);
67+
5968
bool Added = IntTable.add(4, 42);
6069
assert(Added);
6170
assert(!IntTable.add(4, 42));
6271
assert(IntTable.size() == 1);
6372
int Value;
6473
bool Found = IntTable.lookup(4, Value);
6574
assert(Found && Value == 42);
75+
76+
// Test iterator.
77+
IntTable.lock();
78+
for (auto Iter = IntTable.begin(); Iter != IntTable.end();
79+
++Iter, ++Count) {
80+
assert((*Iter).Key == 4);
81+
assert((*Iter).Data == 42);
82+
}
83+
IntTable.unlock();
84+
assert(Count == 1);
85+
assert(Count == IntTable.size());
6686
assert(!IntTable.remove(5));
6787
assert(IntTable.remove(4));
6888

89+
// Test a more complex payload.
6990
__esan::HashTable<int, MyDataPayload> DataTable(4);
7091
MyDataPayload NewData(new MyData("mystring"));
7192
Added = DataTable.add(4, NewData);
@@ -86,6 +107,30 @@ int main()
86107
Found = DataTable.lookup(i+1, FoundData);
87108
assert(Found && strcmp(FoundData.Data->Buf, "delete-at-end") == 0);
88109
}
110+
DataTable.lock();
111+
Count = 0;
112+
for (auto Iter = DataTable.begin(); Iter != DataTable.end();
113+
++Iter, ++Count) {
114+
int Key = (*Iter).Key;
115+
FoundData = (*Iter).Data;
116+
assert(Key >= 1 && Key <= 4);
117+
assert(strcmp(FoundData.Data->Buf, "delete-at-end") == 0);
118+
}
119+
DataTable.unlock();
120+
assert(Count == 4);
121+
assert(Count == DataTable.size());
122+
123+
// Ensure the iterator supports a range-based for loop.
124+
DataTable.lock();
125+
Count = 0;
126+
for (auto Pair : DataTable) {
127+
assert(Pair.Key >= 1 && Pair.Key <= 4);
128+
assert(strcmp(Pair.Data.Data->Buf, "delete-at-end") == 0);
129+
++Count;
130+
}
131+
DataTable.unlock();
132+
assert(Count == 4);
133+
assert(Count == DataTable.size());
89134

90135
// Test payload freeing via smart pointer wrapper.
91136
__esan::HashTable<MyDataPayload, MyDataPayload, true> DataKeyTable;

0 commit comments

Comments
 (0)