P4C
The P4 Compiler
Loading...
Searching...
No Matches
json_generator.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_GENERATOR_H_
18#define IR_JSON_GENERATOR_H_
19
20#include <optional>
21#include <string>
22#include <unordered_set>
23
24#include "ir/node.h"
25#include "lib/bitvec.h"
26#include "lib/cstring.h"
27#include "lib/indent.h"
28#include "lib/ltbitmatrix.h"
29#include "lib/match.h"
30#include "lib/ordered_map.h"
31#include "lib/ordered_set.h"
32#include "lib/safe_vector.h"
33
35 std::unordered_set<int> node_refs;
36 std::ostream &out;
37 bool dumpSourceInfo;
38
39 template <typename T>
40 class has_toJSON {
41 typedef char small;
42 typedef struct {
43 char c[2];
44 } big;
45
46 template <typename C>
47 static small test(decltype(&C::toJSON));
48 template <typename C>
49 static big test(...);
50
51 public:
52 static const bool value = sizeof(test<T>(0)) == sizeof(char);
53 };
54
55 public:
56 indent_t indent;
57
58 explicit JSONGenerator(std::ostream &out, bool dumpSourceInfo = false)
59 : out(out), dumpSourceInfo(dumpSourceInfo) {}
60
61 template <typename T>
62 void generate(const safe_vector<T> &v) {
63 out << "[";
64 if (v.size() > 0) {
65 out << std::endl << ++indent;
66 generate(v[0]);
67 for (size_t i = 1; i < v.size(); i++) {
68 out << "," << std::endl << indent;
69 generate(v[i]);
70 }
71 out << std::endl << --indent;
72 }
73 out << "]";
74 }
75
76 template <typename T>
77 void generate(const std::vector<T> &v) {
78 out << "[";
79 if (v.size() > 0) {
80 out << std::endl << ++indent;
81 generate(v[0]);
82 for (size_t i = 1; i < v.size(); i++) {
83 out << "," << std::endl << indent;
84 generate(v[i]);
85 }
86 out << std::endl << --indent;
87 }
88 out << "]";
89 }
90
91 template <typename T, typename U>
92 void generate(const std::pair<T, U> &v) {
93 ++indent;
94 out << "{" << std::endl;
95 toJSON(v);
96 out << std::endl << --indent << "}";
97 }
98
99 template <typename T, typename U>
100 void toJSON(const std::pair<T, U> &v) {
101 out << indent << "\"first\" : ";
102 generate(v.first);
103 out << "," << std::endl << indent << "\"second\" : ";
104 generate(v.second);
105 }
106
107 template <typename T>
108 void generate(const std::optional<T> &v) {
109 if (!v) {
110 out << "{ \"valid\" : false }";
111 return;
112 }
113 out << "{" << std::endl << ++indent;
114 out << "\"valid\" : true," << std::endl;
115 out << "\"value\" : ";
116 generate(*v);
117 out << std::endl << --indent << "}";
118 }
119
120 template <typename T>
121 void generate(const std::set<T> &v) {
122 out << "[" << std::endl;
123 if (v.size() > 0) {
124 auto it = v.begin();
125 out << ++indent;
126 generate(*it);
127 for (it++; it != v.end(); ++it) {
128 out << "," << std::endl << indent;
129 generate(*it);
130 }
131 out << std::endl << --indent;
132 }
133 out << "]";
134 }
135
136 template <typename T>
137 void generate(const ordered_set<T> &v) {
138 out << "[" << std::endl;
139 if (v.size() > 0) {
140 auto it = v.begin();
141 out << ++indent;
142 generate(*it);
143 for (it++; it != v.end(); ++it) {
144 out << "," << std::endl << indent;
145 generate(*it);
146 }
147 out << std::endl << --indent;
148 }
149 out << "]";
150 }
151
152 template <typename K, typename V>
153 void generate(const std::map<K, V> &v) {
154 out << "[" << std::endl;
155 if (v.size() > 0) {
156 auto it = v.begin();
157 out << ++indent;
158 generate(*it);
159 for (it++; it != v.end(); ++it) {
160 out << "," << std::endl << indent;
161 generate(*it);
162 }
163 out << std::endl << --indent;
164 }
165 out << "]";
166 }
167
168 template <typename K, typename V>
169 void generate(const ordered_map<K, V> &v) {
170 out << "[" << std::endl;
171 if (v.size() > 0) {
172 auto it = v.begin();
173 out << ++indent;
174 generate(*it);
175 for (it++; it != v.end(); ++it) {
176 out << "," << std::endl << indent;
177 generate(*it);
178 }
179 out << std::endl << --indent;
180 }
181 out << "]";
182 }
183
184 void generate(bool v) { out << (v ? "true" : "false"); }
185 template <typename T>
186 typename std::enable_if<std::is_integral<T>::value>::type generate(T v) {
187 out << std::to_string(v);
188 }
189 void generate(double v) { out << std::to_string(v); }
190 template <typename T>
191 typename std::enable_if<std::is_same<T, big_int>::value>::type generate(const T &v) {
192 out << v;
193 }
194
195 void generate(cstring v) {
196 if (v) {
197 out << "\"" << v.escapeJson() << "\"";
198 } else {
199 out << "null";
200 }
201 }
202 template <typename T>
203 typename std::enable_if<std::is_same<T, LTBitMatrix>::value || std::is_enum<T>::value>::type
204 generate(T v) {
205 out << "\"" << v << "\"";
206 }
207
208 void generate(const bitvec &v) { out << "\"" << v << "\""; }
209
210 void generate(const match_t &v) {
211 out << "{" << std::endl
212 << (indent + 1) << "\"word0\" : " << v.word0 << "," << std::endl
213 << (indent + 1) << "\"word1\" : " << v.word1 << std::endl
214 << indent << "}";
215 }
216
217 template <typename T>
218 typename std::enable_if<has_toJSON<T>::value && !std::is_base_of<IR::Node, T>::value>::type
219 generate(const T &v) {
220 ++indent;
221 out << "{" << std::endl;
222 v.toJSON(*this);
223 out << std::endl << --indent << "}";
224 }
225
226 void generate(const IR::Node &v) {
227 out << "{" << std::endl;
228 ++indent;
229 if (node_refs.find(v.id) != node_refs.end()) {
230 out << indent << "\"Node_ID\" : " << v.id;
231 } else {
232 node_refs.insert(v.id);
233 v.toJSON(*this);
234 if (dumpSourceInfo) {
235 v.sourceInfoToJSON(*this);
236 }
237 }
238 out << std::endl << --indent << "}";
239 }
240
241 template <typename T>
242 typename std::enable_if<std::is_pointer<T>::value &&
243 has_toJSON<typename std::remove_pointer<T>::type>::value>::type
244 generate(T v) {
245 if (v)
246 generate(*v);
247 else
248 out << "null";
249 }
250
251 template <typename T, size_t N>
252 void generate(const T (&v)[N]) {
253 out << "[";
254 if (N > 0) {
255 out << std::endl << ++indent;
256 generate(v[0]);
257 for (size_t i = 1; i < N; i++) {
258 out << "," << std::endl << indent;
259 generate(v[i]);
260 }
261 out << std::endl << --indent;
262 }
263 out << "]";
264 }
265
266 JSONGenerator &operator<<(char ch) {
267 out << ch;
268 return *this;
269 }
270 JSONGenerator &operator<<(const char *s) {
271 out << s;
272 return *this;
273 }
274 JSONGenerator &operator<<(indent_t i) {
275 out << i;
276 return *this;
277 }
278 JSONGenerator &operator<<(std::ostream &(*fn)(std::ostream &)) {
279 out << fn;
280 return *this;
281 }
282 template <typename T>
283 JSONGenerator &operator<<(const T &v) {
284 generate(v);
285 return *this;
286 }
287};
288
289#endif /* IR_JSON_GENERATOR_H_ */
Definition node.h:93
Definition json_generator.h:34
Definition bitvec.h:119
Definition cstring.h:80
cstring escapeJson() const
Definition cstring.cpp:269
Definition indent.h:24
Definition ordered_map.h:30
Definition ordered_set.h:30
Definition safe_vector.h:25
Definition match.h:34