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