summaryrefslogtreecommitdiff
path: root/src/expr/node_visitor.h
diff options
context:
space:
mode:
authorAlex Ozdemir <aozdemir@hmc.edu>2020-03-18 14:30:30 -0700
committerGitHub <noreply@github.com>2020-03-18 16:30:30 -0500
commitba3a69d7915292ddb649bdb8b4830623b337818c (patch)
tree43b5b3b2666bb69e621bbe5ab569e239c87590ba /src/expr/node_visitor.h
parent5c825235dd99b7c0767789db9d782e24c581ace5 (diff)
Move node visitor class from smt_util/ to expr/ (#4110)
Done by: Running rg 'smt_util/node_visitor' -l | xargs sed -i 's/smt_util\/node_visitor/expr\/node_visitor/' in src to change the #includes Moving the file Changing src/expr/CMakeLists.txt and src/CMakeLists.txt clang-format, omitting node_visitor.h. In reference to discussion, here.
Diffstat (limited to 'src/expr/node_visitor.h')
-rw-r--r--src/expr/node_visitor.h126
1 files changed, 126 insertions, 0 deletions
diff --git a/src/expr/node_visitor.h b/src/expr/node_visitor.h
new file mode 100644
index 000000000..47ed6eff8
--- /dev/null
+++ b/src/expr/node_visitor.h
@@ -0,0 +1,126 @@
+/********************* */
+/*! \file node_visitor.h
+ ** \verbatim
+ ** Top contributors (to current version):
+ ** Dejan Jovanovic, Morgan Deters, Liana Hadarean
+ ** This file is part of the CVC4 project.
+ ** Copyright (c) 2009-2019 by the authors listed in the file AUTHORS
+ ** in the top-level source directory) and their institutional affiliations.
+ ** All rights reserved. See the file COPYING in the top-level source
+ ** directory for licensing information.\endverbatim
+ **
+ ** \brief A simple visitor for nodes
+ **
+ ** A simple visitor for nodes.
+ **/
+
+#pragma once
+
+#include "cvc4_private.h"
+
+#include <vector>
+
+#include "expr/node.h"
+
+namespace CVC4 {
+
+/**
+ * Traverses the nodes reverse-topologically (children before parents),
+ * calling the visitor in order.
+ */
+template<typename Visitor>
+class NodeVisitor {
+
+ /** For re-entry checking */
+ static thread_local bool s_inRun;
+
+ /**
+ * Guard against NodeVisitor<> being re-entrant.
+ */
+ template <class T>
+ class GuardReentry {
+ T& d_guard;
+ public:
+ GuardReentry(T& guard)
+ : d_guard(guard) {
+ Assert(!d_guard);
+ d_guard = true;
+ }
+ ~GuardReentry() {
+ Assert(d_guard);
+ d_guard = false;
+ }
+ };/* class NodeVisitor<>::GuardReentry */
+
+public:
+
+ /**
+ * Element of the stack.
+ */
+ struct stack_element {
+ /** The node to be visited */
+ TNode d_node;
+ /** The parent of the node */
+ TNode d_parent;
+ /** Have the children been queued up for visitation */
+ bool d_childrenAdded;
+ stack_element(TNode node, TNode parent)
+ : d_node(node), d_parent(parent), d_childrenAdded(false)
+ {
+ }
+ };/* struct preprocess_stack_element */
+
+ /**
+ * Performs the traversal.
+ */
+ static typename Visitor::return_type run(Visitor& visitor, TNode node) {
+
+ GuardReentry<bool> guard(s_inRun);
+
+ // Notify of a start
+ visitor.start(node);
+
+ // Do a reverse-topological sort of the subexpressions
+ std::vector<stack_element> toVisit;
+ toVisit.push_back(stack_element(node, node));
+ while (!toVisit.empty()) {
+ stack_element& stackHead = toVisit.back();
+ // The current node we are processing
+ TNode current = stackHead.d_node;
+ TNode parent = stackHead.d_parent;
+
+ if (visitor.alreadyVisited(current, parent)) {
+ // If already visited, we're done
+ toVisit.pop_back();
+ }
+ else if (stackHead.d_childrenAdded)
+ {
+ // Call the visitor
+ visitor.visit(current, parent);
+ // Done with this node, remove from the stack
+ toVisit.pop_back();
+ }
+ else
+ {
+ // Mark that we have added the children
+ stackHead.d_childrenAdded = true;
+ // We need to add the children
+ for(TNode::iterator child_it = current.begin(); child_it != current.end(); ++ child_it) {
+ TNode childNode = *child_it;
+ if (!visitor.alreadyVisited(childNode, current)) {
+ toVisit.push_back(stack_element(childNode, current));
+ }
+ }
+ }
+ }
+
+ // Notify that we're done
+ return visitor.done(node);
+ }
+
+};/* class NodeVisitor<> */
+
+template <typename Visitor>
+thread_local bool NodeVisitor<Visitor>::s_inRun = false;
+
+}/* CVC4 namespace */
generated by cgit on debian on lair
contact matthew@masot.net with questions or feedback