diff --git a/src/util/state_graph.cpp b/src/util/state_graph.cpp index baaf345d2..1a59dc4af 100644 --- a/src/util/state_graph.cpp +++ b/src/util/state_graph.cpp @@ -264,6 +264,7 @@ void state_graph::add_state(state s) { STRACE("state_graph", tout << "[state_graph] adding state " << s << ": ";); add_state_core(s); CASSERT("state_graph", check_invariant()); + STRACE("state_graph", write_dgml();); STRACE("state_graph", tout << std::endl;); } void state_graph::mark_live(state s) { @@ -273,6 +274,7 @@ void state_graph::mark_live(state s) { if (m_unexplored.contains(s)) mark_unknown_core(s); mark_live_recursive(s); CASSERT("state_graph", check_invariant()); + STRACE("state_graph", write_dgml();); STRACE("state_graph", tout << std::endl;); } void state_graph::add_edge(state s1, state s2, bool maybecycle) { @@ -285,6 +287,7 @@ void state_graph::add_edge(state s1, state s2, bool maybecycle) { add_edge_core(s1, s2, maybecycle); if (m_live.contains(s2)) mark_live(s1); CASSERT("state_graph", check_invariant()); + STRACE("state_graph", write_dgml();); STRACE("state_graph", tout << std::endl;); } void state_graph::mark_done(state s) { @@ -296,6 +299,7 @@ void state_graph::mark_done(state s) { s = merge_all_cycles(s); mark_dead_recursive(s); // check if dead CASSERT("state_graph", check_invariant()); + STRACE("state_graph", write_dgml();); STRACE("state_graph", tout << std::endl;); } @@ -408,3 +412,83 @@ std::ostream& state_graph::display(std::ostream& o) const { return o; } + +/* + Output the whole state graph in dgml format into the file '.z3-state-graph.dgml' + */ +void state_graph::write_dgml() { + std::ofstream dgml(".z3-state-graph.dgml"); + dgml << "" << std::endl + << "" << std::endl + << "" << std::endl; + for (auto s : m_seen) { + if (m_state_ufind.is_root(s)) { + dgml << "" << std::endl; + if (m_live.contains(s)) + dgml << "" << std::endl; + if (m_dead.contains(s)) + dgml << "" << std::endl; + if (m_unexplored.contains(s)) + dgml << "" << std::endl; + dgml << "" << std::endl; + } + } + dgml << "" << std::endl; + dgml << "" << std::endl; + for (auto s : m_seen) { + if (m_state_ufind.is_root(s)) + for (auto t : m_targets[s]) { + dgml << "" << std::endl; + if (!m_sources_maybecycle[t].contains(s)) + dgml << "" << std::endl; + dgml << "" << std::endl; + } + } + dgml << "" << std::endl; + dgml << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl + << "" << std::endl; + dgml.close(); +} diff --git a/src/util/state_graph.h b/src/util/state_graph.h index 190e3261a..9e754048e 100644 --- a/src/util/state_graph.h +++ b/src/util/state_graph.h @@ -114,6 +114,8 @@ private: bool check_invariant() const; #endif + void write_dgml(); + /* 'Core' functions that modify the plain graph, without updating SCCs or propagating live/dead state information.