-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathFormulaTraverser.h
93 lines (73 loc) · 1.67 KB
/
FormulaTraverser.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
#ifndef FORMULATRAVERSER_H
#define FORMULATRAVERSER_H
#include "z3++.h"
#include "FormulaStats.h"
#include <set>
template <class T>
class FormulaTraverser
{
public:
FormulaTraverser() {}
void Traverse(const z3::expr&);
T GetData()
{
return data;
}
private:
T data;
std::set<Z3_ast> processedCache;
};
template<class T>
void FormulaTraverser<T>::Traverse(const z3::expr &expr)
{
auto item = processedCache.find((Z3_ast)expr);
if (item != processedCache.end())
{
return;
}
if (expr.is_numeral())
{
std::stringstream ss;
ss << expr;
data.AddNumeral(ss.str(), expr);
}
else if (expr.is_const())
{
std::stringstream ss;
ss << expr;
if (ss.str() == "true" || ss.str() == "false")
{
return;
}
data.AddConstant(expr, expr.get_sort());
}
else if (expr.is_app())
{
z3::func_decl f = expr.decl();
unsigned num = expr.num_args();
if (num != 0)
{
data.AddFunctionApplication(f.name().str(), expr);
}
for (unsigned i = 0; i < num; i++)
{
Traverse(expr.arg(i));
}
}
else if (expr.is_quantifier())
{
Z3_ast ast = (Z3_ast)expr;
int boundVariables = Z3_get_quantifier_num_bound(expr.ctx(), ast);
for (int i = 0; i < boundVariables; i++)
{
Z3_symbol z3_symbol = Z3_get_quantifier_bound_name(expr.ctx(), ast, i);
Z3_sort z3_sort = Z3_get_quantifier_bound_sort(expr.ctx(), ast, i);
z3::symbol current_symbol(expr.ctx(), z3_symbol);
z3::sort current_sort(expr.ctx(), z3_sort);
data.AddVariable(current_symbol.str(), current_sort);
}
Traverse(expr.body());
}
processedCache.insert((Z3_ast)expr);
}
#endif