3
0
Fork 0
mirror of https://github.com/Z3Prover/z3 synced 2025-04-06 17:44:08 +00:00

added support for saving state graph in dot format (#4621)

* added support for saving state graph in dot format

* moved write_dgml and write_dot under CASSERT

* updated dgml and dot generation a bit so that a state that is both and alive state is detected as having green background but red border when the invariant is vioalted
This commit is contained in:
Margus Veanes 2020-08-09 14:53:29 -07:00 committed by GitHub
parent 3852d4516d
commit e5693b8a98
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 65 additions and 9 deletions

View file

@ -263,8 +263,9 @@ void state_graph::add_state(state s) {
if (m_seen.contains(s)) return;
STRACE("state_graph", tout << "[state_graph] adding state " << s << ": ";);
add_state_core(s);
CASSERT("state_graph", write_dgml());
CASSERT("state_graph", write_dot());
CASSERT("state_graph", check_invariant());
STRACE("state_graph", write_dgml(););
STRACE("state_graph", tout << std::endl;);
}
void state_graph::mark_live(state s) {
@ -273,8 +274,9 @@ void state_graph::mark_live(state s) {
SASSERT(m_state_ufind.is_root(s));
if (m_unexplored.contains(s)) mark_unknown_core(s);
mark_live_recursive(s);
CASSERT("state_graph", write_dgml());
CASSERT("state_graph", write_dot());
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) {
@ -286,8 +288,9 @@ void state_graph::add_edge(state s1, state s2, bool maybecycle) {
s2 = m_state_ufind.find(s2);
add_edge_core(s1, s2, maybecycle);
if (m_live.contains(s2)) mark_live(s1);
CASSERT("state_graph", write_dgml());
CASSERT("state_graph", write_dot());
CASSERT("state_graph", check_invariant());
STRACE("state_graph", write_dgml(););
STRACE("state_graph", tout << std::endl;);
}
void state_graph::mark_done(state s) {
@ -298,8 +301,9 @@ void state_graph::mark_done(state s) {
if (m_unexplored.contains(s)) mark_unknown_core(s);
s = merge_all_cycles(s);
mark_dead_recursive(s); // check if dead
CASSERT("state_graph", write_dgml());
CASSERT("state_graph", write_dot());
CASSERT("state_graph", check_invariant());
STRACE("state_graph", write_dgml(););
STRACE("state_graph", tout << std::endl;);
}
@ -413,10 +417,11 @@ std::ostream& state_graph::display(std::ostream& o) const {
return o;
}
#ifdef Z3DEBUG
/*
Output the whole state graph in dgml format into the file '.z3-state-graph.dgml'
*/
void state_graph::write_dgml() {
bool state_graph::write_dgml() {
std::ofstream dgml(".z3-state-graph.dgml");
dgml << "<?xml version=\"1.0\" encoding=\"utf-8\"?>" << std::endl
<< "<DirectedGraph xmlns=\"http://schemas.microsoft.com/vs/2009/dgml\" GraphDirection=\"TopToBottom\">" << std::endl
@ -432,10 +437,10 @@ void state_graph::write_dgml() {
r = m_state_ufind.next(r);
} while (r != s);
dgml << "\" Category=\"State\">" << std::endl;
if (m_live.contains(s))
dgml << "<Category Ref=\"Alive\"/>" << std::endl;
if (m_dead.contains(s))
dgml << "<Category Ref=\"Dead\"/>" << std::endl;
if (m_live.contains(s))
dgml << "<Category Ref=\"Alive\"/>" << std::endl;
if (m_unexplored.contains(s))
dgml << "<Category Ref=\"Unexplored\"/>" << std::endl;
dgml << "</Node>" << std::endl;
@ -469,6 +474,7 @@ void state_graph::write_dgml() {
<< "<Style TargetType=\"Node\" GroupLabel=\"Dead\" ValueLabel=\"True\">" << std::endl
<< "<Condition Expression=\"HasCategory('Dead')\"/>" << std::endl
<< "<Setter Property=\"Background\" Value=\"tomato\"/>" << std::endl
<< "<Setter Property=\"Stroke\" Value=\"tomato\"/>" << std::endl
<< "</Style>" << std::endl
<< "<Style TargetType=\"Node\" GroupLabel=\"Unexplored\" ValueLabel=\"True\">" << std::endl
<< "<Condition Expression=\"HasCategory('Unexplored')\"/>" << std::endl
@ -491,4 +497,48 @@ void state_graph::write_dgml() {
<< "</Styles>" << std::endl
<< "</DirectedGraph>" << std::endl;
dgml.close();
return true;
}
/*
Output the whole state graph in dot format into the file '.z3-state-graph.dot'
*/
bool state_graph::write_dot() {
std::ofstream dot(".z3-state-graph.dot");
dot << "digraph \"state_graph\" {" << std::endl
<< "rankdir=TB" << std::endl
<< "node [peripheries=1,style=filled,fillcolor=white,fontsize=24]" << std::endl;
for (auto s : m_seen) {
if (m_state_ufind.is_root(s)) {
dot << s << " [label=\"";
auto r = s;
dot << r;
do {
if (r != s)
dot << "," << r;
r = m_state_ufind.next(r);
} while (r != s);
dot << "\"";
if (m_unexplored.contains(s))
dot << ",style=\"dashed,filled\"";
if (m_dead.contains(s))
dot << ",color=tomato,fillcolor=tomato";
if (m_live.contains(s))
dot << ",fillcolor=green";
dot << "]" << std::endl;
}
}
for (auto s : m_seen) {
if (m_state_ufind.is_root(s))
for (auto t : m_targets[s]) {
dot << s << " -> " << t;
if (!m_sources_maybecycle[t].contains(s))
dot << "[style=bold]";
dot << std::endl;
}
}
dot << "}" << std::endl;
dot.close();
return true;
}
#endif

View file

@ -112,10 +112,16 @@ private:
bool is_subset(state_set set1, state_set set2) const;
bool is_disjoint(state_set set1, state_set set2) const;
bool check_invariant() const;
/*
Output the whole state graph in dgml format into the file '.z3-state-graph.dgml'
*/
bool write_dgml();
/*
Output the whole state graph in dot format into the file '.z3-state-graph.dot'
*/
bool write_dot();
#endif
void write_dgml();
/*
'Core' functions that modify the plain graph, without
updating SCCs or propagating live/dead state information.