P4C
The P4 Compiler
Loading...
Searching...
No Matches
ordered_set.h
1/*
2Copyright 2013-present Barefoot Networks, Inc.
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_ORDERED_SET_H_
18#define LIB_ORDERED_SET_H_
19
20#include <cassert>
21#include <functional>
22#include <initializer_list>
23#include <list>
24#include <map>
25#include <set>
26#include <utility>
27
28namespace P4 {
29
30// Remembers items in insertion order
31template <class T, class COMP = std::less<T>, class ALLOC = std::allocator<T>>
33 public:
34 typedef T key_type;
35 typedef T value_type;
36 typedef COMP key_compare;
37 typedef COMP value_compare;
38 typedef ALLOC allocator_type;
39 typedef const T &reference;
40 typedef const T &const_reference;
41
42 private:
43 typedef std::list<T, ALLOC> list_type;
44 typedef typename list_type::iterator list_iterator;
45 list_type data;
46
47 public:
48 // This is a set, so we can't provide any iterator that would make it
49 // possible to modify the values in the set and therefore possibly make the
50 // set contain duplicate values. Therefore we base all our iterators on
51 // list's const_iterator.
52 typedef typename list_type::const_iterator iterator;
53 typedef typename list_type::const_iterator const_iterator;
54 typedef std::reverse_iterator<iterator> reverse_iterator;
55 typedef std::reverse_iterator<const_iterator> const_reverse_iterator;
56
57 private:
58 struct mapcmp {
59 COMP comp;
60 bool operator()(const T *a, const T *b) const { return comp(*a, *b); }
61 };
62 using map_alloc = typename std::allocator_traits<ALLOC>::template rebind_alloc<
63 std::pair<const T *const, list_iterator>>;
64 using map_type = std::map<const T *, list_iterator, mapcmp, map_alloc>;
65 map_type data_map;
66 void init_data_map() {
67 data_map.clear();
68 for (auto it = data.begin(); it != data.end(); it++) data_map.emplace(&*it, it);
69 }
70 iterator tr_iter(typename map_type::iterator i) {
71 if (i == data_map.end()) return data.end();
72 return i->second;
73 }
74 const_iterator tr_iter(typename map_type::const_iterator i) const {
75 if (i == data_map.end()) return data.end();
76 return i->second;
77 }
78
79 public:
80 typedef typename map_type::size_type size_type;
81 class sorted_iterator : public std::iterator<std::bidirectional_iterator_tag, T> {
82 friend class ordered_set;
83 typename map_type::const_iterator iter;
84 sorted_iterator(typename map_type::const_iterator it) // NOLINT(runtime/explicit)
85 : iter(it) {}
86
87 public:
88 const T &operator*() const { return *iter->first; }
89 const T *operator->() const { return iter->first; }
90 sorted_iterator operator++() {
91 ++iter;
92 return *this;
93 }
94 sorted_iterator operator--() {
95 --iter;
96 return *this;
97 }
98 sorted_iterator operator++(int) {
99 auto copy = *this;
100 ++iter;
101 return copy;
102 }
103 sorted_iterator operator--(int) {
104 auto copy = *this;
105 --iter;
106 return copy;
107 }
108 bool operator==(const sorted_iterator i) const { return iter == i.iter; }
109 bool operator!=(const sorted_iterator i) const { return iter != i.iter; }
110 };
111
112 ordered_set() {}
113 ordered_set(const ordered_set &a) : data(a.data) { init_data_map(); }
114 ordered_set(std::initializer_list<T> init) : data(init) { init_data_map(); }
115 template <typename InputIt>
116 ordered_set(InputIt first, InputIt last) : data(first, last) {
117 init_data_map();
118 }
119 ordered_set(ordered_set &&a) = default; /* move is ok? */
120 ordered_set &operator=(const ordered_set &a) {
121 data = a.data;
122 init_data_map();
123 return *this;
124 }
125 ordered_set &operator=(ordered_set &&a) = default; /* move is ok? */
126 bool operator==(const ordered_set &a) const { return data == a.data; }
127 bool operator!=(const ordered_set &a) const { return data != a.data; }
128 bool operator<(const ordered_set &a) const {
129 // we define this to work INDEPENDENT of the order -- so it is possible to have
130 // two ordered_sets where !(a < b) && !(b < a) && !(a == b) -- such sets have the
131 // same elements but in a different order. This is generally what you want if you
132 // have a set of ordered_sets (or use ordered_set as a map key).
133 auto it = a.data_map.begin();
134 for (auto &el : data_map) {
135 if (it == a.data_map.end()) return false;
136 if (mapcmp()(el.first, it->first)) return true;
137 if (mapcmp()(it->first, el.first)) return false;
138 ++it;
139 }
140 return it != a.data_map.end();
141 }
142
143 // FIXME add allocator and comparator ctors...
144
145 iterator begin() noexcept { return data.begin(); }
146 const_iterator begin() const noexcept { return data.begin(); }
147 iterator end() noexcept { return data.end(); }
148 const_iterator end() const noexcept { return data.end(); }
149 reverse_iterator rbegin() noexcept { return data.rbegin(); }
150 const_reverse_iterator rbegin() const noexcept { return data.rbegin(); }
151 reverse_iterator rend() noexcept { return data.rend(); }
152 const_reverse_iterator rend() const noexcept { return data.rend(); }
153 const_iterator cbegin() const noexcept { return data.cbegin(); }
154 const_iterator cend() const noexcept { return data.cend(); }
155 const_reverse_iterator crbegin() const noexcept { return data.crbegin(); }
156 const_reverse_iterator crend() const noexcept { return data.crend(); }
157 sorted_iterator sorted_begin() const noexcept { return data_map.begin(); }
158 sorted_iterator sorted_end() const noexcept { return data_map.end(); }
159
160 reference front() const noexcept { return *data.begin(); }
161 reference back() const noexcept { return *data.rbegin(); }
162
163 bool empty() const noexcept { return data.empty(); }
164 size_type size() const noexcept { return data_map.size(); }
165 size_type max_size() const noexcept { return data_map.max_size(); }
166 void clear() {
167 data.clear();
168 data_map.clear();
169 }
170
171 iterator find(const T &a) { return tr_iter(data_map.find(&a)); }
172 const_iterator find(const T &a) const { return tr_iter(data_map.find(&a)); }
173 size_type count(const T &a) const { return data_map.count(&a); }
174 iterator upper_bound(const T &a) { return tr_iter(data_map.upper_bound(&a)); }
175 const_iterator upper_bound(const T &a) const { return tr_iter(data_map.upper_bound(&a)); }
176 iterator lower_bound(const T &a) { return tr_iter(data_map.lower_bound(&a)); }
177 const_iterator lower_bound(const T &a) const { return tr_iter(data_map.lower_bound(&a)); }
178
179 std::pair<iterator, bool> insert(const T &v) {
180 iterator it = find(v);
181 if (it == data.end()) {
182 list_iterator it = data.insert(data.end(), v);
183 data_map.emplace(&*it, it);
184 return std::make_pair(it, true);
185 }
186 return std::make_pair(it, false);
187 }
188 std::pair<iterator, bool> insert(T &&v) {
189 iterator it = find(v);
190 if (it == data.end()) {
191 list_iterator it = data.insert(data.end(), std::move(v));
192 data_map.emplace(&*it, it);
193 return std::make_pair(it, true);
194 }
195 return std::make_pair(it, false);
196 }
197 void insert(ordered_set::const_iterator begin, ordered_set::const_iterator end) {
198 for (auto it = begin; it != end; ++it) insert(*it);
199 }
200 iterator insert(const_iterator pos, const T &v) {
201 iterator it = find(v);
202 if (it == data.end()) {
203 list_iterator it = data.insert(pos, v);
204 data_map.emplace(&*it, it);
205 return it;
206 }
207 return it;
208 }
209 iterator insert(const_iterator pos, T &&v) {
210 iterator it = find(v);
211 if (it == data.end()) {
212 list_iterator it = data.insert(pos, std::move(v));
213 data_map.emplace(&*it, it);
214 return it;
215 }
216 return it;
217 }
218
219 void push_back(const T &v) {
220 iterator it = find(v);
221 if (it == data.end()) {
222 list_iterator it = data.insert(data.end(), v);
223 data_map.emplace(&*it, it);
224 } else {
225 data.splice(data.end(), data, it);
226 }
227 }
228 void push_back(T &&v) {
229 iterator it = find(v);
230 if (it == data.end()) {
231 list_iterator it = data.insert(data.end(), std::move(v));
232 data_map.emplace(&*it, it);
233 } else {
234 data.splice(data.end(), data, it);
235 }
236 }
237
238 template <class... Args>
239 std::pair<iterator, bool> emplace(Args &&...args) {
240 auto it = data.emplace(data.end(), std::forward<Args>(args)...);
241 auto old = find(*it);
242 if (old == data.end()) {
243 data_map.emplace(&*it, it);
244 return std::make_pair(it, true);
245 } else {
246 data.erase(it);
247 return std::make_pair(old, false);
248 }
249 }
250
251 template <class... Args>
252 std::pair<iterator, bool> emplace_back(Args &&...args) {
253 auto it = data.emplace(data.end(), std::forward<Args>(args)...);
254 auto old = find(*it);
255 if (old != data.end()) {
256 data.erase(old);
257 }
258 data_map.emplace(&*it, it);
259 return std::make_pair(it, true);
260 }
261
262 iterator erase(const_iterator pos) {
263 auto it = data_map.find(&*pos);
264 assert(it != data_map.end());
265 // get the non-const std::list iterator -- libstdc++ is missing
266 // std::list::erase(const_iterator) overload
267 auto list_it = it->second;
268 data_map.erase(it);
269 return data.erase(list_it);
270 }
271 size_type erase(const T &v) {
272 auto it = find(v);
273 if (it != data.end()) {
274 data_map.erase(&v);
275 data.erase(it);
276 return 1;
277 }
278 return 0;
279 }
280};
281
282template <class T, class C1, class A1, class U>
283inline auto operator|=(ordered_set<T, C1, A1> &a, const U &b) -> decltype(b.begin(), a) {
284 for (auto &el : b) a.insert(el);
285 return a;
286}
287template <class T, class C1, class A1, class U>
288inline auto operator-=(ordered_set<T, C1, A1> &a, const U &b) -> decltype(b.begin(), a) {
289 for (auto &el : b) a.erase(el);
290 return a;
291}
292template <class T, class C1, class A1, class U>
293inline auto operator&=(ordered_set<T, C1, A1> &a, const U &b) -> decltype(b.begin(), a) {
294 for (auto it = a.begin(); it != a.end();) {
295 if (b.count(*it))
296 ++it;
297 else
298 it = a.erase(it);
299 }
300 return a;
301}
302
303template <class T, class C1, class A1, class U>
304inline auto contains(const ordered_set<T, C1, A1> &a, const U &b) -> decltype(b.begin(), true) {
305 for (auto &el : b)
306 if (!a.count(el)) return false;
307 return true;
308}
309template <class T, class C1, class A1, class U>
310inline auto intersects(const ordered_set<T, C1, A1> &a, const U &b) -> decltype(b.begin(), true) {
311 for (auto &el : b)
312 if (a.count(el)) return true;
313 return false;
314}
315
316} // namespace P4
317
318#endif /* LIB_ORDERED_SET_H_ */
Definition ordered_set.h:81
Definition ordered_set.h:32
TODO: this is not really specific to BMV2, it should reside somewhere else.
Definition applyOptionsPragmas.cpp:24