mirror of
https://github.com/Z3Prover/z3
synced 2025-04-07 18:05:21 +00:00
556 lines
17 KiB
C++
556 lines
17 KiB
C++
/*++
|
|
Copyright (c) 2011 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
ctx_simplify_tactic.cpp
|
|
|
|
Abstract:
|
|
|
|
Simple context simplifier for propagating constants.
|
|
|
|
Author:
|
|
|
|
Leonardo (leonardo) 2011-10-26
|
|
|
|
Notes:
|
|
|
|
--*/
|
|
#include"ctx_simplify_tactic.h"
|
|
#include"mk_simplified_app.h"
|
|
#include"num_occurs.h"
|
|
#include"cooperate.h"
|
|
#include"ast_ll_pp.h"
|
|
#include"ast_smt2_pp.h"
|
|
|
|
struct ctx_simplify_tactic::imp {
|
|
struct cached_result {
|
|
expr * m_to;
|
|
unsigned m_lvl;
|
|
cached_result * m_next;
|
|
cached_result(expr * t, unsigned lvl, cached_result * next):
|
|
m_to(t),
|
|
m_lvl(lvl),
|
|
m_next(next) {
|
|
}
|
|
};
|
|
|
|
struct cache_cell {
|
|
expr * m_from;
|
|
cached_result * m_result;
|
|
cache_cell():m_from(0), m_result(0) {}
|
|
};
|
|
|
|
ast_manager & m;
|
|
small_object_allocator m_allocator;
|
|
obj_map<expr, expr*> m_assertions;
|
|
ptr_vector<expr> m_trail;
|
|
svector<unsigned> m_scopes;
|
|
svector<cache_cell> m_cache;
|
|
vector<ptr_vector<expr> > m_cache_undo;
|
|
unsigned m_scope_lvl;
|
|
unsigned m_depth;
|
|
unsigned m_num_steps;
|
|
num_occurs m_occs;
|
|
mk_simplified_app m_mk_app;
|
|
unsigned long long m_max_memory;
|
|
unsigned m_max_depth;
|
|
unsigned m_max_steps;
|
|
bool m_bail_on_blowup;
|
|
volatile bool m_cancel;
|
|
|
|
imp(ast_manager & _m, params_ref const & p):
|
|
m(_m),
|
|
m_allocator("context-simplifier"),
|
|
m_occs(true, true),
|
|
m_mk_app(m, p) {
|
|
m_cancel = false;
|
|
m_scope_lvl = 0;
|
|
updt_params(p);
|
|
}
|
|
|
|
void set_cancel(bool f) {
|
|
m_cancel = f;
|
|
}
|
|
|
|
~imp() {
|
|
pop(m_scope_lvl);
|
|
SASSERT(m_scope_lvl == 0);
|
|
restore_cache(0);
|
|
DEBUG_CODE({
|
|
for (unsigned i = 0; i < m_cache.size(); i++) {
|
|
CTRACE("ctx_simplify_tactic_bug", m_cache[i].m_from,
|
|
tout << "i: " << i << "\n" << mk_ismt2_pp(m_cache[i].m_from, m) << "\n";
|
|
tout << "m_result: " << m_cache[i].m_result << "\n";
|
|
if (m_cache[i].m_result) tout << "lvl: " << m_cache[i].m_result->m_lvl << "\n";);
|
|
SASSERT(m_cache[i].m_from == 0);
|
|
SASSERT(m_cache[i].m_result == 0);
|
|
}
|
|
});
|
|
}
|
|
|
|
void updt_params(params_ref const & p) {
|
|
m_max_memory = megabytes_to_bytes(p.get_uint(":max-memory", UINT_MAX));
|
|
m_max_steps = p.get_uint(":max-steps", UINT_MAX);
|
|
m_max_depth = p.get_uint(":max-depth", 1024);
|
|
m_bail_on_blowup = p.get_bool(":bail-on-blowup", false);
|
|
}
|
|
|
|
void checkpoint() {
|
|
cooperate("ctx_simplify_tactic");
|
|
if (memory::get_allocation_size() > m_max_memory)
|
|
throw tactic_exception(TACTIC_MAX_MEMORY_MSG);
|
|
if (m_cancel)
|
|
throw tactic_exception(TACTIC_CANCELED_MSG);
|
|
}
|
|
|
|
bool shared(expr * t) const {
|
|
return t->get_ref_count() > 1 && m_occs.get_num_occs(t) > 1;
|
|
}
|
|
|
|
bool check_cache() {
|
|
for (unsigned i = 0; i < m_cache.size(); i++) {
|
|
cache_cell & cell = m_cache[i];
|
|
if (cell.m_from != 0) {
|
|
SASSERT(cell.m_result != 0);
|
|
cached_result * curr = cell.m_result;
|
|
while (curr) {
|
|
SASSERT(curr->m_lvl <= scope_level());
|
|
curr = curr->m_next;
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void cache_core(expr * from, expr * to) {
|
|
TRACE("ctx_simplify_tactic_cache", tout << "caching\n" << mk_ismt2_pp(from, m) << "\n--->\n" << mk_ismt2_pp(to, m) << "\n";);
|
|
unsigned id = from->get_id();
|
|
m_cache.reserve(id+1);
|
|
cache_cell & cell = m_cache[id];
|
|
void * mem = m_allocator.allocate(sizeof(cached_result));
|
|
if (cell.m_from == 0) {
|
|
// new_entry
|
|
cell.m_from = from;
|
|
cell.m_result = new (mem) cached_result(to, m_scope_lvl, 0);
|
|
m.inc_ref(from);
|
|
m.inc_ref(to);
|
|
}
|
|
else {
|
|
// update
|
|
cell.m_result = new (mem) cached_result(to, m_scope_lvl, cell.m_result);
|
|
m.inc_ref(to);
|
|
}
|
|
m_cache_undo.reserve(m_scope_lvl+1);
|
|
m_cache_undo[m_scope_lvl].push_back(from);
|
|
}
|
|
|
|
void cache(expr * from, expr * to) {
|
|
if (shared(from))
|
|
cache_core(from, to);
|
|
}
|
|
|
|
unsigned scope_level() const {
|
|
return m_scope_lvl;
|
|
}
|
|
|
|
void push() {
|
|
m_scope_lvl++;
|
|
m_scopes.push_back(m_trail.size());
|
|
}
|
|
|
|
void restore_cache(unsigned lvl) {
|
|
if (lvl >= m_cache_undo.size())
|
|
return;
|
|
ptr_vector<expr> & keys = m_cache_undo[lvl];
|
|
ptr_vector<expr>::iterator it = keys.end();
|
|
ptr_vector<expr>::iterator begin = keys.begin();
|
|
while (it != begin) {
|
|
--it;
|
|
expr * key = *it;
|
|
unsigned key_id = key->get_id();
|
|
cache_cell & cell = m_cache[key_id];
|
|
SASSERT(cell.m_from == key);
|
|
SASSERT(cell.m_result != 0);
|
|
m.dec_ref(cell.m_result->m_to);
|
|
cached_result * to_delete = cell.m_result;
|
|
SASSERT(to_delete->m_lvl == lvl);
|
|
TRACE("ctx_simplify_tactic_cache", tout << "uncaching: " << to_delete->m_lvl << "\n" <<
|
|
mk_ismt2_pp(key, m) << "\n--->\n" << mk_ismt2_pp(to_delete->m_to, m) << "\nrestoring:\n";
|
|
if (to_delete->m_next) tout << mk_ismt2_pp(to_delete->m_next->m_to, m); else tout << "<null>";
|
|
tout << "\n";);
|
|
cell.m_result = to_delete->m_next;
|
|
if (cell.m_result == 0) {
|
|
m.dec_ref(cell.m_from);
|
|
cell.m_from = 0;
|
|
}
|
|
m_allocator.deallocate(sizeof(cached_result), to_delete);
|
|
}
|
|
keys.reset();
|
|
}
|
|
|
|
void pop(unsigned num_scopes) {
|
|
if (num_scopes == 0)
|
|
return;
|
|
SASSERT(num_scopes <= m_scope_lvl);
|
|
SASSERT(m_scope_lvl == m_scopes.size());
|
|
|
|
// undo assertions
|
|
unsigned old_trail_size = m_scopes[m_scope_lvl - num_scopes];
|
|
unsigned i = m_trail.size();
|
|
while (i > old_trail_size) {
|
|
--i;
|
|
expr * key = m_trail.back();
|
|
m_assertions.erase(key);
|
|
m_trail.pop_back();
|
|
}
|
|
SASSERT(m_trail.size() == old_trail_size);
|
|
m_scopes.shrink(m_scope_lvl - num_scopes);
|
|
|
|
// restore cache
|
|
for (unsigned i = 0; i < num_scopes; i++) {
|
|
restore_cache(m_scope_lvl);
|
|
m_scope_lvl--;
|
|
}
|
|
CASSERT("ctx_simplify_tactic", check_cache());
|
|
}
|
|
|
|
void assert_eq_core(expr * t, app * val) {
|
|
if (m_assertions.contains(t)) {
|
|
// This branch can only happen when m_max_depth was reached.
|
|
// It can happen when m_assertions contains an entry t->val',
|
|
// but (= t val) was not simplified to (= val' val)
|
|
// because the simplifier stopped at depth m_max_depth
|
|
return;
|
|
}
|
|
|
|
CTRACE("assert_eq_bug", m_assertions.contains(t), tout << "m_depth: " << m_depth << " m_max_depth: " << m_max_depth << "\n"
|
|
<< "t:\n" << mk_ismt2_pp(t, m) << "\nval:\n" << mk_ismt2_pp(val, m) << "\n";
|
|
expr * old_val = 0;
|
|
m_assertions.find(t, old_val);
|
|
tout << "old_val:\n" << mk_ismt2_pp(old_val, m) << "\n";);
|
|
m_assertions.insert(t, val);
|
|
m_trail.push_back(t);
|
|
}
|
|
|
|
void assert_eq_val(expr * t, app * val, bool mk_scope) {
|
|
if (shared(t)) {
|
|
if (mk_scope)
|
|
push();
|
|
assert_eq_core(t, val);
|
|
}
|
|
}
|
|
|
|
void assert_expr(expr * t, bool sign) {
|
|
if (m.is_not(t)) {
|
|
t = to_app(t)->get_arg(0);
|
|
sign = !sign;
|
|
}
|
|
bool mk_scope = true;
|
|
if (shared(t)) {
|
|
push();
|
|
mk_scope = false;
|
|
assert_eq_core(t, sign ? m.mk_false() : m.mk_true());
|
|
}
|
|
expr * lhs, * rhs;
|
|
if (!sign && m.is_eq(t, lhs, rhs)) {
|
|
if (m.is_value(rhs))
|
|
assert_eq_val(lhs, to_app(rhs), mk_scope);
|
|
else if (m.is_value(lhs))
|
|
assert_eq_val(rhs, to_app(lhs), mk_scope);
|
|
}
|
|
}
|
|
|
|
bool is_cached(expr * t, expr_ref & r) {
|
|
unsigned id = t->get_id();
|
|
if (id >= m_cache.size())
|
|
return false;
|
|
cache_cell & cell = m_cache[id];
|
|
SASSERT(cell.m_result == 0 || cell.m_result->m_lvl <= scope_level());
|
|
if (cell.m_result != 0 && cell.m_result->m_lvl == scope_level()) {
|
|
SASSERT(cell.m_from == t);
|
|
SASSERT(cell.m_result->m_to != 0);
|
|
r = cell.m_result->m_to;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void simplify(expr * t, expr_ref & r) {
|
|
r = 0;
|
|
if (m_depth >= m_max_depth || m_num_steps >= m_max_steps || !is_app(t)) {
|
|
r = t;
|
|
return;
|
|
}
|
|
checkpoint();
|
|
TRACE("ctx_simplify_tactic_detail", tout << "processing: " << mk_bounded_pp(t, m) << "\n";);
|
|
expr * _r;
|
|
if (m_assertions.find(t, _r)) {
|
|
r = _r;
|
|
SASSERT(r.get() != 0);
|
|
return;
|
|
}
|
|
if (is_cached(t, r)) {
|
|
SASSERT(r.get() != 0);
|
|
return;
|
|
}
|
|
m_num_steps++;
|
|
m_depth++;
|
|
if (m.is_or(t))
|
|
simplify_or_and<true>(to_app(t), r);
|
|
else if (m.is_and(t))
|
|
simplify_or_and<false>(to_app(t), r);
|
|
else if (m.is_ite(t))
|
|
simplify_ite(to_app(t), r);
|
|
else
|
|
simplify_app(to_app(t), r);
|
|
m_depth--;
|
|
SASSERT(r.get() != 0);
|
|
TRACE("ctx_simplify_tactic_detail", tout << "result:\n" << mk_bounded_pp(t, m) << "\n---->\n" << mk_bounded_pp(r, m) << "\n";);
|
|
}
|
|
|
|
template<bool OR>
|
|
void simplify_or_and(app * t, expr_ref & r) {
|
|
// go forwards
|
|
expr_ref_buffer new_args(m);
|
|
unsigned old_lvl = scope_level();
|
|
bool modified = false;
|
|
unsigned num_args = t->get_num_args();
|
|
for (unsigned i = 0; i < num_args; i++) {
|
|
expr * arg = t->get_arg(i);
|
|
expr_ref new_arg(m);
|
|
simplify(arg, new_arg);
|
|
if (new_arg != arg)
|
|
modified = true;
|
|
if ((OR && m.is_false(new_arg)) ||
|
|
(!OR && m.is_true(new_arg))) {
|
|
modified = true;
|
|
continue;
|
|
}
|
|
if ((OR && m.is_true(new_arg)) ||
|
|
(!OR && m.is_false(new_arg))) {
|
|
r = new_arg;
|
|
pop(scope_level() - old_lvl);
|
|
cache(t, r);
|
|
return;
|
|
}
|
|
new_args.push_back(new_arg);
|
|
if (i < num_args - 1)
|
|
assert_expr(new_arg, OR);
|
|
}
|
|
pop(scope_level() - old_lvl);
|
|
|
|
// go backwards
|
|
expr_ref_buffer new_new_args(m);
|
|
unsigned i = new_args.size();
|
|
while (i > 0) {
|
|
--i;
|
|
expr * arg = new_args[i];
|
|
expr_ref new_arg(m);
|
|
simplify(arg, new_arg);
|
|
if (new_arg != arg)
|
|
modified = true;
|
|
if ((OR && m.is_false(new_arg)) ||
|
|
(!OR && m.is_true(new_arg))) {
|
|
modified = true;
|
|
continue;
|
|
}
|
|
if ((OR && m.is_true(new_arg)) ||
|
|
(!OR && m.is_false(new_arg))) {
|
|
r = new_arg;
|
|
pop(scope_level() - old_lvl);
|
|
cache(t, r);
|
|
return;
|
|
}
|
|
new_new_args.push_back(new_arg);
|
|
if (i > 0)
|
|
assert_expr(new_arg, OR);
|
|
}
|
|
pop(scope_level() - old_lvl);
|
|
|
|
if (!modified) {
|
|
r = t;
|
|
}
|
|
else {
|
|
std::reverse(new_new_args.c_ptr(), new_new_args.c_ptr() + new_new_args.size());
|
|
m_mk_app(t->get_decl(), new_new_args.size(), new_new_args.c_ptr(), r);
|
|
}
|
|
cache(t, r);
|
|
}
|
|
|
|
void simplify_ite(app * ite, expr_ref & r) {
|
|
expr * c = ite->get_arg(0);
|
|
expr * t = ite->get_arg(1);
|
|
expr * e = ite->get_arg(2);
|
|
expr_ref new_c(m);
|
|
unsigned old_lvl = scope_level();
|
|
simplify(c, new_c);
|
|
if (m.is_true(new_c)) {
|
|
simplify(t, r);
|
|
}
|
|
else if (m.is_false(new_c)) {
|
|
simplify(e, r);
|
|
}
|
|
else {
|
|
expr_ref new_t(m);
|
|
expr_ref new_e(m);
|
|
assert_expr(new_c, false);
|
|
simplify(t, new_t);
|
|
pop(scope_level() - old_lvl);
|
|
assert_expr(new_c, true);
|
|
simplify(e, new_e);
|
|
pop(scope_level() - old_lvl);
|
|
if (c == new_c && t == new_t && e == new_e) {
|
|
r = ite;
|
|
}
|
|
else {
|
|
expr * args[3] = { new_c.get(), new_t.get(), new_e.get() };
|
|
TRACE("ctx_simplify_tactic_ite_bug",
|
|
tout << "mk_ite\n" << mk_ismt2_pp(new_c.get(), m) << "\n" << mk_ismt2_pp(new_t.get(), m)
|
|
<< "\n" << mk_ismt2_pp(new_e.get(), m) << "\n";);
|
|
m_mk_app(ite->get_decl(), 3, args, r);
|
|
}
|
|
}
|
|
cache(ite, r);
|
|
}
|
|
|
|
void simplify_app(app * t, expr_ref & r) {
|
|
if (t->get_num_args() == 0) {
|
|
r = t;
|
|
return;
|
|
}
|
|
expr_ref_buffer new_args(m);
|
|
bool modified = false;
|
|
unsigned num_args = t->get_num_args();
|
|
for (unsigned i = 0; i < num_args; i++) {
|
|
expr * arg = t->get_arg(i);
|
|
expr_ref new_arg(m);
|
|
simplify(arg, new_arg);
|
|
CTRACE("ctx_simplify_tactic_bug", new_arg.get() == 0, tout << mk_ismt2_pp(arg, m) << "\n";);
|
|
SASSERT(new_arg);
|
|
if (new_arg != arg)
|
|
modified = true;
|
|
new_args.push_back(new_arg);
|
|
}
|
|
if (!modified) {
|
|
r = t;
|
|
}
|
|
else {
|
|
m_mk_app(t->get_decl(), new_args.size(), new_args.c_ptr(), r);
|
|
}
|
|
}
|
|
|
|
unsigned expr_size(expr* s) {
|
|
ast_mark visit;
|
|
unsigned sz = 0;
|
|
ptr_vector<expr> todo;
|
|
todo.push_back(s);
|
|
while (!todo.empty()) {
|
|
s = todo.back();
|
|
todo.pop_back();
|
|
if (visit.is_marked(s)) {
|
|
continue;
|
|
}
|
|
visit.mark(s, true);
|
|
++sz;
|
|
for (unsigned i = 0; is_app(s) && i < to_app(s)->get_num_args(); ++i) {
|
|
todo.push_back(to_app(s)->get_arg(i));
|
|
}
|
|
}
|
|
return sz;
|
|
}
|
|
|
|
void process(expr * s, expr_ref & r) {
|
|
TRACE("ctx_simplify_tactic", tout << "simplifying:\n" << mk_ismt2_pp(s, m) << "\n";);
|
|
SASSERT(m_scope_lvl == 0);
|
|
m_depth = 0;
|
|
simplify(s, r);
|
|
SASSERT(m_scope_lvl == 0);
|
|
SASSERT(m_depth == 0);
|
|
SASSERT(r.get() != 0);
|
|
TRACE("ctx_simplify_tactic", tout << "result\n" << mk_ismt2_pp(r, m) << " :num-steps " << m_num_steps << "\n";
|
|
tout << "old size: " << expr_size(s) << " new size: " << expr_size(r) << "\n";);
|
|
if (m_bail_on_blowup && expr_size(s) < expr_size(r)) {
|
|
r = s;
|
|
}
|
|
}
|
|
|
|
void operator()(goal & g) {
|
|
SASSERT(g.is_well_sorted());
|
|
bool proofs_enabled = g.proofs_enabled();
|
|
m_occs.reset();
|
|
m_occs(g);
|
|
m_num_steps = 0;
|
|
expr_ref r(m);
|
|
proof * new_pr = 0;
|
|
tactic_report report("ctx-simplify", g);
|
|
unsigned sz = g.size();
|
|
for (unsigned i = 0; i < sz; i++) {
|
|
if (g.inconsistent())
|
|
return;
|
|
expr * t = g.form(i);
|
|
process(t, r);
|
|
if (proofs_enabled) {
|
|
proof * pr = g.pr(i);
|
|
new_pr = m.mk_modus_ponens(pr, m.mk_rewrite_star(t, r, 0, 0)); // TODO :-)
|
|
}
|
|
g.update(i, r, new_pr, g.dep(i));
|
|
}
|
|
IF_VERBOSE(TACTIC_VERBOSITY_LVL, verbose_stream() << "(ctx-simplify :num-steps " << m_num_steps << ")\n";);
|
|
SASSERT(g.is_well_sorted());
|
|
}
|
|
|
|
};
|
|
|
|
ctx_simplify_tactic::ctx_simplify_tactic(ast_manager & m, params_ref const & p):
|
|
m_imp(alloc(imp, m, p)),
|
|
m_params(p) {
|
|
}
|
|
|
|
ctx_simplify_tactic::~ctx_simplify_tactic() {
|
|
dealloc(m_imp);
|
|
}
|
|
|
|
void ctx_simplify_tactic::updt_params(params_ref const & p) {
|
|
m_params = p;
|
|
m_imp->updt_params(p);
|
|
}
|
|
|
|
void ctx_simplify_tactic::get_param_descrs(param_descrs & r) {
|
|
insert_max_memory(r);
|
|
insert_max_steps(r);
|
|
r.insert(":max-depth", CPK_UINT, "(default: 1024) maximum term depth.");
|
|
}
|
|
|
|
void ctx_simplify_tactic::operator()(goal_ref const & in,
|
|
goal_ref_buffer & result,
|
|
model_converter_ref & mc,
|
|
proof_converter_ref & pc,
|
|
expr_dependency_ref & core) {
|
|
mc = 0; pc = 0; core = 0;
|
|
(*m_imp)(*(in.get()));
|
|
in->inc_depth();
|
|
result.push_back(in.get());
|
|
}
|
|
|
|
void ctx_simplify_tactic::set_cancel(bool f) {
|
|
if (m_imp)
|
|
m_imp->set_cancel(f);
|
|
}
|
|
|
|
void ctx_simplify_tactic::cleanup() {
|
|
ast_manager & m = m_imp->m;
|
|
imp * d = m_imp;
|
|
#pragma omp critical (tactic_cancel)
|
|
{
|
|
m_imp = 0;
|
|
}
|
|
dealloc(d);
|
|
d = alloc(imp, m, m_params);
|
|
#pragma omp critical (tactic_cancel)
|
|
{
|
|
m_imp = d;
|
|
}
|
|
}
|
|
|