P4C
The P4 Compiler
Loading...
Searching...
No Matches
hvec_set.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_SET_H_
18#define LIB_HVEC_SET_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 HASH = std::hash<KEY>, class PRED = std::equal_to<KEY>,
28 class ALLOC = std::allocator<KEY>>
30 HASH hf; // FIXME -- use empty base optimization for these?
31 PRED eql;
32
33 public:
34 typedef const KEY value_type;
35 typedef HASH hasher;
36 typedef PRED key_equal;
37 typedef ALLOC allocator_type;
38 typedef value_type *pointer, *const_pointer, &reference, &const_reference;
40
41 explicit hvec_set(size_t icap = 0, const hasher &hf = hasher(),
42 const key_equal &eql = key_equal(),
43 const allocator_type &a = allocator_type())
44 : hash_vector_base(false, false, icap), hf(hf), eql(eql), data(a) {
45 data.reserve(icap);
46 }
47 hvec_set(const hvec_set &) = default;
48 hvec_set(hvec_set &&) = default;
49 hvec_set &operator=(const hvec_set &that) {
50 if (this != std::addressof(that)) {
51 clear();
52 hf = that.hf;
53 eql = that.eql;
54
55 data.reserve(that.size());
56 insert(that.begin(), that.end());
57 }
58
59 return *this;
60 }
61 hvec_set &operator=(hvec_set &&) = default;
62 ~hvec_set() = default;
63 template <class ITER>
64 hvec_set(ITER begin, ITER end, const hasher &hf = hasher(), const key_equal &eql = key_equal(),
65 const allocator_type &a = allocator_type())
66 : hash_vector_base(false, false, end - begin), hf(hf), eql(eql), data(a) {
67 data.reserve(end - begin);
68 for (auto it = begin; it != end; ++it) insert(*it);
69 }
70 hvec_set(std::initializer_list<value_type> il, const hasher &hf = hasher(),
71 const key_equal &eql = key_equal(), const allocator_type &a = allocator_type())
72 : hash_vector_base(false, false, il.size()), hf(hf), eql(eql), data(a) {
73 data.reserve(il.size());
74 for (auto &i : il) insert(i);
75 }
76
77 private:
78 template <class HVM, class VT>
79 class _iter {
80 HVM *self;
81 size_t idx;
82
83 friend class hvec_set;
84 _iter(HVM &s, size_t i) : self(&s), idx(i) {}
85
86 public:
87 using value_type = VT;
88 using difference_type = ssize_t;
89 using pointer = typename HVM::pointer;
90 using reference = typename HVM::reference;
91 using iterator_category = std::bidirectional_iterator_tag;
92 _iter(const _iter &) = default;
93 _iter &operator=(const _iter &a) {
94 self = a.self;
95 idx = a.idx;
96 return *this;
97 }
98 value_type &operator*() const { return self->data[idx]; }
99 value_type *operator->() const { return &self->data[idx]; }
100 _iter &operator++() {
101 do {
102 ++idx;
103 } while (self->erased[idx]);
104 return *this;
105 }
106 _iter &operator--() {
107 do {
108 --idx;
109 } while (self->erased[idx]);
110 return *this;
111 }
112 _iter operator++(int) {
113 auto copy = *this;
114 ++*this;
115 return copy;
116 }
117 _iter operator--(int) {
118 auto copy = *this;
119 --*this;
120 return copy;
121 }
122 bool operator==(const _iter &a) const { return self == a.self && idx == a.idx; }
123 bool operator!=(const _iter &a) const { return self != a.self || idx != a.idx; }
124 operator _iter<const HVM, const VT>() const {
125 return _iter<const HVM, const VT>(*self, idx);
126 }
127 };
128
129 public:
130 typedef _iter<hvec_set, value_type> iterator;
131 typedef _iter<const hvec_set, const value_type> const_iterator;
132 iterator begin() { return iterator(*this, erased.ffz()); }
133 iterator end() { return iterator(*this, data.size()); }
134 const_iterator begin() const { return const_iterator(*this, erased.ffz()); }
135 const_iterator end() const { return const_iterator(*this, data.size()); }
136 const_iterator cbegin() const { return const_iterator(*this, erased.ffz()); }
137 const_iterator cend() const { return const_iterator(*this, data.size()); }
138 value_type &front() const { return *begin(); }
139 value_type &back() const {
140 auto it = end();
141 return *--it;
142 }
143
144 bool empty() const { return inuse == 0; }
145 size_t size() const { return inuse; }
146 size_t max_size() const { return UINT32_MAX; }
147 bool operator==(const hvec_set &a) const {
148 if (inuse != a.inuse) return false;
149 auto it = begin();
150 for (auto &el : a)
151 if (el != *it++) return false;
152 return true;
153 }
154 bool operator!=(const hvec_set &a) const { return !(*this == a); }
155 void clear() {
156 hash_vector_base::clear();
157 data.clear();
158 }
159
160 iterator find(const KEY &k) {
162 size_t idx = hash_vector_base::find(&k, &cache);
163 return idx ? iterator(*this, idx - 1) : end();
164 }
165 const_iterator find(const KEY &k) const {
167 size_t idx = hash_vector_base::find(&k, &cache);
168 return idx ? const_iterator(*this, idx - 1) : end();
169 }
170 size_t count(const KEY &k) const {
172 size_t idx = hash_vector_base::find(&k, &cache);
173 return idx > 0;
174 }
175
176 template <typename... KK>
177 std::pair<iterator, bool> emplace(KK &&...k) {
178 return insert(value_type(std::forward<KK>(k)...));
179 }
180 std::pair<iterator, bool> insert(const value_type &v) {
181 bool new_key = false;
182 size_t idx = hv_insert(&v);
183 if (idx >= data.size()) {
184 idx = data.size();
185 data.push_back(v);
186 new_key = true;
187 } else if ((new_key = erased[idx])) {
188 erased[idx] = 0;
189 data[idx] = v;
190 }
191 return std::make_pair(iterator(*this, idx), new_key);
192 }
193 std::pair<iterator, bool> insert(value_type &&v) {
194 bool new_key = false;
195 size_t idx = hv_insert(&v);
196 if (idx >= data.size()) {
197 idx = data.size();
198 data.push_back(std::move(v));
199 new_key = true;
200 } else if ((new_key = erased[idx])) {
201 erased[idx] = 0;
202 data[idx] = std::move(v);
203 }
204 return std::make_pair(iterator(*this, idx), new_key);
205 }
206
207 template <typename InputIterator>
208 void insert(InputIterator first, InputIterator last) {
209 for (; first != last; ++first) insert(*first);
210 }
211 void insert(std::initializer_list<value_type> vl) { return insert(vl.begin(), vl.end()); }
212 template <class HVM, class VT>
213 _iter<HVM, VT> erase(_iter<HVM, VT> it) {
214 BUG_CHECK(this == it.self, "incorrect iterator for hvec_set::erase");
215 erased[it.idx] = 1;
216 // FIXME -- would be better to call dtor here, but that will cause
217 // problems with the vector when it is resized or destroyed. Could
218 // use raw memory and manual construct instead.
219 data[it.idx] = KEY();
220 ++it;
221 --inuse;
222 return it;
223 }
224 size_t erase(const KEY &k) {
225 size_t idx = remove(&k);
226 if (idx + 1 == 0) return 0;
227 if (idx < data.size()) {
228 data[idx] = KEY();
229 }
230 return 1;
231 }
232#ifdef DEBUG
233 using hash_vector_base::dump;
234#endif
235
236 private:
237 std::vector<KEY, ALLOC> data;
238 size_t hashfn(const void *a) const override { return hf(*static_cast<const KEY *>(a)); }
239 bool cmpfn(const void *a, const void *b) const override {
240 return eql(*static_cast<const KEY *>(a), *static_cast<const KEY *>(b));
241 }
242 bool cmpfn(const void *a, size_t b) const override {
243 return eql(*static_cast<const KEY *>(a), data[b]);
244 }
245 const void *getkey(uint32_t i) const override { return &data[i]; }
246 void *getval(uint32_t) override {
247 BUG("getval in set");
248 return nullptr;
249 }
250 uint32_t limit() override { return data.size(); }
251 void resizedata(size_t sz) override { data.resize(sz); }
252 void moveentry(size_t to, size_t from) override { data[to] = data[from]; }
253};
254
255#endif /* LIB_HVEC_SET_H_ */
Definition hashvec.h:27
Definition hvec_set.h:29
Definition hashvec.h:49