P4C
The P4 Compiler
Loading...
Searching...
No Matches
string_map.h
1/*
2Licensed under the Apache License, Version 2.0 (the "License");
3you may not use this file except in compliance with the License.
4You may obtain a copy of the License at
5
6 http://www.apache.org/licenses/LICENSE-2.0
7
8Unless required by applicable law or agreed to in writing, software
9distributed under the License is distributed on an "AS IS" BASIS,
10WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
11See the License for the specific language governing permissions and
12limitations under the License.
13*/
14
15#ifndef LIB_STRING_MAP_H_
16#define LIB_STRING_MAP_H_
17
18#include <initializer_list>
19#include <list>
20#include <stdexcept>
21#include <utility>
22
23#include "absl/container/flat_hash_map.h"
24#include "cstring.h"
25
38template <class V>
40 public:
41 using key_type = cstring;
42 using mapped_type = V;
43 using value_type = std::pair<cstring, V>;
44 using reference = value_type &;
45 using const_reference = const value_type &;
46
47 private:
48 // TODO: Allow different containers (e.g. vector / small vector)
49 using list_type = std::list<value_type>;
50 using list_iterator = typename list_type::iterator;
51 list_type data;
52
53 public:
54 using iterator = typename list_type::iterator;
55 using const_iterator = typename list_type::const_iterator;
56 using reverse_iterator = std::reverse_iterator<iterator>;
57 using const_reverse_iterator = std::reverse_iterator<const_iterator>;
58
59 private:
60 using map_type = absl::flat_hash_map<cstring, list_iterator>;
61 map_type data_map;
62 void init_data_map() {
63 data_map.clear();
64 for (auto it = data.begin(); it != data.end(); it++) data_map.emplace(it->first, it);
65 }
66 iterator tr_iter(typename map_type::iterator i) {
67 if (i == data_map.end()) return data.end();
68 return i->second;
69 }
70 const_iterator tr_iter(typename map_type::const_iterator i) const {
71 if (i == data_map.end()) return data.end();
72 return i->second;
73 }
74
75 public:
76 using size_type = typename map_type::size_type;
77
78 string_map() = default;
79 string_map(const string_map &a) : data(a.data) { init_data_map(); }
80 template <typename InputIt>
81 string_map(InputIt first, InputIt last) {
82 insert(first, last);
83 }
84 string_map(string_map &&a) = default;
85 string_map &operator=(const string_map &a) {
86 if (this != &a) {
87 data.clear();
88 data.insert(data.end(), a.data.begin(), a.data.end());
89 init_data_map();
90 }
91 return *this;
92 }
93 string_map &operator=(string_map &&a) = default;
94 string_map(std::initializer_list<value_type> il) { insert(il.begin(), il.end()); }
95
96 iterator begin() noexcept { return data.begin(); }
97 const_iterator begin() const noexcept { return data.begin(); }
98 iterator end() noexcept { return data.end(); }
99 const_iterator end() const noexcept { return data.end(); }
100 reverse_iterator rbegin() noexcept { return data.rbegin(); }
101 const_reverse_iterator rbegin() const noexcept { return data.rbegin(); }
102 reverse_iterator rend() noexcept { return data.rend(); }
103 const_reverse_iterator rend() const noexcept { return data.rend(); }
104 const_iterator cbegin() const noexcept { return data.cbegin(); }
105 const_iterator cend() const noexcept { return data.cend(); }
106 const_reverse_iterator crbegin() const noexcept { return data.crbegin(); }
107 const_reverse_iterator crend() const noexcept { return data.crend(); }
108
109 bool empty() const noexcept { return data.empty(); }
110 size_type size() const noexcept { return data_map.size(); }
111 size_type max_size() const noexcept { return data_map.max_size(); }
114 bool operator==(const string_map &a) const { return data == a.data; }
115 bool operator!=(const string_map &a) const { return data != a.data; }
116 void clear() {
117 data.clear();
118 data_map.clear();
119 }
120
121 iterator find(cstring a) { return tr_iter(data_map.find(a)); }
126 iterator find(std::string_view a) {
128 if (key.isNull()) return data.end();
129
130 return tr_iter(data_map.find(key));
131 }
132 const_iterator find(cstring a) const { return tr_iter(data_map.find(a)); }
133 const_iterator find(std::string_view a) const {
135 if (key.isNull()) return data.end();
136
137 return tr_iter(data_map.find(key));
138 }
139
140 size_type count(cstring a) const { return data_map.count(a); }
141 size_type count(std::string_view a) const {
143 if (key.isNull()) return 0;
144
145 return data_map.count(key);
146 }
147
148 bool contains(cstring a) const { return data_map.contains(a); }
149 bool contains(std::string_view a) const {
151 if (key.isNull()) return false;
152
153 return data_map.contains(key);
154 }
155
156 template <typename Key>
157 V &operator[](Key &&k) {
158 auto it = find(key_type(std::forward<Key>(k)));
159 if (it == data.end()) {
160 it = data.emplace(data.end(), std::piecewise_construct, std::forward_as_tuple(k),
161 std::forward_as_tuple());
162 data_map.emplace(it->first, it);
163 }
164 return it->second;
165 }
166
167 template <typename Key>
168 V &at(Key &&k) {
169 auto it = find(std::forward<Key>(k));
170 if (it == data.end()) throw std::out_of_range("string_map::at");
171 return it->second;
172 }
173 template <typename Key>
174 const V &at(Key &&k) const {
175 auto it = find(std::forward<Key>(k));
176 if (it == data.end()) throw std::out_of_range("string_map::at");
177 return it->second;
178 }
179
180 template <typename Key, typename... VV>
181 std::pair<iterator, bool> emplace(Key &&k, VV &&...v) {
182 auto it = find(key_type(std::forward<Key>(k)));
183 if (it == data.end()) {
184 it = data.emplace(data.end(), std::piecewise_construct, std::forward_as_tuple(k),
185 std::forward_as_tuple(std::forward<VV>(v)...));
186 data_map.emplace(it->first, it);
187 return {it, true};
188 }
189 return {it, false};
190 }
191
192 std::pair<iterator, bool> insert(value_type &&value) {
193 return emplace(std::move(value.first), std::move(value.second));
194 }
195 std::pair<iterator, bool> insert(const value_type &value) {
196 return emplace(value.first, value.second);
197 }
198
199 template <class InputIterator>
200 void insert(InputIterator b, InputIterator e) {
201 while (b != e) insert(*b++);
202 }
203
204 iterator erase(const_iterator pos) {
205 auto it = data_map.find(pos->first);
206 assert(it != data_map.end());
207 // get the non-const std::list iterator -- libstdc++ is missing
208 // std::list::erase(const_iterator) overload
209 auto list_it = it->second;
210 data_map.erase(it);
211 return data.erase(list_it);
212 }
213 size_type erase(cstring k) {
214 auto it = find(k);
215 if (it != data.end()) {
216 data_map.erase(k);
217 data.erase(it);
218 return 1;
219 }
220 return 0;
221 }
222 size_type erase(std::string_view k) {
223 auto it = find(k);
224 if (it != data.end()) {
225 data_map.erase(it->first);
226 data.erase(it);
227 return 1;
228 }
229 return 0;
230 }
231
232 void swap(string_map &other) {
233 using std::swap;
234 swap(data, other.data);
235 swap(data_map, other.data_map);
236 }
237};
238
239#endif /* LIB_STRING_MAP_H_ */
Definition cstring.h:80
static cstring get_cached(std::string_view s)
Definition cstring.cpp:188
Definition string_map.h:39
iterator find(std::string_view a)
Definition string_map.h:126
bool operator==(const string_map &a) const
Definition string_map.h:114