P4C
The P4 Compiler
Loading...
Searching...
No Matches
hvec_map.h
1/*
2Copyright 2023-present Intel
3
4Licensed under the Apache License, Version 2.0 (the "License");
5you may not use this file except in compliance with the License.
6You may obtain a copy of the License at
7
8 http://www.apache.org/licenses/LICENSE-2.0
9
10Unless required by applicable law or agreed to in writing, software
11distributed under the License is distributed on an "AS IS" BASIS,
12WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13See the License for the specific language governing permissions and
14limitations under the License.
15*/
16
17#ifndef LIB_HVEC_MAP_H_
18#define LIB_HVEC_MAP_H_
19
20#include <initializer_list>
21#include <tuple>
22#include <vector>
23
24#include "exceptions.h"
25#include "hashvec.h"
26
27template <class KEY, class VAL, class HASH = std::hash<KEY>, class PRED = std::equal_to<KEY>,
28 class ALLOC = std::allocator<std::pair<const KEY, VAL>>>
30 HASH hf; // FIXME -- use empty base optimization for these?
31 PRED eql;
32
33 public:
34 typedef KEY key_type;
35 typedef std::pair<const KEY, VAL> value_type;
36 typedef VAL mapped_type;
37 typedef HASH hasher;
38 typedef PRED key_equal;
39 typedef ALLOC allocator_type;
40
41 typedef typename std::vector<value_type>::pointer pointer;
42 typedef typename std::vector<value_type>::const_pointer const_pointer;
43 typedef typename std::vector<value_type>::reference reference;
44 typedef typename std::vector<value_type>::const_reference const_reference;
45
47
48 explicit hvec_map(size_t icap = 0, const hasher &hf = hasher(),
49 const key_equal &eql = key_equal(),
50 const allocator_type &a = allocator_type())
51 : hash_vector_base(true, false, icap), hf(hf), eql(eql), data(a) {
52 data.reserve(icap);
53 }
54 hvec_map(const hvec_map &) = default;
55 hvec_map(hvec_map &&) = default;
56 hvec_map &operator=(const hvec_map &that) {
57 if (this != std::addressof(that)) {
58 clear();
59 hf = that.hf;
60 eql = that.eql;
61
62 data.reserve(that.size());
63 insert(that.begin(), that.end());
64 }
65
66 return *this;
67 }
68 hvec_map &operator=(hvec_map &&) = default;
69 ~hvec_map() = default;
70 template <class ITER>
71 hvec_map(ITER begin, ITER end, const hasher &hf = hasher(), const key_equal &eql = key_equal(),
72 const allocator_type &a = allocator_type())
73 : hash_vector_base(true, false, end - begin), hf(hf), eql(eql), data(a) {
74 data.reserve(end - begin);
75 for (auto it = begin; it != end; ++it) insert(*it);
76 }
77 hvec_map(std::initializer_list<value_type> il, const hasher &hf = hasher(),
78 const key_equal &eql = key_equal(), const allocator_type &a = allocator_type())
79 : hash_vector_base(true, false, il.size()), hf(hf), eql(eql), data(a) {
80 data.reserve(il.size());
81 for (auto &i : il) insert(i);
82 }
83
84 private:
85 template <class HVM, class VT>
86 class _iter {
87 HVM *self;
88 size_t idx;
89
90 friend class hvec_map;
91 _iter(HVM &s, size_t i) : self(&s), idx(i) {}
92
93 public:
94 using value_type = VT;
95 using difference_type = ssize_t;
96 using pointer = typename HVM::pointer;
97 using reference = typename HVM::reference;
98 using iterator_category = std::bidirectional_iterator_tag;
99 _iter(const _iter &) = default;
100 _iter &operator=(const _iter &a) {
101 self = a.self;
102 idx = a.idx;
103 return *this;
104 }
105 value_type &operator*() const { return self->data[idx]; }
106 value_type *operator->() const { return &self->data[idx]; }
107 _iter &operator++() {
108 do {
109 ++idx;
110 } while (self->erased[idx]);
111 return *this;
112 }
113 _iter &operator--() {
114 do {
115 --idx;
116 } while (self->erased[idx]);
117 return *this;
118 }
119 _iter operator++(int) {
120 auto copy = *this;
121 ++*this;
122 return copy;
123 }
124 _iter operator--(int) {
125 auto copy = *this;
126 --*this;
127 return copy;
128 }
129 bool operator==(const _iter &a) const { return self == a.self && idx == a.idx; }
130 bool operator!=(const _iter &a) const { return self != a.self || idx != a.idx; }
131 operator _iter<const HVM, const VT>() const {
132 return _iter<const HVM, const VT>(*self, idx);
133 }
134 };
135
136 public:
137 typedef _iter<hvec_map, value_type> iterator;
138 typedef _iter<const hvec_map, const value_type> const_iterator;
139 iterator begin() { return iterator(*this, erased.ffz()); }
140 iterator end() { return iterator(*this, data.size()); }
141 const_iterator begin() const { return const_iterator(*this, erased.ffz()); }
142 const_iterator end() const { return const_iterator(*this, data.size()); }
143 const_iterator cbegin() const { return const_iterator(*this, erased.ffz()); }
144 const_iterator cend() const { return const_iterator(*this, data.size()); }
145
146 bool empty() const { return inuse == 0; }
147 size_t size() const { return inuse; }
148 size_t max_size() const { return UINT32_MAX; }
149 bool operator==(const hvec_map &a) const {
150 if (inuse != a.inuse) return false;
151 auto it = begin();
152 for (auto &el : a)
153 if (el != *it++) return false;
154 return true;
155 }
156 bool operator!=(const hvec_map &a) const { return !(*this == a); }
157 void clear() {
158 hash_vector_base::clear();
159 data.clear();
160 }
161
162 iterator find(const KEY &k) {
164 size_t idx = hash_vector_base::find(&k, &cache);
165 return idx ? iterator(*this, idx - 1) : end();
166 }
167 const_iterator find(const KEY &k) const {
169 size_t idx = hash_vector_base::find(&k, &cache);
170 return idx ? const_iterator(*this, idx - 1) : end();
171 }
172 size_t count(const KEY &k) const {
174 size_t idx = hash_vector_base::find(&k, &cache);
175 return idx > 0;
176 }
177
178 VAL &at(const KEY &k) {
180 size_t idx = hash_vector_base::find(&k, &cache);
181 if (!idx || erased[idx - 1]) throw std::out_of_range("hvec_map::at");
182 return data[idx - 1].second;
183 }
184 const VAL &at(const KEY &k) const {
186 size_t idx = hash_vector_base::find(&k, &cache);
187 if (!idx || erased[idx - 1]) throw std::out_of_range("hvec_map::at");
188 return data[idx - 1].second;
189 }
190
191 // FIXME -- how to do this without duplicating the code for lvalue/rvalue?
192 VAL &operator[](const KEY &k) {
193 size_t idx = hv_insert(&k);
194 if (idx >= data.size()) {
195 data.emplace_back(k, VAL());
196 return data.back().second;
197 } else {
198 if (erased[idx]) {
199 erased[idx] = 0;
200 const_cast<KEY &>(data[idx].first) = k;
201 data[idx].second = VAL();
202 }
203 return data[idx].second;
204 }
205 }
206 VAL &operator[](KEY &&k) {
207 size_t idx = hv_insert(&k);
208 if (idx >= data.size()) {
209 data.emplace_back(std::move(k), VAL());
210 return data.back().second;
211 } else {
212 if (erased[idx]) {
213 erased[idx] = 0;
214 const_cast<KEY &>(data[idx].first) = std::move(k);
215 data[idx].second = VAL();
216 }
217 return data[idx].second;
218 }
219 }
220
221 // FIXME -- how to do this without duplicating the code for lvalue/rvalue?
222 template <typename... VV>
223 std::pair<iterator, bool> emplace(const KEY &k, VV &&...v) {
224 bool new_key = false;
225 size_t idx = hv_insert(&k);
226 if (idx >= data.size()) {
227 idx = data.size();
228 data.emplace_back(std::piecewise_construct_t(), std::forward_as_tuple(k),
229 std::forward_as_tuple(std::forward<VV>(v)...));
230 new_key = true;
231 } else {
232 if ((new_key = erased[idx])) {
233 erased[idx] = 0;
234 const_cast<KEY &>(data[idx].first) = k;
235 data[idx].second = VAL(std::forward<VV>(v)...);
236 } else {
237 data[idx].second = VAL(std::forward<VV>(v)...);
238 }
239 }
240 return std::make_pair(iterator(*this, idx), new_key);
241 }
242 template <typename... VV>
243 std::pair<iterator, bool> emplace(KEY &&k, VV &&...v) {
244 bool new_key = false;
245 size_t idx = hv_insert(&k);
246 if (idx >= data.size()) {
247 idx = data.size();
248 data.emplace_back(std::piecewise_construct_t(), std::forward_as_tuple(std::move(k)),
249 std::forward_as_tuple(std::forward<VV>(v)...));
250 new_key = true;
251 } else if ((new_key = erased[idx])) {
252 erased[idx] = 0;
253 const_cast<KEY &>(data[idx].first) = std::move(k);
254 data[idx].second = VAL(std::forward<VV>(v)...);
255 }
256 return std::make_pair(iterator(*this, idx), new_key);
257 }
258 std::pair<iterator, bool> insert(const value_type &v) {
259 bool new_key = false;
260 size_t idx = hv_insert(&v.first);
261 if (idx >= data.size()) {
262 idx = data.size();
263 data.push_back(v);
264 new_key = true;
265 } else if ((new_key = erased[idx])) {
266 erased[idx] = 0;
267 const_cast<KEY &>(data[idx].first) = v.first;
268 data[idx].second = v.second;
269 }
270 return std::make_pair(iterator(*this, idx), new_key);
271 }
272 template <typename InputIterator>
273 void insert(InputIterator first, InputIterator last) {
274 for (; first != last; ++first) insert(*first);
275 }
276 void insert(std::initializer_list<value_type> vl) { return insert(vl.begin(), vl.end()); }
277 template <class HVM, class VT>
278 _iter<HVM, VT> erase(_iter<HVM, VT> it) {
279 BUG_CHECK(this == it.self, "incorrect iterator for hvec_map::erase");
280 erased[it.idx] = 1;
281 // FIXME -- would be better to call dtor here, but that will cause
282 // problems with the vector when it resized or is destroyed. Could
283 // use raw memory and manual construct instead.
284 const_cast<KEY &>(data[it.idx].first) = KEY();
285 data[it.idx].second = VAL();
286 ++it;
287 --inuse;
288 return it;
289 }
290 size_t erase(const KEY &k) {
291 size_t idx = remove(&k);
292 if (idx + 1 == 0) return 0;
293 if (idx < data.size()) {
294 const_cast<KEY &>(data[idx].first) = KEY();
295 data[idx].second = VAL();
296 }
297 return 1;
298 }
299#ifdef DEBUG
300 using hash_vector_base::dump;
301#endif
302
303 private:
304 std::vector<value_type, ALLOC> data;
305 size_t hashfn(const void *a) const override { return hf(*static_cast<const KEY *>(a)); }
306 bool cmpfn(const void *a, const void *b) const override {
307 return eql(*static_cast<const KEY *>(a), *static_cast<const KEY *>(b));
308 }
309 bool cmpfn(const void *a, size_t b) const override {
310 return eql(*static_cast<const KEY *>(a), data[b].first);
311 }
312 const void *getkey(uint32_t i) const override { return &data[i].first; }
313 void *getval(uint32_t i) override { return &data[i].second; }
314 uint32_t limit() override { return data.size(); }
315 void resizedata(size_t sz) override { data.resize(sz); }
316 void moveentry(size_t to, size_t from) override {
317 // can't just assign, as the keys are const -- need to destruct & construct
318 data[to].~value_type();
319 new (&data[to]) value_type(std::move(data[from]));
320 }
321};
322
323template <class K, class T, class V, class Comp, class Alloc>
324inline V get(const hvec_map<K, V, Comp, Alloc> &m, T key, V def = V()) {
325 auto it = m.find(key);
326 if (it != m.end()) return it->second;
327 return def;
328}
329
330template <class K, class T, class V, class Comp, class Alloc>
331inline V *getref(hvec_map<K, V, Comp, Alloc> &m, T key) {
332 auto it = m.find(key);
333 if (it != m.end()) return &it->second;
334 return 0;
335}
336
337template <class K, class T, class V, class Comp, class Alloc>
338inline const V *getref(const hvec_map<K, V, Comp, Alloc> &m, T key) {
339 auto it = m.find(key);
340 if (it != m.end()) return &it->second;
341 return 0;
342}
343
344template <class K, class T, class V, class Comp, class Alloc>
345inline V get(const hvec_map<K, V, Comp, Alloc> *m, T key, V def = V()) {
346 return m ? get(*m, key, def) : def;
347}
348
349template <class K, class T, class V, class Comp, class Alloc>
350inline V *getref(hvec_map<K, V, Comp, Alloc> *m, T key) {
351 return m ? getref(*m, key) : 0;
352}
353
354template <class K, class T, class V, class Comp, class Alloc>
355inline const V *getref(const hvec_map<K, V, Comp, Alloc> *m, T key) {
356 return m ? getref(*m, key) : 0;
357}
358
359#endif /* LIB_HVEC_MAP_H_ */
Definition hashvec.h:27
Definition hvec_map.h:29
Definition hashvec.h:49