P4C
The P4 Compiler
Loading...
Searching...
No Matches
node.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_NODE_H_
18#define IR_NODE_H_
19
20#include <iosfwd>
21
22#include "ir-tree-macros.h"
23#include "ir/gen-tree-macro.h"
24#include "lib/castable.h"
25#include "lib/cstring.h"
26#include "lib/exceptions.h"
27#include "lib/source_file.h"
28
29class Visitor;
30struct Visitor_Context;
31class Inspector;
32class Modifier;
33class Transform;
34class JSONGenerator;
35class JSONLoader;
36
37namespace Util {
38class JsonObject;
39} // namespace Util
40
41namespace IR {
42
43using namespace P4::literals;
44
45class Node;
46class Annotation; // IWYU pragma: keep
47template <class T>
48class Vector; // IWYU pragma: keep
49template <class T>
50class IndexedVector; // IWYU pragma: keep
51
54template <class, class = void>
55struct has_static_type_name : std::false_type {};
56
57template <class T>
58struct has_static_type_name<T, std::void_t<decltype(T::static_type_name())>> : std::true_type {};
59
60template <class T>
61inline constexpr bool has_static_type_name_v = has_static_type_name<T>::value;
62
63// node interface
64class INode : public Util::IHasSourceInfo, public IHasDbPrint, public ICastable {
65 public:
66 virtual ~INode() {}
67 virtual const Node *getNode() const = 0;
68 virtual Node *getNode() = 0;
69 virtual void toJSON(JSONGenerator &) const = 0;
70 virtual cstring node_type_name() const = 0;
71 virtual void validate() const {}
72 virtual const Annotation *getAnnotation(cstring) const { return nullptr; }
73
74 // default checkedTo implementation for nodes: just fallback to generic ICastable method
75 template <typename T>
76 std::enable_if_t<!has_static_type_name_v<T>, const T *> checkedTo() const {
78 }
79
80 // alternative checkedTo implementation that produces slightly better error message
81 // due to node_type_name() / static_type_name() being available
82 template <typename T>
83 std::enable_if_t<has_static_type_name_v<T>, const T *> checkedTo() const {
84 const auto *result = to<T>();
85 BUG_CHECK(result, "Cast failed: %1% with type %2% is not a %3%.", this, node_type_name(),
86 T::static_type_name());
87 return result;
88 }
89
90 DECLARE_TYPEINFO_WITH_TYPEID(INode, NodeKind::INode);
91};
92
93class Node : public virtual INode {
94 public:
95 virtual bool apply_visitor_preorder(Modifier &v);
96 virtual void apply_visitor_postorder(Modifier &v);
97 virtual void apply_visitor_revisit(Modifier &v, const Node *n) const;
98 virtual void apply_visitor_loop_revisit(Modifier &v) const;
99 virtual bool apply_visitor_preorder(Inspector &v) const;
100 virtual void apply_visitor_postorder(Inspector &v) const;
101 virtual void apply_visitor_revisit(Inspector &v) const;
102 virtual void apply_visitor_loop_revisit(Inspector &v) const;
103 virtual const Node *apply_visitor_preorder(Transform &v);
104 virtual const Node *apply_visitor_postorder(Transform &v);
105 virtual void apply_visitor_revisit(Transform &v, const Node *n) const;
106 virtual void apply_visitor_loop_revisit(Transform &v) const;
107 Node &operator=(const Node &) = default;
108 Node &operator=(Node &&) = default;
109
110 protected:
111 static int currentId;
112 void traceVisit(const char *visitor) const;
113 friend class ::Visitor;
114 friend class ::Inspector;
115 friend class ::Modifier;
116 friend class ::Transform;
117 cstring prepareSourceInfoForJSON(Util::SourceInfo &si, unsigned *lineNumber,
118 unsigned *columnNumber) const;
119
120 public:
121 Util::SourceInfo srcInfo;
122 int id; // unique id for each node
123 int clone_id; // unique id this node was cloned from (recursively)
124 void traceCreation() const;
125 Node() : id(currentId++), clone_id(id) { traceCreation(); }
126 explicit Node(Util::SourceInfo si) : srcInfo(si), id(currentId++), clone_id(id) {
127 traceCreation();
128 }
129 Node(const Node &other) : srcInfo(other.srcInfo), id(currentId++), clone_id(other.clone_id) {
130 traceCreation();
131 }
132 virtual ~Node() {}
133 const Node *apply(Visitor &v, const Visitor_Context *ctxt = nullptr) const;
134 const Node *apply(Visitor &&v, const Visitor_Context *ctxt = nullptr) const {
135 return apply(v, ctxt);
136 }
137 virtual Node *clone() const = 0;
138 void dbprint(std::ostream &out) const override;
139 virtual void dump_fields(std::ostream &) const {}
140 const Node *getNode() const final { return this; }
141 Node *getNode() final { return this; }
142 Util::SourceInfo getSourceInfo() const override { return srcInfo; }
143 cstring node_type_name() const override { return "Node"_cs; }
144 static cstring static_type_name() { return "Node"_cs; }
145 virtual int num_children() { return 0; }
146 explicit Node(JSONLoader &json);
147 cstring toString() const override { return node_type_name(); }
148 void toJSON(JSONGenerator &json) const override;
149 void sourceInfoToJSON(JSONGenerator &json) const;
150 Util::JsonObject *sourceInfoJsonObj() const;
151 /* operator== does a 'shallow' comparison, comparing two Node subclass objects for equality,
152 * and comparing pointers in the Node directly for equality */
153 virtual bool operator==(const Node &a) const { return this->typeId() == a.typeId(); }
154 /* 'equiv' does a deep-equals comparison, comparing all non-pointer fields and recursing
155 * though all Node subclass pointers to compare them with 'equiv' as well. */
156 virtual bool equiv(const Node &a) const { return this->typeId() == a.typeId(); }
157#define DEFINE_OPEQ_FUNC(CLASS, BASE) \
158 virtual bool operator==(const CLASS &) const { return false; }
159 IRNODE_ALL_SUBCLASSES(DEFINE_OPEQ_FUNC)
160#undef DEFINE_OPEQ_FUNC
161 virtual void visit_children(Visitor &) {}
162 virtual void visit_children(Visitor &) const {}
163
164 bool operator!=(const Node &n) const { return !operator==(n); }
165
166 DECLARE_TYPEINFO_WITH_TYPEID(Node, NodeKind::Node, INode);
167};
168
169// simple version of dbprint
170cstring dbp(const INode *node);
171
172inline bool equal(const Node *a, const Node *b) { return a == b || (a && b && *a == *b); }
173inline bool equal(const INode *a, const INode *b) {
174 return a == b || (a && b && *a->getNode() == *b->getNode());
175}
176inline bool equiv(const Node *a, const Node *b) { return a == b || (a && b && a->equiv(*b)); }
177inline bool equiv(const INode *a, const INode *b) {
178 return a == b || (a && b && a->getNode()->equiv(*b->getNode()));
179}
180// NOLINTBEGIN(bugprone-macro-parentheses)
181/* common things that ALL Node subclasses must define */
182#define IRNODE_SUBCLASS(T) \
183 public: \
184 T *clone() const override { return new T(*this); } \
185 IRNODE_COMMON_SUBCLASS(T)
186#define IRNODE_ABSTRACT_SUBCLASS(T) \
187 public: \
188 T *clone() const override = 0; \
189 IRNODE_COMMON_SUBCLASS(T)
190
191// NOLINTEND(bugprone-macro-parentheses)
192#define IRNODE_COMMON_SUBCLASS(T) \
193 public: \
194 using Node::operator==; \
195 bool apply_visitor_preorder(Modifier &v) override; \
196 void apply_visitor_postorder(Modifier &v) override; \
197 void apply_visitor_revisit(Modifier &v, const Node *n) const override; \
198 void apply_visitor_loop_revisit(Modifier &v) const override; \
199 bool apply_visitor_preorder(Inspector &v) const override; \
200 void apply_visitor_postorder(Inspector &v) const override; \
201 void apply_visitor_revisit(Inspector &v) const override; \
202 void apply_visitor_loop_revisit(Inspector &v) const override; \
203 const Node *apply_visitor_preorder(Transform &v) override; \
204 const Node *apply_visitor_postorder(Transform &v) override; \
205 void apply_visitor_revisit(Transform &v, const Node *n) const override; \
206 void apply_visitor_loop_revisit(Transform &v) const override;
207
208/* only define 'apply' for a limited number of classes (those we want to call
209 * visitors directly on), as defining it and making it virtual would mean that
210 * NO Transform could transform the class into a sibling class */
211#define IRNODE_DECLARE_APPLY_OVERLOAD(T) \
212 const T *apply(Visitor &v, const Visitor_Context *ctxt = nullptr) const; \
213 const T *apply(Visitor &&v, const Visitor_Context *ctxt = nullptr) const { \
214 return apply(v, ctxt); \
215 }
216#define IRNODE_DEFINE_APPLY_OVERLOAD(CLASS, TEMPLATE, TT) \
217 TEMPLATE \
218 const IR::CLASS TT *IR::CLASS TT::apply(Visitor &v, const Visitor_Context *ctxt) const { \
219 const CLASS *tmp = this; \
220 auto prof = v.init_apply(tmp, ctxt); \
221 v.visit(tmp); \
222 v.end_apply(tmp); \
223 return tmp; \
224 }
225
226} // namespace IR
227
228#endif /* IR_NODE_H_ */
Definition castable.h:34
const T * checkedTo() const
Performs a checked cast. A BUG occurs if the cast fails.
Definition castable.h:52
Definition stringify.h:31
Definition node.h:64
Definition node.h:50
Definition node.h:93
Definition visitor.h:396
Definition json_generator.h:34
Definition json_loader.h:36
Definition json_parser.h:68
Definition visitor.h:368
Definition visitor.h:420
Definition source_file.h:210
Definition json.h:164
Definition source_file.h:123
Definition visitor.h:73
Definition cstring.h:80
Definition cstring.h:76
STL namespace.
Definition node.h:55
T * to() noexcept
Definition rtti.h:226
virtual TypeId typeId() const noexcept=0
Definition visitor.h:45