• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===-- sanitizer_addrhashmap.h ---------------------------------*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // Concurrent uptr->T hashmap.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #ifndef SANITIZER_ADDRHASHMAP_H
15 #define SANITIZER_ADDRHASHMAP_H
16 
17 #include "sanitizer_common.h"
18 #include "sanitizer_mutex.h"
19 #include "sanitizer_atomic.h"
20 #include "sanitizer_allocator_internal.h"
21 
22 namespace __sanitizer {
23 
24 // Concurrent uptr->T hashmap.
25 // T must be a POD type, kSize is preferably a prime but can be any number.
26 // Usage example:
27 //
28 // typedef AddrHashMap<uptr, 11> Map;
29 // Map m;
30 // {
31 //   Map::Handle h(&m, addr);
32 //   use h.operator->() to access the data
33 //   if h.created() then the element was just created, and the current thread
34 //     has exclusive access to it
35 //   otherwise the current thread has only read access to the data
36 // }
37 // {
38 //   Map::Handle h(&m, addr, true);
39 //   this will remove the data from the map in Handle dtor
40 //   the current thread has exclusive access to the data
41 //   if !h.exists() then the element never existed
42 // }
43 template<typename T, uptr kSize>
44 class AddrHashMap {
45  private:
46   struct Cell {
47     atomic_uintptr_t addr;
48     T                val;
49   };
50 
51   struct AddBucket {
52     uptr cap;
53     uptr size;
54     Cell cells[1];  // variable len
55   };
56 
57   static const uptr kBucketSize = 3;
58 
59   struct Bucket {
60     RWMutex          mtx;
61     atomic_uintptr_t add;
62     Cell             cells[kBucketSize];
63   };
64 
65  public:
66   AddrHashMap();
67 
68   class Handle {
69    public:
70     Handle(AddrHashMap<T, kSize> *map, uptr addr);
71     Handle(AddrHashMap<T, kSize> *map, uptr addr, bool remove);
72     Handle(AddrHashMap<T, kSize> *map, uptr addr, bool remove, bool create);
73 
74     ~Handle();
75     T *operator->();
76     bool created() const;
77     bool exists() const;
78 
79    private:
80     friend AddrHashMap<T, kSize>;
81     AddrHashMap<T, kSize> *map_;
82     Bucket                *bucket_;
83     Cell                  *cell_;
84     uptr                   addr_;
85     uptr                   addidx_;
86     bool                   created_;
87     bool                   remove_;
88     bool                   create_;
89   };
90 
91  private:
92   friend class Handle;
93   Bucket *table_;
94 
95   void acquire(Handle *h);
96   void release(Handle *h);
97   uptr calcHash(uptr addr);
98 };
99 
100 template<typename T, uptr kSize>
Handle(AddrHashMap<T,kSize> * map,uptr addr)101 AddrHashMap<T, kSize>::Handle::Handle(AddrHashMap<T, kSize> *map, uptr addr) {
102   map_ = map;
103   addr_ = addr;
104   remove_ = false;
105   create_ = true;
106   map_->acquire(this);
107 }
108 
109 template<typename T, uptr kSize>
Handle(AddrHashMap<T,kSize> * map,uptr addr,bool remove)110 AddrHashMap<T, kSize>::Handle::Handle(AddrHashMap<T, kSize> *map, uptr addr,
111     bool remove) {
112   map_ = map;
113   addr_ = addr;
114   remove_ = remove;
115   create_ = true;
116   map_->acquire(this);
117 }
118 
119 template<typename T, uptr kSize>
Handle(AddrHashMap<T,kSize> * map,uptr addr,bool remove,bool create)120 AddrHashMap<T, kSize>::Handle::Handle(AddrHashMap<T, kSize> *map, uptr addr,
121     bool remove, bool create) {
122   map_ = map;
123   addr_ = addr;
124   remove_ = remove;
125   create_ = create;
126   map_->acquire(this);
127 }
128 
129 template<typename T, uptr kSize>
~Handle()130 AddrHashMap<T, kSize>::Handle::~Handle() {
131   map_->release(this);
132 }
133 
134 template <typename T, uptr kSize>
135 T *AddrHashMap<T, kSize>::Handle::operator->() {
136   return &cell_->val;
137 }
138 
139 template<typename T, uptr kSize>
created()140 bool AddrHashMap<T, kSize>::Handle::created() const {
141   return created_;
142 }
143 
144 template<typename T, uptr kSize>
exists()145 bool AddrHashMap<T, kSize>::Handle::exists() const {
146   return cell_ != nullptr;
147 }
148 
149 template<typename T, uptr kSize>
AddrHashMap()150 AddrHashMap<T, kSize>::AddrHashMap() {
151   table_ = (Bucket*)MmapOrDie(kSize * sizeof(table_[0]), "AddrHashMap");
152 }
153 
154 template<typename T, uptr kSize>
acquire(Handle * h)155 void AddrHashMap<T, kSize>::acquire(Handle *h) {
156   uptr addr = h->addr_;
157   uptr hash = calcHash(addr);
158   Bucket *b = &table_[hash];
159 
160   h->created_ = false;
161   h->addidx_ = -1U;
162   h->bucket_ = b;
163   h->cell_ = nullptr;
164 
165   // If we want to remove the element, we need exclusive access to the bucket,
166   // so skip the lock-free phase.
167   if (h->remove_)
168     goto locked;
169 
170  retry:
171   // First try to find an existing element w/o read mutex.
172   CHECK(!h->remove_);
173   // Check the embed cells.
174   for (uptr i = 0; i < kBucketSize; i++) {
175     Cell *c = &b->cells[i];
176     uptr addr1 = atomic_load(&c->addr, memory_order_acquire);
177     if (addr1 == addr) {
178       h->cell_ = c;
179       return;
180     }
181   }
182 
183   // Check the add cells with read lock.
184   if (atomic_load(&b->add, memory_order_relaxed)) {
185     b->mtx.ReadLock();
186     AddBucket *add = (AddBucket*)atomic_load(&b->add, memory_order_relaxed);
187     for (uptr i = 0; i < add->size; i++) {
188       Cell *c = &add->cells[i];
189       uptr addr1 = atomic_load(&c->addr, memory_order_relaxed);
190       if (addr1 == addr) {
191         h->addidx_ = i;
192         h->cell_ = c;
193         return;
194       }
195     }
196     b->mtx.ReadUnlock();
197   }
198 
199  locked:
200   // Re-check existence under write lock.
201   // Embed cells.
202   b->mtx.Lock();
203   for (uptr i = 0; i < kBucketSize; i++) {
204     Cell *c = &b->cells[i];
205     uptr addr1 = atomic_load(&c->addr, memory_order_relaxed);
206     if (addr1 == addr) {
207       if (h->remove_) {
208         h->cell_ = c;
209         return;
210       }
211       b->mtx.Unlock();
212       goto retry;
213     }
214   }
215 
216   // Add cells.
217   AddBucket *add = (AddBucket*)atomic_load(&b->add, memory_order_relaxed);
218   if (add) {
219     for (uptr i = 0; i < add->size; i++) {
220       Cell *c = &add->cells[i];
221       uptr addr1 = atomic_load(&c->addr, memory_order_relaxed);
222       if (addr1 == addr) {
223         if (h->remove_) {
224           h->addidx_ = i;
225           h->cell_ = c;
226           return;
227         }
228         b->mtx.Unlock();
229         goto retry;
230       }
231     }
232   }
233 
234   // The element does not exist, no need to create it if we want to remove.
235   if (h->remove_ || !h->create_) {
236     b->mtx.Unlock();
237     return;
238   }
239 
240   // Now try to create it under the mutex.
241   h->created_ = true;
242   // See if we have a free embed cell.
243   for (uptr i = 0; i < kBucketSize; i++) {
244     Cell *c = &b->cells[i];
245     uptr addr1 = atomic_load(&c->addr, memory_order_relaxed);
246     if (addr1 == 0) {
247       h->cell_ = c;
248       return;
249     }
250   }
251 
252   // Store in the add cells.
253   if (!add) {
254     // Allocate a new add array.
255     const uptr kInitSize = 64;
256     add = (AddBucket*)InternalAlloc(kInitSize);
257     internal_memset(add, 0, kInitSize);
258     add->cap = (kInitSize - sizeof(*add)) / sizeof(add->cells[0]) + 1;
259     add->size = 0;
260     atomic_store(&b->add, (uptr)add, memory_order_relaxed);
261   }
262   if (add->size == add->cap) {
263     // Grow existing add array.
264     uptr oldsize = sizeof(*add) + (add->cap - 1) * sizeof(add->cells[0]);
265     uptr newsize = oldsize * 2;
266     AddBucket *add1 = (AddBucket*)InternalAlloc(newsize);
267     internal_memset(add1, 0, newsize);
268     add1->cap = (newsize - sizeof(*add)) / sizeof(add->cells[0]) + 1;
269     add1->size = add->size;
270     internal_memcpy(add1->cells, add->cells, add->size * sizeof(add->cells[0]));
271     InternalFree(add);
272     atomic_store(&b->add, (uptr)add1, memory_order_relaxed);
273     add = add1;
274   }
275   // Store.
276   uptr i = add->size++;
277   Cell *c = &add->cells[i];
278   CHECK_EQ(atomic_load(&c->addr, memory_order_relaxed), 0);
279   h->addidx_ = i;
280   h->cell_ = c;
281 }
282 
283 template<typename T, uptr kSize>
release(Handle * h)284 void AddrHashMap<T, kSize>::release(Handle *h) {
285   if (!h->cell_)
286     return;
287   Bucket *b = h->bucket_;
288   Cell *c = h->cell_;
289   uptr addr1 = atomic_load(&c->addr, memory_order_relaxed);
290   if (h->created_) {
291     // Denote completion of insertion.
292     CHECK_EQ(addr1, 0);
293     // After the following store, the element becomes available
294     // for lock-free reads.
295     atomic_store(&c->addr, h->addr_, memory_order_release);
296     b->mtx.Unlock();
297   } else if (h->remove_) {
298     // Denote that the cell is empty now.
299     CHECK_EQ(addr1, h->addr_);
300     atomic_store(&c->addr, 0, memory_order_release);
301     // See if we need to compact the bucket.
302     AddBucket *add = (AddBucket*)atomic_load(&b->add, memory_order_relaxed);
303     if (h->addidx_ == -1U) {
304       // Removed from embed array, move an add element into the freed cell.
305       if (add && add->size != 0) {
306         uptr last = --add->size;
307         Cell *c1 = &add->cells[last];
308         c->val = c1->val;
309         uptr addr1 = atomic_load(&c1->addr, memory_order_relaxed);
310         atomic_store(&c->addr, addr1, memory_order_release);
311         atomic_store(&c1->addr, 0, memory_order_release);
312       }
313     } else {
314       // Removed from add array, compact it.
315       uptr last = --add->size;
316       Cell *c1 = &add->cells[last];
317       if (c != c1) {
318         *c = *c1;
319         atomic_store(&c1->addr, 0, memory_order_relaxed);
320       }
321     }
322     if (add && add->size == 0) {
323       // FIXME(dvyukov): free add?
324     }
325     b->mtx.Unlock();
326   } else {
327     CHECK_EQ(addr1, h->addr_);
328     if (h->addidx_ != -1U)
329       b->mtx.ReadUnlock();
330   }
331 }
332 
333 template<typename T, uptr kSize>
calcHash(uptr addr)334 uptr AddrHashMap<T, kSize>::calcHash(uptr addr) {
335   addr += addr << 10;
336   addr ^= addr >> 6;
337   return addr % kSize;
338 }
339 
340 } // namespace __sanitizer
341 
342 #endif // SANITIZER_ADDRHASHMAP_H
343