P4C
The P4 Compiler
Loading...
Searching...
No Matches
json_loader.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 IR_JSON_LOADER_H_
18#define IR_JSON_LOADER_H_
19
20#include <map>
21#include <optional>
22#include <string>
23#include <unordered_map>
24#include <utility>
25
26#include "ir.h"
27#include "json_parser.h"
28#include "lib/bitvec.h"
29#include "lib/cstring.h"
30#include "lib/ltbitmatrix.h"
31#include "lib/match.h"
32#include "lib/ordered_map.h"
33#include "lib/ordered_set.h"
34#include "lib/safe_vector.h"
35
37 template <typename T>
38 class has_fromJSON {
39 typedef char small;
40 typedef struct {
41 char c[2];
42 } big;
43
44 template <typename C>
45 static small test(decltype(&C::fromJSON));
46 template <typename C>
47 static big test(...);
48
49 public:
50 static const bool value = sizeof(test<T>(0)) == sizeof(char);
51 };
52
53 public:
54 std::unordered_map<int, IR::Node *> &node_refs;
55 JsonData *json = nullptr;
56
57 explicit JSONLoader(std::istream &in)
58 : node_refs(*(new std::unordered_map<int, IR::Node *>())) {
59 in >> json;
60 }
61
62 explicit JSONLoader(JsonData *json)
63 : node_refs(*(new std::unordered_map<int, IR::Node *>())), json(json) {}
64
65 JSONLoader(JsonData *json, std::unordered_map<int, IR::Node *> &refs)
66 : node_refs(refs), json(json) {}
67
68 JSONLoader(const JSONLoader &unpacker, const std::string &field)
69 : node_refs(unpacker.node_refs), json(nullptr) {
70 if (auto *obj = unpacker.json->to<JsonObject>()) json = get(obj, field);
71 }
72
73 private:
74 const IR::Node *get_node() {
75 if (!json || !json->is<JsonObject>()) return nullptr; // invalid json exception?
76 int id = json->as<JsonObject>().get_id();
77 if (id >= 0) {
78 if (node_refs.find(id) == node_refs.end()) {
79 if (auto fn = get(IR::unpacker_table, json->as<JsonObject>().get_type())) {
80 node_refs[id] = fn(*this);
81 // Creating JsonObject from source_info read from jsonFile
82 // and setting SourceInfo for each node
83 // when "--fromJSON" flag is used
84 JsonObject *obj = new JsonObject(json->as<JsonObject>().get_sourceJson());
85 if (obj->hasSrcInfo() == true) {
86 node_refs[id]->srcInfo =
87 Util::SourceInfo(obj->get_filename(), obj->get_line(),
88 obj->get_column(), obj->get_sourceFragment());
89 }
90 } else {
91 return nullptr;
92 } // invalid json exception?
93 }
94 return node_refs[id];
95 }
96 return nullptr; // invalid json exception?
97 }
98
99 template <typename T>
100 void unpack_json(safe_vector<T> &v) {
101 T temp;
102 for (auto e : json->as<JsonVector>()) {
103 load(e, temp);
104 v.push_back(temp);
105 }
106 }
107
108 template <typename T>
109 void unpack_json(std::set<T> &v) {
110 T temp;
111 for (auto e : json->as<JsonVector>()) {
112 load(e, temp);
113 v.insert(temp);
114 }
115 }
116
117 template <typename T>
118 void unpack_json(ordered_set<T> &v) {
119 T temp;
120 for (auto e : json->as<JsonVector>()) {
121 load(e, temp);
122 v.insert(temp);
123 }
124 }
125
126 template <typename T>
127 void unpack_json(IR::Vector<T> &v) {
128 v = *IR::Vector<T>::fromJSON(*this);
129 }
130 template <typename T>
131 void unpack_json(const IR::Vector<T> *&v) {
132 v = IR::Vector<T>::fromJSON(*this);
133 }
134 template <typename T>
135 void unpack_json(IR::IndexedVector<T> &v) {
137 }
138 template <typename T>
139 void unpack_json(const IR::IndexedVector<T> *&v) {
141 }
142 template <class T, template <class K, class V, class COMP, class ALLOC> class MAP, class COMP,
143 class ALLOC>
144 void unpack_json(IR::NameMap<T, MAP, COMP, ALLOC> &m) {
146 }
147 template <class T, template <class K, class V, class COMP, class ALLOC> class MAP, class COMP,
148 class ALLOC>
149 void unpack_json(const IR::NameMap<T, MAP, COMP, ALLOC> *&m) {
151 }
152
153 template <typename K, typename V>
154 void unpack_json(std::map<K, V> &v) {
155 std::pair<K, V> temp;
156 for (auto e : json->as<JsonObject>()) {
157 JsonString *k = new JsonString(e.first);
158 load(k, temp.first);
159 load(e.second, temp.second);
160 v.insert(temp);
161 }
162 }
163 template <typename K, typename V>
164 void unpack_json(ordered_map<K, V> &v) {
165 std::pair<K, V> temp;
166 for (auto e : json->as<JsonObject>()) {
167 JsonString *k = new JsonString(e.first);
168 load(k, temp.first);
169 load(e.second, temp.second);
170 v.insert(temp);
171 }
172 }
173 template <typename V>
174 void unpack_json(string_map<V> &v) {
175 std::pair<cstring, V> temp;
176 for (auto e : json->as<JsonObject>()) {
177 JsonString *k = new JsonString(e.first);
178 load(k, temp.first);
179 load(e.second, temp.second);
180 v.insert(temp);
181 }
182 }
183
184 template <typename K, typename V>
185 void unpack_json(std::multimap<K, V> &v) {
186 std::pair<K, V> temp;
187 for (auto e : json->as<JsonObject>()) {
188 JsonString *k = new JsonString(e.first);
189 load(k, temp.first);
190 load(e.second, temp.second);
191 v.insert(temp);
192 }
193 }
194
195 template <typename T>
196 void unpack_json(std::vector<T> &v) {
197 T temp;
198 for (auto e : json->as<JsonVector>()) {
199 load(e, temp);
200 v.push_back(temp);
201 }
202 }
203
204 template <typename T, typename U>
205 void unpack_json(std::pair<T, U> &v) {
206 const JsonObject *obj = json->checkedTo<JsonObject>();
207 load(::get(obj, "first"), v.first);
208 load(::get(obj, "second"), v.second);
209 }
210
211 template <typename T>
212 void unpack_json(std::optional<T> &v) {
213 const JsonObject *obj = json->checkedTo<JsonObject>();
214 bool isValid = false;
215 load(::get(obj, "valid"), isValid);
216 if (!isValid) {
217 v = std::nullopt;
218 return;
219 }
220 T value;
221 load(::get(obj, "value"), value), v = std::move(value);
222 }
223
224 void unpack_json(bool &v) { v = json->as<JsonBoolean>(); }
225
226 template <typename T>
227 typename std::enable_if<std::is_integral<T>::value>::type unpack_json(T &v) {
228 v = json->as<JsonNumber>();
229 }
230 void unpack_json(big_int &v) { v = json->as<JsonNumber>().val; }
231 void unpack_json(cstring &v) {
232 std::string tmp = json->as<JsonString>();
233 std::string::size_type p = 0;
234 while ((p = tmp.find('\\', p)) != std::string::npos) {
235 tmp.erase(p, 1);
236 switch (tmp[p]) {
237 case 'n':
238 tmp[p] = '\n';
239 break;
240 case 'r':
241 tmp[p] = '\r';
242 break;
243 case 't':
244 tmp[p] = '\t';
245 break;
246 }
247 p++;
248 }
249 if (!json->is<JsonNull>()) v = tmp;
250 }
251 void unpack_json(IR::ID &v) {
252 if (!json->is<JsonNull>()) v.name = json->as<JsonString>();
253 }
254
255 void unpack_json(LTBitMatrix &m) {
256 if (auto *s = json->to<JsonString>()) s->c_str() >> m;
257 }
258
259 void unpack_json(bitvec &v) {
260 if (auto *s = json->to<JsonString>()) s->c_str() >> v;
261 }
262
263 template <typename T>
264 typename std::enable_if<std::is_enum<T>::value>::type unpack_json(T &v) {
265 if (auto *s = json->to<JsonString>()) *s >> v;
266 }
267
268 void unpack_json(match_t &v) {
269 if (auto *s = json->to<JsonString>()) s->c_str() >> v;
270 }
271
272 void unpack_json(UnparsedConstant *&v) {
273 cstring text("");
274 unsigned skip = 0;
275 unsigned base = 0;
276 bool hasWidth = false;
277
278 load("text", text);
279 load("skip", skip);
280 load("base", base);
281 load("hasWidth", hasWidth);
282
283 v = new UnparsedConstant({text, skip, base, hasWidth});
284 }
285
286 template <typename T>
287 typename std::enable_if<
288 has_fromJSON<T>::value && !std::is_base_of<IR::Node, T>::value &&
289 std::is_pointer<decltype(T::fromJSON(std::declval<JSONLoader &>()))>::value>::type
290 unpack_json(T *&v) {
291 v = T::fromJSON(*this);
292 }
293
294 template <typename T>
295 typename std::enable_if<
296 has_fromJSON<T>::value && !std::is_base_of<IR::Node, T>::value &&
297 std::is_pointer<decltype(T::fromJSON(std::declval<JSONLoader &>()))>::value>::type
298 unpack_json(T &v) {
299 v = *(T::fromJSON(*this));
300 }
301
302 template <typename T>
303 typename std::enable_if<
304 has_fromJSON<T>::value && !std::is_base_of<IR::Node, T>::value &&
305 !std::is_pointer<decltype(T::fromJSON(std::declval<JSONLoader &>()))>::value>::type
306 unpack_json(T &v) {
307 v = T::fromJSON(*this);
308 }
309
310 template <typename T>
311 typename std::enable_if<std::is_base_of<IR::INode, T>::value>::type unpack_json(T &v) {
312 v = get_node()->as<T>();
313 }
314 template <typename T>
315 typename std::enable_if<std::is_base_of<IR::INode, T>::value>::type unpack_json(const T *&v) {
316 v = get_node()->checkedTo<T>();
317 }
318
319 template <typename T, size_t N>
320 void unpack_json(T (&v)[N]) {
321 if (auto *j = json->to<JsonVector>()) {
322 for (size_t i = 0; i < N && i < j->size(); ++i) {
323 json = (*j)[i];
324 unpack_json(v[i]);
325 }
326 }
327 }
328
329 public:
330 template <typename T>
331 void load(JsonData *json, T &v) {
332 JSONLoader(json, node_refs).unpack_json(v);
333 }
334
335 template <typename T>
336 void load(const std::string field, T *&v) {
337 JSONLoader loader(*this, field);
338 if (loader.json == nullptr) {
339 v = nullptr;
340 } else {
341 loader.unpack_json(v);
342 }
343 }
344
345 template <typename T>
346 void load(const std::string field, T &v) {
347 JSONLoader loader(*this, field);
348 if (loader.json == nullptr) return;
349 loader.unpack_json(v);
350 }
351
352 template <typename T>
353 JSONLoader &operator>>(T &v) {
354 unpack_json(v);
355 return *this;
356 }
357};
358
359template <class T>
360IR::Vector<T>::Vector(JSONLoader &json) : VectorBase(json) {
361 json.load("vec", vec);
362}
363template <class T>
365 return new Vector<T>(json);
366}
367template <class T>
368IR::IndexedVector<T>::IndexedVector(JSONLoader &json) : Vector<T>(json) {
369 json.load("declarations", declarations);
370}
371template <class T>
373 return new IndexedVector<T>(json);
374}
375template <class T, template <class K, class V, class COMP, class ALLOC> class MAP /*= std::map */,
376 class COMP /*= std::less<cstring>*/,
377 class ALLOC /*= std::allocator<std::pair<cstring, const T*>>*/>
379 json.load("symbols", symbols);
380}
381template <class T, template <class K, class V, class COMP, class ALLOC> class MAP /*= std::map */,
382 class COMP /*= std::less<cstring>*/,
383 class ALLOC /*= std::allocator<std::pair<cstring, const T*>>*/>
385 return new IR::NameMap<T, MAP, COMP, ALLOC>(json);
386}
387
388#endif /* IR_JSON_LOADER_H_ */
const T & as() const
Tries to convert the class to type T. A BUG occurs if the cast fails.
Definition castable.h:40
const T * checkedTo() const
Performs a checked cast. A BUG occurs if the cast fails.
Definition castable.h:52
Definition node.h:50
Definition namemap.h:36
Definition node.h:93
Definition vector.h:56
Definition json_loader.h:36
Definition json_parser.h:34
Definition json_parser.h:13
Definition json_parser.h:90
Definition json_parser.h:25
Definition json_parser.h:68
Definition json_parser.h:43
Definition json_parser.h:57
Definition ltbitmatrix.h:23
Definition source_file.h:123
Definition bitvec.h:119
Definition cstring.h:80
Definition ordered_map.h:30
Definition ordered_set.h:30
Definition safe_vector.h:25
Definition string_map.h:39
Definition id.h:28
T * to() noexcept
Definition rtti.h:226
bool is() const noexcept
Definition rtti.h:216
Definition constantParsing.h:63
Definition match.h:34