@@ -66,19 +66,58 @@ class HashTable {
66
66
// If the table is internally-synchronized, this lock must not be held
67
67
// while a hashtable function is called as it will deadlock: the lock
68
68
// is not recursive. This is meant for use with externally-synchronized
69
- // tables.
69
+ // tables or with an iterator .
70
70
void lock ();
71
71
void unlock ();
72
72
73
73
private:
74
- void resize ();
75
-
76
74
struct HashEntry {
77
75
KeyTy Key;
78
76
DataTy Payload;
79
77
HashEntry *Next;
80
78
};
81
79
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
+
82
121
HashEntry **Table;
83
122
u32 Capacity;
84
123
u32 Entries;
@@ -247,4 +286,96 @@ void HashTable<KeyTy, DataTy, ExternalLock, HashFuncTy, EqualFuncTy>::unlock() {
247
286
Mutex.Unlock ();
248
287
}
249
288
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
+
250
381
} // namespace __esan
0 commit comments