P4C
The P4 Compiler
Loading...
Searching...
No Matches
visitor.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_VISITOR_H_
18#define IR_VISITOR_H_
19
20#include <cassert>
21#include <cstddef>
22#include <cstdint>
23#include <functional>
24#include <iosfwd>
25#include <map>
26#include <memory>
27#include <type_traits>
28#include <typeinfo>
29#include <unordered_map>
30#include <utility>
31
32#include "absl/time/time.h"
33#include "ir/gen-tree-macro.h"
34#include "ir/ir-tree-macros.h"
35#include "ir/node.h"
36#include "ir/vector.h"
37#include "lib/castable.h"
38#include "lib/cstring.h"
39#include "lib/error.h"
40#include "lib/exceptions.h"
41#include "lib/null.h"
42#include "lib/source_file.h"
43
44// declare this outside of Visitor so it can be forward declared in node.h
46 // We maintain a linked list of Context structures on the stack
47 // in the Visitor::apply_visitor functions as we do the recursive
48 // descent traversal. pre/postorder function can access this
49 // context via getContext/findContext
50 const Visitor_Context *parent;
51 const IR::Node *node, *original;
52 mutable const char *child_name;
53 mutable int child_index;
54 int depth;
55 template <class T>
56 inline const T *findContext(const Visitor_Context *&c) const {
57 c = this;
58 while ((c = c->parent))
59 if (auto *rv = c->node->to<T>()) return rv;
60 return nullptr;
61 }
62 template <class T>
63 inline const T *findContext() const {
64 const Visitor_Context *c = this;
65 return findContext<T>(c);
66 }
67};
68
71class Inspector;
72
73class Visitor {
74 public:
76 class profile_t {
77 // for profiling -- a profile_t object is created when a pass
78 // starts and destroyed when it ends. Moveable but not copyable.
79 Visitor &v;
80 absl::Time start;
81 explicit profile_t(Visitor &);
82 friend class Visitor;
83
84 public:
85 profile_t() = delete;
86 profile_t(const profile_t &) = delete;
87 profile_t &operator=(const profile_t &) = delete;
88 profile_t &operator=(profile_t &&) = delete;
89
90 ~profile_t();
92 };
93 virtual ~Visitor() = default;
94
95 mutable cstring internalName;
96 // Some visitors are created and applied by other visitors.
97 // This field keeps track of the caller.
98 const Visitor *called_by = nullptr;
99
100 const Visitor &setCalledBy(const Visitor *visitor) {
101 called_by = visitor;
102 return *this;
103 }
104 // init_apply is called (once) when apply is called on an IR tree
105 // it expects to allocate a profile record which will be destroyed
106 // when the traversal completes. Visitor subclasses may extend this
107 // to do any additional initialization they need. They should call their
108 // parent's init_apply to do further initialization
109 virtual profile_t init_apply(const IR::Node *root);
110 profile_t init_apply(const IR::Node *root, const Context *parent_context);
111 // End_apply is called symmetrically with init_apply, after the visit
112 // is completed. Both functions will be called in the event of a normal
113 // completion, but only the 0-argument version will be called in the event
114 // of an exception, as there is no root in that case.
115 virtual void end_apply();
116 virtual void end_apply(const IR::Node *root);
117
118 // apply_visitor is the main traversal function that manages the
119 // depth-first recursive traversal. `visit` is a convenience function
120 // that calls it on a variable.
121 virtual const IR::Node *apply_visitor(const IR::Node *n, const char *name = 0) = 0;
122 void visit(const IR::Node *&n, const char *name = 0) { n = apply_visitor(n, name); }
123 void visit(const IR::Node *const &n, const char *name = 0) {
124 auto t = apply_visitor(n, name);
125 if (t != n) visitor_const_error();
126 }
127 void visit(const IR::Node *&n, const char *name, int cidx) {
128 ctxt->child_index = cidx;
129 n = apply_visitor(n, name);
130 }
131 void visit(const IR::Node *const &n, const char *name, int cidx) {
132 ctxt->child_index = cidx;
133 auto t = apply_visitor(n, name);
134 if (t != n) visitor_const_error();
135 }
136 void visit(IR::Node *&, const char * = 0, int = 0) { BUG("Can't visit non-const pointer"); }
137#define DECLARE_VISIT_FUNCTIONS(CLASS, BASE) \
138 void visit(const IR::CLASS *&n, const char *name = 0); \
139 void visit(const IR::CLASS *const &n, const char *name = 0); \
140 void visit(const IR::CLASS *&n, const char *name, int cidx); \
141 void visit(const IR::CLASS *const &n, const char *name, int cidx); \
142 void visit(IR::CLASS *&, const char * = 0, int = 0) { BUG("Can't visit non-const pointer"); }
143 IRNODE_ALL_SUBCLASSES(DECLARE_VISIT_FUNCTIONS)
144#undef DECLARE_VISIT_FUNCTIONS
145 void visit(IR::Node &n, const char *name = 0) {
146 if (name && ctxt) ctxt->child_name = name;
147 n.visit_children(*this);
148 }
149 void visit(const IR::Node &n, const char *name = 0) {
150 if (name && ctxt) ctxt->child_name = name;
151 n.visit_children(*this);
152 }
153 void visit(IR::Node &n, const char *name, int cidx) {
154 if (ctxt) {
155 ctxt->child_name = name;
156 ctxt->child_index = cidx;
157 }
158 n.visit_children(*this);
159 }
160 void visit(const IR::Node &n, const char *name, int cidx) {
161 if (ctxt) {
162 ctxt->child_name = name;
163 ctxt->child_index = cidx;
164 }
165 n.visit_children(*this);
166 }
167 template <class T>
168 void parallel_visit(IR::Vector<T> &v, const char *name = 0) {
169 if (name && ctxt) ctxt->child_name = name;
170 v.parallel_visit_children(*this);
171 }
172 template <class T>
173 void parallel_visit(const IR::Vector<T> &v, const char *name = 0) {
174 if (name && ctxt) ctxt->child_name = name;
175 v.parallel_visit_children(*this);
176 }
177 template <class T>
178 void parallel_visit(IR::Vector<T> &v, const char *name, int cidx) {
179 if (ctxt) {
180 ctxt->child_name = name;
181 ctxt->child_index = cidx;
182 }
183 v.parallel_visit_children(*this);
184 }
185 template <class T>
186 void parallel_visit(const IR::Vector<T> &v, const char *name, int cidx) {
187 if (ctxt) {
188 ctxt->child_name = name;
189 ctxt->child_index = cidx;
190 }
191 v.parallel_visit_children(*this);
192 }
193
194 virtual Visitor *clone() const {
195 BUG("need %s::clone method", name());
196 return nullptr;
197 }
198 virtual bool check_clone(const Visitor *a) { return typeid(*this) == typeid(*a); }
199
200 // Functions for IR visit_children to call for ControlFlowVisitors.
201 virtual ControlFlowVisitor *controlFlowVisitor() { return nullptr; }
202 virtual Visitor &flow_clone() { return *this; }
203 // all flow_clones share a split_link chain to allow stack walking
204 SplitFlowVisit_base *split_link_mem = nullptr, *&split_link;
205 Visitor() : split_link(split_link_mem) {}
206
210 virtual void flow_merge(Visitor &) {}
211 virtual bool flow_merge_closure(Visitor &) { BUG("%s pass does not support loops", name()); }
214 virtual void flow_merge_global_from(cstring) {}
215 virtual void erase_global(cstring) {}
216 virtual bool check_global(cstring) { return false; }
217 virtual void clear_globals() {}
218 virtual bool has_flow_joins() const { return false; }
219
220 static cstring demangle(const char *);
221 virtual const char *name() const {
222 if (!internalName) internalName = demangle(typeid(*this).name());
223 return internalName.c_str();
224 }
225 void setName(const char *name) { internalName = cstring(name); }
226 void print_context() const; // for debugging; can be called from debugger
227
228 // Context access/search functions. getContext returns the context
229 // that refers to the immediate parent of the node currently being
230 // visited. findContext searches up the context for a (grand)parent
231 // of a specific type. Orig versions return the original node (before
232 // any cloning or modifications)
233 const IR::Node *getOriginal() const { return ctxt->original; }
234 template <class T>
235 const T *getOriginal() const {
236 CHECK_NULL(ctxt->original);
237 BUG_CHECK(ctxt->original->is<T>(), "%1% does not have the expected type %2%",
238 ctxt->original, demangle(typeid(T).name()));
239 return ctxt->original->to<T>();
240 }
241 const Context *getChildContext() const { return ctxt; }
242 const Context *getContext() const { return ctxt->parent; }
243 template <class T>
244 const T *getParent() const {
245 return ctxt->parent ? ctxt->parent->node->to<T>() : nullptr;
246 }
247 int getChildrenVisited() const { return ctxt->child_index; }
248 int getContextDepth() const { return ctxt->depth - 1; }
249 template <class T>
250 inline const T *findContext(const Context *&c) const {
251 if (!c) c = ctxt;
252 if (!c) return nullptr;
253 while ((c = c->parent))
254 if (auto *rv = c->node->to<T>()) return rv;
255 return nullptr;
256 }
257 template <class T>
258 inline const T *findContext() const {
259 const Context *c = ctxt;
260 return findContext<T>(c);
261 }
262 template <class T>
263 inline const T *findOrigCtxt(const Context *&c) const {
264 if (!c) c = ctxt;
265 if (!c) return nullptr;
266 while ((c = c->parent))
267 if (auto *rv = c->original->to<T>()) return rv;
268 return nullptr;
269 }
270 template <class T>
271 inline const T *findOrigCtxt() const {
272 const Context *c = ctxt;
273 return findOrigCtxt<T>(c);
274 }
275 inline bool isInContext(const IR::Node *n) const {
276 for (auto *c = ctxt; c; c = c->parent) {
277 if (c->node == n || c->original == n) return true;
278 }
279 return false;
280 }
281
285 const IR::Node *getCurrentNode() const { return ctxt->node; }
286 template <class T>
287 const T *getCurrentNode() const {
288 return ctxt->node ? ctxt->node->to<T>() : nullptr;
289 }
290
294 bool warning_enabled(int warning_kind) const { return warning_enabled(this, warning_kind); }
297 static bool warning_enabled(const Visitor *visitor, int warning_kind);
298 template <class T, typename = std::enable_if_t<Util::has_SourceInfo_v<T>>, class... Args>
299 void warn(const int kind, const char *format, const T *node, Args &&...args) {
300 if (warning_enabled(kind)) ::warning(kind, format, node, std::forward<Args>(args)...);
301 }
302
304 template <class T, typename = std::enable_if_t<Util::has_SourceInfo_v<T>>, class... Args>
305 void warn(const int kind, const char *format, const T &node, Args &&...args) {
306 if (warning_enabled(kind)) ::warning(kind, format, node, std::forward<Args>(args)...);
307 }
308
309 protected:
310 // if visitDagOnce is set to 'false' (usually in the derived Visitor
311 // class constructor), nodes that appear multiple times in the tree
312 // will be visited repeatedly. If this is done in a Modifier or Transform
313 // pass, this will result in them being duplicated if they are modified.
314 bool visitDagOnce = true;
315 bool dontForwardChildrenBeforePreorder = false;
316 // if joinFlows is 'true', Visitor will track nodes with more than one parent and
317 // flow_merge the visitor from all the parents before visiting the node and its
318 // children. This only works for Inspector (not Modifier/Transform) currently.
319 bool joinFlows = false;
320
321 virtual void init_join_flows(const IR::Node *) {
322 BUG("joinFlows only supported in ControlFlowVisitor currently");
323 }
324
345 virtual bool join_flows(const IR::Node *) { return false; }
348 virtual void post_join_flows(const IR::Node *, const IR::Node *) {}
349
350 void visit_children(const IR::Node *, std::function<void()> fn) { fn(); }
351 class Tracker; // used by Inspector -- private to it
352 class ChangeTracker; // used by Modifier and Transform -- private to them
353 // This overrides visitDagOnce for a single node -- can only be called from
354 // preorder and postorder functions
355 // FIXME: It would be better named visitCurrentOnce() / visitCurrenAgain()
356 virtual void visitOnce() const { BUG("do not know how to handle request"); }
357 virtual void visitAgain() const { BUG("do not know how to handle request"); }
358
359 private:
360 virtual void visitor_const_error();
361 const Context *ctxt = nullptr; // should be readonly to subclasses
362 friend class Inspector;
363 friend class Modifier;
364 friend class Transform;
365 friend class ControlFlowVisitor;
366};
367
368class Modifier : public virtual Visitor {
369 std::shared_ptr<ChangeTracker> visited;
370 void visitor_const_error() override;
371 bool check_clone(const Visitor *) override;
372
373 public:
374 profile_t init_apply(const IR::Node *root) override;
375 const IR::Node *apply_visitor(const IR::Node *n, const char *name = 0) override;
376 virtual bool preorder(IR::Node *) { return true; }
377 virtual void postorder(IR::Node *) {}
378 virtual void revisit(const IR::Node *, const IR::Node *) {}
379 virtual void loop_revisit(const IR::Node *) { BUG("IR loop detected"); }
380#define DECLARE_VISIT_FUNCTIONS(CLASS, BASE) \
381 virtual bool preorder(IR::CLASS *); \
382 virtual void postorder(IR::CLASS *); \
383 virtual void revisit(const IR::CLASS *, const IR::CLASS *); \
384 virtual void loop_revisit(const IR::CLASS *);
385 IRNODE_ALL_SUBCLASSES(DECLARE_VISIT_FUNCTIONS)
386#undef DECLARE_VISIT_FUNCTIONS
387 void revisit_visited();
388 bool visit_in_progress(const IR::Node *) const;
389 void visitOnce() const override;
390 void visitAgain() const override;
391
392 protected:
393 bool forceClone = false; // force clone whole tree even if unchanged
394};
395
396class Inspector : public virtual Visitor {
397 std::shared_ptr<Tracker> visited;
398 bool check_clone(const Visitor *) override;
399
400 public:
401 profile_t init_apply(const IR::Node *root) override;
402 const IR::Node *apply_visitor(const IR::Node *, const char *name = 0) override;
403 virtual bool preorder(const IR::Node *) { return true; } // return 'false' to prune
404 virtual void postorder(const IR::Node *) {}
405 virtual void revisit(const IR::Node *) {}
406 virtual void loop_revisit(const IR::Node *) { BUG("IR loop detected"); }
407#define DECLARE_VISIT_FUNCTIONS(CLASS, BASE) \
408 virtual bool preorder(const IR::CLASS *); \
409 virtual void postorder(const IR::CLASS *); \
410 virtual void revisit(const IR::CLASS *); \
411 virtual void loop_revisit(const IR::CLASS *);
412 IRNODE_ALL_SUBCLASSES(DECLARE_VISIT_FUNCTIONS)
413#undef DECLARE_VISIT_FUNCTIONS
414 void revisit_visited();
415 bool visit_in_progress(const IR::Node *n) const;
416 void visitOnce() const override;
417 void visitAgain() const override;
418};
419
420class Transform : public virtual Visitor {
421 std::shared_ptr<ChangeTracker> visited;
422 bool prune_flag = false;
423 void visitor_const_error() override;
424 bool check_clone(const Visitor *) override;
425
426 public:
427 profile_t init_apply(const IR::Node *root) override;
428 const IR::Node *apply_visitor(const IR::Node *, const char *name = 0) override;
429 virtual const IR::Node *preorder(IR::Node *n) { return n; }
430 virtual const IR::Node *postorder(IR::Node *n) { return n; }
431 virtual void revisit(const IR::Node *, const IR::Node *) {}
432 virtual void loop_revisit(const IR::Node *) { BUG("IR loop detected"); }
433#define DECLARE_VISIT_FUNCTIONS(CLASS, BASE) \
434 virtual const IR::Node *preorder(IR::CLASS *); \
435 virtual const IR::Node *postorder(IR::CLASS *); \
436 virtual void revisit(const IR::CLASS *, const IR::Node *); \
437 virtual void loop_revisit(const IR::CLASS *);
438 IRNODE_ALL_SUBCLASSES(DECLARE_VISIT_FUNCTIONS)
439#undef DECLARE_VISIT_FUNCTIONS
440 void revisit_visited();
441 bool visit_in_progress(const IR::Node *) const;
442 void visitOnce() const override;
443 void visitAgain() const override;
444 // can only be called usefully from a 'preorder' function (directly or indirectly)
445 void prune() { prune_flag = true; }
446
447 protected:
448 const IR::Node *transform_child(const IR::Node *child) {
449 auto *rv = apply_visitor(child);
450 prune_flag = true;
451 return rv;
452 }
453 bool forceClone = false; // force clone whole tree even if unchanged
454};
455
456// turn this on for extra info tracking control joinFlows for debugging
457#define DEBUG_FLOW_JOIN 0
458
459class ControlFlowVisitor : public virtual Visitor {
460 std::map<cstring, ControlFlowVisitor &> &globals;
461
462 protected:
463 ControlFlowVisitor *clone() const override = 0;
465 ControlFlowVisitor *vclone = nullptr; // a clone to accumulate info into
466 int count = 0; // additional parents needed -1
467 bool done = false;
468#if DEBUG_FLOW_JOIN
469 struct ctrs_t {
470 int exist = 0, visited = 0;
471 };
472 std::map<const IR::Node *, ctrs_t> parents;
473#endif
474 };
475 typedef std::map<const IR::Node *, flow_join_info_t> flow_join_points_t;
476 friend std::ostream &operator<<(std::ostream &, const flow_join_info_t &);
477 friend std::ostream &operator<<(std::ostream &, const flow_join_points_t &);
478 friend void dump(const flow_join_info_t &);
479 friend void dump(const flow_join_info_t *);
480 friend void dump(const flow_join_points_t &);
481 friend void dump(const flow_join_points_t *);
482
483 // Flag set by visit_children of nodes that are unconditional branches, to denote that
484 // the control flow is currently unreachable. Future flow_merges to this visitor should
485 // clear this if merging a reachable state.
486 bool unreachable = false;
487
488 flow_join_points_t *flow_join_points = 0;
489 class SetupJoinPoints : public Inspector {
490 protected:
491 flow_join_points_t &join_points;
492 bool preorder(const IR::Node *n) override {
493 BUG_CHECK(join_points.count(n) == 0, "oops");
494#if DEBUG_FLOW_JOIN
495 auto *ctxt = getContext();
496 join_points[n].parents[ctxt ? ctxt->original : nullptr].exist++;
497#endif
498 return true;
499 }
500 void revisit(const IR::Node *n) override {
501 ++join_points[n].count;
502#if DEBUG_FLOW_JOIN
503 auto *ctxt = getContext();
504 join_points[n].parents[ctxt ? ctxt->original : nullptr].exist++;
505#endif
506 }
507
508 public:
509 explicit SetupJoinPoints(flow_join_points_t &fjp) : join_points(fjp) {}
510 };
511 bool BackwardsCompatibleBroken = false; // enable old broken behavior for code that
512 // depends on it. Should not be set for new uses.
513
514 virtual void applySetupJoinPoints(const IR::Node *root) {
515 root->apply(SetupJoinPoints(*flow_join_points));
516 }
517 void init_join_flows(const IR::Node *root) override;
518 bool join_flows(const IR::Node *n) override;
519 void post_join_flows(const IR::Node *, const IR::Node *) override;
520
526 virtual bool filter_join_point(const IR::Node *) { return false; }
527 ControlFlowVisitor() : globals(*new std::map<cstring, ControlFlowVisitor &>) {}
528
529 public:
530 ControlFlowVisitor *controlFlowVisitor() override { return this; }
531 ControlFlowVisitor &flow_clone() override;
532 void flow_merge(Visitor &) override = 0;
533 virtual void flow_copy(ControlFlowVisitor &) = 0;
534 virtual bool operator==(const ControlFlowVisitor &) const {
535 BUG("%s pass does not support loops", name());
536 }
537 bool operator!=(const ControlFlowVisitor &v) const { return !(*this == v); }
538 void setUnreachable() { unreachable = true; }
539 bool isUnreachable() { return unreachable; }
540 void flow_merge_global_to(cstring key) override {
541 if (globals.count(key))
542 globals.at(key).flow_merge(*this);
543 else
544 globals.emplace(key, flow_clone());
545 }
546 void flow_merge_global_from(cstring key) override {
547 if (globals.count(key)) flow_merge(globals.at(key));
548 }
549 void erase_global(cstring key) override { globals.erase(key); }
550 bool check_global(cstring key) override { return globals.count(key) != 0; }
551 void clear_globals() override { globals.clear(); }
552 std::pair<cstring, ControlFlowVisitor *> save_global(cstring key) {
553 ControlFlowVisitor *cfv = nullptr;
554 if (auto i = globals.find(key); i != globals.end()) {
555 cfv = &i->second;
556 globals.erase(i);
557 }
558 return std::make_pair(key, cfv);
559 }
560 void restore_global(std::pair<cstring, ControlFlowVisitor *> saved) {
561 globals.erase(saved.first);
562 if (saved.second) globals.emplace(saved.first, *saved.second);
563 }
564
567 Visitor &self;
568 cstring key;
569
570 public:
571 GuardGlobal(Visitor &self, cstring key) : self(self), key(key) {
572 BUG_CHECK(!self.check_global(key), "ControlFlowVisitor global %s in use", key);
573 }
574 ~GuardGlobal() { self.erase_global(key); }
575 };
578 ControlFlowVisitor &self;
579 std::vector<std::pair<cstring, ControlFlowVisitor *>> saved;
580
581 public:
582 SaveGlobal(ControlFlowVisitor &self, cstring key) : self(self) {
583 saved.push_back(self.save_global(key));
584 }
585 SaveGlobal(ControlFlowVisitor &self, cstring k1, cstring k2) : self(self) {
586 saved.push_back(self.save_global(k1));
587 saved.push_back(self.save_global(k2));
588 }
589 ~SaveGlobal() {
590 for (auto it = saved.rbegin(); it != saved.rend(); ++it) self.restore_global(*it);
591 }
592 };
593
594 bool has_flow_joins() const override { return !!flow_join_points; }
595 const flow_join_info_t *flow_join_status(const IR::Node *n) const {
596 if (!flow_join_points || !flow_join_points->count(n)) return nullptr;
597 return &flow_join_points->at(n);
598 }
599};
600
610 protected:
611 Visitor &v;
613 std::vector<Visitor *> visitors;
614 int visit_next = 0, start_index = 0;
615 bool paused = false;
616 friend ControlFlowVisitor;
617
618 explicit SplitFlowVisit_base(Visitor &v) : v(v) {
619 prev = v.split_link;
620 v.split_link = this;
621 }
622 ~SplitFlowVisit_base() { v.split_link = prev; }
623 void *operator new(size_t); // declared and not defined, as this class can
624 // only be instantiated on the stack. Trying to allocate one on the heap will
625 // cause a linker error.
626
627 public:
628 bool finished() { return size_t(visit_next) >= visitors.size(); }
629 virtual bool ready() { return !finished() && !paused; }
630 void pause() { paused = true; }
631 void unpause() { paused = false; }
632 virtual void do_visit() = 0;
633 virtual void run_visit() {
634 auto *ctxt = v.getChildContext();
635 start_index = ctxt ? ctxt->child_index : 0;
636 paused = false;
637 while (!finished()) do_visit();
638 for (auto *cl : visitors) {
639 if (cl && cl != &v) v.flow_merge(*cl);
640 }
641 }
642 virtual void dbprint(std::ostream &) const = 0;
643 friend void dump(const SplitFlowVisit_base *);
644};
645
646template <class N>
648 std::vector<const N **> nodes;
649 std::vector<const N *const *> const_nodes;
650
651 public:
652 explicit SplitFlowVisit(Visitor &v) : SplitFlowVisit_base(v) {}
653 void addNode(const N *&node) {
654 BUG_CHECK(const_nodes.empty(), "Mixing const and non-const in SplitFlowVisit");
655 BUG_CHECK(visitors.size() == nodes.size(), "size mismatch in SplitFlowVisit");
656 BUG_CHECK(visit_next == 0, "Can't addNode to SplitFlowVisit after visiting started");
657 visitors.push_back(visitors.empty() ? &v : &v.flow_clone());
658 nodes.emplace_back(&node);
659 }
660 void addNode(const N *const &node) {
661 BUG_CHECK(nodes.empty(), "Mixing const and non-const in SplitFlowVisit");
662 BUG_CHECK(visitors.size() == const_nodes.size(), "size mismatch in SplitFlowVisit");
663 BUG_CHECK(visit_next == 0, "Can't addNode to SplitFlowVisit after visiting started");
664 visitors.push_back(visitors.empty() ? &v : &v.flow_clone());
665 const_nodes.emplace_back(&node);
666 }
667 template <class T1, class T2, class... Args>
668 void addNode(T1 &&t1, T2 &&t2, Args &&...args) {
669 addNode(std::forward<T1>(t1));
670 addNode(std::forward<T2>(t2), std::forward<Args>(args)...);
671 }
672 template <class... Args>
673 explicit SplitFlowVisit(Visitor &v, Args &&...args) : SplitFlowVisit(v) {
674 addNode(std::forward<Args>(args)...);
675 }
676 void do_visit() override {
677 if (!finished()) {
678 BUG_CHECK(!paused, "trying to visit paused split_flow_visitor");
679 int idx = visit_next++;
680 if (nodes.empty())
681 visitors.at(idx)->visit(*const_nodes.at(idx), nullptr, start_index + idx);
682 else
683 visitors.at(idx)->visit(*nodes.at(idx), nullptr, start_index + idx);
684 }
685 }
686 void dbprint(std::ostream &out) const override {
687 out << "SplitFlowVisit processed " << visit_next << " of " << visitors.size();
688 }
689};
690
691template <class N>
693 IR::Vector<N> *vec = nullptr;
694 const IR::Vector<N> *const_vec = nullptr;
695 std::vector<const IR::Node *> result;
696 void init_visit(size_t size) {
697 if (size > 0) visitors.push_back(&v);
698 while (visitors.size() < size) visitors.push_back(&v.flow_clone());
699 }
700
701 public:
703 result.resize(vec.size());
704 init_visit(vec.size());
705 }
707 : SplitFlowVisit_base(v), const_vec(&vec) {
708 init_visit(vec.size());
709 }
710 void do_visit() override {
711 if (!finished()) {
712 BUG_CHECK(!paused, "trying to visit paused split_flow_visitor");
713 int idx = visit_next++;
714 if (vec)
715 result[idx] = visitors.at(idx)->apply_visitor(vec->at(idx));
716 else
717 visitors.at(idx)->visit(const_vec->at(idx), nullptr, start_index + idx);
718 }
719 }
720 void run_visit() override {
721 SplitFlowVisit_base::run_visit();
722 if (vec) {
723 int idx = 0;
724 for (auto i = vec->begin(); i != vec->end(); ++idx) {
725 if (!result[idx] && *i) {
726 i = vec->erase(i);
727 } else if (result[idx] == *i) {
728 ++i;
729 } else if (auto l = result[idx]->template to<IR::Vector<N>>()) {
730 i = vec->erase(i);
731 i = vec->insert(i, l->begin(), l->end());
732 i += l->size();
733 } else if (auto v = result[idx]->template to<IR::VectorBase>()) {
734 if (v->empty()) {
735 i = vec->erase(i);
736 } else {
737 i = vec->insert(i, v->size() - 1, nullptr);
738 for (auto el : *v) {
739 CHECK_NULL(el);
740 if (auto e = el->template to<N>())
741 *i++ = e;
742 else
743 BUG("visitor returned invalid type %s for Vector<%s>",
744 el->node_type_name(), N::static_type_name());
745 }
746 }
747 } else if (auto e = result[idx]->template to<N>()) {
748 *i++ = e;
749 } else {
750 CHECK_NULL(result[idx]);
751 BUG("visitor returned invalid type %s for Vector<%s>",
752 result[idx]->node_type_name(), N::static_type_name());
753 }
754 }
755 }
756 }
757 void dbprint(std::ostream &out) const override {
758 out << "SplitFlowVisitVector processed " << visit_next << " of " << visitors.size();
759 }
760};
761
762class Backtrack : public virtual Visitor {
763 public:
764 struct trigger : public ICastable {
765 enum type_t { OK, OTHER } type;
766 explicit trigger(type_t t) : type(t) {}
767 virtual ~trigger();
768 virtual void dbprint(std::ostream &out) const { out << demangle(typeid(*this).name()); }
769
770 protected:
771 // must call this from the constructor if a trigger subclass contains pointers
772 // or references to GC objects
773 void register_for_gc(size_t);
774
775 DECLARE_TYPEINFO(trigger);
776 };
777 virtual bool backtrack(trigger &trig) = 0;
778 virtual bool never_backtracks() { return false; } // generally not overridden
779 // returns true for passes that will never catch a trigger (backtrack() is always false)
780};
781
782class P4WriteContext : public virtual Visitor {
783 public:
784 bool isWrite(bool root_value = false); // might write based on context
785 bool isRead(bool root_value = false); // might read based on context
786 // note that the context might (conservatively) return true for BOTH isWrite and isRead,
787 // as it might be an 'inout' access or it might be unable to decide.
788};
789
794template <typename NodeType, typename Func>
795void forAllMatching(const IR::Node *root, Func &&function) {
796 struct NodeVisitor : public Inspector {
797 explicit NodeVisitor(Func &&function) : function(function) {}
798 Func function;
799 void postorder(const NodeType *node) override { function(node); }
800 };
801 root->apply(NodeVisitor(std::forward<Func>(function)));
802}
803
810template <typename NodeType, typename RootType, typename Func>
811const RootType *modifyAllMatching(const RootType *root, Func &&function) {
812 struct NodeVisitor : public Modifier {
813 explicit NodeVisitor(Func &&function) : function(function) {}
814 Func function;
815 void postorder(NodeType *node) override { function(node); }
816 };
817 return root->apply(NodeVisitor(std::forward<Func>(function)));
818}
819
826template <typename NodeType, typename Func>
827const IR::Node *transformAllMatching(const IR::Node *root, Func &&function) {
828 struct NodeVisitor : public Transform {
829 explicit NodeVisitor(Func &&function) : function(function) {}
830 Func function;
831 const IR::Node *postorder(NodeType *node) override { return function(node); }
832 };
833 return root->apply(NodeVisitor(std::forward<Func>(function)));
834}
835
836#endif /* IR_VISITOR_H_ */
Definition visitor.h:762
RAII class to ensure global key is only used in one place.
Definition visitor.h:566
RAII class to save and restore one or more global keys.
Definition visitor.h:577
Definition visitor.h:489
Definition visitor.h:459
void flow_merge_global_to(cstring key) override
Definition visitor.h:540
void flow_merge(Visitor &) override=0
bool join_flows(const IR::Node *n) override
Definition visitor.cpp:652
virtual bool filter_join_point(const IR::Node *)
Definition visitor.h:526
void post_join_flows(const IR::Node *, const IR::Node *) override
Definition visitor.cpp:712
Definition visitor.h:464
Definition castable.h:34
Definition node.h:93
Definition vector.h:56
Definition visitor.h:396
Definition visitor.h:368
Definition visitor.h:782
Definition visitor.h:609
Definition visitor.h:647
Definition visitor.h:692
Definition visitor.h:420
Definition visitor.h:76
Definition visitor.h:73
virtual void post_join_flows(const IR::Node *, const IR::Node *)
Definition visitor.h:348
bool warning_enabled(int warning_kind) const
Definition visitor.h:294
virtual void flow_merge(Visitor &)
Definition visitor.h:210
const IR::Node * getCurrentNode() const
Definition visitor.h:285
virtual void flow_merge_global_to(cstring)
Definition visitor.h:213
virtual bool join_flows(const IR::Node *)
Definition visitor.h:345
void warn(const int kind, const char *format, const T &node, Args &&...args)
The const ref variant of the above.
Definition visitor.h:305
Definition cstring.h:80
STL namespace.
Definition visitor.h:764
T * to() noexcept
Definition rtti.h:226
bool is() const noexcept
Definition rtti.h:216
Definition visitor.h:45