P4C
The P4 Compiler
Loading...
Searching...
No Matches
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_MAP_H_
18#define LIB_MAP_H_
19
20#include <iterator>
21#include <optional>
22
25template <typename Map, typename Key>
26typename Map::mapped_type get(const Map &m, const Key &key) {
27 auto it = m.find(key);
28 return it != m.end() ? it->second : typename Map::mapped_type{};
29}
30template <class Map, typename Key, typename Value>
31typename Map::mapped_type get(const Map &m, const Key &key, Value &&def) {
32 using M = typename Map::mapped_type;
33 auto it = m.find(key);
34 return it != m.end() ? it->second : static_cast<M>(std::forward<Value>(def));
35}
36
39template <typename Map, typename Key>
40const typename Map::mapped_type *getref(const Map &m, const Key &key) {
41 auto it = m.find(key);
42 return it != m.end() ? &it->second : nullptr;
43}
44
45template <typename Map, typename Key>
46typename Map::mapped_type *getref(Map &m, const Key &key) {
47 auto it = m.find(key);
48 return it != m.end() ? &it->second : nullptr;
49}
50
51// Given a map and a key, return a optional<V> if the key exists and None if the
52// key does not exist in the map.
53template <class Map, typename Key>
54std::optional<typename Map::mapped_type> get_optional(const Map &m, const Key &key) {
55 auto it = m.find(key);
56 if (it != m.end()) return std::optional<typename Map::mapped_type>(it->second);
57
58 return {};
59}
60
61template <typename Map, typename Key>
62typename Map::mapped_type get(const Map *m, const Key &key) {
63 return m ? get(*m, key) : typename Map::mapped_type{};
64}
65
66template <class Map, typename Key, typename Value>
67typename Map::mapped_type get(const Map *m, const Key &key, Value &&def) {
68 return m ? get(*m, key, std::forward(def)) : typename Map::mapped_type{};
69}
70
71template <typename Map, typename Key>
72const typename Map::mapped_type *getref(const Map *m, const Key &key) {
73 return m ? getref(*m, key) : nullptr;
74}
75
76template <typename Map, typename Key>
77typename Map::mapped_type *getref(Map *m, const Key &key) {
78 return m ? getref(*m, key) : nullptr;
79}
80
81/* iterate over the keys in a map */
82template <class PairIter>
83class IterKeys {
84 class iterator {
85 PairIter it;
86
87 public:
88 using iterator_category = typename std::iterator_traits<PairIter>::iterator_category;
89 using value_type = typename std::iterator_traits<PairIter>::value_type::first_type;
90 using difference_type = typename std::iterator_traits<PairIter>::difference_type;
91 using pointer = decltype(&it->first);
92 using reference = decltype(*&it->first);
93
94 explicit iterator(PairIter i) : it(i) {}
95 iterator &operator++() {
96 ++it;
97 return *this;
98 }
99 iterator &operator--() {
100 --it;
101 return *this;
102 }
103 iterator operator++(int) {
104 auto copy = *this;
105 ++it;
106 return copy;
107 }
108 iterator operator--(int) {
109 auto copy = *this;
110 --it;
111 return copy;
112 }
113 bool operator==(const iterator &i) const { return it == i.it; }
114 bool operator!=(const iterator &i) const { return it != i.it; }
115 reference operator*() const { return it->first; }
116 pointer operator->() const { return &it->first; }
117 } b, e;
118
119 public:
120 template <class U>
121 explicit IterKeys(U &map) : b(map.begin()), e(map.end()) {}
122 IterKeys(PairIter b, PairIter e) : b(b), e(e) {}
123 iterator begin() const { return b; }
124 iterator end() const { return e; }
125};
126
127template <class Map>
130}
131
132template <class Map>
135}
136
137template <class PairIter>
138IterKeys<PairIter> Keys(std::pair<PairIter, PairIter> range) {
139 return IterKeys<PairIter>(range.first, range.second);
140}
141
142/* iterate over the values in a map */
143template <class PairIter>
145 class iterator {
146 PairIter it;
147
148 public:
149 using iterator_category = typename std::iterator_traits<PairIter>::iterator_category;
150 using value_type = typename std::iterator_traits<PairIter>::value_type::second_type;
151 using difference_type = typename std::iterator_traits<PairIter>::difference_type;
152 using pointer = decltype(&it->second);
153 using reference = decltype(*&it->second);
154
155 explicit iterator(PairIter i) : it(i) {}
156 iterator &operator++() {
157 ++it;
158 return *this;
159 }
160 iterator &operator--() {
161 --it;
162 return *this;
163 }
164 iterator operator++(int) {
165 auto copy = *this;
166 ++it;
167 return copy;
168 }
169 iterator operator--(int) {
170 auto copy = *this;
171 --it;
172 return copy;
173 }
174 bool operator==(const iterator &i) const { return it == i.it; }
175 bool operator!=(const iterator &i) const { return it != i.it; }
176 reference operator*() const { return it->second; }
177 pointer operator->() const { return &it->second; }
178 } b, e;
179
180 public:
181 using value_type = typename std::iterator_traits<PairIter>::value_type::second_type;
182
183 template <class U>
184 explicit IterValues(U &map) : b(map.begin()), e(map.end()) {}
185 IterValues(PairIter b, PairIter e) : b(b), e(e) {}
186 iterator begin() const { return b; }
187 iterator end() const { return e; }
188};
189
190template <class Map>
193}
194
195template <class Map>
196IterValues<typename Map::const_iterator> Values(const Map &m) {
198}
199
200template <class PairIter>
201IterValues<PairIter> Values(std::pair<PairIter, PairIter> range) {
202 return IterValues<PairIter>(range.first, range.second);
203}
204
205/* iterate over the values for a single key in a multimap */
206template <class M>
208 M &map;
209 typename M::key_type key;
210 class iterator {
211 using MapIt = decltype(map.begin());
212
213 const MapForKey &self;
214 MapIt it;
215
216 public:
217 using iterator_category = std::forward_iterator_tag;
218 using value_type = typename M::value_type;
219 using difference_type = typename std::iterator_traits<MapIt>::difference_type;
220 using pointer = decltype(&it->second);
221 using reference = decltype(*&it->second);
222
223 iterator(const MapForKey &s, MapIt i) : self(s), it(std::move(i)) {}
224 iterator &operator++() {
225 if (++it != self.map.end() && it->first != self.key) it = self.map.end();
226 return *this;
227 }
228 iterator operator++(int) {
229 auto copy = *this;
230 ++*this;
231 return copy;
232 }
233 bool operator==(const iterator &i) const { return it == i.it; }
234 bool operator!=(const iterator &i) const { return it != i.it; }
235 reference operator*() const { return it->second; }
236 pointer operator->() const { return &it->second; }
237 };
238
239 public:
240 MapForKey(M &m, typename M::key_type k) : map(m), key(k) {}
241 iterator begin() const { return iterator(*this, map.find(key)); }
242 iterator end() const { return iterator(*this, map.end()); }
243};
244
245template <class M>
246MapForKey<M> ValuesForKey(M &m, typename M::key_type k) {
247 return MapForKey<M>(m, k);
248}
249
250#endif /* LIB_MAP_H_ */
Definition map.h:83
Definition map.h:144
Definition map.h:207