P4C
The P4 Compiler
All Classes Namespaces Functions Variables Typedefs Enumerations Enumerator Friends Modules Pages
ordered_map.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_MAP_H_
18#define LIB_ORDERED_MAP_H_
19
20#include <cassert>
21#include <functional>
22#include <initializer_list>
23#include <list>
24#include <map>
25#include <utility>
26
27// Map is ordered by order of element insertion.
28template <class K, class V, class COMP = std::less<K>,
29 class ALLOC = std::allocator<std::pair<const K, V>>>
31 public:
32 typedef K key_type;
33 typedef V mapped_type;
34 typedef std::pair<const K, V> value_type;
35 typedef COMP key_compare;
36 typedef ALLOC allocator_type;
37 typedef value_type &reference;
38 typedef const value_type &const_reference;
39
40 private:
41 typedef std::list<value_type, ALLOC> list_type;
42 typedef typename list_type::iterator list_iterator;
43 list_type data;
44
45 public:
46 typedef typename list_type::iterator iterator;
47 typedef typename list_type::const_iterator const_iterator;
48 typedef std::reverse_iterator<iterator> reverse_iterator;
49 typedef std::reverse_iterator<const_iterator> const_reverse_iterator;
50
52 friend class ordered_map;
53
54 protected:
55 COMP comp;
56 explicit value_compare(COMP c) : comp(c) {}
57
58 public:
59 bool operator()(const value_type &a, const value_type &b) const {
60 return comp(a.first, b.first);
61 }
62 };
63
64 private:
65 struct mapcmp {
66 COMP comp;
67 bool operator()(const K *a, const K *b) const { return comp(*a, *b); }
68 };
69 using map_alloc = typename std::allocator_traits<ALLOC>::template rebind_alloc<
70 std::pair<const K *const, list_iterator>>;
71 using map_type = std::map<const K *, list_iterator, mapcmp, map_alloc>;
72 map_type data_map;
73 void init_data_map() {
74 data_map.clear();
75 for (auto it = data.begin(); it != data.end(); it++) data_map.emplace(&it->first, it);
76 }
77 iterator tr_iter(typename map_type::iterator i) {
78 if (i == data_map.end()) return data.end();
79 return i->second;
80 }
81 const_iterator tr_iter(typename map_type::const_iterator i) const {
82 if (i == data_map.end()) return data.end();
83 return i->second;
84 }
85
86 public:
87 typedef typename map_type::size_type size_type;
88
89 ordered_map() {}
90 ordered_map(const ordered_map &a) : data(a.data) { init_data_map(); }
91 template <typename InputIt>
92 ordered_map(InputIt first, InputIt last) : data(first, last) {
93 init_data_map();
94 }
95 ordered_map(ordered_map &&a) = default; /* move is ok? */
96 ordered_map &operator=(const ordered_map &a) {
97 /* std::list assignment broken by spec if elements are const... */
98 if (this != &a) {
99 data.clear();
100 for (auto &el : a.data) data.push_back(el);
101 init_data_map();
102 }
103 return *this;
104 }
105 ordered_map &operator=(ordered_map &&a) = default; /* move is ok? */
106 ordered_map(std::initializer_list<value_type> il) : data(il) { init_data_map(); }
107 // FIXME add allocator and comparator ctors...
108
109 iterator begin() noexcept { return data.begin(); }
110 const_iterator begin() const noexcept { return data.begin(); }
111 iterator end() noexcept { return data.end(); }
112 const_iterator end() const noexcept { return data.end(); }
113 reverse_iterator rbegin() noexcept { return data.rbegin(); }
114 const_reverse_iterator rbegin() const noexcept { return data.rbegin(); }
115 reverse_iterator rend() noexcept { return data.rend(); }
116 const_reverse_iterator rend() const noexcept { return data.rend(); }
117 const_iterator cbegin() const noexcept { return data.cbegin(); }
118 const_iterator cend() const noexcept { return data.cend(); }
119 const_reverse_iterator crbegin() const noexcept { return data.crbegin(); }
120 const_reverse_iterator crend() const noexcept { return data.crend(); }
121
122 bool empty() const noexcept { return data.empty(); }
123 size_type size() const noexcept { return data_map.size(); }
124 size_type max_size() const noexcept { return data_map.max_size(); }
125 bool operator==(const ordered_map &a) const { return data == a.data; }
126 bool operator!=(const ordered_map &a) const { return data != a.data; }
127 void clear() {
128 data.clear();
129 data_map.clear();
130 }
131
132 iterator find(const key_type &a) { return tr_iter(data_map.find(&a)); }
133 const_iterator find(const key_type &a) const { return tr_iter(data_map.find(&a)); }
134 size_type count(const key_type &a) const { return data_map.count(&a); }
135 iterator lower_bound(const key_type &a) { return tr_iter(data_map.lower_bound(&a)); }
136 const_iterator lower_bound(const key_type &a) const {
137 return tr_iter(data_map.lower_bound(&a));
138 }
139 iterator upper_bound(const key_type &a) { return tr_iter(data_map.upper_bound(&a)); }
140 const_iterator upper_bound(const key_type &a) const {
141 return tr_iter(data_map.upper_bound(&a));
142 }
143 iterator upper_bound_pred(const key_type &a) {
144 auto ub = data_map.upper_bound(&a);
145 if (ub == data_map.begin()) return end();
146 return tr_iter(--ub);
147 }
148 const_iterator upper_bound_pred(const key_type &a) const {
149 auto ub = data_map.upper_bound(&a);
150 if (ub == data_map.begin()) return end();
151 return tr_iter(--ub);
152 }
153
154 V &operator[](const K &x) {
155 auto it = find(x);
156 if (it == data.end()) {
157 it = data.emplace(data.end(), x, V());
158 data_map.emplace(&it->first, it);
159 }
160 return it->second;
161 }
162 V &operator[](K &&x) {
163 auto it = find(x);
164 if (it == data.end()) {
165 it = data.emplace(data.end(), std::move(x), V());
166 data_map.emplace(&it->first, it);
167 }
168 return it->second;
169 }
170 V &at(const K &x) { return data_map.at(&x)->second; }
171 const V &at(const K &x) const { return data_map.at(&x)->second; }
172
173 template <typename KK, typename... VV>
174 std::pair<iterator, bool> emplace(KK &&k, VV &&...v) {
175 auto it = find(k);
176 if (it == data.end()) {
177 it = data.emplace(data.end(), std::piecewise_construct_t(), std::forward_as_tuple(k),
178 std::forward_as_tuple(std::forward<VV>(v)...));
179 data_map.emplace(&it->first, it);
180 return std::make_pair(it, true);
181 }
182 return std::make_pair(it, false);
183 }
184 template <typename KK, typename... VV>
185 std::pair<iterator, bool> emplace_hint(iterator pos, KK &&k, VV &&...v) {
186 /* should be const_iterator pos, but glibc++ std::list is broken */
187 auto it = find(k);
188 if (it == data.end()) {
189 it = data.emplace(pos, std::piecewise_construct_t(), std::forward_as_tuple(k),
190 std::forward_as_tuple(std::forward<VV>(v)...));
191 data_map.emplace(&it->first, it);
192 return std::make_pair(it, true);
193 }
194 return std::make_pair(it, false);
195 }
196
197 std::pair<iterator, bool> insert(const value_type &v) {
198 auto it = find(v.first);
199 if (it == data.end()) {
200 it = data.insert(data.end(), v);
201 data_map.emplace(&it->first, it);
202 return std::make_pair(it, true);
203 }
204 return std::make_pair(it, false);
205 }
206
207 /* TODO: should not exist, does not make sense for map that preserves
208 * insertion order. This function does not preserve it. */
209 std::pair<iterator, bool> insert(iterator pos, const value_type &v) {
210 /* should be const_iterator pos, but glibc++ std::list is broken */
211 auto it = find(v.first);
212 if (it == data.end()) {
213 it = data.insert(pos, v);
214 data_map.emplace(&it->first, it);
215 return std::make_pair(it, true);
216 }
217 return std::make_pair(it, false);
218 }
219 template <class InputIterator>
220 void insert(InputIterator b, InputIterator e) {
221 while (b != e) insert(*b++);
222 }
223
224 /* TODO: should not exist, does not make sense for map that preserves
225 * insertion order. This function does not preserve it. */
226 template <class InputIterator>
227 void insert(iterator pos, InputIterator b, InputIterator e) {
228 /* should be const_iterator pos, but glibc++ std::list is broken */
229 while (b != e) insert(pos, *b++);
230 }
231
232 iterator erase(const_iterator pos) {
233 auto it = data_map.find(&pos->first);
234 assert(it != data_map.end());
235 // get the non-const std::list iterator -- libstdc++ is missing
236 // std::list::erase(const_iterator) overload
237 auto list_it = it->second;
238 data_map.erase(it);
239 return data.erase(list_it);
240 }
241 size_type erase(const K &k) {
242 auto it = find(k);
243 if (it != data.end()) {
244 data_map.erase(&k);
245 data.erase(it);
246 return 1;
247 }
248 return 0;
249 }
250
251 template <class Compare>
252 void sort(Compare comp) {
253 data.sort(comp);
254 }
255};
256
257#endif /* LIB_ORDERED_MAP_H_ */
Definition ordered_map.h:51
Definition ordered_map.h:30