From c6e66ebc3a92bd77f33a2227df088c473a616b46 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Tue, 15 Dec 2015 18:43:36 +0000 Subject: [PATCH 01/31] Adding ackr component. --- scripts/mk_project.py | 3 +- src/ackr/ackr.pyg | 8 + src/ackr/ackr_info.h | 106 +++++++++ src/ackr/lackr.cpp | 337 +++++++++++++++++++++++++++ src/ackr/lackr.h | 106 +++++++++ src/ackr/lackr_model_constructor.cpp | 313 +++++++++++++++++++++++++ src/ackr/lackr_model_constructor.h | 40 ++++ src/ackr/lackr_model_converter.cpp | 138 +++++++++++ src/ackr/lackr_model_converter.h | 22 ++ src/ackr/lackr_tactic.cpp | 136 +++++++++++ src/ackr/lackr_tactic.h | 28 +++ 11 files changed, 1236 insertions(+), 1 deletion(-) create mode 100644 src/ackr/ackr.pyg create mode 100644 src/ackr/ackr_info.h create mode 100644 src/ackr/lackr.cpp create mode 100644 src/ackr/lackr.h create mode 100644 src/ackr/lackr_model_constructor.cpp create mode 100644 src/ackr/lackr_model_constructor.h create mode 100644 src/ackr/lackr_model_converter.cpp create mode 100644 src/ackr/lackr_model_converter.h create mode 100644 src/ackr/lackr_tactic.cpp create mode 100644 src/ackr/lackr_tactic.h diff --git a/scripts/mk_project.py b/scripts/mk_project.py index eb7218d4b..12e8fea3c 100644 --- a/scripts/mk_project.py +++ b/scripts/mk_project.py @@ -76,10 +76,11 @@ def init_project_def(): add_lib('portfolio', ['smtlogic_tactics', 'sat_solver', 'ufbv_tactic', 'fpa_tactics', 'aig_tactic', 'fp', 'qe','sls_tactic', 'subpaving_tactic'], 'tactic/portfolio') add_lib('smtparser', ['portfolio'], 'parsers/smt') add_lib('opt', ['smt', 'smtlogic_tactics', 'sls_tactic', 'sat_solver'], 'opt') + add_lib('ackr', ['smt', 'smtlogic_tactics', 'sat_solver'], 'ackr') API_files = ['z3_api.h', 'z3_ast_containers.h', 'z3_algebraic.h', 'z3_polynomial.h', 'z3_rcf.h', 'z3_fixedpoint.h', 'z3_optimization.h', 'z3_interp.h', 'z3_fpa.h'] add_lib('api', ['portfolio', 'smtparser', 'realclosure', 'interp', 'opt'], includes2install=['z3.h', 'z3_v1.h', 'z3_macros.h'] + API_files) - add_exe('shell', ['api', 'sat', 'extra_cmds','opt'], exe_name='z3') + add_exe('shell', ['api', 'sat', 'extra_cmds','opt','ackr'], exe_name='z3') add_exe('test', ['api', 'fuzzing', 'simplex'], exe_name='test-z3', install=False) _libz3Component = add_dll('api_dll', ['api', 'sat', 'extra_cmds'], 'api/dll', reexports=['api'], diff --git a/src/ackr/ackr.pyg b/src/ackr/ackr.pyg new file mode 100644 index 000000000..4b3d6f5f7 --- /dev/null +++ b/src/ackr/ackr.pyg @@ -0,0 +1,8 @@ +def_module_params('ackr', + description='solving UF via ackermannization (currently for QF_AUFBV)', + export=True, + params=( + ('eager', BOOL, True, 'eagerly instantiate all congruence rules'), + ('sat_backend', BOOL, False, 'use SAT rather than SMT'), + )) + diff --git a/src/ackr/ackr_info.h b/src/ackr/ackr_info.h new file mode 100644 index 000000000..6d2b596fe --- /dev/null +++ b/src/ackr/ackr_info.h @@ -0,0 +1,106 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +ackr_info.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#ifndef ACKR_INFO_H_12278 +#define ACKR_INFO_H_12278 +#include"obj_hashtable.h" +#include"ast.h" +#include"ref.h" +#include"expr_replacer.h" + +// +// Information about how a formula was extracted into +// a formula without uninterpreted function symbols. +// +// The intended use is that new terms are added via set_abstr. +// Once all terms are abstracted, call seal. +// abstract may only be called when sealed. +// +// The class enables reference counting. +class ackr_info { + public: + ackr_info(ast_manager& m) + : m_m(m) + , m_consts(m) + , m_er(mk_default_expr_replacer(m)) + , m_subst(m_m) + , m_ref_count(0) + , m_sealed(false) + {} + + virtual ~ackr_info() { + m_consts.reset(); + } + + inline void set_abstr(app* term, app* c) { + SASSERT(!m_sealed); + SASSERT(c); + m_t2c.insert(term,c); + m_c2t.insert(c->get_decl(),term); + m_subst.insert(term, c); + m_consts.push_back(c); + } + + inline void abstract(expr * e, expr_ref& res) { + SASSERT(m_sealed); + (*m_er)(e, res); + } + + inline app* find_term(func_decl* c) const { + app * rv = 0; + m_c2t.find(c,rv); + return rv; + } + + inline app* get_abstr(app* term) const { + app * const rv = m_t2c.find(term); + SASSERT(rv); + return rv; + } + + inline void seal() { + m_sealed=true; + m_er->set_substitution(&m_subst); + } + + // + // Reference counting + // + void inc_ref() { ++m_ref_count; } + void dec_ref() { + --m_ref_count; + if (m_ref_count == 0) { + dealloc(this); + } + } + private: + typedef obj_map t2ct; + typedef obj_map c2tt; + ast_manager& m_m; + + t2ct m_t2c; // terms to constants + c2tt m_c2t; // constants to terms (inversion of m_t2c) + expr_ref_vector m_consts; // the constants introduced during abstraction + + // replacer and substitution used to compute abstractions + scoped_ptr m_er; + expr_substitution m_subst; + + bool m_sealed; // debugging + unsigned m_ref_count; // reference counting +}; + +typedef ref ackr_info_ref; +#endif /* ACKR_INFO_H_12278 */ diff --git a/src/ackr/lackr.cpp b/src/ackr/lackr.cpp new file mode 100644 index 000000000..99e5cf0b5 --- /dev/null +++ b/src/ackr/lackr.cpp @@ -0,0 +1,337 @@ +/*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ +/////////////// +#include"lackr.h" +#include"ackr_params.hpp" +#include"tactic.h" +#include"lackr_model_constructor.h" +#include"ackr_info.h" +#include"for_each_expr.h" +/////////////// +#include"array_decl_plugin.h" +#include"simplifier_plugin.h" +#include"basic_simplifier_plugin.h" +#include"array_simplifier_params.h" +#include"array_simplifier_plugin.h" +#include"bv_simplifier_plugin.h" +#include"bool_rewriter.h" +/////////////// +#include"th_rewriter.h" +/////////////// +#include"cooperate.h" +/////////////// +#include"model_smt2_pp.h" +/////////////// + +struct simp_wrap { + inline void operator() (expr * s, expr_ref & r) { + proof_ref dummy(m); + simp(s, r, dummy); + } + simp_wrap(ast_manager& m) + : m(m) + , simp(m) + , bsp(m) + , bvsp(m, bsp, bv_par) + , asp(m, bsp, simp, ar_par) + { + params_ref p; + p.set_bool("local_ctx", true); + p.set_uint("local_ctx_limit", 10000000); + p.set_bool("ite_extra_rules", true); + bsp.get_rewriter().updt_params(p); + + simp.register_plugin(&bsp); + simp.register_plugin(&bvsp); + } + + ~simp_wrap() { + simp.release_plugins(); + } + + ast_manager& m; + simplifier simp; + basic_simplifier_plugin bsp; + bv_simplifier_params bv_par; + bv_simplifier_plugin bvsp; + array_simplifier_params ar_par; + array_simplifier_plugin asp; +}; + + +lackr::lackr(ast_manager& m, params_ref p, expr_ref _f) + : m_m(m) + , m_p(p) + , m_fla(m) + , m_abstr(m) + , m_sat(0) + , m_bvutil(m) + , m_simp(m) + , m_ackrs(m) + , m_cancel(0) +{ + m_fla = _f; + updt_params(p); +} + +lackr::~lackr() { + const fun2terms_map::iterator e = m_fun2terms.end(); + for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { + dealloc(i->get_value()); + } +} + + + +lbool lackr::operator() () { + setup_sat(); + const bool ok = init(); + if (!ok) return l_undef; + TRACE("lackr", tout << "sat goal\n"; m_sat->display(tout);); + const lbool rv = m_eager ? eager() : lazy(); + if (rv == l_true) m_sat->get_model(m_model); + TRACE("lackr", tout << "sat:" << rv << '\n'; ); + CTRACE("lackr", rv == l_true, + model_smt2_pp(tout << "abstr_model(\n", m_m, *(m_model.get()), 2); tout << ")\n"; ); + return rv; +} + + +bool lackr::init() { + params_ref simp_p(m_p); + m_simp.updt_params(simp_p); + m_info = alloc(ackr_info, m_m); + bool iok = collect_terms() && abstract(); + if (!iok) return false; + return true; +} + +// +// Introduce ackermann lemma for the two given terms. +// +bool lackr::ackr(app * const t1, app * const t2) { + TRACE("lackr", tout << "ackr " + << mk_ismt2_pp(t1, m_m, 2) + << " , " + << mk_ismt2_pp(t2, m_m, 2) << "\n";); + const unsigned sz = t1->get_num_args(); + expr_ref_vector eqs(m_m); + for (unsigned gi = 0; gi < sz; ++gi) { + expr * const arg1 = t1->get_arg(gi); + expr * const arg2 = t2->get_arg(gi); + if (arg1 == arg2) continue; + if (m_bvutil.is_numeral(arg1) && m_bvutil.is_numeral(arg2)) { + SASSERT(arg1 != arg2); + TRACE("lackr", tout << "never eq\n";); + return true; + } + eqs.push_back(m_m.mk_eq(arg1, arg2)); + } + app * const a1 = m_info->get_abstr(t1); + app * const a2 = m_info->get_abstr(t2); + SASSERT(a1); + SASSERT(a2); + TRACE("lackr", tout << "abstr1 " << mk_ismt2_pp(a1, m_m, 2) << "\n";); + TRACE("lackr", tout << "abstr2 " << mk_ismt2_pp(a2, m_m, 2) << "\n";); + expr_ref lhs(m_m.mk_and(eqs.size(), eqs.c_ptr()), m_m); + TRACE("lackr", tout << "ackr constr lhs" << mk_ismt2_pp(lhs, m_m, 2) << "\n";); + expr_ref rhs(m_m.mk_eq(a1, a2),m_m); + TRACE("lackr", tout << "ackr constr rhs" << mk_ismt2_pp(rhs, m_m, 2) << "\n";); + expr_ref cg(m_m.mk_implies(lhs, rhs), m_m); + TRACE("lackr", tout << "ackr constr" << mk_ismt2_pp(cg, m_m, 2) << "\n";); + expr_ref cga(m_m); + m_info->abstract(cg, cga); + m_simp(cga); + TRACE("lackr", tout << "ackr constr abs:" << mk_ismt2_pp(cga, m_m, 2) << "\n";); + if (!m_m.is_true(cga)) m_ackrs.push_back(cga); + return true; +} + +// +// Introduce the ackermann lemma for each pair of terms. +// +bool lackr::eager_enc() { + const fun2terms_map::iterator e = m_fun2terms.end(); + for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { + checkpoint(); + func_decl* const fd = i->m_key; + app_set * const ts = i->get_value(); + const app_set::iterator r = ts->end(); + for (app_set::iterator j = ts->begin(); j != r; ++j) { + app_set::iterator k = j; + ++k; + for (; k != r; ++k) { + app * const t1 = *j; + app * const t2 = *k; + SASSERT(t1->get_decl() == fd); + SASSERT(t2->get_decl() == fd); + if (t1 == t2) continue; + if (!ackr(t1,t2)) return false; + } + } + } + return true; +} + +bool lackr::abstract() { + const fun2terms_map::iterator e = m_fun2terms.end(); + for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { + func_decl* const fd = i->m_key; + app_set * const ts = i->get_value(); + sort* const s = fd->get_range(); + const app_set::iterator r = ts->end(); + for (app_set::iterator j = ts->begin(); j != r; ++j) { + app * const fc = m_m.mk_fresh_const(fd->get_name().str().c_str(), s); + app * const t = *j; + SASSERT(t->get_decl() == fd); + m_info->set_abstr(t, fc); + TRACE("lackr", tout << "abstr term " + << mk_ismt2_pp(t, m_m, 2) + << " -> " + << mk_ismt2_pp(fc, m_m, 2) + << "\n";); + } + } + m_info->seal(); + m_info->abstract(m_fla.get(), m_abstr); + TRACE("lackr", tout << "abs(\n" << mk_ismt2_pp(m_abstr.get(), m_m, 2) << ")\n";); + return true; +} + +void lackr::add_term(app* a) { + //TRACE("lackr", tout << "inspecting term(\n" << mk_ismt2_pp(a, m_m, 2) << ")\n";); + if (a->get_num_args() == 0) return; + func_decl* const fd = a->get_decl(); + if (!is_uninterp(a)) return; + SASSERT(m_bvutil.is_bv_sort(fd->get_range()) || m_m.is_bool(a)); + app_set* ts = 0; + if (!m_fun2terms.find(fd, ts)) { + ts = alloc(app_set); + m_fun2terms.insert(fd, ts); + } + TRACE("lackr", tout << "term(" << mk_ismt2_pp(a, m_m, 2) << ")\n";); + ts->insert(a); +} + + +lbool lackr::eager() { + if (!eager_enc()) return l_undef; + m_sat->assert_expr(m_abstr); + TRACE("lackr", tout << "run sat 0\n"; ); + if (m_sat->check_sat(0, 0) == l_false) + return l_false; + checkpoint(); + expr_ref all(m_m); + all = m_m.mk_and(m_ackrs.size(), m_ackrs.c_ptr()); + m_simp(all); + TRACE("lackr", tout << "run sat\n"; ); + m_sat->assert_expr(all); + return m_sat->check_sat(0, 0); +} + + +lbool lackr::lazy() { + lackr_model_constructor mc(m_m, m_info); + m_sat->assert_expr(m_abstr); + unsigned ackr_head = 0; + unsigned it = 0; + while (1) { + checkpoint(); + //std::cout << "it: " << ++it << "\n"; + TRACE("lackr", tout << "lazy check\n";); + const lbool r = m_sat->check_sat(0, 0); + if (r == l_undef) return l_undef; // give up + if (r == l_false) return l_false; // abstraction unsat + // reconstruct model + model_ref am; + m_sat->get_model(am); + const bool mc_res = mc.check(am); + if (mc_res) return l_true; // model okay + // refine abstraction + const lackr_model_constructor::conflict_list conflicts = mc.get_conflicts(); + for (lackr_model_constructor::conflict_list::const_iterator i = conflicts.begin(); + i != conflicts.end(); ++i) { + ackr(i->first, i->second); + } + while (ackr_head < m_ackrs.size()) { + m_sat->assert_expr(m_ackrs.get(ackr_head++)); + } + } +} + +void lackr::setup_sat() { + if (m_use_sat) { + //std::cout << "; qfbv sat\n"; + tactic_ref t = mk_qfbv_tactic(m_m, m_p); + m_sat = mk_tactic2solver(m_m, t.get(), m_p); + } + else { + //std::cout << "; smt sat\n"; + tactic_ref t = mk_qfaufbv_tactic(m_m, m_p); + m_sat = mk_tactic2solver(m_m, t.get(), m_p); + } + SASSERT(m_sat); + m_sat->set_produce_models(true); +} + + +// +// Collect all uninterpreted terms. +// +bool lackr::collect_terms() { + ptr_vector stack; + expr * curr; + expr_mark visited; + stack.push_back(m_fla.get()); + while (!stack.empty()) { + curr = stack.back(); + if (visited.is_marked(curr)) { + stack.pop_back(); + continue; + } + + switch (curr->get_kind()) { + case AST_VAR: + visited.mark(curr, true); + stack.pop_back(); + break; + + case AST_APP: + if (for_each_expr_args(stack, visited, + to_app(curr)->get_num_args(), to_app(curr)->get_args())) { + visited.mark(curr, true); + stack.pop_back(); + add_term(to_app(curr)); + } + break; + case AST_QUANTIFIER: + if (visited.is_marked(to_quantifier(curr)->get_expr())) { + visited.mark(curr, true); + stack.pop_back(); + } + else { + stack.push_back(to_quantifier(curr)->get_expr()); + } + break; + default: + UNREACHABLE(); + return false; + } + } + visited.reset(); + return true; +} diff --git a/src/ackr/lackr.h b/src/ackr/lackr.h new file mode 100644 index 000000000..2f685a001 --- /dev/null +++ b/src/ackr/lackr.h @@ -0,0 +1,106 @@ + /*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr.h + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef LACKR_H_15079 +#define LACKR_H_15079 +/////////////// +#include"inc_sat_solver.h" +#include"qfaufbv_tactic.h" +#include"qfbv_tactic.h" +#include"tactic2solver.h" +#include"ackr_info.h" +#include"ackr_params.hpp" +#include"tactic_exception.h" +#include"th_rewriter.h" +#include"bv_decl_plugin.h" +#include"cooperate.h" + +class lackr { + public: + lackr(ast_manager& m, params_ref p, expr_ref _f); + lbool operator() (); + ~lackr(); + inline ackr_info_ref get_info() { return m_info; } + inline model_ref get_model() { return m_model; } + + void updt_params(params_ref const & _p) { + ackr_params p(_p); + m_eager = p.eager(); + m_use_sat = p.sat_backend(); + } + + // + // timeout mechanisms + // + + void checkpoint() { + if (m_cancel) + throw tactic_exception(TACTIC_CANCELED_MSG); + cooperate("lackr"); + } + + //virtual void set_cancel(bool f) { + // //#pragma omp critical (lackr_cancel) + // { + // m_cancel = f; + // if (m_sat == NULL) return; + // if (f) m_sat->cancel(); + // else m_sat->reset_cancel(); + // } + //} + private: + typedef obj_hashtable app_set; + typedef obj_map fun2terms_map; + ast_manager& m_m; + params_ref m_p; + expr_ref m_fla; + expr_ref m_abstr; + fun2terms_map m_fun2terms; + ackr_info_ref m_info; + scoped_ptr m_sat; + bv_util m_bvutil; + th_rewriter m_simp; + expr_ref_vector m_ackrs; + model_ref m_model; + volatile bool m_cancel; + bool m_eager; + bool m_use_sat; + + bool init(); + void setup_sat(); + lbool eager(); + lbool lazy(); + + // + // Introduce ackermann lemma for the two given terms. + // + bool ackr(app * const t1, app * const t2); + + // + // Introduce the ackermann lemma for each pair of terms. + // + bool eager_enc(); + + bool abstract(); + + void add_term(app* a); + + // + // Collect all uninterpreted terms. + // + bool collect_terms(); +}; +#endif /* LACKR_H_15079 */ diff --git a/src/ackr/lackr_model_constructor.cpp b/src/ackr/lackr_model_constructor.cpp new file mode 100644 index 000000000..5f0060ad5 --- /dev/null +++ b/src/ackr/lackr_model_constructor.cpp @@ -0,0 +1,313 @@ +/*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + model_constructor.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ +#include"lackr_model_constructor.h" +#include"model_evaluator.h" +#include"ast_smt2_pp.h" +#include"ackr_info.h" +#include"for_each_expr.h" +#include"bv_rewriter.h" +#include"bool_rewriter.h" +struct lackr_model_constructor::imp { + public: + imp(ast_manager& m, + ackr_info_ref info, + model_ref& abstr_model, + vector>& conflicts) + : m_m(m) + , m_info(info) + , m_abstr_model(abstr_model) + , m_conflicts(conflicts) + , m_b_rw(m) + , m_bv_rw(m) + , m_empty_model(m) + {} + + ~imp() { + { + values2val_t::iterator i = m_values2val.begin(); + const values2val_t::iterator e = m_values2val.end(); + for (; i != e; ++i) { + m_m.dec_ref(i->m_key); + m_m.dec_ref(i->m_value.value); + m_m.dec_ref(i->m_value.source_term); + } + } + { + app2val_t::iterator i = m_app2val.begin(); + const app2val_t::iterator e = m_app2val.end(); + for (; i != e; ++i) { + m_m.dec_ref(i->m_value); + m_m.dec_ref(i->m_key); + } + } + } + + // + // Returns true iff model was successfully constructed. + // + bool check() { + for (unsigned i = 0; i < m_abstr_model->get_num_constants(); i++) { + func_decl * const c = m_abstr_model->get_constant(i); + app * const term = m_info->find_term(c); + if (term) m_stack.push_back(term); + else m_stack.push_back(m_m.mk_const(c)); + } + return run(); + } + private: + ast_manager& m_m; + ackr_info_ref m_info; + model_ref& m_abstr_model; + vector>& m_conflicts; + bool_rewriter m_b_rw; + bv_rewriter m_bv_rw; + scoped_ptr m_evaluator; + model m_empty_model; + private: + struct val_info { expr * value; app * source_term; }; + typedef obj_map app2val_t; + typedef obj_map values2val_t; + values2val_t m_values2val; + app2val_t m_app2val; + ptr_vector m_stack; + + static inline val_info mk_val_info(expr* value, app* source_term) { + val_info rv; + rv.value = value; + rv.source_term = source_term; + return rv; + } + + // + // Performs congruence check on terms on the stack. + // (Currently stops upon the first failure). + // Returns true if and only if congruence check succeeded. + bool run() { + m_evaluator = alloc(model_evaluator, m_empty_model); + expr_mark visited; + expr * curr; + while (!m_stack.empty()) { + curr = m_stack.back(); + if (visited.is_marked(curr)) { + m_stack.pop_back(); + continue; + } + + switch (curr->get_kind()) { + case AST_VAR: + UNREACHABLE(); + return false; + case AST_APP: { + app * a = to_app(curr); + if (for_each_expr_args(m_stack, visited, a->get_num_args(), a->get_args())) { + visited.mark(a, true); + m_stack.pop_back(); + if (!mk_value(a)) return false; + } + } + break; + case AST_QUANTIFIER: + UNREACHABLE(); + return false; + default: + UNREACHABLE(); + return false; + } + } + return true; + } + + inline bool is_val(expr * e) { + if (!is_app(e)) return false; + return is_val(to_app(e)); + } + + inline bool is_val(app * a) { + const family_id fid = a->get_decl()->get_family_id(); + const bool rv = fid != null_family_id && a->get_num_args() == 0; + SASSERT(rv == (m_bv_rw.is_numeral(a) || m_m.is_true(a) || m_m.is_false(a))); + return rv; + } + + inline bool eval_cached(app * a, expr *& val) { + if (is_val(a)) { val = a; return true; } + return m_app2val.find(a, val); + } + + bool evaluate(app * const a, expr_ref& result) { + SASSERT(!is_val(a)); + const unsigned num = a->get_num_args(); + if (num == 0) { // handle constants + make_value_constant(a, result); + return true; + } + // evaluate arguments + expr_ref_vector values(m_m); + values.reserve(num); + expr * const * args = a->get_args(); + for (unsigned i = 0; i < num; ++i) { + expr * val; + const bool b = eval_cached(to_app(args[i]), val); // TODO: OK conversion to_app? + CTRACE("model_constructor", !b, tout << "fail arg val(\n" << mk_ismt2_pp(args[i], m_m, 2); ); + TRACE("model_constructor", tout << + "arg val " << i << "(\n" << mk_ismt2_pp(args[i], m_m, 2) + << " : " << mk_ismt2_pp(val, m_m, 2) << '\n'; ); + SASSERT(b); + values[i] = val; + } + // handle functions + if (a->get_family_id() == null_family_id) { // handle uninterpreted + app_ref key(m_m.mk_app(a->get_decl(), values.c_ptr()), m_m); + if (!make_value_uninterpreted_function(a, values, key.get(), result)) { + return false; + } + } + else { // handle interpreted + make_value_interpreted_function(a, values, result); + } + return true; + } + + // + // Check and record the value for a given term, given that all arguments are already checked. + // + bool mk_value(app * a) { + if (is_val(a)) return true; // skip numerals + TRACE("model_constructor", tout << "mk_value(\n" << mk_ismt2_pp(a, m_m, 2) << ")\n";); + SASSERT(!m_app2val.contains(a)); + const unsigned num = a->get_num_args(); + expr_ref result(m_m); + if (!evaluate(a, result)) return false; + SASSERT(is_val(result)); + TRACE("model_constructor", + tout << "map term(\n" << mk_ismt2_pp(a, m_m, 2) << "\n->" + << mk_ismt2_pp(result.get(), m_m, 2)<< ")\n"; ); + CTRACE("model_constructor", + !is_val(result.get()), + tout << "eval fail\n" << mk_ismt2_pp(a, m_m, 2) << mk_ismt2_pp(result, m_m, 2) << "\n"; + ); + SASSERT(is_val(result.get())); + m_app2val.insert(a, result.get()); // memoize + m_m.inc_ref(a); + m_m.inc_ref(result.get()); + return true; + } + + // Constants from the abstract model are directly mapped to the concrete one. + void make_value_constant(app * const a, expr_ref& result) { + SASSERT(a->get_num_args() == 0); + func_decl * const fd = a->get_decl(); + expr * val = m_abstr_model->get_const_interp(fd); + if (val == 0) { // TODO: avoid model completetion? + sort * s = fd->get_range(); + val = m_abstr_model->get_some_value(s); + } + result = val; + } + + bool make_value_uninterpreted_function(app* a, + expr_ref_vector& values, + app* key, + expr_ref& result) { + // get ackermann constant + app * const ac = m_info->get_abstr(a); + func_decl * const a_fd = a->get_decl(); + SASSERT(ac->get_num_args() == 0); + SASSERT(a_fd->get_range() == ac->get_decl()->get_range()); + expr_ref value(m_m); + value = m_abstr_model->get_const_interp(ac->get_decl()); + // get ackermann constant's interpretation + if (value.get() == 0) { // TODO: avoid model completion? + sort * s = a_fd->get_range(); + value = m_abstr_model->get_some_value(s); + } + // check congruence + val_info vi; + if(m_values2val.find(key,vi)) { // already is mapped to a value + SASSERT(vi.source_term); + const bool ok = vi.value == value; + if (!ok) { + TRACE("model_constructor", + tout << "already mapped by(\n" << mk_ismt2_pp(vi.source_term, m_m, 2) << "\n->" + << mk_ismt2_pp(vi.value, m_m, 2) << ")\n"; ); + m_conflicts.push_back(std::make_pair(a, vi.source_term)); + } + result = vi.value; + return ok; + } else { // new value + result = value; + vi.value = value; + vi.source_term = a; + m_values2val.insert(key,vi); + m_m.inc_ref(vi.source_term); + m_m.inc_ref(vi.value); + m_m.inc_ref(key); + return true; + } + UNREACHABLE(); + } + + void make_value_interpreted_function(app* a, + expr_ref_vector& values, + expr_ref& result) { + const unsigned num = values.size(); + func_decl * const fd = a->get_decl(); + const family_id fid = fd->get_family_id(); + expr_ref term(m_m); + term = m_m.mk_app(a->get_decl(), num, values.c_ptr()); + m_evaluator->operator() (term, result); + TRACE("model_constructor", + tout << "eval(\n" << mk_ismt2_pp(term.get(), m_m, 2) << "\n->" + << mk_ismt2_pp(result.get(), m_m, 2) << ")\n"; ); + return; + if (fid == m_b_rw.get_fid()) { + decl_kind k = fd->get_decl_kind(); + if (k == OP_EQ) { + // theory dispatch for = + SASSERT(num == 2); + family_id s_fid = m_m.get_sort(values.get(0))->get_family_id(); + br_status st = BR_FAILED; + if (s_fid == m_bv_rw.get_fid()) + st = m_bv_rw.mk_eq_core(values.get(0), values.get(1), result); + } else { + br_status st = m_b_rw.mk_app_core(fd, num, values.c_ptr(), result); + } + } else { + br_status st = BR_FAILED; + if (fid == m_bv_rw.get_fid()) { + st = m_bv_rw.mk_app_core(fd, num, values.c_ptr(), result); + } + else { + UNREACHABLE(); + } + } + } +}; + +lackr_model_constructor::lackr_model_constructor(ast_manager& m, ackr_info_ref info) + : m(m) + , state(UNKNOWN) + , info(info) +{} + +bool lackr_model_constructor::check(model_ref& abstr_model) { + conflicts.reset(); + lackr_model_constructor::imp i(m, info, abstr_model, conflicts); + const bool rv = i.check(); + state = rv ? CHECKED : CONFLICT; + return rv; +} \ No newline at end of file diff --git a/src/ackr/lackr_model_constructor.h b/src/ackr/lackr_model_constructor.h new file mode 100644 index 000000000..5b2385c2c --- /dev/null +++ b/src/ackr/lackr_model_constructor.h @@ -0,0 +1,40 @@ + /*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + model_constructor.h + + Abstract: + Given a propositional abstraction, attempt to construct a model. + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef LACKR_MODEL_CONSTRUCTOR_H_626 +#define LACKR_MODEL_CONSTRUCTOR_H_626 +#include"ast.h" +#include"ackr_info.h" +#include"model.h" +class lackr_model_constructor { + public: + typedef std::pair app_pair; + typedef vector conflict_list; + lackr_model_constructor(ast_manager& m, ackr_info_ref info); + bool check(model_ref& abstr_model); + const conflict_list& get_conflicts() { + SASSERT(state == CONFLICT); + return conflicts; + } + private: + struct imp; + enum {CHECKED, CONFLICT, UNKNOWN} state; + conflict_list conflicts; + ast_manager& m; + const ackr_info_ref info; +}; +#endif /* MODEL_CONSTRUCTOR_H_626 */ diff --git a/src/ackr/lackr_model_converter.cpp b/src/ackr/lackr_model_converter.cpp new file mode 100644 index 000000000..05db4e789 --- /dev/null +++ b/src/ackr/lackr_model_converter.cpp @@ -0,0 +1,138 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +lackr_model_converter.cpp + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#include"lackr_model_converter.h" +#include"model_evaluator.h" +#include"ast_smt2_pp.h" +#include"ackr_info.h" + + +class lackr_model_converter : public model_converter { +public: + lackr_model_converter(ast_manager & m, + const ackr_info_ref& info, + model_ref& abstr_model) + : m(m) + , info(info) + , abstr_model(abstr_model) + { } + + virtual ~lackr_model_converter() { } + + virtual void operator()(model_ref & md, unsigned goal_idx) { + SASSERT(goal_idx == 0); + SASSERT(md.get() == 0 || (!md->get_num_constants() && !md->get_num_functions())); + SASSERT(abstr_model.get()); + model * new_model = alloc(model, m); + convert(abstr_model.get(), new_model); + md = new_model; + } + + virtual void operator()(model_ref & md) { + operator()(md, 0); + } + + //void display(std::ostream & out); + + virtual model_converter * translate(ast_translation & translator) { + NOT_IMPLEMENTED_YET(); + } +protected: + ast_manager& m; + const ackr_info_ref info; + model_ref abstr_model; + void convert(model * source, model * destination); + void add_entry(model_evaluator & evaluator, + app* term, expr* value, + obj_map& interpretations); + void convert_sorts(model * source, model * destination); + void convert_constants(model * source, model * destination); +}; + +void lackr_model_converter::convert(model * source, model * destination) { + SASSERT(source->get_num_functions() == 0); + convert_constants(source,destination); + convert_sorts(source,destination); +} + +void lackr_model_converter::convert_constants(model * source, model * destination) { + TRACE("lackr_model", tout << "converting constants\n";); + obj_map interpretations; + model_evaluator evaluator(*source); + for (unsigned i = 0; i < source->get_num_constants(); i++) { + func_decl * const c = source->get_constant(i); + app * const term = info->find_term(c); + expr * value = source->get_const_interp(c); + if(!term) { + destination->register_decl(c, value); + } else { + add_entry(evaluator, term, value, interpretations); + } + } + + obj_map::iterator e = interpretations.end(); + for (obj_map::iterator i = interpretations.begin(); + i!=e; ++i) { + func_decl* const fd = i->m_key; + func_interp* const fi = i->get_value(); + fi->set_else(m.get_some_value(fd->get_range())); + destination->register_decl(fd, fi); + } +} + +void lackr_model_converter::add_entry(model_evaluator & evaluator, + app* term, expr* value, + obj_map& interpretations) { + TRACE("lackr_model", tout << "add_entry" + << mk_ismt2_pp(term, m, 2) + << "->" + << mk_ismt2_pp(value, m, 2) << "\n"; + ); + + func_interp* fi = 0; + func_decl * const declaration = term->get_decl(); + const unsigned sz = declaration->get_arity(); + SASSERT(sz == term->get_num_args()); + if (!interpretations.find(declaration, fi)) { + fi = alloc(func_interp,m,sz); + interpretations.insert(declaration, fi); + } + expr_ref_vector args(m); + for (unsigned gi = 0; gi < sz; ++gi) { + expr * const arg = term->get_arg(gi); + expr_ref aarg(m); + info->abstract(arg, aarg); + expr_ref arg_value(m); + evaluator(aarg,arg_value); + args.push_back(arg_value); + } + if (fi->get_entry(args.c_ptr()) == 0) { + fi->insert_new_entry(args.c_ptr(), value); + } else { + TRACE("lackr_model", tout << "entry already present\n";); + } +} + +void lackr_model_converter::convert_sorts(model * source, model * destination) { + for (unsigned i = 0; i < source->get_num_uninterpreted_sorts(); i++) { + sort * const s = source->get_uninterpreted_sort(i); + ptr_vector u = source->get_universe(s); + destination->register_usort(s, u.size(), u.c_ptr()); + } +} + +model_converter * mk_lackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model) { + return alloc(lackr_model_converter, m, info, abstr_model); +} diff --git a/src/ackr/lackr_model_converter.h b/src/ackr/lackr_model_converter.h new file mode 100644 index 000000000..ec59d7572 --- /dev/null +++ b/src/ackr/lackr_model_converter.h @@ -0,0 +1,22 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +lackr_model_converter.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#ifndef LACKR_MODEL_CONVERTER_H_5814 +#define LACKR_MODEL_CONVERTER_H_5814 +#include"model_converter.h" +#include"ackr_info.h" + +model_converter * mk_lackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); +#endif /* LACKR_MODEL_CONVERTER_H_5814 */ diff --git a/src/ackr/lackr_tactic.cpp b/src/ackr/lackr_tactic.cpp new file mode 100644 index 000000000..c5368940b --- /dev/null +++ b/src/ackr/lackr_tactic.cpp @@ -0,0 +1,136 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +lackr_tactic.cpp + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#include"tactical.h" +/////////////// +#include"solve_eqs_tactic.h" +#include"simplify_tactic.h" +#include"propagate_values_tactic.h" +#include"bit_blaster_tactic.h" +#include"elim_uncnstr_tactic.h" +#include"max_bv_sharing_tactic.h" +#include"bv_size_reduction_tactic.h" +#include"ctx_simplify_tactic.h" +#include"nnf_tactic.h" +/////////////// +#include"model_smt2_pp.h" +#include"cooperate.h" +#include"lackr.h" +#include"lackr_model_converter.h" + +class lackr_tactic : public tactic { +public: + lackr_tactic(ast_manager& m, params_ref const& p) + : m_m(m) + , m_p(p) + , m_imp(0) + {} + + virtual ~lackr_tactic() { + if (m_imp) dealloc(m_imp); + } + + virtual void operator()(goal_ref const & g, + goal_ref_buffer & result, + model_converter_ref & mc, + proof_converter_ref & pc, + expr_dependency_ref & core) { + mc = 0; + ast_manager& m(g->m()); + TRACE("lackr", g->display(tout << "Goal:\n");); + // conflate all assertions into one conjunction + ptr_vector flas; + g->get_formulas(flas); + expr_ref f(m); + f = m.mk_and(flas.size(), flas.c_ptr()); + // running implementation + m_imp = alloc(lackr, m, m_p, f); + const lbool o = m_imp->operator()(); + flas.reset(); + // report result + goal_ref resg(alloc(goal, *g, true)); + if (o == l_false) resg->assert_expr(m.mk_false()); + if (o != l_undef) result.push_back(resg.get()); + // report model + if (g->models_enabled() && (o == l_true)) { + model_ref abstr_model = m_imp->get_model(); + mc = mk_lackr_model_converter(m, m_imp->get_info(), abstr_model); + } + // clenup + lackr * d = m_imp; + #pragma omp critical (lackr) + { + m_imp = 0; + } + dealloc(d); + } + + virtual void cleanup() { } + + virtual tactic* translate(ast_manager& m) { + return alloc(lackr_tactic, m, m_p); + } + + // Currently tactics are not cancelable. + //virtual void set_cancel(bool f) { + // if (m_imp) m_imp->set_cancel(f); + //} +private: + ast_manager& m_m; + params_ref m_p; + lackr* m_imp; +}; + +tactic * mk_lackr_tactic(ast_manager & m, params_ref const & p) { + //return and_then(mk_nnf_tactic(m_m, m_p), alloc(lackr_tactic, m_m, m_p)); + //return alloc(lackr_tactic, m_m, m_p); + //params_ref main_p; + //main_p.set_bool("elim_and", true); + //main_p.set_bool("sort_store", true); + //main_p.set_bool("expand_select_store", true); + //main_p.set_bool("expand_store_eq", true); + + params_ref simp2_p = p; + simp2_p.set_bool("som", true); + simp2_p.set_bool("pull_cheap_ite", true); + simp2_p.set_bool("push_ite_bv", false); + simp2_p.set_bool("local_ctx", true); + simp2_p.set_uint("local_ctx_limit", 10000000); + + simp2_p.set_bool("ite_extra_rules", true); + //simp2_p.set_bool("blast_eq_value", true); + //simp2_p.set_bool("bv_sort_ac", true); + + params_ref ctx_simp_p; + ctx_simp_p.set_uint("max_depth", 32); + ctx_simp_p.set_uint("max_steps", 5000000); + + tactic * const preamble_t = and_then( + mk_simplify_tactic(m), + mk_propagate_values_tactic(m), + //using_params(mk_ctx_simplify_tactic(m_m), ctx_simp_p), + mk_solve_eqs_tactic(m), + mk_elim_uncnstr_tactic(m), + if_no_proofs(if_no_unsat_cores(mk_bv_size_reduction_tactic(m))), + mk_max_bv_sharing_tactic(m), + //mk_macro_finder_tactic(m, p), + using_params(mk_simplify_tactic(m), simp2_p) + //mk_nnf_tactic(m_m, m_p) + ); + + return and_then( + preamble_t, + alloc(lackr_tactic, m, p)); +} diff --git a/src/ackr/lackr_tactic.h b/src/ackr/lackr_tactic.h new file mode 100644 index 000000000..792fa2072 --- /dev/null +++ b/src/ackr/lackr_tactic.h @@ -0,0 +1,28 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +lackr_tactic.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ + +#ifndef _LACKR_TACTIC_H_ +#define _LACKR_TACTIC_H_ +#include"tactical.h" + +tactic * mk_lackr_tactic(ast_manager & m, params_ref const & p); + +/* +ADD_TACTIC("lackr", "lackr.", "mk_lackr_tactic(m, p)") +*/ + +#endif + From 4cc1640a45d50ef35ff691f6984968b780d3e5ab Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Wed, 16 Dec 2015 14:41:54 +0000 Subject: [PATCH 02/31] Adding stats to lackr. --- src/ackr/lackr.cpp | 16 +++++++++------- src/ackr/lackr.h | 26 ++++++++++++++++++-------- src/ackr/lackr_tactic.cpp | 14 ++++++++++++-- 3 files changed, 39 insertions(+), 17 deletions(-) diff --git a/src/ackr/lackr.cpp b/src/ackr/lackr.cpp index 99e5cf0b5..4fa154c34 100644 --- a/src/ackr/lackr.cpp +++ b/src/ackr/lackr.cpp @@ -73,7 +73,7 @@ struct simp_wrap { }; -lackr::lackr(ast_manager& m, params_ref p, expr_ref _f) +lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref _f) : m_m(m) , m_p(p) , m_fla(m) @@ -83,6 +83,7 @@ lackr::lackr(ast_manager& m, params_ref p, expr_ref _f) , m_simp(m) , m_ackrs(m) , m_cancel(0) + , m_st(st) { m_fla = _f; updt_params(p); @@ -157,7 +158,9 @@ bool lackr::ackr(app * const t1, app * const t2) { m_info->abstract(cg, cga); m_simp(cga); TRACE("lackr", tout << "ackr constr abs:" << mk_ismt2_pp(cga, m_m, 2) << "\n";); - if (!m_m.is_true(cga)) m_ackrs.push_back(cga); + if (m_m.is_true(cga)) return true; + m_st.m_ackrs_sz++; + m_ackrs.push_back(cga); return true; } @@ -229,12 +232,12 @@ void lackr::add_term(app* a) { lbool lackr::eager() { - if (!eager_enc()) return l_undef; m_sat->assert_expr(m_abstr); TRACE("lackr", tout << "run sat 0\n"; ); if (m_sat->check_sat(0, 0) == l_false) return l_false; checkpoint(); + if (!eager_enc()) return l_undef; expr_ref all(m_m); all = m_m.mk_and(m_ackrs.size(), m_ackrs.c_ptr()); m_simp(all); @@ -247,12 +250,11 @@ lbool lackr::eager() { lbool lackr::lazy() { lackr_model_constructor mc(m_m, m_info); m_sat->assert_expr(m_abstr); - unsigned ackr_head = 0; - unsigned it = 0; + unsigned ackr_head = 0; while (1) { + m_st.m_it++; checkpoint(); - //std::cout << "it: " << ++it << "\n"; - TRACE("lackr", tout << "lazy check\n";); + TRACE("lackr", tout << "lazy check: " << m_st.m_it << "\n";); const lbool r = m_sat->check_sat(0, 0); if (r == l_undef) return l_undef; // give up if (r == l_false) return l_false; // abstraction unsat diff --git a/src/ackr/lackr.h b/src/ackr/lackr.h index 2f685a001..f1e3de884 100644 --- a/src/ackr/lackr.h +++ b/src/ackr/lackr.h @@ -28,27 +28,36 @@ #include"bv_decl_plugin.h" #include"cooperate.h" +struct lackr_stats { + lackr_stats() : m_it(0), m_ackrs_sz(0) {} + void reset() { m_it = m_ackrs_sz = 0; } + unsigned m_it; // number of lazy iterations + unsigned m_ackrs_sz; // number of congruence constraints +}; + class lackr { public: - lackr(ast_manager& m, params_ref p, expr_ref _f); - lbool operator() (); + lackr(ast_manager& m, params_ref p,lackr_stats& st, + expr_ref _f); ~lackr(); - inline ackr_info_ref get_info() { return m_info; } - inline model_ref get_model() { return m_model; } - void updt_params(params_ref const & _p) { ackr_params p(_p); m_eager = p.eager(); m_use_sat = p.sat_backend(); } + lbool operator() (); + + // + // getters + // + inline ackr_info_ref get_info() { return m_info; } + inline model_ref get_model() { return m_model; } // // timeout mechanisms // - void checkpoint() { - if (m_cancel) - throw tactic_exception(TACTIC_CANCELED_MSG); + if (m_cancel) throw tactic_exception(TACTIC_CANCELED_MSG); cooperate("lackr"); } @@ -78,6 +87,7 @@ class lackr { volatile bool m_cancel; bool m_eager; bool m_use_sat; + lackr_stats& m_st; bool init(); void setup_sat(); diff --git a/src/ackr/lackr_tactic.cpp b/src/ackr/lackr_tactic.cpp index c5368940b..7f2c6f95b 100644 --- a/src/ackr/lackr_tactic.cpp +++ b/src/ackr/lackr_tactic.cpp @@ -28,6 +28,7 @@ Revision History: #include"model_smt2_pp.h" #include"cooperate.h" #include"lackr.h" +#include"ackr_params.hpp" #include"lackr_model_converter.h" class lackr_tactic : public tactic { @@ -56,7 +57,7 @@ public: expr_ref f(m); f = m.mk_and(flas.size(), flas.c_ptr()); // running implementation - m_imp = alloc(lackr, m, m_p, f); + m_imp = alloc(lackr, m, m_p, m_st, f); const lbool o = m_imp->operator()(); flas.reset(); // report result @@ -68,7 +69,7 @@ public: model_ref abstr_model = m_imp->get_model(); mc = mk_lackr_model_converter(m, m_imp->get_info(), abstr_model); } - // clenup + // cleanup lackr * d = m_imp; #pragma omp critical (lackr) { @@ -77,6 +78,14 @@ public: dealloc(d); } + virtual void collect_statistics(statistics & st) const { + ackr_params p(m_p); + if (!p.eager()) st.update("lackr-its", m_st.m_it); + st.update("ackr-constraints", m_st.m_ackrs_sz); + } + + virtual void reset_statistics() { m_st.reset(); } + virtual void cleanup() { } virtual tactic* translate(ast_manager& m) { @@ -91,6 +100,7 @@ private: ast_manager& m_m; params_ref m_p; lackr* m_imp; + lackr_stats m_st; }; tactic * mk_lackr_tactic(ast_manager & m, params_ref const & p) { From 3dbc307ecd11d20137ab9027f5117515830a5663 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Wed, 16 Dec 2015 20:10:14 +0000 Subject: [PATCH 03/31] Setting up the lackr branch. --- src/ackr/ackr_info.h | 2 +- src/ackr/lackr.cpp | 22 +++++++++++++++------- src/ackr/lackr.h | 6 +++++- src/ackr/lackr_model_constructor.cpp | 8 ++++---- src/muz/README | 2 +- src/tactic/bv/bvarray2uf_tactic.h | 2 +- src/tactic/fpa/fpa2bv_model_converter.h | 2 +- 7 files changed, 28 insertions(+), 16 deletions(-) diff --git a/src/ackr/ackr_info.h b/src/ackr/ackr_info.h index 6d2b596fe..3c92f0ab3 100644 --- a/src/ackr/ackr_info.h +++ b/src/ackr/ackr_info.h @@ -98,8 +98,8 @@ class ackr_info { scoped_ptr m_er; expr_substitution m_subst; - bool m_sealed; // debugging unsigned m_ref_count; // reference counting + bool m_sealed; // debugging }; typedef ref ackr_info_ref; diff --git a/src/ackr/lackr.cpp b/src/ackr/lackr.cpp index 4fa154c34..6081633c1 100644 --- a/src/ackr/lackr.cpp +++ b/src/ackr/lackr.cpp @@ -104,6 +104,7 @@ lbool lackr::operator() () { if (!ok) return l_undef; TRACE("lackr", tout << "sat goal\n"; m_sat->display(tout);); const lbool rv = m_eager ? eager() : lazy(); + std::cout << "res: " << rv << "\n"; if (rv == l_true) m_sat->get_model(m_model); TRACE("lackr", tout << "sat:" << rv << '\n'; ); CTRACE("lackr", rv == l_true, @@ -234,16 +235,22 @@ void lackr::add_term(app* a) { lbool lackr::eager() { m_sat->assert_expr(m_abstr); TRACE("lackr", tout << "run sat 0\n"; ); - if (m_sat->check_sat(0, 0) == l_false) - return l_false; + std::cout << "++sat call\n"; + const lbool rv0 = m_sat->check_sat(0, 0); + std::cout << "--sat call\n"; + if (rv0 == l_false) return l_false; checkpoint(); if (!eager_enc()) return l_undef; + checkpoint(); expr_ref all(m_m); all = m_m.mk_and(m_ackrs.size(), m_ackrs.c_ptr()); m_simp(all); TRACE("lackr", tout << "run sat\n"; ); m_sat->assert_expr(all); - return m_sat->check_sat(0, 0); + std::cout << "++sat call\n"; + const lbool rv = m_sat->check_sat(0, 0); + std::cout << "--sat call\n"; + return rv; } @@ -276,10 +283,10 @@ lbool lackr::lazy() { } void lackr::setup_sat() { - if (m_use_sat) { - //std::cout << "; qfbv sat\n"; - tactic_ref t = mk_qfbv_tactic(m_m, m_p); - m_sat = mk_tactic2solver(m_m, t.get(), m_p); + if (m_use_sat) { + //tactic_ref t = mk_qfbv_tactic(m_m, m_p); + //m_sat = mk_tactic2solver(m_m, t.get(), m_p); + m_sat = mk_inc_sat_solver(m_m, m_p); } else { //std::cout << "; smt sat\n"; @@ -318,6 +325,7 @@ bool lackr::collect_terms() { visited.mark(curr, true); stack.pop_back(); add_term(to_app(curr)); + checkpoint(); } break; case AST_QUANTIFIER: diff --git a/src/ackr/lackr.h b/src/ackr/lackr.h index f1e3de884..38f810d04 100644 --- a/src/ackr/lackr.h +++ b/src/ackr/lackr.h @@ -57,7 +57,11 @@ class lackr { // timeout mechanisms // void checkpoint() { - if (m_cancel) throw tactic_exception(TACTIC_CANCELED_MSG); + //std::cout << "chk\n"; + if (m_m.canceled()) { + std::cout << "canceled\n"; + throw tactic_exception(TACTIC_CANCELED_MSG); + } cooperate("lackr"); } diff --git a/src/ackr/lackr_model_constructor.cpp b/src/ackr/lackr_model_constructor.cpp index 5f0060ad5..3f727cd92 100644 --- a/src/ackr/lackr_model_constructor.cpp +++ b/src/ackr/lackr_model_constructor.cpp @@ -26,7 +26,7 @@ struct lackr_model_constructor::imp { imp(ast_manager& m, ackr_info_ref info, model_ref& abstr_model, - vector>& conflicts) + conflict_list& conflicts) : m_m(m) , m_info(info) , m_abstr_model(abstr_model) @@ -45,7 +45,7 @@ struct lackr_model_constructor::imp { m_m.dec_ref(i->m_value.value); m_m.dec_ref(i->m_value.source_term); } - } + } { app2val_t::iterator i = m_app2val.begin(); const app2val_t::iterator e = m_app2val.end(); @@ -72,7 +72,7 @@ struct lackr_model_constructor::imp { ast_manager& m_m; ackr_info_ref m_info; model_ref& m_abstr_model; - vector>& m_conflicts; + conflict_list& m_conflicts; bool_rewriter m_b_rw; bv_rewriter m_bv_rw; scoped_ptr m_evaluator; @@ -310,4 +310,4 @@ bool lackr_model_constructor::check(model_ref& abstr_model) { const bool rv = i.check(); state = rv ? CHECKED : CONFLICT; return rv; -} \ No newline at end of file +} diff --git a/src/muz/README b/src/muz/README index c7d5a9665..3cebe98f6 100644 --- a/src/muz/README +++ b/src/muz/README @@ -9,4 +9,4 @@ solving Datalog programs. - clp - Dart/Symbolic execution-based solver - tab - Tabulation based solver - bmc - Bounded model checking based solver -- fp - main exported routines \ No newline at end of file +- fp - main exported routines diff --git a/src/tactic/bv/bvarray2uf_tactic.h b/src/tactic/bv/bvarray2uf_tactic.h index 608a831e0..336911bf7 100644 --- a/src/tactic/bv/bvarray2uf_tactic.h +++ b/src/tactic/bv/bvarray2uf_tactic.h @@ -30,4 +30,4 @@ tactic * mk_bvarray2uf_tactic(ast_manager & m, params_ref const & p = params_ref */ -#endif \ No newline at end of file +#endif diff --git a/src/tactic/fpa/fpa2bv_model_converter.h b/src/tactic/fpa/fpa2bv_model_converter.h index 534fb6e1c..854543f24 100644 --- a/src/tactic/fpa/fpa2bv_model_converter.h +++ b/src/tactic/fpa/fpa2bv_model_converter.h @@ -107,4 +107,4 @@ protected: model_converter * mk_fpa2bv_model_converter(ast_manager & m, fpa2bv_converter const & conv); -#endif \ No newline at end of file +#endif From 0fd61eee1f5c295b4d42bbeaed43e5a4e13e2560 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Thu, 17 Dec 2015 13:25:56 +0000 Subject: [PATCH 04/31] Cleanup in lackr. --- src/ackr/lackr.cpp | 147 ++++++++++---------------------------- src/ackr/lackr.h | 44 ++++-------- src/ackr/lackr_tactic.cpp | 26 ++----- 3 files changed, 56 insertions(+), 161 deletions(-) diff --git a/src/ackr/lackr.cpp b/src/ackr/lackr.cpp index 6081633c1..aa99f347e 100644 --- a/src/ackr/lackr.cpp +++ b/src/ackr/lackr.cpp @@ -22,57 +22,13 @@ #include"ackr_info.h" #include"for_each_expr.h" /////////////// -#include"array_decl_plugin.h" -#include"simplifier_plugin.h" -#include"basic_simplifier_plugin.h" -#include"array_simplifier_params.h" -#include"array_simplifier_plugin.h" -#include"bv_simplifier_plugin.h" -#include"bool_rewriter.h" -/////////////// -#include"th_rewriter.h" -/////////////// -#include"cooperate.h" +#include"inc_sat_solver.h" +#include"qfaufbv_tactic.h" +#include"qfbv_tactic.h" +#include"tactic2solver.h" /////////////// #include"model_smt2_pp.h" /////////////// - -struct simp_wrap { - inline void operator() (expr * s, expr_ref & r) { - proof_ref dummy(m); - simp(s, r, dummy); - } - simp_wrap(ast_manager& m) - : m(m) - , simp(m) - , bsp(m) - , bvsp(m, bsp, bv_par) - , asp(m, bsp, simp, ar_par) - { - params_ref p; - p.set_bool("local_ctx", true); - p.set_uint("local_ctx_limit", 10000000); - p.set_bool("ite_extra_rules", true); - bsp.get_rewriter().updt_params(p); - - simp.register_plugin(&bsp); - simp.register_plugin(&bvsp); - } - - ~simp_wrap() { - simp.release_plugins(); - } - - ast_manager& m; - simplifier simp; - basic_simplifier_plugin bsp; - bv_simplifier_params bv_par; - bv_simplifier_plugin bvsp; - array_simplifier_params ar_par; - array_simplifier_plugin asp; -}; - - lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref _f) : m_m(m) , m_p(p) @@ -81,8 +37,7 @@ lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref _f) , m_sat(0) , m_bvutil(m) , m_simp(m) - , m_ackrs(m) - , m_cancel(0) + , m_ackrs(m) , m_st(st) { m_fla = _f; @@ -96,30 +51,22 @@ lackr::~lackr() { } } - - lbool lackr::operator() () { setup_sat(); - const bool ok = init(); - if (!ok) return l_undef; - TRACE("lackr", tout << "sat goal\n"; m_sat->display(tout);); + init(); const lbool rv = m_eager ? eager() : lazy(); - std::cout << "res: " << rv << "\n"; if (rv == l_true) m_sat->get_model(m_model); - TRACE("lackr", tout << "sat:" << rv << '\n'; ); CTRACE("lackr", rv == l_true, model_smt2_pp(tout << "abstr_model(\n", m_m, *(m_model.get()), 2); tout << ")\n"; ); return rv; } - -bool lackr::init() { +void lackr::init() { params_ref simp_p(m_p); m_simp.updt_params(simp_p); m_info = alloc(ackr_info, m_m); - bool iok = collect_terms() && abstract(); - if (!iok) return false; - return true; + collect_terms(); + abstract(); } // @@ -127,10 +74,9 @@ bool lackr::init() { // bool lackr::ackr(app * const t1, app * const t2) { TRACE("lackr", tout << "ackr " - << mk_ismt2_pp(t1, m_m, 2) - << " , " - << mk_ismt2_pp(t2, m_m, 2) << "\n";); + << mk_ismt2_pp(t1, m_m, 2) << " , " << mk_ismt2_pp(t2, m_m, 2) << "\n";); const unsigned sz = t1->get_num_args(); + SASSERT(t2->get_num_args() == sz); expr_ref_vector eqs(m_m); for (unsigned gi = 0; gi < sz; ++gi) { expr * const arg1 = t1->get_arg(gi); @@ -139,14 +85,13 @@ bool lackr::ackr(app * const t1, app * const t2) { if (m_bvutil.is_numeral(arg1) && m_bvutil.is_numeral(arg2)) { SASSERT(arg1 != arg2); TRACE("lackr", tout << "never eq\n";); - return true; + return false; } eqs.push_back(m_m.mk_eq(arg1, arg2)); } app * const a1 = m_info->get_abstr(t1); app * const a2 = m_info->get_abstr(t2); - SASSERT(a1); - SASSERT(a2); + SASSERT(a1 && a2); TRACE("lackr", tout << "abstr1 " << mk_ismt2_pp(a1, m_m, 2) << "\n";); TRACE("lackr", tout << "abstr2 " << mk_ismt2_pp(a2, m_m, 2) << "\n";); expr_ref lhs(m_m.mk_and(eqs.size(), eqs.c_ptr()), m_m); @@ -156,10 +101,10 @@ bool lackr::ackr(app * const t1, app * const t2) { expr_ref cg(m_m.mk_implies(lhs, rhs), m_m); TRACE("lackr", tout << "ackr constr" << mk_ismt2_pp(cg, m_m, 2) << "\n";); expr_ref cga(m_m); - m_info->abstract(cg, cga); + m_info->abstract(cg, cga); // constraint needs abstraction due to nested applications m_simp(cga); TRACE("lackr", tout << "ackr constr abs:" << mk_ismt2_pp(cga, m_m, 2) << "\n";); - if (m_m.is_true(cga)) return true; + if (m_m.is_true(cga)) return false; m_st.m_ackrs_sz++; m_ackrs.push_back(cga); return true; @@ -168,12 +113,12 @@ bool lackr::ackr(app * const t1, app * const t2) { // // Introduce the ackermann lemma for each pair of terms. // -bool lackr::eager_enc() { +void lackr::eager_enc() { const fun2terms_map::iterator e = m_fun2terms.end(); for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { checkpoint(); func_decl* const fd = i->m_key; - app_set * const ts = i->get_value(); + app_set * const ts = i->get_value(); const app_set::iterator r = ts->end(); for (app_set::iterator j = ts->begin(); j != r; ++j) { app_set::iterator k = j; @@ -184,14 +129,13 @@ bool lackr::eager_enc() { SASSERT(t1->get_decl() == fd); SASSERT(t2->get_decl() == fd); if (t1 == t2) continue; - if (!ackr(t1,t2)) return false; + ackr(t1,t2); } } } - return true; } -bool lackr::abstract() { +void lackr::abstract() { const fun2terms_map::iterator e = m_fun2terms.end(); for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { func_decl* const fd = i->m_key; @@ -212,12 +156,10 @@ bool lackr::abstract() { } m_info->seal(); m_info->abstract(m_fla.get(), m_abstr); - TRACE("lackr", tout << "abs(\n" << mk_ismt2_pp(m_abstr.get(), m_m, 2) << ")\n";); - return true; + TRACE("lackr", tout << "abs(\n" << mk_ismt2_pp(m_abstr.get(), m_m, 2) << ")\n";); } void lackr::add_term(app* a) { - //TRACE("lackr", tout << "inspecting term(\n" << mk_ismt2_pp(a, m_m, 2) << ")\n";); if (a->get_num_args() == 0) return; func_decl* const fd = a->get_decl(); if (!is_uninterp(a)) return; @@ -235,25 +177,17 @@ void lackr::add_term(app* a) { lbool lackr::eager() { m_sat->assert_expr(m_abstr); TRACE("lackr", tout << "run sat 0\n"; ); - std::cout << "++sat call\n"; const lbool rv0 = m_sat->check_sat(0, 0); - std::cout << "--sat call\n"; if (rv0 == l_false) return l_false; - checkpoint(); - if (!eager_enc()) return l_undef; - checkpoint(); + eager_enc(); expr_ref all(m_m); all = m_m.mk_and(m_ackrs.size(), m_ackrs.c_ptr()); - m_simp(all); - TRACE("lackr", tout << "run sat\n"; ); + m_simp(all); m_sat->assert_expr(all); - std::cout << "++sat call\n"; - const lbool rv = m_sat->check_sat(0, 0); - std::cout << "--sat call\n"; - return rv; + TRACE("lackr", tout << "run sat all\n"; ); + return m_sat->check_sat(0, 0); } - lbool lackr::lazy() { lackr_model_constructor mc(m_m, m_info); m_sat->assert_expr(m_abstr); @@ -284,12 +218,10 @@ lbool lackr::lazy() { void lackr::setup_sat() { if (m_use_sat) { - //tactic_ref t = mk_qfbv_tactic(m_m, m_p); - //m_sat = mk_tactic2solver(m_m, t.get(), m_p); - m_sat = mk_inc_sat_solver(m_m, m_p); + tactic_ref t = mk_qfbv_tactic(m_m, m_p); + m_sat = mk_tactic2solver(m_m, t.get(), m_p); } else { - //std::cout << "; smt sat\n"; tactic_ref t = mk_qfaufbv_tactic(m_m, m_p); m_sat = mk_tactic2solver(m_m, t.get(), m_p); } @@ -299,9 +231,9 @@ void lackr::setup_sat() { // -// Collect all uninterpreted terms. +// Collect all uninterpreted terms, skipping 0-arity. // -bool lackr::collect_terms() { +void lackr::collect_terms() { ptr_vector stack; expr * curr; expr_mark visited; @@ -318,30 +250,23 @@ bool lackr::collect_terms() { visited.mark(curr, true); stack.pop_back(); break; - case AST_APP: - if (for_each_expr_args(stack, visited, - to_app(curr)->get_num_args(), to_app(curr)->get_args())) { + { + app * const a = to_app(curr); + if (for_each_expr_args(stack, visited, a->get_num_args(), a->get_args())) { visited.mark(curr, true); stack.pop_back(); - add_term(to_app(curr)); - checkpoint(); + add_term(a); } + } break; case AST_QUANTIFIER: - if (visited.is_marked(to_quantifier(curr)->get_expr())) { - visited.mark(curr, true); - stack.pop_back(); - } - else { - stack.push_back(to_quantifier(curr)->get_expr()); - } + UNREACHABLE(); break; default: UNREACHABLE(); - return false; + return; } } - visited.reset(); - return true; + visited.reset(); } diff --git a/src/ackr/lackr.h b/src/ackr/lackr.h index 38f810d04..ea1ea7ce2 100644 --- a/src/ackr/lackr.h +++ b/src/ackr/lackr.h @@ -17,16 +17,16 @@ #ifndef LACKR_H_15079 #define LACKR_H_15079 /////////////// -#include"inc_sat_solver.h" -#include"qfaufbv_tactic.h" -#include"qfbv_tactic.h" -#include"tactic2solver.h" #include"ackr_info.h" #include"ackr_params.hpp" -#include"tactic_exception.h" #include"th_rewriter.h" -#include"bv_decl_plugin.h" #include"cooperate.h" +#include"bv_decl_plugin.h" +#include"lbool.h" +#include"model.h" +#include"solver.h" +#include"util.h" +#include"tactic_exception.h" struct lackr_stats { lackr_stats() : m_it(0), m_ackrs_sz(0) {} @@ -37,8 +37,7 @@ struct lackr_stats { class lackr { public: - lackr(ast_manager& m, params_ref p,lackr_stats& st, - expr_ref _f); + lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref _f); ~lackr(); void updt_params(params_ref const & _p) { ackr_params p(_p); @@ -54,26 +53,14 @@ class lackr { inline model_ref get_model() { return m_model; } // - // timeout mechanisms + // timeout mechanism // void checkpoint() { - //std::cout << "chk\n"; - if (m_m.canceled()) { - std::cout << "canceled\n"; + if (m_m.canceled()) { throw tactic_exception(TACTIC_CANCELED_MSG); } cooperate("lackr"); } - - //virtual void set_cancel(bool f) { - // //#pragma omp critical (lackr_cancel) - // { - // m_cancel = f; - // if (m_sat == NULL) return; - // if (f) m_sat->cancel(); - // else m_sat->reset_cancel(); - // } - //} private: typedef obj_hashtable app_set; typedef obj_map fun2terms_map; @@ -88,33 +75,32 @@ class lackr { th_rewriter m_simp; expr_ref_vector m_ackrs; model_ref m_model; - volatile bool m_cancel; bool m_eager; bool m_use_sat; lackr_stats& m_st; - bool init(); + void init(); void setup_sat(); lbool eager(); lbool lazy(); // - // Introduce ackermann lemma for the two given terms. + // Introduce congruence ackermann lemma for the two given terms. // bool ackr(app * const t1, app * const t2); // // Introduce the ackermann lemma for each pair of terms. // - bool eager_enc(); + void eager_enc(); - bool abstract(); + void abstract(); void add_term(app* a); // - // Collect all uninterpreted terms. + // Collect all uninterpreted terms, skipping 0-arity. // - bool collect_terms(); + void collect_terms(); }; #endif /* LACKR_H_15079 */ diff --git a/src/ackr/lackr_tactic.cpp b/src/ackr/lackr_tactic.cpp index 7f2c6f95b..5e2b88e70 100644 --- a/src/ackr/lackr_tactic.cpp +++ b/src/ackr/lackr_tactic.cpp @@ -36,12 +36,9 @@ public: lackr_tactic(ast_manager& m, params_ref const& p) : m_m(m) , m_p(p) - , m_imp(0) {} - virtual ~lackr_tactic() { - if (m_imp) dealloc(m_imp); - } + virtual ~lackr_tactic() { } virtual void operator()(goal_ref const & g, goal_ref_buffer & result, @@ -57,8 +54,8 @@ public: expr_ref f(m); f = m.mk_and(flas.size(), flas.c_ptr()); // running implementation - m_imp = alloc(lackr, m, m_p, m_st, f); - const lbool o = m_imp->operator()(); + scoped_ptr imp = alloc(lackr, m, m_p, m_st, f); + const lbool o = imp->operator()(); flas.reset(); // report result goal_ref resg(alloc(goal, *g, true)); @@ -66,16 +63,9 @@ public: if (o != l_undef) result.push_back(resg.get()); // report model if (g->models_enabled() && (o == l_true)) { - model_ref abstr_model = m_imp->get_model(); - mc = mk_lackr_model_converter(m, m_imp->get_info(), abstr_model); + model_ref abstr_model = imp->get_model(); + mc = mk_lackr_model_converter(m, imp->get_info(), abstr_model); } - // cleanup - lackr * d = m_imp; - #pragma omp critical (lackr) - { - m_imp = 0; - } - dealloc(d); } virtual void collect_statistics(statistics & st) const { @@ -91,15 +81,9 @@ public: virtual tactic* translate(ast_manager& m) { return alloc(lackr_tactic, m, m_p); } - - // Currently tactics are not cancelable. - //virtual void set_cancel(bool f) { - // if (m_imp) m_imp->set_cancel(f); - //} private: ast_manager& m_m; params_ref m_p; - lackr* m_imp; lackr_stats m_st; }; From 5706df30c6b1f969575954879b8ab32bee35ee0d Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Fri, 8 Jan 2016 16:17:34 +0000 Subject: [PATCH 05/31] Fixing soft timeout for check-sat-using. --- src/ackr/lackr_tactic.cpp | 1 + src/cmd_context/tactic_cmds.cpp | 4 ++-- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/src/ackr/lackr_tactic.cpp b/src/ackr/lackr_tactic.cpp index 5e2b88e70..27cd12415 100644 --- a/src/ackr/lackr_tactic.cpp +++ b/src/ackr/lackr_tactic.cpp @@ -121,6 +121,7 @@ tactic * mk_lackr_tactic(ast_manager & m, params_ref const & p) { mk_max_bv_sharing_tactic(m), //mk_macro_finder_tactic(m, p), using_params(mk_simplify_tactic(m), simp2_p) + //, mk_simplify_tactic(m) //mk_nnf_tactic(m_m, m_p) ); diff --git a/src/cmd_context/tactic_cmds.cpp b/src/cmd_context/tactic_cmds.cpp index 81bf2136b..1c5da3829 100644 --- a/src/cmd_context/tactic_cmds.cpp +++ b/src/cmd_context/tactic_cmds.cpp @@ -187,8 +187,8 @@ public: tactic_ref tref = using_params(sexpr2tactic(ctx, m_tactic), p); tref->set_logic(ctx.get_logic()); ast_manager & m = ctx.m(); - unsigned timeout = p.get_uint("timeout", UINT_MAX); - unsigned rlimit = p.get_uint("rlimit", 0); + unsigned timeout = p.get_uint("timeout", ctx.params().m_timeout); + unsigned rlimit = p.get_uint("rlimit", ctx.params().m_rlimit); goal_ref g = alloc(goal, m, ctx.produce_proofs(), ctx.produce_models(), ctx.produce_unsat_cores()); assert_exprs_from(ctx, *g); TRACE("check_sat_using", g->display(tout);); From b46f31211505688c3f8702558a5c55ba6a53d0d9 Mon Sep 17 00:00:00 2001 From: mikolas Date: Mon, 11 Jan 2016 17:18:22 +0000 Subject: [PATCH 06/31] A proper model converter for the lazy mode. --- src/ackr/lackr_model_constructor.cpp | 80 ++++++++++++++++++++++--- src/ackr/lackr_model_constructor.h | 29 +++++++-- src/ackr/lackr_model_converter_lazy.cpp | 60 +++++++++++++++++++ src/ackr/lackr_model_converter_lazy.h | 23 +++++++ 4 files changed, 179 insertions(+), 13 deletions(-) create mode 100644 src/ackr/lackr_model_converter_lazy.cpp create mode 100644 src/ackr/lackr_model_converter_lazy.h diff --git a/src/ackr/lackr_model_constructor.cpp b/src/ackr/lackr_model_constructor.cpp index 3f727cd92..bcc036d2e 100644 --- a/src/ackr/lackr_model_constructor.cpp +++ b/src/ackr/lackr_model_constructor.cpp @@ -68,6 +68,57 @@ struct lackr_model_constructor::imp { } return run(); } + + + void make_model(model_ref& destination) { + { + for (unsigned i = 0; i < m_abstr_model->get_num_uninterpreted_sorts(); i++) { + sort * const s = m_abstr_model->get_uninterpreted_sort(i); + ptr_vector u = m_abstr_model->get_universe(s); + destination->register_usort(s, u.size(), u.c_ptr()); + } + } + { + const app2val_t::iterator e = m_app2val.end(); + app2val_t::iterator i = m_app2val.end(); + for (; i != e; ++i) { + app * a = i->m_key; + if (a->get_num_args()) continue; + destination->register_decl(a->get_decl(), i->m_value); + } + } + + obj_map interpretations; + { + const values2val_t::iterator e = m_values2val.end(); + values2val_t::iterator i = m_values2val.end(); + for (; i != e; ++i) add_entry(i->m_key, i->m_value.value, interpretations); + } + + { + obj_map::iterator ie = interpretations.end(); + obj_map::iterator ii = interpretations.begin(); + for (; ii != ie; ++ii) { + func_decl* const fd = ii->m_key; + func_interp* const fi = ii->get_value(); + fi->set_else(m_m.get_some_value(fd->get_range())); + destination->register_decl(fd, fi); + } + } + } + + void add_entry(app* term, expr* value, + obj_map& interpretations) { + func_interp* fi = 0; + func_decl * const declaration = term->get_decl(); + const unsigned sz = declaration->get_arity(); + SASSERT(sz == term->get_num_args()); + if (!interpretations.find(declaration, fi)) { + fi = alloc(func_interp, m_m, sz); + interpretations.insert(declaration, fi); + } + fi->insert_new_entry(term->get_args(), value); + } private: ast_manager& m_m; ackr_info_ref m_info; @@ -299,15 +350,30 @@ struct lackr_model_constructor::imp { }; lackr_model_constructor::lackr_model_constructor(ast_manager& m, ackr_info_ref info) - : m(m) - , state(UNKNOWN) - , info(info) + : m_imp(0) + , m_m(m) + , m_state(UNKNOWN) + , m_info(info) + , m_ref_count(0) {} +lackr_model_constructor::~lackr_model_constructor() { + if (m_imp) dealloc(m_imp); +} + bool lackr_model_constructor::check(model_ref& abstr_model) { - conflicts.reset(); - lackr_model_constructor::imp i(m, info, abstr_model, conflicts); - const bool rv = i.check(); - state = rv ? CHECKED : CONFLICT; + m_conflicts.reset(); + if (m_imp) { + dealloc(m_imp); + m_imp = 0; + } + m_imp = alloc(lackr_model_constructor::imp, m_m, m_info, abstr_model, m_conflicts); + const bool rv = m_imp->check(); + m_state = rv ? CHECKED : CONFLICT; return rv; } + +void lackr_model_constructor::make_model(model_ref& model) { + SASSERT(m_state == CHECKED); + m_imp->make_model(model); +} diff --git a/src/ackr/lackr_model_constructor.h b/src/ackr/lackr_model_constructor.h index 5b2385c2c..478efe97f 100644 --- a/src/ackr/lackr_model_constructor.h +++ b/src/ackr/lackr_model_constructor.h @@ -25,16 +25,33 @@ class lackr_model_constructor { typedef std::pair app_pair; typedef vector conflict_list; lackr_model_constructor(ast_manager& m, ackr_info_ref info); + ~lackr_model_constructor(); bool check(model_ref& abstr_model); const conflict_list& get_conflicts() { - SASSERT(state == CONFLICT); - return conflicts; + SASSERT(m_state == CONFLICT); + return m_conflicts; + } + void make_model(model_ref& model); + + // + // Reference counting + // + void inc_ref() { ++m_ref_count; } + void dec_ref() { + --m_ref_count; + if (m_ref_count == 0) { + dealloc(this); + } } private: struct imp; - enum {CHECKED, CONFLICT, UNKNOWN} state; - conflict_list conflicts; - ast_manager& m; - const ackr_info_ref info; + imp * m_imp; + enum {CHECKED, CONFLICT, UNKNOWN} m_state; + conflict_list m_conflicts; + ast_manager& m_m; + const ackr_info_ref m_info; + unsigned m_ref_count; // reference counting }; + +typedef ref lackr_model_constructor_ref; #endif /* MODEL_CONSTRUCTOR_H_626 */ diff --git a/src/ackr/lackr_model_converter_lazy.cpp b/src/ackr/lackr_model_converter_lazy.cpp new file mode 100644 index 000000000..dc3a2b40e --- /dev/null +++ b/src/ackr/lackr_model_converter_lazy.cpp @@ -0,0 +1,60 @@ +/*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr_model_converter_lazy.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ +#include"lackr_model_converter_lazy.h" +#include"model_evaluator.h" +#include"ast_smt2_pp.h" +#include"ackr_info.h" +#include"lackr_model_constructor.h" + + +class lackr_model_converter_lazy : public model_converter { +public: + lackr_model_converter_lazy(ast_manager & m, + const lackr_model_constructor_ref& lmc) + : m(m) + , model_constructor(lmc) + { } + + virtual ~lackr_model_converter_lazy() { } + + virtual void operator()(model_ref & md, unsigned goal_idx) { + SASSERT(goal_idx == 0); + SASSERT(md.get() == 0 || (!md->get_num_constants() && !md->get_num_functions())); + SASSERT(model_constructor.get()); + model * new_model = alloc(model, m); + md = new_model; + model_constructor->make_model(md); + } + + virtual void operator()(model_ref & md) { + operator()(md, 0); + } + + //void display(std::ostream & out); + + virtual model_converter * translate(ast_translation & translator) { + NOT_IMPLEMENTED_YET(); + } +protected: + ast_manager& m; + const lackr_model_constructor_ref model_constructor; +}; + +model_converter * mk_lackr_model_converter_lazy(ast_manager & m, + const lackr_model_constructor_ref& model_constructor) { + return alloc(lackr_model_converter_lazy, m, model_constructor); +} diff --git a/src/ackr/lackr_model_converter_lazy.h b/src/ackr/lackr_model_converter_lazy.h new file mode 100644 index 000000000..354e1c40b --- /dev/null +++ b/src/ackr/lackr_model_converter_lazy.h @@ -0,0 +1,23 @@ + /*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr_model_converter_lazy.h + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef LACKR_MODEL_CONVERTER_LAZY_H_14201 +#define LACKR_MODEL_CONVERTER_LAZY_H_14201 +#include"model_converter.h" +#include"ackr_info.h" + +model_converter * mk_lackr_model_converter_lazy(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); +#endif /* LACKR_MODEL_CONVERTER_LAZY_H_14201 */ From d97e2b432cdb8bdd3c67d371b7c8a89577a60a3a Mon Sep 17 00:00:00 2001 From: mikolas Date: Thu, 14 Jan 2016 14:11:11 +0000 Subject: [PATCH 07/31] Ackermann run on separate assertions rather than an AND thereof. --- src/ackr/lackr.cpp | 30 ++++++++++++++++++++++-------- src/ackr/lackr.h | 8 +++++--- src/ackr/lackr_tactic.cpp | 10 ++++------ 3 files changed, 31 insertions(+), 17 deletions(-) diff --git a/src/ackr/lackr.cpp b/src/ackr/lackr.cpp index aa99f347e..3bbfed791 100644 --- a/src/ackr/lackr.cpp +++ b/src/ackr/lackr.cpp @@ -29,10 +29,10 @@ /////////////// #include"model_smt2_pp.h" /////////////// -lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref _f) +lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas) : m_m(m) , m_p(p) - , m_fla(m) + , m_formulas(formulas) , m_abstr(m) , m_sat(0) , m_bvutil(m) @@ -40,7 +40,6 @@ lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref _f) , m_ackrs(m) , m_st(st) { - m_fla = _f; updt_params(p); } @@ -135,6 +134,7 @@ void lackr::eager_enc() { } } + void lackr::abstract() { const fun2terms_map::iterator e = m_fun2terms.end(); for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { @@ -155,8 +155,13 @@ void lackr::abstract() { } } m_info->seal(); - m_info->abstract(m_fla.get(), m_abstr); - TRACE("lackr", tout << "abs(\n" << mk_ismt2_pp(m_abstr.get(), m_m, 2) << ")\n";); + // perform abstraction of the formulas + const unsigned sz = m_formulas.size(); + for (unsigned i = 0; i < sz; ++i) { + expr_ref a(m_m); + m_info->abstract(m_formulas.get(i), a); + m_abstr.push_back(a); + } } void lackr::add_term(app* a) { @@ -173,9 +178,15 @@ void lackr::add_term(app* a) { ts->insert(a); } +void lackr::push_abstraction() { + const unsigned sz = m_abstr.size(); + for (unsigned i = 0; i < sz; ++i) { + m_sat->assert_expr(m_abstr.get(i)); + } +} lbool lackr::eager() { - m_sat->assert_expr(m_abstr); + push_abstraction(); TRACE("lackr", tout << "run sat 0\n"; ); const lbool rv0 = m_sat->check_sat(0, 0); if (rv0 == l_false) return l_false; @@ -190,7 +201,7 @@ lbool lackr::eager() { lbool lackr::lazy() { lackr_model_constructor mc(m_m, m_info); - m_sat->assert_expr(m_abstr); + push_abstraction(); unsigned ackr_head = 0; while (1) { m_st.m_it++; @@ -237,7 +248,10 @@ void lackr::collect_terms() { ptr_vector stack; expr * curr; expr_mark visited; - stack.push_back(m_fla.get()); + for(unsigned i = 0; i < m_formulas.size(); ++i) { + stack.push_back(m_formulas.get(i)); + } + while (!stack.empty()) { curr = stack.back(); if (visited.is_marked(curr)) { diff --git a/src/ackr/lackr.h b/src/ackr/lackr.h index ea1ea7ce2..e0a5394ab 100644 --- a/src/ackr/lackr.h +++ b/src/ackr/lackr.h @@ -37,7 +37,7 @@ struct lackr_stats { class lackr { public: - lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref _f); + lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas); ~lackr(); void updt_params(params_ref const & _p) { ackr_params p(_p); @@ -66,8 +66,8 @@ class lackr { typedef obj_map fun2terms_map; ast_manager& m_m; params_ref m_p; - expr_ref m_fla; - expr_ref m_abstr; + expr_ref_vector m_formulas; + expr_ref_vector m_abstr; fun2terms_map m_fun2terms; ackr_info_ref m_info; scoped_ptr m_sat; @@ -96,6 +96,8 @@ class lackr { void abstract(); + void push_abstraction(); + void add_term(app* a); // diff --git a/src/ackr/lackr_tactic.cpp b/src/ackr/lackr_tactic.cpp index 27cd12415..2302e859f 100644 --- a/src/ackr/lackr_tactic.cpp +++ b/src/ackr/lackr_tactic.cpp @@ -48,13 +48,11 @@ public: mc = 0; ast_manager& m(g->m()); TRACE("lackr", g->display(tout << "Goal:\n");); - // conflate all assertions into one conjunction - ptr_vector flas; - g->get_formulas(flas); - expr_ref f(m); - f = m.mk_and(flas.size(), flas.c_ptr()); // running implementation - scoped_ptr imp = alloc(lackr, m, m_p, m_st, f); + expr_ref_vector flas(m); + const unsigned sz = g->size(); + for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); + scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas); const lbool o = imp->operator()(); flas.reset(); // report result From d32c9449b23d2320761868f43fe543aeb5201afd Mon Sep 17 00:00:00 2001 From: mikolas Date: Tue, 26 Jan 2016 11:53:47 +0000 Subject: [PATCH 08/31] Editing some comments and also enabling to export the ackermannized formula as a gole. --- src/ackr/ackr_info.h | 19 ++++++++++--------- src/ackr/lackr.cpp | 8 ++++++++ src/ackr/lackr.h | 17 +++++++++++++++++ src/ackr/lackr_tactic.h | 2 +- 4 files changed, 36 insertions(+), 10 deletions(-) diff --git a/src/ackr/ackr_info.h b/src/ackr/ackr_info.h index 3c92f0ab3..1b21f0c91 100644 --- a/src/ackr/ackr_info.h +++ b/src/ackr/ackr_info.h @@ -20,15 +20,16 @@ Revision History: #include"ref.h" #include"expr_replacer.h" -// -// Information about how a formula was extracted into -// a formula without uninterpreted function symbols. -// -// The intended use is that new terms are added via set_abstr. -// Once all terms are abstracted, call seal. -// abstract may only be called when sealed. -// -// The class enables reference counting. +/** \brief + Information about how a formula is being converted into + a formula without uninterpreted function symbols via ackermannization. + + The intended use is that new terms are added via set_abstr. + Once all terms are abstracted, call seal. + The function abstract may only be called when sealed. + + The class enables reference counting. +**/ class ackr_info { public: ackr_info(ast_manager& m) diff --git a/src/ackr/lackr.cpp b/src/ackr/lackr.cpp index 3bbfed791..c1138c7a5 100644 --- a/src/ackr/lackr.cpp +++ b/src/ackr/lackr.cpp @@ -60,6 +60,13 @@ lbool lackr::operator() () { return rv; } +void lackr::mk_ackermann(/*out*/goal_ref& g) { + init(); + eager_enc(); + for (unsigned i = 0; i < m_abstr.size(); ++i) g->assert_expr(m_abstr.get(i)); + for (unsigned i = 0; i < m_ackrs.size(); ++i) g->assert_expr(m_ackrs.get(i)); +} + void lackr::init() { params_ref simp_p(m_p); m_simp.updt_params(simp_p); @@ -82,6 +89,7 @@ bool lackr::ackr(app * const t1, app * const t2) { expr * const arg2 = t2->get_arg(gi); if (arg1 == arg2) continue; if (m_bvutil.is_numeral(arg1) && m_bvutil.is_numeral(arg2)) { + // quickly abort if there are two distinct numerals SASSERT(arg1 != arg2); TRACE("lackr", tout << "never eq\n";); return false; diff --git a/src/ackr/lackr.h b/src/ackr/lackr.h index e0a5394ab..efeaf0de2 100644 --- a/src/ackr/lackr.h +++ b/src/ackr/lackr.h @@ -27,6 +27,7 @@ #include"solver.h" #include"util.h" #include"tactic_exception.h" +#include"goal.h" struct lackr_stats { lackr_stats() : m_it(0), m_ackrs_sz(0) {} @@ -35,6 +36,10 @@ struct lackr_stats { unsigned m_ackrs_sz; // number of congruence constraints }; +/** \brief + A class to encode or directly solve problems with uninterpreted functions via ackermannization. + Currently, solving is supported only for QF_UFBV. +**/ class lackr { public: lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas); @@ -44,8 +49,20 @@ class lackr { m_eager = p.eager(); m_use_sat = p.sat_backend(); } + + /** \brief + * Solve the formula that the class was initialized with. + **/ lbool operator() (); + + /** \brief + * Converts function occurrences to constants and encodes all congruence ackermann lemmas. + * This guarantees a equisatisfiability with the input formula. It has a worst-case quadratic blowup. + **/ + void mk_ackermann(/*out*/goal_ref& g); + + // // getters // diff --git a/src/ackr/lackr_tactic.h b/src/ackr/lackr_tactic.h index 792fa2072..c578bed92 100644 --- a/src/ackr/lackr_tactic.h +++ b/src/ackr/lackr_tactic.h @@ -21,7 +21,7 @@ Revision History: tactic * mk_lackr_tactic(ast_manager & m, params_ref const & p); /* -ADD_TACTIC("lackr", "lackr.", "mk_lackr_tactic(m, p)") +ADD_TACTIC("lackr", "A tactic for solving QF_UFBV based on Ackermannization.", "mk_lackr_tactic(m, p)") */ #endif From c63f9f49129014ba1b242c2310370d72aa52a13c Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Tue, 26 Jan 2016 16:50:00 +0000 Subject: [PATCH 09/31] Moving things around. Adding tactic just for ackermannization. --- src/tactic/ackr/ackermannize_tactic.cpp | 73 ++++ src/tactic/ackr/ackermannize_tactic.h | 28 ++ src/tactic/ackr/ackr.pyg | 8 + src/tactic/ackr/ackr_info.h | 107 +++++ src/tactic/ackr/ackr_model_converter.cpp | 148 +++++++ src/tactic/ackr/ackr_model_converter.h | 23 ++ src/tactic/ackr/lackr.cpp | 298 ++++++++++++++ src/tactic/ackr/lackr.h | 126 ++++++ src/tactic/ackr/lackr_model_constructor.cpp | 379 ++++++++++++++++++ src/tactic/ackr/lackr_model_constructor.h | 57 +++ .../ackr/lackr_model_converter_lazy.cpp | 60 +++ src/tactic/ackr/lackr_model_converter_lazy.h | 23 ++ src/tactic/ackr/qfufbv_ackr_tactic.cpp | 117 ++++++ src/tactic/ackr/qfufbv_ackr_tactic.h | 28 ++ 14 files changed, 1475 insertions(+) create mode 100644 src/tactic/ackr/ackermannize_tactic.cpp create mode 100644 src/tactic/ackr/ackermannize_tactic.h create mode 100644 src/tactic/ackr/ackr.pyg create mode 100644 src/tactic/ackr/ackr_info.h create mode 100644 src/tactic/ackr/ackr_model_converter.cpp create mode 100644 src/tactic/ackr/ackr_model_converter.h create mode 100644 src/tactic/ackr/lackr.cpp create mode 100644 src/tactic/ackr/lackr.h create mode 100644 src/tactic/ackr/lackr_model_constructor.cpp create mode 100644 src/tactic/ackr/lackr_model_constructor.h create mode 100644 src/tactic/ackr/lackr_model_converter_lazy.cpp create mode 100644 src/tactic/ackr/lackr_model_converter_lazy.h create mode 100644 src/tactic/ackr/qfufbv_ackr_tactic.cpp create mode 100644 src/tactic/ackr/qfufbv_ackr_tactic.h diff --git a/src/tactic/ackr/ackermannize_tactic.cpp b/src/tactic/ackr/ackermannize_tactic.cpp new file mode 100644 index 000000000..6d52c6fb3 --- /dev/null +++ b/src/tactic/ackr/ackermannize_tactic.cpp @@ -0,0 +1,73 @@ +/*++ +Copyright (c) 2016 Microsoft Corporation + +Module Name: + +ackermannize_tactic.cpp + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#include"tactical.h" +#include"lackr.h" +#include"ackr_params.hpp" +#include"ackr_model_converter.h" +#include"model_smt2_pp.h" + +class ackermannize_tactic : public tactic { +public: + ackermannize_tactic(ast_manager& m, params_ref const& p) + : m_m(m) + , m_p(p) + {} + + virtual ~ackermannize_tactic() { } + + virtual void operator()(goal_ref const & g, + goal_ref_buffer & result, + model_converter_ref & mc, + proof_converter_ref & pc, + expr_dependency_ref & core) { + mc = 0; + ast_manager& m(g->m()); + expr_ref_vector flas(m); + const unsigned sz = g->size(); + for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); + scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas); + flas.reset(); + // mk result + goal_ref resg(alloc(goal, *g, true)); + imp->mk_ackermann(resg); + result.push_back(resg.get()); + // report model + if (g->models_enabled()) { + model_ref abstr_model = imp->get_model(); + mc = mk_ackr_model_converter(m, imp->get_info(), abstr_model); + } + } + + virtual void collect_statistics(statistics & st) const { + st.update("ackr-constraints", m_st.m_ackrs_sz); + } + + virtual void reset_statistics() { m_st.reset(); } + + virtual void cleanup() { } + + virtual tactic* translate(ast_manager& m) { + return alloc(ackermannize_tactic, m, m_p); + } +private: + ast_manager& m_m; + params_ref m_p; + lackr_stats m_st; +}; + +tactic * mk_ackermannize_tactic(ast_manager & m, params_ref const & p) { + return alloc(ackermannize_tactic, m, p); +} diff --git a/src/tactic/ackr/ackermannize_tactic.h b/src/tactic/ackr/ackermannize_tactic.h new file mode 100644 index 000000000..07c68d5e5 --- /dev/null +++ b/src/tactic/ackr/ackermannize_tactic.h @@ -0,0 +1,28 @@ +/*++ +Copyright (c) 2016 Microsoft Corporation + +Module Name: + +ackermannize_tactic.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ + +#ifndef _ACKERMANNIZE_TACTIC_H_ +#define _ACKERMANNIZE_TACTIC_H +#include"tactical.h" + +tactic * mk_ackermannize_tactic(ast_manager & m, params_ref const & p); + +/* +ADD_TACTIC("ackermannize", "A tactic for performing full Ackermannization.", "mk_ackermannize_tactic(m, p)") +*/ + +#endif + diff --git a/src/tactic/ackr/ackr.pyg b/src/tactic/ackr/ackr.pyg new file mode 100644 index 000000000..d1ec8efe2 --- /dev/null +++ b/src/tactic/ackr/ackr.pyg @@ -0,0 +1,8 @@ +def_module_params('ackr', + description='solving UF via ackermannization (currently for QF_UFBV)', + export=True, + params=( + ('eager', BOOL, True, 'eagerly instantiate all congruence rules'), + ('sat_backend', BOOL, False, 'use SAT rather than SMT'), + )) + diff --git a/src/tactic/ackr/ackr_info.h b/src/tactic/ackr/ackr_info.h new file mode 100644 index 000000000..1b21f0c91 --- /dev/null +++ b/src/tactic/ackr/ackr_info.h @@ -0,0 +1,107 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +ackr_info.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#ifndef ACKR_INFO_H_12278 +#define ACKR_INFO_H_12278 +#include"obj_hashtable.h" +#include"ast.h" +#include"ref.h" +#include"expr_replacer.h" + +/** \brief + Information about how a formula is being converted into + a formula without uninterpreted function symbols via ackermannization. + + The intended use is that new terms are added via set_abstr. + Once all terms are abstracted, call seal. + The function abstract may only be called when sealed. + + The class enables reference counting. +**/ +class ackr_info { + public: + ackr_info(ast_manager& m) + : m_m(m) + , m_consts(m) + , m_er(mk_default_expr_replacer(m)) + , m_subst(m_m) + , m_ref_count(0) + , m_sealed(false) + {} + + virtual ~ackr_info() { + m_consts.reset(); + } + + inline void set_abstr(app* term, app* c) { + SASSERT(!m_sealed); + SASSERT(c); + m_t2c.insert(term,c); + m_c2t.insert(c->get_decl(),term); + m_subst.insert(term, c); + m_consts.push_back(c); + } + + inline void abstract(expr * e, expr_ref& res) { + SASSERT(m_sealed); + (*m_er)(e, res); + } + + inline app* find_term(func_decl* c) const { + app * rv = 0; + m_c2t.find(c,rv); + return rv; + } + + inline app* get_abstr(app* term) const { + app * const rv = m_t2c.find(term); + SASSERT(rv); + return rv; + } + + inline void seal() { + m_sealed=true; + m_er->set_substitution(&m_subst); + } + + // + // Reference counting + // + void inc_ref() { ++m_ref_count; } + void dec_ref() { + --m_ref_count; + if (m_ref_count == 0) { + dealloc(this); + } + } + private: + typedef obj_map t2ct; + typedef obj_map c2tt; + ast_manager& m_m; + + t2ct m_t2c; // terms to constants + c2tt m_c2t; // constants to terms (inversion of m_t2c) + expr_ref_vector m_consts; // the constants introduced during abstraction + + // replacer and substitution used to compute abstractions + scoped_ptr m_er; + expr_substitution m_subst; + + unsigned m_ref_count; // reference counting + bool m_sealed; // debugging +}; + +typedef ref ackr_info_ref; +#endif /* ACKR_INFO_H_12278 */ diff --git a/src/tactic/ackr/ackr_model_converter.cpp b/src/tactic/ackr/ackr_model_converter.cpp new file mode 100644 index 000000000..c21a6240c --- /dev/null +++ b/src/tactic/ackr/ackr_model_converter.cpp @@ -0,0 +1,148 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +ackr_model_converter.cpp + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#include"ackr_model_converter.h" +#include"model_evaluator.h" +#include"ast_smt2_pp.h" +#include"ackr_info.h" + + +class ackr_model_converter : public model_converter { +public: + ackr_model_converter(ast_manager & m, + const ackr_info_ref& info, + model_ref& abstr_model) + : m(m) + , info(info) + , abstr_model(abstr_model) + , fixed_model(true) + { } + + ackr_model_converter(ast_manager & m, + const ackr_info_ref& info) + : m(m) + , info(info) + , fixed_model(false) + { } + + virtual ~ackr_model_converter() { } + + virtual void operator()(model_ref & md, unsigned goal_idx) { + SASSERT(goal_idx == 0); + SASSERT(!fixed_model || md.get() == 0 || (!md->get_num_constants() && !md->get_num_functions())); + model_ref& old_model = fixed_model ? abstr_model : md; + SASSERT(old_model.get()); + model * new_model = alloc(model, m); + convert(old_model.get(), new_model); + md = new_model; + } + + virtual void operator()(model_ref & md) { operator()(md, 0); } + + //void display(std::ostream & out); + + virtual model_converter * translate(ast_translation & translator) {NOT_IMPLEMENTED_YET();} +protected: + ast_manager& m; + const ackr_info_ref info; + model_ref abstr_model; + bool fixed_model; + void convert(model * source, model * destination); + void add_entry(model_evaluator & evaluator, + app* term, expr* value, + obj_map& interpretations); + void convert_sorts(model * source, model * destination); + void convert_constants(model * source, model * destination); +}; + +void ackr_model_converter::convert(model * source, model * destination) { + SASSERT(source->get_num_functions() == 0); + convert_constants(source,destination); + convert_sorts(source,destination); +} + +void ackr_model_converter::convert_constants(model * source, model * destination) { + TRACE("ackr_model", tout << "converting constants\n";); + obj_map interpretations; + model_evaluator evaluator(*source); + for (unsigned i = 0; i < source->get_num_constants(); i++) { + func_decl * const c = source->get_constant(i); + app * const term = info->find_term(c); + expr * value = source->get_const_interp(c); + if(!term) { + destination->register_decl(c, value); + } else { + add_entry(evaluator, term, value, interpretations); + } + } + + obj_map::iterator e = interpretations.end(); + for (obj_map::iterator i = interpretations.begin(); + i!=e; ++i) { + func_decl* const fd = i->m_key; + func_interp* const fi = i->get_value(); + fi->set_else(m.get_some_value(fd->get_range())); + destination->register_decl(fd, fi); + } +} + +void ackr_model_converter::add_entry(model_evaluator & evaluator, + app* term, expr* value, + obj_map& interpretations) { + TRACE("ackr_model", tout << "add_entry" + << mk_ismt2_pp(term, m, 2) + << "->" + << mk_ismt2_pp(value, m, 2) << "\n"; + ); + + func_interp* fi = 0; + func_decl * const declaration = term->get_decl(); + const unsigned sz = declaration->get_arity(); + SASSERT(sz == term->get_num_args()); + if (!interpretations.find(declaration, fi)) { + fi = alloc(func_interp,m,sz); + interpretations.insert(declaration, fi); + } + expr_ref_vector args(m); + for (unsigned gi = 0; gi < sz; ++gi) { + expr * const arg = term->get_arg(gi); + expr_ref aarg(m); + info->abstract(arg, aarg); + expr_ref arg_value(m); + evaluator(aarg,arg_value); + args.push_back(arg_value); + } + if (fi->get_entry(args.c_ptr()) == 0) { + fi->insert_new_entry(args.c_ptr(), value); + } else { + TRACE("ackr_model", tout << "entry already present\n";); + } +} + +void ackr_model_converter::convert_sorts(model * source, model * destination) { + for (unsigned i = 0; i < source->get_num_uninterpreted_sorts(); i++) { + sort * const s = source->get_uninterpreted_sort(i); + ptr_vector u = source->get_universe(s); + destination->register_usort(s, u.size(), u.c_ptr()); + } +} + +model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info) { + return alloc(ackr_model_converter, m, info); +} + +model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model) { + return alloc(ackr_model_converter, m, info, abstr_model); +} diff --git a/src/tactic/ackr/ackr_model_converter.h b/src/tactic/ackr/ackr_model_converter.h new file mode 100644 index 000000000..611203242 --- /dev/null +++ b/src/tactic/ackr/ackr_model_converter.h @@ -0,0 +1,23 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +ackr_model_converter.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#ifndef ACKR_MODEL_CONVERTER_H_5814 +#define ACKR_MODEL_CONVERTER_H_5814 +#include"model_converter.h" +#include"ackr_info.h" + +model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); +model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info); +#endif /* LACKR_MODEL_CONVERTER_H_5814 */ diff --git a/src/tactic/ackr/lackr.cpp b/src/tactic/ackr/lackr.cpp new file mode 100644 index 000000000..fc36d4e1d --- /dev/null +++ b/src/tactic/ackr/lackr.cpp @@ -0,0 +1,298 @@ +/*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ +/////////////// +#include"lackr.h" +#include"ackr_params.hpp" +#include"tactic.h" +#include"lackr_model_constructor.h" +#include"ackr_info.h" +#include"for_each_expr.h" +/////////////// +#include"inc_sat_solver.h" +#include"qfaufbv_tactic.h" +#include"qfbv_tactic.h" +#include"tactic2solver.h" +/////////////// +#include"model_smt2_pp.h" +/////////////// +lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas) + : m_m(m) + , m_p(p) + , m_formulas(formulas) + , m_abstr(m) + , m_sat(NULL) + , m_bvutil(m) + , m_simp(m) + , m_ackrs(m) + , m_st(st) + , m_is_init(false) +{ + updt_params(p); +} + +lackr::~lackr() { + const fun2terms_map::iterator e = m_fun2terms.end(); + for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { + dealloc(i->get_value()); + } +} + +lbool lackr::operator() () { + setup_sat(); + init(); + const lbool rv = m_eager ? eager() : lazy(); + if (rv == l_true) m_sat->get_model(m_model); + CTRACE("lackr", rv == l_true, + model_smt2_pp(tout << "abstr_model(\n", m_m, *(m_model.get()), 2); tout << ")\n"; ); + return rv; +} + +void lackr::mk_ackermann(/*out*/goal_ref& g) { + init(); + eager_enc(); + for (unsigned i = 0; i < m_abstr.size(); ++i) g->assert_expr(m_abstr.get(i)); + for (unsigned i = 0; i < m_ackrs.size(); ++i) g->assert_expr(m_ackrs.get(i)); +} + +void lackr::init() { + SASSERT(!m_is_init); + m_is_init = true; + params_ref simp_p(m_p); + m_simp.updt_params(simp_p); + m_info = alloc(ackr_info, m_m); + collect_terms(); + abstract(); +} + +// +// Introduce ackermann lemma for the two given terms. +// +bool lackr::ackr(app * const t1, app * const t2) { + TRACE("lackr", tout << "ackr " + << mk_ismt2_pp(t1, m_m, 2) << " , " << mk_ismt2_pp(t2, m_m, 2) << "\n";); + const unsigned sz = t1->get_num_args(); + SASSERT(t2->get_num_args() == sz); + expr_ref_vector eqs(m_m); + for (unsigned i = 0; i < sz; ++i) { + expr * const arg1 = t1->get_arg(i); + expr * const arg2 = t2->get_arg(i); + if (arg1 == arg2) continue; // quickly skip syntactically equal + if (m_bvutil.is_numeral(arg1) && m_bvutil.is_numeral(arg2)) { + // quickly abort if there are two distinct numerals + SASSERT(arg1 != arg2); + TRACE("lackr", tout << "never eq\n";); + return false; + } + eqs.push_back(m_m.mk_eq(arg1, arg2)); + } + app * const a1 = m_info->get_abstr(t1); + app * const a2 = m_info->get_abstr(t2); + SASSERT(a1 && a2); + TRACE("lackr", tout << "abstr1 " << mk_ismt2_pp(a1, m_m, 2) << "\n";); + TRACE("lackr", tout << "abstr2 " << mk_ismt2_pp(a2, m_m, 2) << "\n";); + expr_ref lhs(m_m.mk_and(eqs.size(), eqs.c_ptr()), m_m); + TRACE("lackr", tout << "ackr constr lhs" << mk_ismt2_pp(lhs, m_m, 2) << "\n";); + expr_ref rhs(m_m.mk_eq(a1, a2),m_m); + TRACE("lackr", tout << "ackr constr rhs" << mk_ismt2_pp(rhs, m_m, 2) << "\n";); + expr_ref cg(m_m.mk_implies(lhs, rhs), m_m); + TRACE("lackr", tout << "ackr constr" << mk_ismt2_pp(cg, m_m, 2) << "\n";); + expr_ref cga(m_m); + m_info->abstract(cg, cga); // constraint needs abstraction due to nested applications + m_simp(cga); + TRACE("lackr", tout << "ackr constr abs:" << mk_ismt2_pp(cga, m_m, 2) << "\n";); + if (m_m.is_true(cga)) return false; + m_st.m_ackrs_sz++; + m_ackrs.push_back(cga); + return true; +} + +// +// Introduce the ackermann lemma for each pair of terms. +// +void lackr::eager_enc() { + const fun2terms_map::iterator e = m_fun2terms.end(); + for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { + checkpoint(); + func_decl* const fd = i->m_key; + app_set * const ts = i->get_value(); + const app_set::iterator r = ts->end(); + for (app_set::iterator j = ts->begin(); j != r; ++j) { + app_set::iterator k = j; + ++k; + for (; k != r; ++k) { + app * const t1 = *j; + app * const t2 = *k; + SASSERT(t1->get_decl() == fd); + SASSERT(t2->get_decl() == fd); + if (t1 == t2) continue; + ackr(t1,t2); + } + } + } +} + + +void lackr::abstract() { + const fun2terms_map::iterator e = m_fun2terms.end(); + for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { + func_decl* const fd = i->m_key; + app_set * const ts = i->get_value(); + sort* const s = fd->get_range(); + const app_set::iterator r = ts->end(); + for (app_set::iterator j = ts->begin(); j != r; ++j) { + app * const fc = m_m.mk_fresh_const(fd->get_name().str().c_str(), s); + app * const t = *j; + SASSERT(t->get_decl() == fd); + m_info->set_abstr(t, fc); + TRACE("lackr", tout << "abstr term " + << mk_ismt2_pp(t, m_m, 2) + << " -> " + << mk_ismt2_pp(fc, m_m, 2) + << "\n";); + } + } + m_info->seal(); + // perform abstraction of the formulas + const unsigned sz = m_formulas.size(); + for (unsigned i = 0; i < sz; ++i) { + expr_ref a(m_m); + m_info->abstract(m_formulas.get(i), a); + m_abstr.push_back(a); + } +} + +void lackr::add_term(app* a) { + if (a->get_num_args() == 0) return; + func_decl* const fd = a->get_decl(); + if (!is_uninterp(a)) return; + SASSERT(m_bvutil.is_bv_sort(fd->get_range()) || m_m.is_bool(a)); + app_set* ts = 0; + if (!m_fun2terms.find(fd, ts)) { + ts = alloc(app_set); + m_fun2terms.insert(fd, ts); + } + TRACE("lackr", tout << "term(" << mk_ismt2_pp(a, m_m, 2) << ")\n";); + ts->insert(a); +} + +void lackr::push_abstraction() { + const unsigned sz = m_abstr.size(); + for (unsigned i = 0; i < sz; ++i) { + m_sat->assert_expr(m_abstr.get(i)); + } +} + +lbool lackr::eager() { + push_abstraction(); + TRACE("lackr", tout << "run sat 0\n"; ); + const lbool rv0 = m_sat->check_sat(0, 0); + if (rv0 == l_false) return l_false; + eager_enc(); + expr_ref all(m_m); + all = m_m.mk_and(m_ackrs.size(), m_ackrs.c_ptr()); + m_simp(all); + m_sat->assert_expr(all); + TRACE("lackr", tout << "run sat all\n"; ); + return m_sat->check_sat(0, 0); +} + +lbool lackr::lazy() { + lackr_model_constructor mc(m_m, m_info); + push_abstraction(); + unsigned ackr_head = 0; + while (1) { + m_st.m_it++; + checkpoint(); + TRACE("lackr", tout << "lazy check: " << m_st.m_it << "\n";); + const lbool r = m_sat->check_sat(0, 0); + if (r == l_undef) return l_undef; // give up + if (r == l_false) return l_false; // abstraction unsat + // reconstruct model + model_ref am; + m_sat->get_model(am); + const bool mc_res = mc.check(am); + if (mc_res) return l_true; // model okay + // refine abstraction + const lackr_model_constructor::conflict_list conflicts = mc.get_conflicts(); + for (lackr_model_constructor::conflict_list::const_iterator i = conflicts.begin(); + i != conflicts.end(); ++i) { + ackr(i->first, i->second); + } + while (ackr_head < m_ackrs.size()) { + m_sat->assert_expr(m_ackrs.get(ackr_head++)); + } + } +} + +void lackr::setup_sat() { + SASSERT(m_sat == NULL); + if (m_use_sat) { + tactic_ref t = mk_qfbv_tactic(m_m, m_p); + m_sat = mk_tactic2solver(m_m, t.get(), m_p); + } + else { + tactic_ref t = mk_qfaufbv_tactic(m_m, m_p); + m_sat = mk_tactic2solver(m_m, t.get(), m_p); + } + SASSERT(m_sat != NULL); + m_sat->set_produce_models(true); +} + + +// +// Collect all uninterpreted terms, skipping 0-arity. +// +void lackr::collect_terms() { + ptr_vector stack; + expr * curr; + expr_mark visited; + for(unsigned i = 0; i < m_formulas.size(); ++i) { + stack.push_back(m_formulas.get(i)); + } + + while (!stack.empty()) { + curr = stack.back(); + if (visited.is_marked(curr)) { + stack.pop_back(); + continue; + } + + switch (curr->get_kind()) { + case AST_VAR: + visited.mark(curr, true); + stack.pop_back(); + break; + case AST_APP: + { + app * const a = to_app(curr); + if (for_each_expr_args(stack, visited, a->get_num_args(), a->get_args())) { + visited.mark(curr, true); + stack.pop_back(); + add_term(a); + } + } + break; + case AST_QUANTIFIER: + UNREACHABLE(); + break; + default: + UNREACHABLE(); + return; + } + } + visited.reset(); +} diff --git a/src/tactic/ackr/lackr.h b/src/tactic/ackr/lackr.h new file mode 100644 index 000000000..b108653ac --- /dev/null +++ b/src/tactic/ackr/lackr.h @@ -0,0 +1,126 @@ + /*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr.h + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef LACKR_H_15079 +#define LACKR_H_15079 +/////////////// +#include"ackr_info.h" +#include"ackr_params.hpp" +#include"th_rewriter.h" +#include"cooperate.h" +#include"bv_decl_plugin.h" +#include"lbool.h" +#include"model.h" +#include"solver.h" +#include"util.h" +#include"tactic_exception.h" +#include"goal.h" + +struct lackr_stats { + lackr_stats() : m_it(0), m_ackrs_sz(0) {} + void reset() { m_it = m_ackrs_sz = 0; } + unsigned m_it; // number of lazy iterations + unsigned m_ackrs_sz; // number of congruence constraints +}; + +/** \brief + A class to encode or directly solve problems with uninterpreted functions via ackermannization. + Currently, solving is supported only for QF_UFBV. +**/ +class lackr { + public: + lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas); + ~lackr(); + void updt_params(params_ref const & _p) { + ackr_params p(_p); + m_eager = p.eager(); + m_use_sat = p.sat_backend(); + } + + /** \brief + * Solve the formula that the class was initialized with. + **/ + lbool operator() (); + + + /** \brief + * Converts function occurrences to constants and encodes all congruence ackermann lemmas. + * This guarantees a equisatisfiability with the input formula. It has a worst-case quadratic blowup. + **/ + void mk_ackermann(/*out*/goal_ref& g); + + + // + // getters + // + inline ackr_info_ref get_info() { return m_info; } + inline model_ref get_model() { return m_model; } + + // + // timeout mechanism + // + void checkpoint() { + if (m_m.canceled()) { + throw tactic_exception(TACTIC_CANCELED_MSG); + } + cooperate("lackr"); + } + private: + typedef obj_hashtable app_set; + typedef obj_map fun2terms_map; + ast_manager& m_m; + params_ref m_p; + expr_ref_vector m_formulas; + expr_ref_vector m_abstr; + fun2terms_map m_fun2terms; + ackr_info_ref m_info; + scoped_ptr m_sat; + bv_util m_bvutil; + th_rewriter m_simp; + expr_ref_vector m_ackrs; + model_ref m_model; + bool m_eager; + bool m_use_sat; + lackr_stats& m_st; + bool m_is_init; + + void init(); + void setup_sat(); + lbool eager(); + lbool lazy(); + + // + // Introduce congruence ackermann lemma for the two given terms. + // + bool ackr(app * const t1, app * const t2); + + // + // Introduce the ackermann lemma for each pair of terms. + // + void eager_enc(); + + void abstract(); + + void push_abstraction(); + + void add_term(app* a); + + // + // Collect all uninterpreted terms, skipping 0-arity. + // + void collect_terms(); +}; +#endif /* LACKR_H_15079 */ diff --git a/src/tactic/ackr/lackr_model_constructor.cpp b/src/tactic/ackr/lackr_model_constructor.cpp new file mode 100644 index 000000000..ad64fc679 --- /dev/null +++ b/src/tactic/ackr/lackr_model_constructor.cpp @@ -0,0 +1,379 @@ +/*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + model_constructor.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ +#include"lackr_model_constructor.h" +#include"model_evaluator.h" +#include"ast_smt2_pp.h" +#include"ackr_info.h" +#include"for_each_expr.h" +#include"bv_rewriter.h" +#include"bool_rewriter.h" +struct lackr_model_constructor::imp { + public: + imp(ast_manager& m, + ackr_info_ref info, + model_ref& abstr_model, + conflict_list& conflicts) + : m_m(m) + , m_info(info) + , m_abstr_model(abstr_model) + , m_conflicts(conflicts) + , m_b_rw(m) + , m_bv_rw(m) + , m_empty_model(m) + {} + + ~imp() { + { + values2val_t::iterator i = m_values2val.begin(); + const values2val_t::iterator e = m_values2val.end(); + for (; i != e; ++i) { + m_m.dec_ref(i->m_key); + m_m.dec_ref(i->m_value.value); + m_m.dec_ref(i->m_value.source_term); + } + } + { + app2val_t::iterator i = m_app2val.begin(); + const app2val_t::iterator e = m_app2val.end(); + for (; i != e; ++i) { + m_m.dec_ref(i->m_value); + m_m.dec_ref(i->m_key); + } + } + } + + // + // Returns true iff model was successfully constructed. + // + bool check() { + for (unsigned i = 0; i < m_abstr_model->get_num_constants(); i++) { + func_decl * const c = m_abstr_model->get_constant(i); + app * const term = m_info->find_term(c); + if (term) m_stack.push_back(term); + else m_stack.push_back(m_m.mk_const(c)); + } + return run(); + } + + + void make_model(model_ref& destination) { + { + for (unsigned i = 0; i < m_abstr_model->get_num_uninterpreted_sorts(); i++) { + sort * const s = m_abstr_model->get_uninterpreted_sort(i); + ptr_vector u = m_abstr_model->get_universe(s); + destination->register_usort(s, u.size(), u.c_ptr()); + } + } + { + const app2val_t::iterator e = m_app2val.end(); + app2val_t::iterator i = m_app2val.end(); + for (; i != e; ++i) { + app * a = i->m_key; + if (a->get_num_args()) continue; + destination->register_decl(a->get_decl(), i->m_value); + } + } + + obj_map interpretations; + { + const values2val_t::iterator e = m_values2val.end(); + values2val_t::iterator i = m_values2val.end(); + for (; i != e; ++i) add_entry(i->m_key, i->m_value.value, interpretations); + } + + { + obj_map::iterator ie = interpretations.end(); + obj_map::iterator ii = interpretations.begin(); + for (; ii != ie; ++ii) { + func_decl* const fd = ii->m_key; + func_interp* const fi = ii->get_value(); + fi->set_else(m_m.get_some_value(fd->get_range())); + destination->register_decl(fd, fi); + } + } + } + + void add_entry(app* term, expr* value, + obj_map& interpretations) { + func_interp* fi = 0; + func_decl * const declaration = term->get_decl(); + const unsigned sz = declaration->get_arity(); + SASSERT(sz == term->get_num_args()); + if (!interpretations.find(declaration, fi)) { + fi = alloc(func_interp, m_m, sz); + interpretations.insert(declaration, fi); + } + fi->insert_new_entry(term->get_args(), value); + } + private: + ast_manager& m_m; + ackr_info_ref m_info; + model_ref& m_abstr_model; + conflict_list& m_conflicts; + bool_rewriter m_b_rw; + bv_rewriter m_bv_rw; + scoped_ptr m_evaluator; + model m_empty_model; + private: + struct val_info { expr * value; app * source_term; }; + typedef obj_map app2val_t; + typedef obj_map values2val_t; + values2val_t m_values2val; + app2val_t m_app2val; + ptr_vector m_stack; + + static inline val_info mk_val_info(expr* value, app* source_term) { + val_info rv; + rv.value = value; + rv.source_term = source_term; + return rv; + } + + // + // Performs congruence check on terms on the stack. + // (Currently stops upon the first failure). + // Returns true if and only if congruence check succeeded. + bool run() { + m_evaluator = alloc(model_evaluator, m_empty_model); + expr_mark visited; + expr * curr; + while (!m_stack.empty()) { + curr = m_stack.back(); + if (visited.is_marked(curr)) { + m_stack.pop_back(); + continue; + } + + switch (curr->get_kind()) { + case AST_VAR: + UNREACHABLE(); + return false; + case AST_APP: { + app * a = to_app(curr); + if (for_each_expr_args(m_stack, visited, a->get_num_args(), a->get_args())) { + visited.mark(a, true); + m_stack.pop_back(); + if (!mk_value(a)) return false; + } + } + break; + case AST_QUANTIFIER: + UNREACHABLE(); + return false; + default: + UNREACHABLE(); + return false; + } + } + return true; + } + + inline bool is_val(expr * e) { + if (!is_app(e)) return false; + return is_val(to_app(e)); + } + + inline bool is_val(app * a) { + const family_id fid = a->get_decl()->get_family_id(); + const bool rv = fid != null_family_id && a->get_num_args() == 0; + SASSERT(rv == (m_bv_rw.is_numeral(a) || m_m.is_true(a) || m_m.is_false(a))); + return rv; + } + + inline bool eval_cached(app * a, expr *& val) { + if (is_val(a)) { val = a; return true; } + return m_app2val.find(a, val); + } + + bool evaluate(app * const a, expr_ref& result) { + SASSERT(!is_val(a)); + const unsigned num = a->get_num_args(); + if (num == 0) { // handle constants + make_value_constant(a, result); + return true; + } + // evaluate arguments + expr_ref_vector values(m_m); + values.reserve(num); + expr * const * args = a->get_args(); + for (unsigned i = 0; i < num; ++i) { + expr * val; + const bool b = eval_cached(to_app(args[i]), val); // TODO: OK conversion to_app? + CTRACE("model_constructor", !b, tout << "fail arg val(\n" << mk_ismt2_pp(args[i], m_m, 2); ); + TRACE("model_constructor", tout << + "arg val " << i << "(\n" << mk_ismt2_pp(args[i], m_m, 2) + << " : " << mk_ismt2_pp(val, m_m, 2) << '\n'; ); + SASSERT(b); + values[i] = val; + } + // handle functions + if (a->get_family_id() == null_family_id) { // handle uninterpreted + app_ref key(m_m.mk_app(a->get_decl(), values.c_ptr()), m_m); + if (!make_value_uninterpreted_function(a, values, key.get(), result)) { + return false; + } + } + else { // handle interpreted + make_value_interpreted_function(a, values, result); + } + return true; + } + + // + // Check and record the value for a given term, given that all arguments are already checked. + // + bool mk_value(app * a) { + if (is_val(a)) return true; // skip numerals + TRACE("model_constructor", tout << "mk_value(\n" << mk_ismt2_pp(a, m_m, 2) << ")\n";); + SASSERT(!m_app2val.contains(a)); + const unsigned num = a->get_num_args(); + expr_ref result(m_m); + if (!evaluate(a, result)) return false; + SASSERT(is_val(result)); + TRACE("model_constructor", + tout << "map term(\n" << mk_ismt2_pp(a, m_m, 2) << "\n->" + << mk_ismt2_pp(result.get(), m_m, 2)<< ")\n"; ); + CTRACE("model_constructor", + !is_val(result.get()), + tout << "eval fail\n" << mk_ismt2_pp(a, m_m, 2) << mk_ismt2_pp(result, m_m, 2) << "\n"; + ); + SASSERT(is_val(result.get())); + m_app2val.insert(a, result.get()); // memoize + m_m.inc_ref(a); + m_m.inc_ref(result.get()); + return true; + } + + // Constants from the abstract model are directly mapped to the concrete one. + void make_value_constant(app * const a, expr_ref& result) { + SASSERT(a->get_num_args() == 0); + func_decl * const fd = a->get_decl(); + expr * val = m_abstr_model->get_const_interp(fd); + if (val == 0) { // TODO: avoid model completetion? + sort * s = fd->get_range(); + val = m_abstr_model->get_some_value(s); + } + result = val; + } + + bool make_value_uninterpreted_function(app* a, + expr_ref_vector& values, + app* key, + expr_ref& result) { + // get ackermann constant + app * const ac = m_info->get_abstr(a); + func_decl * const a_fd = a->get_decl(); + SASSERT(ac->get_num_args() == 0); + SASSERT(a_fd->get_range() == ac->get_decl()->get_range()); + expr_ref value(m_m); + value = m_abstr_model->get_const_interp(ac->get_decl()); + // get ackermann constant's interpretation + if (value.get() == 0) { // TODO: avoid model completion? + sort * s = a_fd->get_range(); + value = m_abstr_model->get_some_value(s); + } + // check congruence + val_info vi; + if(m_values2val.find(key,vi)) { // already is mapped to a value + SASSERT(vi.source_term); + const bool ok = vi.value == value; + if (!ok) { + TRACE("model_constructor", + tout << "already mapped by(\n" << mk_ismt2_pp(vi.source_term, m_m, 2) << "\n->" + << mk_ismt2_pp(vi.value, m_m, 2) << ")\n"; ); + m_conflicts.push_back(std::make_pair(a, vi.source_term)); + } + result = vi.value; + return ok; + } else { // new value + result = value; + vi.value = value; + vi.source_term = a; + m_values2val.insert(key,vi); + m_m.inc_ref(vi.source_term); + m_m.inc_ref(vi.value); + m_m.inc_ref(key); + return true; + } + UNREACHABLE(); + } + + void make_value_interpreted_function(app* a, + expr_ref_vector& values, + expr_ref& result) { + const unsigned num = values.size(); + func_decl * const fd = a->get_decl(); + const family_id fid = fd->get_family_id(); + expr_ref term(m_m); + term = m_m.mk_app(a->get_decl(), num, values.c_ptr()); + m_evaluator->operator() (term, result); + TRACE("model_constructor", + tout << "eval(\n" << mk_ismt2_pp(term.get(), m_m, 2) << "\n->" + << mk_ismt2_pp(result.get(), m_m, 2) << ")\n"; ); + return; + if (fid == m_b_rw.get_fid()) { + decl_kind k = fd->get_decl_kind(); + if (k == OP_EQ) { + // theory dispatch for = + SASSERT(num == 2); + family_id s_fid = m_m.get_sort(values.get(0))->get_family_id(); + br_status st = BR_FAILED; + if (s_fid == m_bv_rw.get_fid()) + st = m_bv_rw.mk_eq_core(values.get(0), values.get(1), result); + } else { + br_status st = m_b_rw.mk_app_core(fd, num, values.c_ptr(), result); + } + } else { + br_status st = BR_FAILED; + if (fid == m_bv_rw.get_fid()) { + st = m_bv_rw.mk_app_core(fd, num, values.c_ptr(), result); + } + else { + UNREACHABLE(); + } + } + } +}; + +lackr_model_constructor::lackr_model_constructor(ast_manager& m, ackr_info_ref info) + : m_imp(0) + , m_m(m) + , m_state(UNKNOWN) + , m_info(info) + , m_ref_count(0) +{} + +lackr_model_constructor::~lackr_model_constructor() { + if (m_imp) dealloc(m_imp); +} + +bool lackr_model_constructor::check(model_ref& abstr_model) { + m_conflicts.reset(); + if (m_imp) { + dealloc(m_imp); + m_imp = 0; + } + m_imp = alloc(lackr_model_constructor::imp, m_m, m_info, abstr_model, m_conflicts); + const bool rv = m_imp->check(); + m_state = rv ? CHECKED : CONFLICT; + return rv; +} + +void lackr_model_constructor::make_model(model_ref& model) { + SASSERT(m_state == CHECKED); + m_imp->make_model(model); +} diff --git a/src/tactic/ackr/lackr_model_constructor.h b/src/tactic/ackr/lackr_model_constructor.h new file mode 100644 index 000000000..478efe97f --- /dev/null +++ b/src/tactic/ackr/lackr_model_constructor.h @@ -0,0 +1,57 @@ + /*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + model_constructor.h + + Abstract: + Given a propositional abstraction, attempt to construct a model. + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef LACKR_MODEL_CONSTRUCTOR_H_626 +#define LACKR_MODEL_CONSTRUCTOR_H_626 +#include"ast.h" +#include"ackr_info.h" +#include"model.h" +class lackr_model_constructor { + public: + typedef std::pair app_pair; + typedef vector conflict_list; + lackr_model_constructor(ast_manager& m, ackr_info_ref info); + ~lackr_model_constructor(); + bool check(model_ref& abstr_model); + const conflict_list& get_conflicts() { + SASSERT(m_state == CONFLICT); + return m_conflicts; + } + void make_model(model_ref& model); + + // + // Reference counting + // + void inc_ref() { ++m_ref_count; } + void dec_ref() { + --m_ref_count; + if (m_ref_count == 0) { + dealloc(this); + } + } + private: + struct imp; + imp * m_imp; + enum {CHECKED, CONFLICT, UNKNOWN} m_state; + conflict_list m_conflicts; + ast_manager& m_m; + const ackr_info_ref m_info; + unsigned m_ref_count; // reference counting +}; + +typedef ref lackr_model_constructor_ref; +#endif /* MODEL_CONSTRUCTOR_H_626 */ diff --git a/src/tactic/ackr/lackr_model_converter_lazy.cpp b/src/tactic/ackr/lackr_model_converter_lazy.cpp new file mode 100644 index 000000000..30dba8a04 --- /dev/null +++ b/src/tactic/ackr/lackr_model_converter_lazy.cpp @@ -0,0 +1,60 @@ +/*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr_model_converter_lazy.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ +#include"lackr_model_converter_lazy.h" +#include"model_evaluator.h" +#include"ast_smt2_pp.h" +#include"ackr_info.h" +#include"lackr_model_constructor.h" + + +class lackr_model_converter_lazy : public model_converter { +public: + lackr_model_converter_lazy(ast_manager & m, + const lackr_model_constructor_ref& lmc) + : m(m) + , model_constructor(lmc) + { } + + virtual ~lackr_model_converter_lazy() { } + + virtual void operator()(model_ref & md, unsigned goal_idx) { + SASSERT(goal_idx == 0); + SASSERT(md.get() == 0 || (!md->get_num_constants() && !md->get_num_functions())); + SASSERT(model_constructor.get()); + model * new_model = alloc(model, m); + md = new_model; + model_constructor->make_model(md); + } + + virtual void operator()(model_ref & md) { + operator()(md, 0); + } + + //void display(std::ostream & out); + + virtual model_converter * translate(ast_translation & translator) { + NOT_IMPLEMENTED_YET(); + } +protected: + ast_manager& m; + const lackr_model_constructor_ref model_constructor; +}; + +model_converter * mk_lackr_model_converter_lazy(ast_manager & m, + const lackr_model_constructor_ref& model_constructor) { + return alloc(lackr_model_converter_lazy, m, model_constructor); +} diff --git a/src/tactic/ackr/lackr_model_converter_lazy.h b/src/tactic/ackr/lackr_model_converter_lazy.h new file mode 100644 index 000000000..354e1c40b --- /dev/null +++ b/src/tactic/ackr/lackr_model_converter_lazy.h @@ -0,0 +1,23 @@ + /*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr_model_converter_lazy.h + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef LACKR_MODEL_CONVERTER_LAZY_H_14201 +#define LACKR_MODEL_CONVERTER_LAZY_H_14201 +#include"model_converter.h" +#include"ackr_info.h" + +model_converter * mk_lackr_model_converter_lazy(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); +#endif /* LACKR_MODEL_CONVERTER_LAZY_H_14201 */ diff --git a/src/tactic/ackr/qfufbv_ackr_tactic.cpp b/src/tactic/ackr/qfufbv_ackr_tactic.cpp new file mode 100644 index 000000000..9974cc07b --- /dev/null +++ b/src/tactic/ackr/qfufbv_ackr_tactic.cpp @@ -0,0 +1,117 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +qfufbv_ackr_tactic.cpp + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#include"tactical.h" +/////////////// +#include"solve_eqs_tactic.h" +#include"simplify_tactic.h" +#include"propagate_values_tactic.h" +#include"bit_blaster_tactic.h" +#include"elim_uncnstr_tactic.h" +#include"max_bv_sharing_tactic.h" +#include"bv_size_reduction_tactic.h" +#include"ctx_simplify_tactic.h" +#include"nnf_tactic.h" +/////////////// +#include"model_smt2_pp.h" +#include"cooperate.h" +#include"lackr.h" +#include"ackr_params.hpp" +#include"ackr_model_converter.h" + +class qfufbv_ackr_tactic : public tactic { +public: + qfufbv_ackr_tactic(ast_manager& m, params_ref const& p) + : m_m(m) + , m_p(p) + {} + + virtual ~qfufbv_ackr_tactic() { } + + virtual void operator()(goal_ref const & g, + goal_ref_buffer & result, + model_converter_ref & mc, + proof_converter_ref & pc, + expr_dependency_ref & core) { + mc = 0; + ast_manager& m(g->m()); + TRACE("lackr", g->display(tout << "goal:\n");); + // running implementation + expr_ref_vector flas(m); + const unsigned sz = g->size(); + for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); + scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas); + const lbool o = imp->operator()(); + flas.reset(); + // report result + goal_ref resg(alloc(goal, *g, true)); + if (o == l_false) resg->assert_expr(m.mk_false()); + if (o != l_undef) result.push_back(resg.get()); + // report model + if (g->models_enabled() && (o == l_true)) { + model_ref abstr_model = imp->get_model(); + mc = mk_ackr_model_converter(m, imp->get_info(), abstr_model); + } + } + + virtual void collect_statistics(statistics & st) const { + ackr_params p(m_p); + if (!p.eager()) st.update("lackr-its", m_st.m_it); + st.update("ackr-constraints", m_st.m_ackrs_sz); + } + + virtual void reset_statistics() { m_st.reset(); } + + virtual void cleanup() { } + + virtual tactic* translate(ast_manager& m) { + return alloc(qfufbv_ackr_tactic, m, m_p); + } +private: + ast_manager& m_m; + params_ref m_p; + lackr_stats m_st; +}; + +tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p) { + params_ref simp2_p = p; + simp2_p.set_bool("som", true); + simp2_p.set_bool("pull_cheap_ite", true); + simp2_p.set_bool("push_ite_bv", false); + simp2_p.set_bool("local_ctx", true); + simp2_p.set_uint("local_ctx_limit", 10000000); + + simp2_p.set_bool("ite_extra_rules", true); + simp2_p.set_bool("mul2concat", true); + + params_ref ctx_simp_p; + ctx_simp_p.set_uint("max_depth", 32); + ctx_simp_p.set_uint("max_steps", 5000000); + + tactic * const preamble_t = and_then( + mk_simplify_tactic(m), + mk_propagate_values_tactic(m), + //using_params(mk_ctx_simplify_tactic(m_m), ctx_simp_p), + mk_solve_eqs_tactic(m), + mk_elim_uncnstr_tactic(m), + if_no_proofs(if_no_unsat_cores(mk_bv_size_reduction_tactic(m))), + mk_max_bv_sharing_tactic(m), + using_params(mk_simplify_tactic(m), simp2_p) + ); + + return and_then( + preamble_t, + alloc(qfufbv_ackr_tactic, m, p)); +} diff --git a/src/tactic/ackr/qfufbv_ackr_tactic.h b/src/tactic/ackr/qfufbv_ackr_tactic.h new file mode 100644 index 000000000..bf7d42bc2 --- /dev/null +++ b/src/tactic/ackr/qfufbv_ackr_tactic.h @@ -0,0 +1,28 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +qfufbv_ackr_tactic.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ + +#ifndef _QFUFBF_ACKR_TACTIC_H_ +#define _QFUFBF_ACKR_TACTIC_H_ +#include"tactical.h" + +tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p); + +/* +ADD_TACTIC("qfufbv_ackr", "A tactic for solving QF_UFBV based on Ackermannization.", "mk_qfufbv_ackr_tactic(m, p)") +*/ + +#endif + From 0dc8dc4d8e36da3bee3c77693df9672e9495d3f1 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Tue, 26 Jan 2016 17:02:30 +0000 Subject: [PATCH 10/31] Moving things around. Adding tactic just for ackermannization. --- src/ackr/ackr.pyg | 8 - src/ackr/ackr_info.h | 107 ------- src/ackr/lackr.cpp | 294 ------------------ src/ackr/lackr.h | 125 -------- src/ackr/lackr_model_constructor.cpp | 379 ------------------------ src/ackr/lackr_model_constructor.h | 57 ---- src/ackr/lackr_model_converter.cpp | 138 --------- src/ackr/lackr_model_converter.h | 22 -- src/ackr/lackr_model_converter_lazy.cpp | 60 ---- src/ackr/lackr_model_converter_lazy.h | 23 -- src/ackr/lackr_tactic.cpp | 129 -------- src/ackr/lackr_tactic.h | 28 -- 12 files changed, 1370 deletions(-) delete mode 100644 src/ackr/ackr.pyg delete mode 100644 src/ackr/ackr_info.h delete mode 100644 src/ackr/lackr.cpp delete mode 100644 src/ackr/lackr.h delete mode 100644 src/ackr/lackr_model_constructor.cpp delete mode 100644 src/ackr/lackr_model_constructor.h delete mode 100644 src/ackr/lackr_model_converter.cpp delete mode 100644 src/ackr/lackr_model_converter.h delete mode 100644 src/ackr/lackr_model_converter_lazy.cpp delete mode 100644 src/ackr/lackr_model_converter_lazy.h delete mode 100644 src/ackr/lackr_tactic.cpp delete mode 100644 src/ackr/lackr_tactic.h diff --git a/src/ackr/ackr.pyg b/src/ackr/ackr.pyg deleted file mode 100644 index 4b3d6f5f7..000000000 --- a/src/ackr/ackr.pyg +++ /dev/null @@ -1,8 +0,0 @@ -def_module_params('ackr', - description='solving UF via ackermannization (currently for QF_AUFBV)', - export=True, - params=( - ('eager', BOOL, True, 'eagerly instantiate all congruence rules'), - ('sat_backend', BOOL, False, 'use SAT rather than SMT'), - )) - diff --git a/src/ackr/ackr_info.h b/src/ackr/ackr_info.h deleted file mode 100644 index 1b21f0c91..000000000 --- a/src/ackr/ackr_info.h +++ /dev/null @@ -1,107 +0,0 @@ -/*++ -Copyright (c) 2015 Microsoft Corporation - -Module Name: - -ackr_info.h - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ -#ifndef ACKR_INFO_H_12278 -#define ACKR_INFO_H_12278 -#include"obj_hashtable.h" -#include"ast.h" -#include"ref.h" -#include"expr_replacer.h" - -/** \brief - Information about how a formula is being converted into - a formula without uninterpreted function symbols via ackermannization. - - The intended use is that new terms are added via set_abstr. - Once all terms are abstracted, call seal. - The function abstract may only be called when sealed. - - The class enables reference counting. -**/ -class ackr_info { - public: - ackr_info(ast_manager& m) - : m_m(m) - , m_consts(m) - , m_er(mk_default_expr_replacer(m)) - , m_subst(m_m) - , m_ref_count(0) - , m_sealed(false) - {} - - virtual ~ackr_info() { - m_consts.reset(); - } - - inline void set_abstr(app* term, app* c) { - SASSERT(!m_sealed); - SASSERT(c); - m_t2c.insert(term,c); - m_c2t.insert(c->get_decl(),term); - m_subst.insert(term, c); - m_consts.push_back(c); - } - - inline void abstract(expr * e, expr_ref& res) { - SASSERT(m_sealed); - (*m_er)(e, res); - } - - inline app* find_term(func_decl* c) const { - app * rv = 0; - m_c2t.find(c,rv); - return rv; - } - - inline app* get_abstr(app* term) const { - app * const rv = m_t2c.find(term); - SASSERT(rv); - return rv; - } - - inline void seal() { - m_sealed=true; - m_er->set_substitution(&m_subst); - } - - // - // Reference counting - // - void inc_ref() { ++m_ref_count; } - void dec_ref() { - --m_ref_count; - if (m_ref_count == 0) { - dealloc(this); - } - } - private: - typedef obj_map t2ct; - typedef obj_map c2tt; - ast_manager& m_m; - - t2ct m_t2c; // terms to constants - c2tt m_c2t; // constants to terms (inversion of m_t2c) - expr_ref_vector m_consts; // the constants introduced during abstraction - - // replacer and substitution used to compute abstractions - scoped_ptr m_er; - expr_substitution m_subst; - - unsigned m_ref_count; // reference counting - bool m_sealed; // debugging -}; - -typedef ref ackr_info_ref; -#endif /* ACKR_INFO_H_12278 */ diff --git a/src/ackr/lackr.cpp b/src/ackr/lackr.cpp deleted file mode 100644 index c1138c7a5..000000000 --- a/src/ackr/lackr.cpp +++ /dev/null @@ -1,294 +0,0 @@ -/*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - lackr.cpp - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: ---*/ -/////////////// -#include"lackr.h" -#include"ackr_params.hpp" -#include"tactic.h" -#include"lackr_model_constructor.h" -#include"ackr_info.h" -#include"for_each_expr.h" -/////////////// -#include"inc_sat_solver.h" -#include"qfaufbv_tactic.h" -#include"qfbv_tactic.h" -#include"tactic2solver.h" -/////////////// -#include"model_smt2_pp.h" -/////////////// -lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas) - : m_m(m) - , m_p(p) - , m_formulas(formulas) - , m_abstr(m) - , m_sat(0) - , m_bvutil(m) - , m_simp(m) - , m_ackrs(m) - , m_st(st) -{ - updt_params(p); -} - -lackr::~lackr() { - const fun2terms_map::iterator e = m_fun2terms.end(); - for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { - dealloc(i->get_value()); - } -} - -lbool lackr::operator() () { - setup_sat(); - init(); - const lbool rv = m_eager ? eager() : lazy(); - if (rv == l_true) m_sat->get_model(m_model); - CTRACE("lackr", rv == l_true, - model_smt2_pp(tout << "abstr_model(\n", m_m, *(m_model.get()), 2); tout << ")\n"; ); - return rv; -} - -void lackr::mk_ackermann(/*out*/goal_ref& g) { - init(); - eager_enc(); - for (unsigned i = 0; i < m_abstr.size(); ++i) g->assert_expr(m_abstr.get(i)); - for (unsigned i = 0; i < m_ackrs.size(); ++i) g->assert_expr(m_ackrs.get(i)); -} - -void lackr::init() { - params_ref simp_p(m_p); - m_simp.updt_params(simp_p); - m_info = alloc(ackr_info, m_m); - collect_terms(); - abstract(); -} - -// -// Introduce ackermann lemma for the two given terms. -// -bool lackr::ackr(app * const t1, app * const t2) { - TRACE("lackr", tout << "ackr " - << mk_ismt2_pp(t1, m_m, 2) << " , " << mk_ismt2_pp(t2, m_m, 2) << "\n";); - const unsigned sz = t1->get_num_args(); - SASSERT(t2->get_num_args() == sz); - expr_ref_vector eqs(m_m); - for (unsigned gi = 0; gi < sz; ++gi) { - expr * const arg1 = t1->get_arg(gi); - expr * const arg2 = t2->get_arg(gi); - if (arg1 == arg2) continue; - if (m_bvutil.is_numeral(arg1) && m_bvutil.is_numeral(arg2)) { - // quickly abort if there are two distinct numerals - SASSERT(arg1 != arg2); - TRACE("lackr", tout << "never eq\n";); - return false; - } - eqs.push_back(m_m.mk_eq(arg1, arg2)); - } - app * const a1 = m_info->get_abstr(t1); - app * const a2 = m_info->get_abstr(t2); - SASSERT(a1 && a2); - TRACE("lackr", tout << "abstr1 " << mk_ismt2_pp(a1, m_m, 2) << "\n";); - TRACE("lackr", tout << "abstr2 " << mk_ismt2_pp(a2, m_m, 2) << "\n";); - expr_ref lhs(m_m.mk_and(eqs.size(), eqs.c_ptr()), m_m); - TRACE("lackr", tout << "ackr constr lhs" << mk_ismt2_pp(lhs, m_m, 2) << "\n";); - expr_ref rhs(m_m.mk_eq(a1, a2),m_m); - TRACE("lackr", tout << "ackr constr rhs" << mk_ismt2_pp(rhs, m_m, 2) << "\n";); - expr_ref cg(m_m.mk_implies(lhs, rhs), m_m); - TRACE("lackr", tout << "ackr constr" << mk_ismt2_pp(cg, m_m, 2) << "\n";); - expr_ref cga(m_m); - m_info->abstract(cg, cga); // constraint needs abstraction due to nested applications - m_simp(cga); - TRACE("lackr", tout << "ackr constr abs:" << mk_ismt2_pp(cga, m_m, 2) << "\n";); - if (m_m.is_true(cga)) return false; - m_st.m_ackrs_sz++; - m_ackrs.push_back(cga); - return true; -} - -// -// Introduce the ackermann lemma for each pair of terms. -// -void lackr::eager_enc() { - const fun2terms_map::iterator e = m_fun2terms.end(); - for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { - checkpoint(); - func_decl* const fd = i->m_key; - app_set * const ts = i->get_value(); - const app_set::iterator r = ts->end(); - for (app_set::iterator j = ts->begin(); j != r; ++j) { - app_set::iterator k = j; - ++k; - for (; k != r; ++k) { - app * const t1 = *j; - app * const t2 = *k; - SASSERT(t1->get_decl() == fd); - SASSERT(t2->get_decl() == fd); - if (t1 == t2) continue; - ackr(t1,t2); - } - } - } -} - - -void lackr::abstract() { - const fun2terms_map::iterator e = m_fun2terms.end(); - for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { - func_decl* const fd = i->m_key; - app_set * const ts = i->get_value(); - sort* const s = fd->get_range(); - const app_set::iterator r = ts->end(); - for (app_set::iterator j = ts->begin(); j != r; ++j) { - app * const fc = m_m.mk_fresh_const(fd->get_name().str().c_str(), s); - app * const t = *j; - SASSERT(t->get_decl() == fd); - m_info->set_abstr(t, fc); - TRACE("lackr", tout << "abstr term " - << mk_ismt2_pp(t, m_m, 2) - << " -> " - << mk_ismt2_pp(fc, m_m, 2) - << "\n";); - } - } - m_info->seal(); - // perform abstraction of the formulas - const unsigned sz = m_formulas.size(); - for (unsigned i = 0; i < sz; ++i) { - expr_ref a(m_m); - m_info->abstract(m_formulas.get(i), a); - m_abstr.push_back(a); - } -} - -void lackr::add_term(app* a) { - if (a->get_num_args() == 0) return; - func_decl* const fd = a->get_decl(); - if (!is_uninterp(a)) return; - SASSERT(m_bvutil.is_bv_sort(fd->get_range()) || m_m.is_bool(a)); - app_set* ts = 0; - if (!m_fun2terms.find(fd, ts)) { - ts = alloc(app_set); - m_fun2terms.insert(fd, ts); - } - TRACE("lackr", tout << "term(" << mk_ismt2_pp(a, m_m, 2) << ")\n";); - ts->insert(a); -} - -void lackr::push_abstraction() { - const unsigned sz = m_abstr.size(); - for (unsigned i = 0; i < sz; ++i) { - m_sat->assert_expr(m_abstr.get(i)); - } -} - -lbool lackr::eager() { - push_abstraction(); - TRACE("lackr", tout << "run sat 0\n"; ); - const lbool rv0 = m_sat->check_sat(0, 0); - if (rv0 == l_false) return l_false; - eager_enc(); - expr_ref all(m_m); - all = m_m.mk_and(m_ackrs.size(), m_ackrs.c_ptr()); - m_simp(all); - m_sat->assert_expr(all); - TRACE("lackr", tout << "run sat all\n"; ); - return m_sat->check_sat(0, 0); -} - -lbool lackr::lazy() { - lackr_model_constructor mc(m_m, m_info); - push_abstraction(); - unsigned ackr_head = 0; - while (1) { - m_st.m_it++; - checkpoint(); - TRACE("lackr", tout << "lazy check: " << m_st.m_it << "\n";); - const lbool r = m_sat->check_sat(0, 0); - if (r == l_undef) return l_undef; // give up - if (r == l_false) return l_false; // abstraction unsat - // reconstruct model - model_ref am; - m_sat->get_model(am); - const bool mc_res = mc.check(am); - if (mc_res) return l_true; // model okay - // refine abstraction - const lackr_model_constructor::conflict_list conflicts = mc.get_conflicts(); - for (lackr_model_constructor::conflict_list::const_iterator i = conflicts.begin(); - i != conflicts.end(); ++i) { - ackr(i->first, i->second); - } - while (ackr_head < m_ackrs.size()) { - m_sat->assert_expr(m_ackrs.get(ackr_head++)); - } - } -} - -void lackr::setup_sat() { - if (m_use_sat) { - tactic_ref t = mk_qfbv_tactic(m_m, m_p); - m_sat = mk_tactic2solver(m_m, t.get(), m_p); - } - else { - tactic_ref t = mk_qfaufbv_tactic(m_m, m_p); - m_sat = mk_tactic2solver(m_m, t.get(), m_p); - } - SASSERT(m_sat); - m_sat->set_produce_models(true); -} - - -// -// Collect all uninterpreted terms, skipping 0-arity. -// -void lackr::collect_terms() { - ptr_vector stack; - expr * curr; - expr_mark visited; - for(unsigned i = 0; i < m_formulas.size(); ++i) { - stack.push_back(m_formulas.get(i)); - } - - while (!stack.empty()) { - curr = stack.back(); - if (visited.is_marked(curr)) { - stack.pop_back(); - continue; - } - - switch (curr->get_kind()) { - case AST_VAR: - visited.mark(curr, true); - stack.pop_back(); - break; - case AST_APP: - { - app * const a = to_app(curr); - if (for_each_expr_args(stack, visited, a->get_num_args(), a->get_args())) { - visited.mark(curr, true); - stack.pop_back(); - add_term(a); - } - } - break; - case AST_QUANTIFIER: - UNREACHABLE(); - break; - default: - UNREACHABLE(); - return; - } - } - visited.reset(); -} diff --git a/src/ackr/lackr.h b/src/ackr/lackr.h deleted file mode 100644 index efeaf0de2..000000000 --- a/src/ackr/lackr.h +++ /dev/null @@ -1,125 +0,0 @@ - /*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - lackr.h - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: - --*/ -#ifndef LACKR_H_15079 -#define LACKR_H_15079 -/////////////// -#include"ackr_info.h" -#include"ackr_params.hpp" -#include"th_rewriter.h" -#include"cooperate.h" -#include"bv_decl_plugin.h" -#include"lbool.h" -#include"model.h" -#include"solver.h" -#include"util.h" -#include"tactic_exception.h" -#include"goal.h" - -struct lackr_stats { - lackr_stats() : m_it(0), m_ackrs_sz(0) {} - void reset() { m_it = m_ackrs_sz = 0; } - unsigned m_it; // number of lazy iterations - unsigned m_ackrs_sz; // number of congruence constraints -}; - -/** \brief - A class to encode or directly solve problems with uninterpreted functions via ackermannization. - Currently, solving is supported only for QF_UFBV. -**/ -class lackr { - public: - lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas); - ~lackr(); - void updt_params(params_ref const & _p) { - ackr_params p(_p); - m_eager = p.eager(); - m_use_sat = p.sat_backend(); - } - - /** \brief - * Solve the formula that the class was initialized with. - **/ - lbool operator() (); - - - /** \brief - * Converts function occurrences to constants and encodes all congruence ackermann lemmas. - * This guarantees a equisatisfiability with the input formula. It has a worst-case quadratic blowup. - **/ - void mk_ackermann(/*out*/goal_ref& g); - - - // - // getters - // - inline ackr_info_ref get_info() { return m_info; } - inline model_ref get_model() { return m_model; } - - // - // timeout mechanism - // - void checkpoint() { - if (m_m.canceled()) { - throw tactic_exception(TACTIC_CANCELED_MSG); - } - cooperate("lackr"); - } - private: - typedef obj_hashtable app_set; - typedef obj_map fun2terms_map; - ast_manager& m_m; - params_ref m_p; - expr_ref_vector m_formulas; - expr_ref_vector m_abstr; - fun2terms_map m_fun2terms; - ackr_info_ref m_info; - scoped_ptr m_sat; - bv_util m_bvutil; - th_rewriter m_simp; - expr_ref_vector m_ackrs; - model_ref m_model; - bool m_eager; - bool m_use_sat; - lackr_stats& m_st; - - void init(); - void setup_sat(); - lbool eager(); - lbool lazy(); - - // - // Introduce congruence ackermann lemma for the two given terms. - // - bool ackr(app * const t1, app * const t2); - - // - // Introduce the ackermann lemma for each pair of terms. - // - void eager_enc(); - - void abstract(); - - void push_abstraction(); - - void add_term(app* a); - - // - // Collect all uninterpreted terms, skipping 0-arity. - // - void collect_terms(); -}; -#endif /* LACKR_H_15079 */ diff --git a/src/ackr/lackr_model_constructor.cpp b/src/ackr/lackr_model_constructor.cpp deleted file mode 100644 index bcc036d2e..000000000 --- a/src/ackr/lackr_model_constructor.cpp +++ /dev/null @@ -1,379 +0,0 @@ -/*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - model_constructor.cpp - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: ---*/ -#include"lackr_model_constructor.h" -#include"model_evaluator.h" -#include"ast_smt2_pp.h" -#include"ackr_info.h" -#include"for_each_expr.h" -#include"bv_rewriter.h" -#include"bool_rewriter.h" -struct lackr_model_constructor::imp { - public: - imp(ast_manager& m, - ackr_info_ref info, - model_ref& abstr_model, - conflict_list& conflicts) - : m_m(m) - , m_info(info) - , m_abstr_model(abstr_model) - , m_conflicts(conflicts) - , m_b_rw(m) - , m_bv_rw(m) - , m_empty_model(m) - {} - - ~imp() { - { - values2val_t::iterator i = m_values2val.begin(); - const values2val_t::iterator e = m_values2val.end(); - for (; i != e; ++i) { - m_m.dec_ref(i->m_key); - m_m.dec_ref(i->m_value.value); - m_m.dec_ref(i->m_value.source_term); - } - } - { - app2val_t::iterator i = m_app2val.begin(); - const app2val_t::iterator e = m_app2val.end(); - for (; i != e; ++i) { - m_m.dec_ref(i->m_value); - m_m.dec_ref(i->m_key); - } - } - } - - // - // Returns true iff model was successfully constructed. - // - bool check() { - for (unsigned i = 0; i < m_abstr_model->get_num_constants(); i++) { - func_decl * const c = m_abstr_model->get_constant(i); - app * const term = m_info->find_term(c); - if (term) m_stack.push_back(term); - else m_stack.push_back(m_m.mk_const(c)); - } - return run(); - } - - - void make_model(model_ref& destination) { - { - for (unsigned i = 0; i < m_abstr_model->get_num_uninterpreted_sorts(); i++) { - sort * const s = m_abstr_model->get_uninterpreted_sort(i); - ptr_vector u = m_abstr_model->get_universe(s); - destination->register_usort(s, u.size(), u.c_ptr()); - } - } - { - const app2val_t::iterator e = m_app2val.end(); - app2val_t::iterator i = m_app2val.end(); - for (; i != e; ++i) { - app * a = i->m_key; - if (a->get_num_args()) continue; - destination->register_decl(a->get_decl(), i->m_value); - } - } - - obj_map interpretations; - { - const values2val_t::iterator e = m_values2val.end(); - values2val_t::iterator i = m_values2val.end(); - for (; i != e; ++i) add_entry(i->m_key, i->m_value.value, interpretations); - } - - { - obj_map::iterator ie = interpretations.end(); - obj_map::iterator ii = interpretations.begin(); - for (; ii != ie; ++ii) { - func_decl* const fd = ii->m_key; - func_interp* const fi = ii->get_value(); - fi->set_else(m_m.get_some_value(fd->get_range())); - destination->register_decl(fd, fi); - } - } - } - - void add_entry(app* term, expr* value, - obj_map& interpretations) { - func_interp* fi = 0; - func_decl * const declaration = term->get_decl(); - const unsigned sz = declaration->get_arity(); - SASSERT(sz == term->get_num_args()); - if (!interpretations.find(declaration, fi)) { - fi = alloc(func_interp, m_m, sz); - interpretations.insert(declaration, fi); - } - fi->insert_new_entry(term->get_args(), value); - } - private: - ast_manager& m_m; - ackr_info_ref m_info; - model_ref& m_abstr_model; - conflict_list& m_conflicts; - bool_rewriter m_b_rw; - bv_rewriter m_bv_rw; - scoped_ptr m_evaluator; - model m_empty_model; - private: - struct val_info { expr * value; app * source_term; }; - typedef obj_map app2val_t; - typedef obj_map values2val_t; - values2val_t m_values2val; - app2val_t m_app2val; - ptr_vector m_stack; - - static inline val_info mk_val_info(expr* value, app* source_term) { - val_info rv; - rv.value = value; - rv.source_term = source_term; - return rv; - } - - // - // Performs congruence check on terms on the stack. - // (Currently stops upon the first failure). - // Returns true if and only if congruence check succeeded. - bool run() { - m_evaluator = alloc(model_evaluator, m_empty_model); - expr_mark visited; - expr * curr; - while (!m_stack.empty()) { - curr = m_stack.back(); - if (visited.is_marked(curr)) { - m_stack.pop_back(); - continue; - } - - switch (curr->get_kind()) { - case AST_VAR: - UNREACHABLE(); - return false; - case AST_APP: { - app * a = to_app(curr); - if (for_each_expr_args(m_stack, visited, a->get_num_args(), a->get_args())) { - visited.mark(a, true); - m_stack.pop_back(); - if (!mk_value(a)) return false; - } - } - break; - case AST_QUANTIFIER: - UNREACHABLE(); - return false; - default: - UNREACHABLE(); - return false; - } - } - return true; - } - - inline bool is_val(expr * e) { - if (!is_app(e)) return false; - return is_val(to_app(e)); - } - - inline bool is_val(app * a) { - const family_id fid = a->get_decl()->get_family_id(); - const bool rv = fid != null_family_id && a->get_num_args() == 0; - SASSERT(rv == (m_bv_rw.is_numeral(a) || m_m.is_true(a) || m_m.is_false(a))); - return rv; - } - - inline bool eval_cached(app * a, expr *& val) { - if (is_val(a)) { val = a; return true; } - return m_app2val.find(a, val); - } - - bool evaluate(app * const a, expr_ref& result) { - SASSERT(!is_val(a)); - const unsigned num = a->get_num_args(); - if (num == 0) { // handle constants - make_value_constant(a, result); - return true; - } - // evaluate arguments - expr_ref_vector values(m_m); - values.reserve(num); - expr * const * args = a->get_args(); - for (unsigned i = 0; i < num; ++i) { - expr * val; - const bool b = eval_cached(to_app(args[i]), val); // TODO: OK conversion to_app? - CTRACE("model_constructor", !b, tout << "fail arg val(\n" << mk_ismt2_pp(args[i], m_m, 2); ); - TRACE("model_constructor", tout << - "arg val " << i << "(\n" << mk_ismt2_pp(args[i], m_m, 2) - << " : " << mk_ismt2_pp(val, m_m, 2) << '\n'; ); - SASSERT(b); - values[i] = val; - } - // handle functions - if (a->get_family_id() == null_family_id) { // handle uninterpreted - app_ref key(m_m.mk_app(a->get_decl(), values.c_ptr()), m_m); - if (!make_value_uninterpreted_function(a, values, key.get(), result)) { - return false; - } - } - else { // handle interpreted - make_value_interpreted_function(a, values, result); - } - return true; - } - - // - // Check and record the value for a given term, given that all arguments are already checked. - // - bool mk_value(app * a) { - if (is_val(a)) return true; // skip numerals - TRACE("model_constructor", tout << "mk_value(\n" << mk_ismt2_pp(a, m_m, 2) << ")\n";); - SASSERT(!m_app2val.contains(a)); - const unsigned num = a->get_num_args(); - expr_ref result(m_m); - if (!evaluate(a, result)) return false; - SASSERT(is_val(result)); - TRACE("model_constructor", - tout << "map term(\n" << mk_ismt2_pp(a, m_m, 2) << "\n->" - << mk_ismt2_pp(result.get(), m_m, 2)<< ")\n"; ); - CTRACE("model_constructor", - !is_val(result.get()), - tout << "eval fail\n" << mk_ismt2_pp(a, m_m, 2) << mk_ismt2_pp(result, m_m, 2) << "\n"; - ); - SASSERT(is_val(result.get())); - m_app2val.insert(a, result.get()); // memoize - m_m.inc_ref(a); - m_m.inc_ref(result.get()); - return true; - } - - // Constants from the abstract model are directly mapped to the concrete one. - void make_value_constant(app * const a, expr_ref& result) { - SASSERT(a->get_num_args() == 0); - func_decl * const fd = a->get_decl(); - expr * val = m_abstr_model->get_const_interp(fd); - if (val == 0) { // TODO: avoid model completetion? - sort * s = fd->get_range(); - val = m_abstr_model->get_some_value(s); - } - result = val; - } - - bool make_value_uninterpreted_function(app* a, - expr_ref_vector& values, - app* key, - expr_ref& result) { - // get ackermann constant - app * const ac = m_info->get_abstr(a); - func_decl * const a_fd = a->get_decl(); - SASSERT(ac->get_num_args() == 0); - SASSERT(a_fd->get_range() == ac->get_decl()->get_range()); - expr_ref value(m_m); - value = m_abstr_model->get_const_interp(ac->get_decl()); - // get ackermann constant's interpretation - if (value.get() == 0) { // TODO: avoid model completion? - sort * s = a_fd->get_range(); - value = m_abstr_model->get_some_value(s); - } - // check congruence - val_info vi; - if(m_values2val.find(key,vi)) { // already is mapped to a value - SASSERT(vi.source_term); - const bool ok = vi.value == value; - if (!ok) { - TRACE("model_constructor", - tout << "already mapped by(\n" << mk_ismt2_pp(vi.source_term, m_m, 2) << "\n->" - << mk_ismt2_pp(vi.value, m_m, 2) << ")\n"; ); - m_conflicts.push_back(std::make_pair(a, vi.source_term)); - } - result = vi.value; - return ok; - } else { // new value - result = value; - vi.value = value; - vi.source_term = a; - m_values2val.insert(key,vi); - m_m.inc_ref(vi.source_term); - m_m.inc_ref(vi.value); - m_m.inc_ref(key); - return true; - } - UNREACHABLE(); - } - - void make_value_interpreted_function(app* a, - expr_ref_vector& values, - expr_ref& result) { - const unsigned num = values.size(); - func_decl * const fd = a->get_decl(); - const family_id fid = fd->get_family_id(); - expr_ref term(m_m); - term = m_m.mk_app(a->get_decl(), num, values.c_ptr()); - m_evaluator->operator() (term, result); - TRACE("model_constructor", - tout << "eval(\n" << mk_ismt2_pp(term.get(), m_m, 2) << "\n->" - << mk_ismt2_pp(result.get(), m_m, 2) << ")\n"; ); - return; - if (fid == m_b_rw.get_fid()) { - decl_kind k = fd->get_decl_kind(); - if (k == OP_EQ) { - // theory dispatch for = - SASSERT(num == 2); - family_id s_fid = m_m.get_sort(values.get(0))->get_family_id(); - br_status st = BR_FAILED; - if (s_fid == m_bv_rw.get_fid()) - st = m_bv_rw.mk_eq_core(values.get(0), values.get(1), result); - } else { - br_status st = m_b_rw.mk_app_core(fd, num, values.c_ptr(), result); - } - } else { - br_status st = BR_FAILED; - if (fid == m_bv_rw.get_fid()) { - st = m_bv_rw.mk_app_core(fd, num, values.c_ptr(), result); - } - else { - UNREACHABLE(); - } - } - } -}; - -lackr_model_constructor::lackr_model_constructor(ast_manager& m, ackr_info_ref info) - : m_imp(0) - , m_m(m) - , m_state(UNKNOWN) - , m_info(info) - , m_ref_count(0) -{} - -lackr_model_constructor::~lackr_model_constructor() { - if (m_imp) dealloc(m_imp); -} - -bool lackr_model_constructor::check(model_ref& abstr_model) { - m_conflicts.reset(); - if (m_imp) { - dealloc(m_imp); - m_imp = 0; - } - m_imp = alloc(lackr_model_constructor::imp, m_m, m_info, abstr_model, m_conflicts); - const bool rv = m_imp->check(); - m_state = rv ? CHECKED : CONFLICT; - return rv; -} - -void lackr_model_constructor::make_model(model_ref& model) { - SASSERT(m_state == CHECKED); - m_imp->make_model(model); -} diff --git a/src/ackr/lackr_model_constructor.h b/src/ackr/lackr_model_constructor.h deleted file mode 100644 index 478efe97f..000000000 --- a/src/ackr/lackr_model_constructor.h +++ /dev/null @@ -1,57 +0,0 @@ - /*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - model_constructor.h - - Abstract: - Given a propositional abstraction, attempt to construct a model. - - - Author: - - Mikolas Janota - - Revision History: - --*/ -#ifndef LACKR_MODEL_CONSTRUCTOR_H_626 -#define LACKR_MODEL_CONSTRUCTOR_H_626 -#include"ast.h" -#include"ackr_info.h" -#include"model.h" -class lackr_model_constructor { - public: - typedef std::pair app_pair; - typedef vector conflict_list; - lackr_model_constructor(ast_manager& m, ackr_info_ref info); - ~lackr_model_constructor(); - bool check(model_ref& abstr_model); - const conflict_list& get_conflicts() { - SASSERT(m_state == CONFLICT); - return m_conflicts; - } - void make_model(model_ref& model); - - // - // Reference counting - // - void inc_ref() { ++m_ref_count; } - void dec_ref() { - --m_ref_count; - if (m_ref_count == 0) { - dealloc(this); - } - } - private: - struct imp; - imp * m_imp; - enum {CHECKED, CONFLICT, UNKNOWN} m_state; - conflict_list m_conflicts; - ast_manager& m_m; - const ackr_info_ref m_info; - unsigned m_ref_count; // reference counting -}; - -typedef ref lackr_model_constructor_ref; -#endif /* MODEL_CONSTRUCTOR_H_626 */ diff --git a/src/ackr/lackr_model_converter.cpp b/src/ackr/lackr_model_converter.cpp deleted file mode 100644 index 05db4e789..000000000 --- a/src/ackr/lackr_model_converter.cpp +++ /dev/null @@ -1,138 +0,0 @@ -/*++ -Copyright (c) 2015 Microsoft Corporation - -Module Name: - -lackr_model_converter.cpp - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ -#include"lackr_model_converter.h" -#include"model_evaluator.h" -#include"ast_smt2_pp.h" -#include"ackr_info.h" - - -class lackr_model_converter : public model_converter { -public: - lackr_model_converter(ast_manager & m, - const ackr_info_ref& info, - model_ref& abstr_model) - : m(m) - , info(info) - , abstr_model(abstr_model) - { } - - virtual ~lackr_model_converter() { } - - virtual void operator()(model_ref & md, unsigned goal_idx) { - SASSERT(goal_idx == 0); - SASSERT(md.get() == 0 || (!md->get_num_constants() && !md->get_num_functions())); - SASSERT(abstr_model.get()); - model * new_model = alloc(model, m); - convert(abstr_model.get(), new_model); - md = new_model; - } - - virtual void operator()(model_ref & md) { - operator()(md, 0); - } - - //void display(std::ostream & out); - - virtual model_converter * translate(ast_translation & translator) { - NOT_IMPLEMENTED_YET(); - } -protected: - ast_manager& m; - const ackr_info_ref info; - model_ref abstr_model; - void convert(model * source, model * destination); - void add_entry(model_evaluator & evaluator, - app* term, expr* value, - obj_map& interpretations); - void convert_sorts(model * source, model * destination); - void convert_constants(model * source, model * destination); -}; - -void lackr_model_converter::convert(model * source, model * destination) { - SASSERT(source->get_num_functions() == 0); - convert_constants(source,destination); - convert_sorts(source,destination); -} - -void lackr_model_converter::convert_constants(model * source, model * destination) { - TRACE("lackr_model", tout << "converting constants\n";); - obj_map interpretations; - model_evaluator evaluator(*source); - for (unsigned i = 0; i < source->get_num_constants(); i++) { - func_decl * const c = source->get_constant(i); - app * const term = info->find_term(c); - expr * value = source->get_const_interp(c); - if(!term) { - destination->register_decl(c, value); - } else { - add_entry(evaluator, term, value, interpretations); - } - } - - obj_map::iterator e = interpretations.end(); - for (obj_map::iterator i = interpretations.begin(); - i!=e; ++i) { - func_decl* const fd = i->m_key; - func_interp* const fi = i->get_value(); - fi->set_else(m.get_some_value(fd->get_range())); - destination->register_decl(fd, fi); - } -} - -void lackr_model_converter::add_entry(model_evaluator & evaluator, - app* term, expr* value, - obj_map& interpretations) { - TRACE("lackr_model", tout << "add_entry" - << mk_ismt2_pp(term, m, 2) - << "->" - << mk_ismt2_pp(value, m, 2) << "\n"; - ); - - func_interp* fi = 0; - func_decl * const declaration = term->get_decl(); - const unsigned sz = declaration->get_arity(); - SASSERT(sz == term->get_num_args()); - if (!interpretations.find(declaration, fi)) { - fi = alloc(func_interp,m,sz); - interpretations.insert(declaration, fi); - } - expr_ref_vector args(m); - for (unsigned gi = 0; gi < sz; ++gi) { - expr * const arg = term->get_arg(gi); - expr_ref aarg(m); - info->abstract(arg, aarg); - expr_ref arg_value(m); - evaluator(aarg,arg_value); - args.push_back(arg_value); - } - if (fi->get_entry(args.c_ptr()) == 0) { - fi->insert_new_entry(args.c_ptr(), value); - } else { - TRACE("lackr_model", tout << "entry already present\n";); - } -} - -void lackr_model_converter::convert_sorts(model * source, model * destination) { - for (unsigned i = 0; i < source->get_num_uninterpreted_sorts(); i++) { - sort * const s = source->get_uninterpreted_sort(i); - ptr_vector u = source->get_universe(s); - destination->register_usort(s, u.size(), u.c_ptr()); - } -} - -model_converter * mk_lackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model) { - return alloc(lackr_model_converter, m, info, abstr_model); -} diff --git a/src/ackr/lackr_model_converter.h b/src/ackr/lackr_model_converter.h deleted file mode 100644 index ec59d7572..000000000 --- a/src/ackr/lackr_model_converter.h +++ /dev/null @@ -1,22 +0,0 @@ -/*++ -Copyright (c) 2015 Microsoft Corporation - -Module Name: - -lackr_model_converter.h - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ -#ifndef LACKR_MODEL_CONVERTER_H_5814 -#define LACKR_MODEL_CONVERTER_H_5814 -#include"model_converter.h" -#include"ackr_info.h" - -model_converter * mk_lackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); -#endif /* LACKR_MODEL_CONVERTER_H_5814 */ diff --git a/src/ackr/lackr_model_converter_lazy.cpp b/src/ackr/lackr_model_converter_lazy.cpp deleted file mode 100644 index dc3a2b40e..000000000 --- a/src/ackr/lackr_model_converter_lazy.cpp +++ /dev/null @@ -1,60 +0,0 @@ -/*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - lackr_model_converter_lazy.cpp - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: ---*/ -#include"lackr_model_converter_lazy.h" -#include"model_evaluator.h" -#include"ast_smt2_pp.h" -#include"ackr_info.h" -#include"lackr_model_constructor.h" - - -class lackr_model_converter_lazy : public model_converter { -public: - lackr_model_converter_lazy(ast_manager & m, - const lackr_model_constructor_ref& lmc) - : m(m) - , model_constructor(lmc) - { } - - virtual ~lackr_model_converter_lazy() { } - - virtual void operator()(model_ref & md, unsigned goal_idx) { - SASSERT(goal_idx == 0); - SASSERT(md.get() == 0 || (!md->get_num_constants() && !md->get_num_functions())); - SASSERT(model_constructor.get()); - model * new_model = alloc(model, m); - md = new_model; - model_constructor->make_model(md); - } - - virtual void operator()(model_ref & md) { - operator()(md, 0); - } - - //void display(std::ostream & out); - - virtual model_converter * translate(ast_translation & translator) { - NOT_IMPLEMENTED_YET(); - } -protected: - ast_manager& m; - const lackr_model_constructor_ref model_constructor; -}; - -model_converter * mk_lackr_model_converter_lazy(ast_manager & m, - const lackr_model_constructor_ref& model_constructor) { - return alloc(lackr_model_converter_lazy, m, model_constructor); -} diff --git a/src/ackr/lackr_model_converter_lazy.h b/src/ackr/lackr_model_converter_lazy.h deleted file mode 100644 index 354e1c40b..000000000 --- a/src/ackr/lackr_model_converter_lazy.h +++ /dev/null @@ -1,23 +0,0 @@ - /*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - lackr_model_converter_lazy.h - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: - --*/ -#ifndef LACKR_MODEL_CONVERTER_LAZY_H_14201 -#define LACKR_MODEL_CONVERTER_LAZY_H_14201 -#include"model_converter.h" -#include"ackr_info.h" - -model_converter * mk_lackr_model_converter_lazy(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); -#endif /* LACKR_MODEL_CONVERTER_LAZY_H_14201 */ diff --git a/src/ackr/lackr_tactic.cpp b/src/ackr/lackr_tactic.cpp deleted file mode 100644 index 2302e859f..000000000 --- a/src/ackr/lackr_tactic.cpp +++ /dev/null @@ -1,129 +0,0 @@ -/*++ -Copyright (c) 2015 Microsoft Corporation - -Module Name: - -lackr_tactic.cpp - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ -#include"tactical.h" -/////////////// -#include"solve_eqs_tactic.h" -#include"simplify_tactic.h" -#include"propagate_values_tactic.h" -#include"bit_blaster_tactic.h" -#include"elim_uncnstr_tactic.h" -#include"max_bv_sharing_tactic.h" -#include"bv_size_reduction_tactic.h" -#include"ctx_simplify_tactic.h" -#include"nnf_tactic.h" -/////////////// -#include"model_smt2_pp.h" -#include"cooperate.h" -#include"lackr.h" -#include"ackr_params.hpp" -#include"lackr_model_converter.h" - -class lackr_tactic : public tactic { -public: - lackr_tactic(ast_manager& m, params_ref const& p) - : m_m(m) - , m_p(p) - {} - - virtual ~lackr_tactic() { } - - virtual void operator()(goal_ref const & g, - goal_ref_buffer & result, - model_converter_ref & mc, - proof_converter_ref & pc, - expr_dependency_ref & core) { - mc = 0; - ast_manager& m(g->m()); - TRACE("lackr", g->display(tout << "Goal:\n");); - // running implementation - expr_ref_vector flas(m); - const unsigned sz = g->size(); - for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); - scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas); - const lbool o = imp->operator()(); - flas.reset(); - // report result - goal_ref resg(alloc(goal, *g, true)); - if (o == l_false) resg->assert_expr(m.mk_false()); - if (o != l_undef) result.push_back(resg.get()); - // report model - if (g->models_enabled() && (o == l_true)) { - model_ref abstr_model = imp->get_model(); - mc = mk_lackr_model_converter(m, imp->get_info(), abstr_model); - } - } - - virtual void collect_statistics(statistics & st) const { - ackr_params p(m_p); - if (!p.eager()) st.update("lackr-its", m_st.m_it); - st.update("ackr-constraints", m_st.m_ackrs_sz); - } - - virtual void reset_statistics() { m_st.reset(); } - - virtual void cleanup() { } - - virtual tactic* translate(ast_manager& m) { - return alloc(lackr_tactic, m, m_p); - } -private: - ast_manager& m_m; - params_ref m_p; - lackr_stats m_st; -}; - -tactic * mk_lackr_tactic(ast_manager & m, params_ref const & p) { - //return and_then(mk_nnf_tactic(m_m, m_p), alloc(lackr_tactic, m_m, m_p)); - //return alloc(lackr_tactic, m_m, m_p); - //params_ref main_p; - //main_p.set_bool("elim_and", true); - //main_p.set_bool("sort_store", true); - //main_p.set_bool("expand_select_store", true); - //main_p.set_bool("expand_store_eq", true); - - params_ref simp2_p = p; - simp2_p.set_bool("som", true); - simp2_p.set_bool("pull_cheap_ite", true); - simp2_p.set_bool("push_ite_bv", false); - simp2_p.set_bool("local_ctx", true); - simp2_p.set_uint("local_ctx_limit", 10000000); - - simp2_p.set_bool("ite_extra_rules", true); - //simp2_p.set_bool("blast_eq_value", true); - //simp2_p.set_bool("bv_sort_ac", true); - - params_ref ctx_simp_p; - ctx_simp_p.set_uint("max_depth", 32); - ctx_simp_p.set_uint("max_steps", 5000000); - - tactic * const preamble_t = and_then( - mk_simplify_tactic(m), - mk_propagate_values_tactic(m), - //using_params(mk_ctx_simplify_tactic(m_m), ctx_simp_p), - mk_solve_eqs_tactic(m), - mk_elim_uncnstr_tactic(m), - if_no_proofs(if_no_unsat_cores(mk_bv_size_reduction_tactic(m))), - mk_max_bv_sharing_tactic(m), - //mk_macro_finder_tactic(m, p), - using_params(mk_simplify_tactic(m), simp2_p) - //, mk_simplify_tactic(m) - //mk_nnf_tactic(m_m, m_p) - ); - - return and_then( - preamble_t, - alloc(lackr_tactic, m, p)); -} diff --git a/src/ackr/lackr_tactic.h b/src/ackr/lackr_tactic.h deleted file mode 100644 index c578bed92..000000000 --- a/src/ackr/lackr_tactic.h +++ /dev/null @@ -1,28 +0,0 @@ -/*++ -Copyright (c) 2015 Microsoft Corporation - -Module Name: - -lackr_tactic.h - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ - -#ifndef _LACKR_TACTIC_H_ -#define _LACKR_TACTIC_H_ -#include"tactical.h" - -tactic * mk_lackr_tactic(ast_manager & m, params_ref const & p); - -/* -ADD_TACTIC("lackr", "A tactic for solving QF_UFBV based on Ackermannization.", "mk_lackr_tactic(m, p)") -*/ - -#endif - From e7477e2f6a7bf082c9c7b5b9d3b942bdf265e827 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Tue, 26 Jan 2016 17:02:51 +0000 Subject: [PATCH 11/31] Moving things around. Adding tactic just for ackermannization. --- scripts/mk_project.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/scripts/mk_project.py b/scripts/mk_project.py index b1b3fdef8..6c45256fc 100644 --- a/scripts/mk_project.py +++ b/scripts/mk_project.py @@ -77,7 +77,7 @@ def init_project_def(): add_lib('portfolio', ['smtlogic_tactics', 'sat_solver', 'ufbv_tactic', 'fpa_tactics', 'aig_tactic', 'fp', 'qe','sls_tactic', 'subpaving_tactic'], 'tactic/portfolio') add_lib('smtparser', ['portfolio'], 'parsers/smt') add_lib('opt', ['smt', 'smtlogic_tactics', 'sls_tactic', 'sat_solver'], 'opt') - add_lib('ackr', ['smt', 'smtlogic_tactics', 'sat_solver'], 'ackr') + add_lib('ackr', ['smt', 'smtlogic_tactics', 'sat_solver'], 'tactic/ackr') API_files = ['z3_api.h', 'z3_ast_containers.h', 'z3_algebraic.h', 'z3_polynomial.h', 'z3_rcf.h', 'z3_fixedpoint.h', 'z3_optimization.h', 'z3_interp.h', 'z3_fpa.h'] add_lib('api', ['portfolio', 'smtparser', 'realclosure', 'interp', 'opt'], includes2install=['z3.h', 'z3_v1.h', 'z3_macros.h'] + API_files) From 4b37140780d9de8d9b7d477da81b3e13a0d8ce92 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Tue, 26 Jan 2016 18:11:33 +0000 Subject: [PATCH 12/31] small fix --- src/tactic/ackr/ackermannize_tactic.cpp | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/tactic/ackr/ackermannize_tactic.cpp b/src/tactic/ackr/ackermannize_tactic.cpp index 6d52c6fb3..a1d8917df 100644 --- a/src/tactic/ackr/ackermannize_tactic.cpp +++ b/src/tactic/ackr/ackermannize_tactic.cpp @@ -46,8 +46,7 @@ public: result.push_back(resg.get()); // report model if (g->models_enabled()) { - model_ref abstr_model = imp->get_model(); - mc = mk_ackr_model_converter(m, imp->get_info(), abstr_model); + mc = mk_ackr_model_converter(m, imp->get_info()); } } From 956d7742996d7c33f494bc4bb1ab01091c847c90 Mon Sep 17 00:00:00 2001 From: mikolas Date: Wed, 27 Jan 2016 16:22:28 +0000 Subject: [PATCH 13/31] Detecting OP_BSDIV0, etc. as uninterpreted functions in ackermannization. --- src/tactic/ackr/ackr_model_converter.cpp | 7 ++++++- src/tactic/ackr/lackr.cpp | 2 +- src/tactic/ackr/lackr.h | 19 +++++++++++++++++++ src/tactic/ackr/lackr_model_constructor.cpp | 6 ++++++ 4 files changed, 32 insertions(+), 2 deletions(-) diff --git a/src/tactic/ackr/ackr_model_converter.cpp b/src/tactic/ackr/ackr_model_converter.cpp index c21a6240c..7967cdb63 100644 --- a/src/tactic/ackr/ackr_model_converter.cpp +++ b/src/tactic/ackr/ackr_model_converter.cpp @@ -68,7 +68,12 @@ protected: }; void ackr_model_converter::convert(model * source, model * destination) { - SASSERT(source->get_num_functions() == 0); + //SASSERT(source->get_num_functions() == 0); + for (unsigned i = 0; i < source->get_num_functions(); i++) { + func_decl * const fd = source->get_function(i); + func_interp * const fi = source->get_func_interp(fd); + destination->register_decl(fd, fi); + } convert_constants(source,destination); convert_sorts(source,destination); } diff --git a/src/tactic/ackr/lackr.cpp b/src/tactic/ackr/lackr.cpp index fc36d4e1d..e363b96a2 100644 --- a/src/tactic/ackr/lackr.cpp +++ b/src/tactic/ackr/lackr.cpp @@ -177,8 +177,8 @@ void lackr::abstract() { void lackr::add_term(app* a) { if (a->get_num_args() == 0) return; + if (!should_ackermannize(a)) return; func_decl* const fd = a->get_decl(); - if (!is_uninterp(a)) return; SASSERT(m_bvutil.is_bv_sort(fd->get_range()) || m_m.is_bool(a)); app_set* ts = 0; if (!m_fun2terms.find(fd, ts)) { diff --git a/src/tactic/ackr/lackr.h b/src/tactic/ackr/lackr.h index b108653ac..3258b9d6c 100644 --- a/src/tactic/ackr/lackr.h +++ b/src/tactic/ackr/lackr.h @@ -122,5 +122,24 @@ class lackr { // Collect all uninterpreted terms, skipping 0-arity. // void collect_terms(); + + inline bool should_ackermannize(app const * a) const; }; + +inline bool lackr::should_ackermannize(app const * a) const { + if (a->get_family_id() == m_bvutil.get_family_id()) { + switch (a->get_decl_kind()) { + case OP_BSDIV0: + case OP_BUDIV0: + case OP_BSREM0: + case OP_BUREM0: + case OP_BSMOD0: + return true; + default: + return is_uninterp(a); + } + } + return (is_uninterp(a)); +} + #endif /* LACKR_H_15079 */ diff --git a/src/tactic/ackr/lackr_model_constructor.cpp b/src/tactic/ackr/lackr_model_constructor.cpp index ad64fc679..22eb4b40b 100644 --- a/src/tactic/ackr/lackr_model_constructor.cpp +++ b/src/tactic/ackr/lackr_model_constructor.cpp @@ -78,6 +78,12 @@ struct lackr_model_constructor::imp { destination->register_usort(s, u.size(), u.c_ptr()); } } + for (unsigned i = 0; i < m_abstr_model->get_num_functions(); i++) { + func_decl * const fd = m_abstr_model->get_function(i); + func_interp * const fi = m_abstr_model->get_func_interp(fd); + destination->register_decl(fd, fi); + } + { const app2val_t::iterator e = m_app2val.end(); app2val_t::iterator i = m_app2val.end(); From e318d460d7298a2d1114f53b52dc34d52f3d5d56 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Wed, 27 Jan 2016 16:27:31 +0000 Subject: [PATCH 14/31] dbg printing --- src/tactic/ackr/lackr.cpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/tactic/ackr/lackr.cpp b/src/tactic/ackr/lackr.cpp index fc36d4e1d..5f9ef5dba 100644 --- a/src/tactic/ackr/lackr.cpp +++ b/src/tactic/ackr/lackr.cpp @@ -124,6 +124,7 @@ bool lackr::ackr(app * const t1, app * const t2) { // Introduce the ackermann lemma for each pair of terms. // void lackr::eager_enc() { + TRACE("lackr", tout << "#funs: " << m_fun2terms.size() << std::endl;); const fun2terms_map::iterator e = m_fun2terms.end(); for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { checkpoint(); @@ -262,6 +263,7 @@ void lackr::collect_terms() { expr_mark visited; for(unsigned i = 0; i < m_formulas.size(); ++i) { stack.push_back(m_formulas.get(i)); + TRACE("lackr", tout << "infla: " < Date: Thu, 28 Jan 2016 11:46:31 +0000 Subject: [PATCH 15/31] Adding a probe for qf_ufbv and applying it in the qfufbv_ackr_tactic. --- src/tactic/ackr/qfufbv_ackr_tactic.cpp | 8 ++--- src/tactic/probe.cpp | 41 ++++++++++++++++++++++++++ src/tactic/probe.h | 1 + 3 files changed, 46 insertions(+), 4 deletions(-) diff --git a/src/tactic/ackr/qfufbv_ackr_tactic.cpp b/src/tactic/ackr/qfufbv_ackr_tactic.cpp index 9974cc07b..28a801ce1 100644 --- a/src/tactic/ackr/qfufbv_ackr_tactic.cpp +++ b/src/tactic/ackr/qfufbv_ackr_tactic.cpp @@ -23,7 +23,7 @@ Revision History: #include"max_bv_sharing_tactic.h" #include"bv_size_reduction_tactic.h" #include"ctx_simplify_tactic.h" -#include"nnf_tactic.h" +#include"smt_tactic.h" /////////////// #include"model_smt2_pp.h" #include"cooperate.h" @@ -111,7 +111,7 @@ tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p) { using_params(mk_simplify_tactic(m), simp2_p) ); - return and_then( - preamble_t, - alloc(qfufbv_ackr_tactic, m, p)); + tactic * const actual_tactic = alloc(qfufbv_ackr_tactic, m, p); + return and_then(preamble_t, + cond(mk_is_qfufbv_probe(), actual_tactic, mk_smt_tactic())); } diff --git a/src/tactic/probe.cpp b/src/tactic/probe.cpp index dd27691d3..bb9a8fb32 100644 --- a/src/tactic/probe.cpp +++ b/src/tactic/probe.cpp @@ -301,6 +301,7 @@ public: } }; + class is_qfbv_probe : public probe { public: virtual result operator()(goal const & g) { @@ -356,6 +357,46 @@ probe * mk_is_qfaufbv_probe() { return alloc(is_qfaufbv_probe); } + +struct is_non_qfufbv_predicate { + struct found {}; + ast_manager & m; + bv_util m_bv_util; + + is_non_qfufbv_predicate(ast_manager & _m) : m(_m), m_bv_util(_m) {} + + void operator()(var *) { throw found(); } + + void operator()(quantifier *) { throw found(); } + + void operator()(app * n) { + if (!m.is_bool(n) && !m_bv_util.is_bv(n)) + throw found(); + family_id fid = n->get_family_id(); + if (fid == m.get_basic_family_id()) + return; + if (fid == m_bv_util.get_family_id()) + return; + if (is_uninterp(n)) + return; + + throw found(); + } +}; + +class is_qfufbv_probe : public probe { +public: + virtual result operator()(goal const & g) { + return !test(g); + } +}; + +probe * mk_is_qfufbv_probe() { + return alloc(is_qfufbv_probe); +} + + + class num_consts_probe : public probe { bool m_bool; // If true, track only boolean constants. Otherwise, track only non boolean constants. char const * m_family; // (Ignored if m_bool == true), if != 0 and m_bool == true, then track only constants of the given family. diff --git a/src/tactic/probe.h b/src/tactic/probe.h index 56269094e..a5bc5c0a3 100644 --- a/src/tactic/probe.h +++ b/src/tactic/probe.h @@ -112,6 +112,7 @@ probe * mk_div(probe * p1, probe * p2); probe * mk_is_propositional_probe(); probe * mk_is_qfbv_probe(); probe * mk_is_qfaufbv_probe(); +probe * mk_is_qfufbv_probe(); /* ADD_PROBE("is-propositional", "true if the goal is in propositional logic.", "mk_is_propositional_probe()") From 3e94a445402deb7b7310c057337ace484688bbf6 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Thu, 28 Jan 2016 18:18:42 +0000 Subject: [PATCH 16/31] Refactoring ackermannization functionality. --- src/tactic/ackr/ackermannize_tactic.cpp | 2 +- src/tactic/ackr/ackr.pyg | 3 +- src/tactic/ackr/ackr_helper.h | 43 +++++++++++++++++++ src/tactic/ackr/lackr.cpp | 34 +++------------ src/tactic/ackr/lackr.h | 30 +++---------- src/tactic/ackr/lackr_model_constructor.cpp | 4 +- src/tactic/ackr/lackr_model_constructor.h | 3 ++ src/tactic/ackr_tactics/ackr_tactics.pyg | 7 +++ .../qfufbv_ackr_tactic.cpp | 38 ++++++++++++++-- .../qfufbv_ackr_tactic.h | 4 +- src/tactic/smtlogics/qfbv_tactic.cpp | 6 ++- 11 files changed, 111 insertions(+), 63 deletions(-) create mode 100644 src/tactic/ackr/ackr_helper.h create mode 100644 src/tactic/ackr_tactics/ackr_tactics.pyg rename src/tactic/{ackr => ackr_tactics}/qfufbv_ackr_tactic.cpp (76%) rename src/tactic/{ackr => ackr_tactics}/qfufbv_ackr_tactic.h (85%) diff --git a/src/tactic/ackr/ackermannize_tactic.cpp b/src/tactic/ackr/ackermannize_tactic.cpp index a1d8917df..6e95d564b 100644 --- a/src/tactic/ackr/ackermannize_tactic.cpp +++ b/src/tactic/ackr/ackermannize_tactic.cpp @@ -38,7 +38,7 @@ public: expr_ref_vector flas(m); const unsigned sz = g->size(); for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); - scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas); + scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas, NULL); flas.reset(); // mk result goal_ref resg(alloc(goal, *g, true)); diff --git a/src/tactic/ackr/ackr.pyg b/src/tactic/ackr/ackr.pyg index d1ec8efe2..f2e244cfd 100644 --- a/src/tactic/ackr/ackr.pyg +++ b/src/tactic/ackr/ackr.pyg @@ -1,8 +1,7 @@ def_module_params('ackr', - description='solving UF via ackermannization (currently for QF_UFBV)', + description='solving UF via ackermannization', export=True, params=( ('eager', BOOL, True, 'eagerly instantiate all congruence rules'), - ('sat_backend', BOOL, False, 'use SAT rather than SMT'), )) diff --git a/src/tactic/ackr/ackr_helper.h b/src/tactic/ackr/ackr_helper.h new file mode 100644 index 000000000..60f9639a4 --- /dev/null +++ b/src/tactic/ackr/ackr_helper.h @@ -0,0 +1,43 @@ + /*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackr_helper.h + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef ACKR_HELPER_H_6475 +#define ACKR_HELPER_H_6475 +#include"bv_decl_plugin.h" +class ackr_helper { + public: + ackr_helper(ast_manager& m) : m_bvutil(m) {} + inline bool should_ackermannize(app const * a) const { + if (a->get_family_id() == m_bvutil.get_family_id()) { + switch (a->get_decl_kind()) { + case OP_BSDIV0: + case OP_BUDIV0: + case OP_BSREM0: + case OP_BUREM0: + case OP_BSMOD0: + return true; + default: + return is_uninterp(a); + } + } + return (is_uninterp(a)); + } + + bv_util& bvutil() { return m_bvutil; } + private: + bv_util m_bvutil; +}; +#endif /* ACKR_HELPER_H_6475 */ diff --git a/src/tactic/ackr/lackr.cpp b/src/tactic/ackr/lackr.cpp index cb1d5e559..05c87bb58 100644 --- a/src/tactic/ackr/lackr.cpp +++ b/src/tactic/ackr/lackr.cpp @@ -22,20 +22,16 @@ #include"ackr_info.h" #include"for_each_expr.h" /////////////// -#include"inc_sat_solver.h" -#include"qfaufbv_tactic.h" -#include"qfbv_tactic.h" -#include"tactic2solver.h" -/////////////// #include"model_smt2_pp.h" /////////////// -lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas) +lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas, + solver * uffree_solver) : m_m(m) , m_p(p) , m_formulas(formulas) , m_abstr(m) - , m_sat(NULL) - , m_bvutil(m) + , m_sat(uffree_solver) + , m_ackr_helper(m) , m_simp(m) , m_ackrs(m) , m_st(st) @@ -52,7 +48,7 @@ lackr::~lackr() { } lbool lackr::operator() () { - setup_sat(); + SASSERT(m_sat); init(); const lbool rv = m_eager ? eager() : lazy(); if (rv == l_true) m_sat->get_model(m_model); @@ -91,7 +87,7 @@ bool lackr::ackr(app * const t1, app * const t2) { expr * const arg1 = t1->get_arg(i); expr * const arg2 = t2->get_arg(i); if (arg1 == arg2) continue; // quickly skip syntactically equal - if (m_bvutil.is_numeral(arg1) && m_bvutil.is_numeral(arg2)) { + if (m_ackr_helper.bvutil().is_numeral(arg1) && m_ackr_helper.bvutil().is_numeral(arg2)) { // quickly abort if there are two distinct numerals SASSERT(arg1 != arg2); TRACE("lackr", tout << "never eq\n";); @@ -178,9 +174,8 @@ void lackr::abstract() { void lackr::add_term(app* a) { if (a->get_num_args() == 0) return; - if (!should_ackermannize(a)) return; + if (!m_ackr_helper.should_ackermannize(a)) return; func_decl* const fd = a->get_decl(); - SASSERT(m_bvutil.is_bv_sort(fd->get_range()) || m_m.is_bool(a)); app_set* ts = 0; if (!m_fun2terms.find(fd, ts)) { ts = alloc(app_set); @@ -239,21 +234,6 @@ lbool lackr::lazy() { } } -void lackr::setup_sat() { - SASSERT(m_sat == NULL); - if (m_use_sat) { - tactic_ref t = mk_qfbv_tactic(m_m, m_p); - m_sat = mk_tactic2solver(m_m, t.get(), m_p); - } - else { - tactic_ref t = mk_qfaufbv_tactic(m_m, m_p); - m_sat = mk_tactic2solver(m_m, t.get(), m_p); - } - SASSERT(m_sat != NULL); - m_sat->set_produce_models(true); -} - - // // Collect all uninterpreted terms, skipping 0-arity. // diff --git a/src/tactic/ackr/lackr.h b/src/tactic/ackr/lackr.h index 3258b9d6c..f684b4b6a 100644 --- a/src/tactic/ackr/lackr.h +++ b/src/tactic/ackr/lackr.h @@ -18,6 +18,7 @@ #define LACKR_H_15079 /////////////// #include"ackr_info.h" +#include"ackr_helper.h" #include"ackr_params.hpp" #include"th_rewriter.h" #include"cooperate.h" @@ -42,12 +43,12 @@ struct lackr_stats { **/ class lackr { public: - lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas); + lackr(ast_manager& m, params_ref p, lackr_stats& st, + expr_ref_vector& formulas, solver * uffree_solver); ~lackr(); void updt_params(params_ref const & _p) { ackr_params p(_p); m_eager = p.eager(); - m_use_sat = p.sat_backend(); } /** \brief @@ -87,18 +88,16 @@ class lackr { expr_ref_vector m_abstr; fun2terms_map m_fun2terms; ackr_info_ref m_info; - scoped_ptr m_sat; - bv_util m_bvutil; + solver* m_sat; + ackr_helper m_ackr_helper; th_rewriter m_simp; expr_ref_vector m_ackrs; model_ref m_model; bool m_eager; - bool m_use_sat; lackr_stats& m_st; bool m_is_init; void init(); - void setup_sat(); lbool eager(); lbool lazy(); @@ -122,24 +121,5 @@ class lackr { // Collect all uninterpreted terms, skipping 0-arity. // void collect_terms(); - - inline bool should_ackermannize(app const * a) const; }; - -inline bool lackr::should_ackermannize(app const * a) const { - if (a->get_family_id() == m_bvutil.get_family_id()) { - switch (a->get_decl_kind()) { - case OP_BSDIV0: - case OP_BUDIV0: - case OP_BSREM0: - case OP_BUREM0: - case OP_BSMOD0: - return true; - default: - return is_uninterp(a); - } - } - return (is_uninterp(a)); -} - #endif /* LACKR_H_15079 */ diff --git a/src/tactic/ackr/lackr_model_constructor.cpp b/src/tactic/ackr/lackr_model_constructor.cpp index 22eb4b40b..2a38a1a4e 100644 --- a/src/tactic/ackr/lackr_model_constructor.cpp +++ b/src/tactic/ackr/lackr_model_constructor.cpp @@ -34,6 +34,7 @@ struct lackr_model_constructor::imp { , m_b_rw(m) , m_bv_rw(m) , m_empty_model(m) + , m_ackr_helper(m) {} ~imp() { @@ -141,6 +142,7 @@ struct lackr_model_constructor::imp { values2val_t m_values2val; app2val_t m_app2val; ptr_vector m_stack; + ackr_helper m_ackr_helper; static inline val_info mk_val_info(expr* value, app* source_term) { val_info rv; @@ -227,7 +229,7 @@ struct lackr_model_constructor::imp { values[i] = val; } // handle functions - if (a->get_family_id() == null_family_id) { // handle uninterpreted + if (m_ackr_helper.should_ackermannize(a)) { // handle uninterpreted app_ref key(m_m.mk_app(a->get_decl(), values.c_ptr()), m_m); if (!make_value_uninterpreted_function(a, values, key.get(), result)) { return false; diff --git a/src/tactic/ackr/lackr_model_constructor.h b/src/tactic/ackr/lackr_model_constructor.h index 478efe97f..f74c3c90d 100644 --- a/src/tactic/ackr/lackr_model_constructor.h +++ b/src/tactic/ackr/lackr_model_constructor.h @@ -19,7 +19,9 @@ #define LACKR_MODEL_CONSTRUCTOR_H_626 #include"ast.h" #include"ackr_info.h" +#include"ackr_helper.h" #include"model.h" + class lackr_model_constructor { public: typedef std::pair app_pair; @@ -50,6 +52,7 @@ class lackr_model_constructor { conflict_list m_conflicts; ast_manager& m_m; const ackr_info_ref m_info; + unsigned m_ref_count; // reference counting }; diff --git a/src/tactic/ackr_tactics/ackr_tactics.pyg b/src/tactic/ackr_tactics/ackr_tactics.pyg new file mode 100644 index 000000000..8f391bc91 --- /dev/null +++ b/src/tactic/ackr_tactics/ackr_tactics.pyg @@ -0,0 +1,7 @@ +def_module_params('ackr_tactics', + description='tactics based on solving UF-theories via ackermannization (see also ackr module)', + export=True, + params=( + ('sat_backend', BOOL, False, 'use SAT rather than SMT in qfufbv_ackr_tactic'), + )) + diff --git a/src/tactic/ackr/qfufbv_ackr_tactic.cpp b/src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp similarity index 76% rename from src/tactic/ackr/qfufbv_ackr_tactic.cpp rename to src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp index 28a801ce1..4eb3792ff 100644 --- a/src/tactic/ackr/qfufbv_ackr_tactic.cpp +++ b/src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp @@ -28,14 +28,22 @@ Revision History: #include"model_smt2_pp.h" #include"cooperate.h" #include"lackr.h" -#include"ackr_params.hpp" +#include"ackr_tactics_params.hpp" #include"ackr_model_converter.h" +/////////////// +#include"inc_sat_solver.h" +#include"qfaufbv_tactic.h" +#include"qfbv_tactic.h" +#include"tactic2solver.h" +/////////////// + class qfufbv_ackr_tactic : public tactic { public: qfufbv_ackr_tactic(ast_manager& m, params_ref const& p) : m_m(m) , m_p(p) + , m_use_sat(false) {} virtual ~qfufbv_ackr_tactic() { } @@ -47,12 +55,13 @@ public: expr_dependency_ref & core) { mc = 0; ast_manager& m(g->m()); - TRACE("lackr", g->display(tout << "goal:\n");); + TRACE("qfufbv_ackr_tactic", g->display(tout << "goal:\n");); // running implementation expr_ref_vector flas(m); const unsigned sz = g->size(); for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); - scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas); + scoped_ptr uffree_solver = setup_sat(); + scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas, uffree_solver.get()); const lbool o = imp->operator()(); flas.reset(); // report result @@ -66,6 +75,11 @@ public: } } + void updt_params(params_ref const & _p) { + ackr_tactics_params p(_p); + m_use_sat = p.sat_backend(); + } + virtual void collect_statistics(statistics & st) const { ackr_params p(m_p); if (!p.eager()) st.update("lackr-its", m_st.m_it); @@ -83,6 +97,24 @@ private: ast_manager& m_m; params_ref m_p; lackr_stats m_st; + bool m_use_sat; + + solver* setup_sat() { + solver * sat(NULL); + if (m_use_sat) { + tactic_ref t = mk_qfbv_tactic(m_m, m_p); + sat = mk_tactic2solver(m_m, t.get(), m_p); + } + else { + tactic_ref t = mk_qfaufbv_tactic(m_m, m_p); + sat = mk_tactic2solver(m_m, t.get(), m_p); + } + SASSERT(sat != NULL); + sat->set_produce_models(true); + return sat; + } + + }; tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p) { diff --git a/src/tactic/ackr/qfufbv_ackr_tactic.h b/src/tactic/ackr_tactics/qfufbv_ackr_tactic.h similarity index 85% rename from src/tactic/ackr/qfufbv_ackr_tactic.h rename to src/tactic/ackr_tactics/qfufbv_ackr_tactic.h index bf7d42bc2..699500be6 100644 --- a/src/tactic/ackr/qfufbv_ackr_tactic.h +++ b/src/tactic/ackr_tactics/qfufbv_ackr_tactic.h @@ -14,8 +14,8 @@ Mikolas Janota Revision History: --*/ -#ifndef _QFUFBF_ACKR_TACTIC_H_ -#define _QFUFBF_ACKR_TACTIC_H_ +#ifndef _QFUFBV_ACKR_TACTIC_H_ +#define _QFUFBV_ACKR_TACTIC_H_ #include"tactical.h" tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p); diff --git a/src/tactic/smtlogics/qfbv_tactic.cpp b/src/tactic/smtlogics/qfbv_tactic.cpp index 4e017e9e6..4ebcd8b50 100644 --- a/src/tactic/smtlogics/qfbv_tactic.cpp +++ b/src/tactic/smtlogics/qfbv_tactic.cpp @@ -28,6 +28,7 @@ Notes: #include"bv_size_reduction_tactic.h" #include"aig_tactic.h" #include"sat_tactic.h" +#include"ackermannize_tactic.h" #define MEMLIMIT 300 @@ -66,7 +67,9 @@ tactic * mk_qfbv_preamble(ast_manager& m, params_ref const& p) { // We should decide later, if we keep it or not. // using_params(mk_simplify_tactic(m), hoist_p), - mk_max_bv_sharing_tactic(m)); + mk_max_bv_sharing_tactic(m), + mk_ackermannize_tactic(m,p) + ); } static tactic * main_p(tactic* t) { @@ -98,7 +101,6 @@ tactic * mk_qfbv_tactic(ast_manager& m, params_ref const & p, tactic* sat, tacti big_aig_p.set_bool("aig_per_assertion", false); tactic* preamble_st = mk_qfbv_preamble(m, p); - tactic * st = main_p(and_then(preamble_st, // If the user sets HI_DIV0=false, then the formula may contain uninterpreted function // symbols. In this case, we should not use From 2141a075ba85a5d443e5bad0e9f9a583d79385a3 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Thu, 28 Jan 2016 18:24:54 +0000 Subject: [PATCH 17/31] Refactoring ackermannization functionality. --- scripts/mk_project.py | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/scripts/mk_project.py b/scripts/mk_project.py index 6c45256fc..d99956652 100644 --- a/scripts/mk_project.py +++ b/scripts/mk_project.py @@ -52,6 +52,7 @@ def init_project_def(): add_lib('proto_model', ['model', 'simplifier', 'smt_params'], 'smt/proto_model') add_lib('smt', ['bit_blaster', 'macros', 'normal_forms', 'cmd_context', 'proto_model', 'substitution', 'grobner', 'euclid', 'simplex', 'proof_checker', 'pattern', 'parser_util', 'fpa']) + add_lib('ackr', ['smt'], 'tactic/ackr') add_lib('bv_tactics', ['tactic', 'bit_blaster'], 'tactic/bv') add_lib('fuzzing', ['ast'], 'test/fuzzing') add_lib('smt_tactic', ['smt'], 'smt/tactic') @@ -70,18 +71,18 @@ def init_project_def(): add_lib('duality_intf', ['muz', 'transforms', 'duality'], 'muz/duality') add_lib('fp', ['muz', 'pdr', 'clp', 'tab', 'rel', 'bmc', 'duality_intf', 'ddnf'], 'muz/fp') add_lib('nlsat_smt_tactic', ['nlsat_tactic', 'smt_tactic'], 'tactic/nlsat_smt') - add_lib('smtlogic_tactics', ['arith_tactics', 'bv_tactics', 'nlsat_tactic', 'smt_tactic', 'aig_tactic', 'fp', 'muz','qe','nlsat_smt_tactic'], 'tactic/smtlogics') + add_lib('smtlogic_tactics', ['ackr', 'arith_tactics', 'bv_tactics', 'nlsat_tactic', 'smt_tactic', 'aig_tactic', 'fp', 'muz','qe','nlsat_smt_tactic'], 'tactic/smtlogics') add_lib('fpa_tactics', ['fpa', 'core_tactics', 'bv_tactics', 'sat_tactic', 'smt_tactic', 'arith_tactics', 'smtlogic_tactics'], 'tactic/fpa') add_lib('ufbv_tactic', ['normal_forms', 'core_tactics', 'macros', 'smt_tactic', 'rewriter'], 'tactic/ufbv') add_lib('sat_solver', ['solver', 'core_tactics', 'aig_tactic', 'bv_tactics', 'arith_tactics', 'sat_tactic'], 'sat/sat_solver') + add_lib('ackr_tactics', ['bv_tactics', 'smt_tactic', 'aig_tactic', 'sat_solver', 'ackr', 'smtlogic_tactics'], 'tactic/ackr_tactics') add_lib('portfolio', ['smtlogic_tactics', 'sat_solver', 'ufbv_tactic', 'fpa_tactics', 'aig_tactic', 'fp', 'qe','sls_tactic', 'subpaving_tactic'], 'tactic/portfolio') add_lib('smtparser', ['portfolio'], 'parsers/smt') add_lib('opt', ['smt', 'smtlogic_tactics', 'sls_tactic', 'sat_solver'], 'opt') - add_lib('ackr', ['smt', 'smtlogic_tactics', 'sat_solver'], 'tactic/ackr') API_files = ['z3_api.h', 'z3_ast_containers.h', 'z3_algebraic.h', 'z3_polynomial.h', 'z3_rcf.h', 'z3_fixedpoint.h', 'z3_optimization.h', 'z3_interp.h', 'z3_fpa.h'] add_lib('api', ['portfolio', 'smtparser', 'realclosure', 'interp', 'opt'], includes2install=['z3.h', 'z3_v1.h', 'z3_macros.h'] + API_files) - add_exe('shell', ['api', 'sat', 'extra_cmds','opt','ackr'], exe_name='z3') + add_exe('shell', ['api', 'sat', 'extra_cmds','opt','ackr_tactics'], exe_name='z3') add_exe('test', ['api', 'fuzzing', 'simplex'], exe_name='test-z3', install=False) _libz3Component = add_dll('api_dll', ['api', 'sat', 'extra_cmds'], 'api/dll', reexports=['api'], From 2ce7dc68ad41f8d4cc1e2a89baccb5ab37b2b185 Mon Sep 17 00:00:00 2001 From: mikolas Date: Fri, 29 Jan 2016 15:37:10 +0000 Subject: [PATCH 18/31] Adding a probe for estimating the number of Ackermann congruence lemas. --- src/tactic/ackr/ackr_bound_probe.cpp | 78 ++++++++++++++++++++++++++++ src/tactic/ackr/ackr_bound_probe.h | 26 ++++++++++ src/tactic/smtlogics/qfbv_tactic.cpp | 3 +- 3 files changed, 106 insertions(+), 1 deletion(-) create mode 100644 src/tactic/ackr/ackr_bound_probe.cpp create mode 100644 src/tactic/ackr/ackr_bound_probe.h diff --git a/src/tactic/ackr/ackr_bound_probe.cpp b/src/tactic/ackr/ackr_bound_probe.cpp new file mode 100644 index 000000000..2a527ecb4 --- /dev/null +++ b/src/tactic/ackr/ackr_bound_probe.cpp @@ -0,0 +1,78 @@ +/*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackr_bound_probe.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ +#include"ackr_helper.h" +#include"ackr_bound_probe.h" + +/** \brief + * For each function f, calculate the number of its occurrences o_f and compute "o_f choose 2". + * The probe then sums up these for all functions. + * This upper bound might be crude because some congruence lemmas trivially simplify to true. + */ +class ackr_bound_probe : public probe { + struct proc { + typedef obj_hashtable app_set; + typedef obj_map fun2terms_map; + ast_manager & m_m; + fun2terms_map m_fun2terms; // a map from functions to occurrences + ackr_helper m_ackr_helper; + + proc(ast_manager & m) : m_m(m), m_ackr_helper(m) { } + + ~proc() { + fun2terms_map::iterator it = m_fun2terms.begin(); + const fun2terms_map::iterator end = m_fun2terms.end(); + for (; it != end; ++it) dealloc(it->get_value()); + } + + void operator()(quantifier *) {} + void operator()(var *) {} + void operator()(app * a) { + if (a->get_num_args() == 0) return; + if (!m_ackr_helper.should_ackermannize(a)) return; + func_decl* const fd = a->get_decl(); + app_set* ts = 0; + if (!m_fun2terms.find(fd, ts)) { + ts = alloc(app_set); + m_fun2terms.insert(fd, ts); + } + ts->insert(a); + } + }; + +public: + ackr_bound_probe() {} + + virtual result operator()(goal const & g) { + proc p(g.m()); + unsigned sz = g.size(); + expr_fast_mark1 visited; + for (unsigned i = 0; i < sz; i++) { + for_each_expr_core(p, visited, g.form(i)); + } + proc::fun2terms_map::iterator it = p.m_fun2terms.begin(); + proc::fun2terms_map::iterator end = p.m_fun2terms.end(); + unsigned total = 0; + for (; it != end; ++it) total += n_choose_2(it->m_value->size()); + return result(total); + } + + inline static unsigned n_choose_2(unsigned n) { return n & 1 ? (n * (n >> 1)) : (n >> 1) * (n - 1); } +}; + +probe * mk_ackr_bound_probe() { + return alloc(ackr_bound_probe); +} diff --git a/src/tactic/ackr/ackr_bound_probe.h b/src/tactic/ackr/ackr_bound_probe.h new file mode 100644 index 000000000..9c966d46e --- /dev/null +++ b/src/tactic/ackr/ackr_bound_probe.h @@ -0,0 +1,26 @@ + /*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackr_bound_probe.h + + Abstract: + + A probe to give an upper bound of Ackermann congruence lemmas that a formula might generate. + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef ACKR_BOUND_PROBE_H_15037 +#define ACKR_BOUND_PROBE_H_15037 +#include"probe.h" +probe * mk_ackr_bound_probe(); + +/* +ADD_PROBE("ackr-bound-probe", "A probe to give an upper bound of Ackermann congruence lemmas that a formula might generate.", "mk_ackr_bound_probe()") +*/ +#endif /* ACKR_BOUND_PROBE_H_15037 */ diff --git a/src/tactic/smtlogics/qfbv_tactic.cpp b/src/tactic/smtlogics/qfbv_tactic.cpp index 4ebcd8b50..a03744474 100644 --- a/src/tactic/smtlogics/qfbv_tactic.cpp +++ b/src/tactic/smtlogics/qfbv_tactic.cpp @@ -29,6 +29,7 @@ Notes: #include"aig_tactic.h" #include"sat_tactic.h" #include"ackermannize_tactic.h" +#include"ackr_bound_probe.h" #define MEMLIMIT 300 @@ -68,7 +69,7 @@ tactic * mk_qfbv_preamble(ast_manager& m, params_ref const& p) { // using_params(mk_simplify_tactic(m), hoist_p), mk_max_bv_sharing_tactic(m), - mk_ackermannize_tactic(m,p) + when(mk_lt(mk_ackr_bound_probe(), mk_const_probe(static_cast(100))), mk_ackermannize_tactic(m,p)) ); } From 470b5c20feda863b4fc04cab119fa84272a2be12 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Fri, 29 Jan 2016 16:43:18 +0000 Subject: [PATCH 19/31] Small modifs in ackermannization. --- src/tactic/ackr/ackr_bound_probe.cpp | 9 ++++++++- src/tactic/ackr/lackr_model_constructor.cpp | 12 +----------- src/tactic/smtlogics/qfbv_tactic.cpp | 4 +++- 3 files changed, 12 insertions(+), 13 deletions(-) diff --git a/src/tactic/ackr/ackr_bound_probe.cpp b/src/tactic/ackr/ackr_bound_probe.cpp index 2a527ecb4..cca2d6a90 100644 --- a/src/tactic/ackr/ackr_bound_probe.cpp +++ b/src/tactic/ackr/ackr_bound_probe.cpp @@ -16,6 +16,7 @@ --*/ #include"ackr_helper.h" #include"ackr_bound_probe.h" +#include"ast_smt2_pp.h" /** \brief * For each function f, calculate the number of its occurrences o_f and compute "o_f choose 2". @@ -66,7 +67,13 @@ public: proc::fun2terms_map::iterator it = p.m_fun2terms.begin(); proc::fun2terms_map::iterator end = p.m_fun2terms.end(); unsigned total = 0; - for (; it != end; ++it) total += n_choose_2(it->m_value->size()); + for (; it != end; ++it) { + const unsigned fsz = it->m_value->size(); + const unsigned n2 = n_choose_2(fsz); + TRACE("ackr_bound_probe", tout << mk_ismt2_pp(it->m_key, g.m(), 0) << " #" << fsz << " n_choose_2=" << n2 << std::endl;); + total += n2; + } + TRACE("ackr_bound_probe", tout << "total=" << total << std::endl;); return result(total); } diff --git a/src/tactic/ackr/lackr_model_constructor.cpp b/src/tactic/ackr/lackr_model_constructor.cpp index 2a38a1a4e..f9eded93b 100644 --- a/src/tactic/ackr/lackr_model_constructor.cpp +++ b/src/tactic/ackr/lackr_model_constructor.cpp @@ -190,17 +190,7 @@ struct lackr_model_constructor::imp { return true; } - inline bool is_val(expr * e) { - if (!is_app(e)) return false; - return is_val(to_app(e)); - } - - inline bool is_val(app * a) { - const family_id fid = a->get_decl()->get_family_id(); - const bool rv = fid != null_family_id && a->get_num_args() == 0; - SASSERT(rv == (m_bv_rw.is_numeral(a) || m_m.is_true(a) || m_m.is_false(a))); - return rv; - } + inline bool is_val(expr * e) { return m_m.is_value(e); } inline bool eval_cached(app * a, expr *& val) { if (is_val(a)) { val = a; return true; } diff --git a/src/tactic/smtlogics/qfbv_tactic.cpp b/src/tactic/smtlogics/qfbv_tactic.cpp index a03744474..56d8b1c88 100644 --- a/src/tactic/smtlogics/qfbv_tactic.cpp +++ b/src/tactic/smtlogics/qfbv_tactic.cpp @@ -33,6 +33,8 @@ Notes: #define MEMLIMIT 300 +#define ACKRLIMIT 1000 + tactic * mk_qfbv_preamble(ast_manager& m, params_ref const& p) { params_ref solve_eq_p; @@ -69,7 +71,7 @@ tactic * mk_qfbv_preamble(ast_manager& m, params_ref const& p) { // using_params(mk_simplify_tactic(m), hoist_p), mk_max_bv_sharing_tactic(m), - when(mk_lt(mk_ackr_bound_probe(), mk_const_probe(static_cast(100))), mk_ackermannize_tactic(m,p)) + when(mk_lt(mk_ackr_bound_probe(), mk_const_probe(static_cast(ACKRLIMIT))), mk_ackermannize_tactic(m,p)) ); } From c9799b143d7a1f4053d382c78a927777bced9b90 Mon Sep 17 00:00:00 2001 From: mikolas Date: Fri, 29 Jan 2016 17:18:21 +0000 Subject: [PATCH 20/31] Adding parameters to Ackermannization in qfbv_tactic. --- src/tactic/smtlogics/qfbv_tactic.cpp | 15 ++++++++++++--- 1 file changed, 12 insertions(+), 3 deletions(-) diff --git a/src/tactic/smtlogics/qfbv_tactic.cpp b/src/tactic/smtlogics/qfbv_tactic.cpp index 56d8b1c88..fa188dffd 100644 --- a/src/tactic/smtlogics/qfbv_tactic.cpp +++ b/src/tactic/smtlogics/qfbv_tactic.cpp @@ -30,11 +30,10 @@ Notes: #include"sat_tactic.h" #include"ackermannize_tactic.h" #include"ackr_bound_probe.h" +#include"qfbv_tactic_params.hpp" #define MEMLIMIT 300 -#define ACKRLIMIT 1000 - tactic * mk_qfbv_preamble(ast_manager& m, params_ref const& p) { params_ref solve_eq_p; @@ -51,10 +50,20 @@ tactic * mk_qfbv_preamble(ast_manager& m, params_ref const& p) { simp2_p.set_bool("hoist_mul", false); // required by som + qfbv_tactic_params my_params(p); + params_ref hoist_p; hoist_p.set_bool("hoist_mul", true); hoist_p.set_bool("som", false); + const double should_ackermannize = static_cast(my_params.div0ackermann()); + const double ackermannize_limit = static_cast(my_params.div0_ackermann_limit()); + probe * const should_ackermann_p = mk_and( + mk_const_probe(should_ackermannize), + mk_lt(mk_ackr_bound_probe(), mk_const_probe(ackermannize_limit)) + ); + + return and_then( mk_simplify_tactic(m), @@ -71,7 +80,7 @@ tactic * mk_qfbv_preamble(ast_manager& m, params_ref const& p) { // using_params(mk_simplify_tactic(m), hoist_p), mk_max_bv_sharing_tactic(m), - when(mk_lt(mk_ackr_bound_probe(), mk_const_probe(static_cast(ACKRLIMIT))), mk_ackermannize_tactic(m,p)) + when(should_ackermann_p, mk_ackermannize_tactic(m,p)) ); } From de28e57deea98d5406d6317b26c616917b8c2419 Mon Sep 17 00:00:00 2001 From: mikolas Date: Fri, 29 Jan 2016 17:21:21 +0000 Subject: [PATCH 21/31] Adding parameters to Ackermannization in qfbv_tactic. --- src/tactic/smtlogics/qfbv_tactic_params.pyg | 6 ++++++ 1 file changed, 6 insertions(+) create mode 100644 src/tactic/smtlogics/qfbv_tactic_params.pyg diff --git a/src/tactic/smtlogics/qfbv_tactic_params.pyg b/src/tactic/smtlogics/qfbv_tactic_params.pyg new file mode 100644 index 000000000..8dccfc650 --- /dev/null +++ b/src/tactic/smtlogics/qfbv_tactic_params.pyg @@ -0,0 +1,6 @@ +def_module_params(class_name='qfbv_tactic_params', + module_name="smtlogic_tactic", + export=True, + params=(('div0ackermann', BOOL, False, "if true, then try to Ackermannize div etc functions."), + ("div0_ackermann_limit", UINT, 1000, "a bound for number of congregants Ackerman lemmas") + )) From 0b298b4df9bbbfd49d55f6376f3e4c761b4dff1b Mon Sep 17 00:00:00 2001 From: "Christoph M. Wintersteiger" Date: Mon, 1 Feb 2016 18:04:19 +0000 Subject: [PATCH 22/31] Minor fixes for QF_BV div0 ackermannization --- src/tactic/ackr/ackermannize_tactic.cpp | 8 ++++++++ src/tactic/probe.cpp | 15 ++++++++++----- 2 files changed, 18 insertions(+), 5 deletions(-) diff --git a/src/tactic/ackr/ackermannize_tactic.cpp b/src/tactic/ackr/ackermannize_tactic.cpp index 6e95d564b..1345b9415 100644 --- a/src/tactic/ackr/ackermannize_tactic.cpp +++ b/src/tactic/ackr/ackermannize_tactic.cpp @@ -35,6 +35,10 @@ public: expr_dependency_ref & core) { mc = 0; ast_manager& m(g->m()); + tactic_report report("ackermannize", *g); + fail_if_unsat_core_generation("ackermannize", g); + fail_if_proof_generation("ackermannize", g); + expr_ref_vector flas(m); const unsigned sz = g->size(); for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); @@ -48,6 +52,10 @@ public: if (g->models_enabled()) { mc = mk_ackr_model_converter(m, imp->get_info()); } + + resg->inc_depth(); + TRACE("ackermannize", resg->display(tout);); + SASSERT(resg->is_well_sorted()); } virtual void collect_statistics(statistics & st) const { diff --git a/src/tactic/probe.cpp b/src/tactic/probe.cpp index bb9a8fb32..a12d2c90d 100644 --- a/src/tactic/probe.cpp +++ b/src/tactic/probe.cpp @@ -285,9 +285,16 @@ struct is_non_qfbv_predicate { throw found(); family_id fid = n->get_family_id(); if (fid == m.get_basic_family_id()) - return; - if (fid == u.get_family_id()) - return; + return; + if (fid == u.get_family_id()) { + if (n->get_decl_kind() == OP_BSDIV0 || + n->get_decl_kind() == OP_BUDIV0 || + n->get_decl_kind() == OP_BSREM0 || + n->get_decl_kind() == OP_BUREM0 || + n->get_decl_kind() == OP_BSMOD0) + throw found(); + return; + } if (is_uninterp_const(n)) return; throw found(); @@ -305,8 +312,6 @@ public: class is_qfbv_probe : public probe { public: virtual result operator()(goal const & g) { - bv_rewriter rw(g.m()); - if (!rw.hi_div0()) return false; return !test(g); } }; From 3f6a1eb8c5877f041941e667888f9d519809ec9f Mon Sep 17 00:00:00 2001 From: "Christoph M. Wintersteiger" Date: Tue, 2 Feb 2016 13:01:32 +0000 Subject: [PATCH 23/31] Fix for QF_BV core theory detection. --- src/tactic/bv/bv1_blaster_tactic.cpp | 2 -- src/tactic/smtlogics/qfbv_tactic.cpp | 15 ++++++++------- 2 files changed, 8 insertions(+), 9 deletions(-) diff --git a/src/tactic/bv/bv1_blaster_tactic.cpp b/src/tactic/bv/bv1_blaster_tactic.cpp index 2e142cb13..e7327e7a9 100644 --- a/src/tactic/bv/bv1_blaster_tactic.cpp +++ b/src/tactic/bv/bv1_blaster_tactic.cpp @@ -481,8 +481,6 @@ tactic * mk_bv1_blaster_tactic(ast_manager & m, params_ref const & p) { class is_qfbv_eq_probe : public probe { public: virtual result operator()(goal const & g) { - bv_rewriter rw(g.m()); - if (!rw.hi_div0()) return false; bv1_blaster_tactic t(g.m()); return t.is_target(g); diff --git a/src/tactic/smtlogics/qfbv_tactic.cpp b/src/tactic/smtlogics/qfbv_tactic.cpp index fa188dffd..aeee6dc54 100644 --- a/src/tactic/smtlogics/qfbv_tactic.cpp +++ b/src/tactic/smtlogics/qfbv_tactic.cpp @@ -115,11 +115,12 @@ tactic * mk_qfbv_tactic(ast_manager& m, params_ref const & p, tactic* sat, tacti tactic* preamble_st = mk_qfbv_preamble(m, p); tactic * st = main_p(and_then(preamble_st, // If the user sets HI_DIV0=false, then the formula may contain uninterpreted function - // symbols. In this case, we should not use - cond(mk_is_qfbv_probe(), - cond(mk_is_qfbv_eq_probe(), - and_then(mk_bv1_blaster_tactic(m), - using_params(smt, solver_p)), + // symbols. In this case, we should not use the `sat', but instead `smt'. Alternatively, + // the UFs can be eliminated by eager ackermannization in the preamble. + cond(mk_is_qfbv_eq_probe(), + and_then(mk_bv1_blaster_tactic(m), + using_params(smt, solver_p)), + cond(mk_is_qfbv_probe(), and_then(mk_bit_blaster_tactic(m), when(mk_lt(mk_memory_probe(), mk_const_probe(MEMLIMIT)), and_then(using_params(and_then(mk_simplify_tactic(m), @@ -129,8 +130,8 @@ tactic * mk_qfbv_tactic(ast_manager& m, params_ref const & p, tactic* sat, tacti mk_aig_tactic(), using_params(mk_aig_tactic(), big_aig_p))))), - sat)), - smt))); + sat), + smt)))); st->updt_params(p); return st; From bcab9a3600ba6f0a13d51f99e50d128128a4a677 Mon Sep 17 00:00:00 2001 From: mikolas Date: Tue, 2 Feb 2016 15:04:20 +0000 Subject: [PATCH 24/31] re-factoring --- src/tactic/ackr/ackermannize_tactic.cpp | 22 +++++++++++++++---- src/tactic/ackr/ackermannize_tactic.h | 4 +++- src/tactic/ackr_tactics/ackr_tactics.pyg | 2 ++ .../ackr_tactics/qfufbv_ackr_tactic.cpp | 4 ++++ src/tactic/ackr_tactics/qfufbv_ackr_tactic.h | 2 +- src/tactic/smtlogics/qfbv_tactic.cpp | 15 +------------ src/tactic/smtlogics/qfbv_tactic_params.pyg | 6 ----- 7 files changed, 29 insertions(+), 26 deletions(-) delete mode 100644 src/tactic/smtlogics/qfbv_tactic_params.pyg diff --git a/src/tactic/ackr/ackermannize_tactic.cpp b/src/tactic/ackr/ackermannize_tactic.cpp index 1345b9415..6f67ff40b 100644 --- a/src/tactic/ackr/ackermannize_tactic.cpp +++ b/src/tactic/ackr/ackermannize_tactic.cpp @@ -18,6 +18,8 @@ Revision History: #include"ackr_params.hpp" #include"ackr_model_converter.h" #include"model_smt2_pp.h" +#include"ackr_bound_probe.h" +#include"ackr_tactics_params.hpp" class ackermannize_tactic : public tactic { public: @@ -35,10 +37,10 @@ public: expr_dependency_ref & core) { mc = 0; ast_manager& m(g->m()); - tactic_report report("ackermannize", *g); - fail_if_unsat_core_generation("ackermannize", g); - fail_if_proof_generation("ackermannize", g); - + tactic_report report("ackermannize", *g); + fail_if_unsat_core_generation("ackermannize", g); + fail_if_proof_generation("ackermannize", g); + expr_ref_vector flas(m); const unsigned sz = g->size(); for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); @@ -78,3 +80,15 @@ private: tactic * mk_ackermannize_tactic(ast_manager & m, params_ref const & p) { return alloc(ackermannize_tactic, m, p); } + +tactic * mk_ackermannize_bounded_tactic(ast_manager & m, params_ref const & p) { + ackr_tactics_params my_params(p); + const double should_ackermannize = static_cast(my_params.div0ackermann()); + const double ackermannize_limit = static_cast(my_params.div0_ackermann_limit()); + probe * const should_ackermann_p = mk_and( + mk_const_probe(should_ackermannize), + mk_lt(mk_ackr_bound_probe(), mk_const_probe(ackermannize_limit)) + ); + tactic * const actual_tactic = mk_ackermannize_tactic(m, p); + return when(should_ackermann_p, actual_tactic); +} \ No newline at end of file diff --git a/src/tactic/ackr/ackermannize_tactic.h b/src/tactic/ackr/ackermannize_tactic.h index 07c68d5e5..18ea20683 100644 --- a/src/tactic/ackr/ackermannize_tactic.h +++ b/src/tactic/ackr/ackermannize_tactic.h @@ -18,10 +18,12 @@ Revision History: #define _ACKERMANNIZE_TACTIC_H #include"tactical.h" +tactic * mk_ackermannize_bounded_tactic(ast_manager & m, params_ref const & p); tactic * mk_ackermannize_tactic(ast_manager & m, params_ref const & p); /* -ADD_TACTIC("ackermannize", "A tactic for performing full Ackermannization.", "mk_ackermannize_tactic(m, p)") + ADD_TACTIC("ackermannize", "A tactic for performing full Ackermannization.", "mk_ackermannize_tactic(m, p)") + ADD_TACTIC("ackermannize_bounded", "A tactic for performing full Ackermannization where Ackermannization is invoked only if bounds given by the parameters of the tactic are not exceeded.", "mk_ackermannize_bounded_tactic(m, p)") */ #endif diff --git a/src/tactic/ackr_tactics/ackr_tactics.pyg b/src/tactic/ackr_tactics/ackr_tactics.pyg index 8f391bc91..1fccd1ad4 100644 --- a/src/tactic/ackr_tactics/ackr_tactics.pyg +++ b/src/tactic/ackr_tactics/ackr_tactics.pyg @@ -3,5 +3,7 @@ def_module_params('ackr_tactics', export=True, params=( ('sat_backend', BOOL, False, 'use SAT rather than SMT in qfufbv_ackr_tactic'), + ('div0ackermann', BOOL, False, "if true, then try to Ackermannize div etc functions."), + ("div0_ackermann_limit", UINT, 1000, "a bound for number of congruence Ackermann lemmas") )) diff --git a/src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp b/src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp index 4eb3792ff..a8a76e698 100644 --- a/src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp +++ b/src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp @@ -55,6 +55,10 @@ public: expr_dependency_ref & core) { mc = 0; ast_manager& m(g->m()); + tactic_report report("qfufbv_ackr", *g); + fail_if_unsat_core_generation("qfufbv_ackr", g); + fail_if_proof_generation("qfufbv_ackr", g); + TRACE("qfufbv_ackr_tactic", g->display(tout << "goal:\n");); // running implementation expr_ref_vector flas(m); diff --git a/src/tactic/ackr_tactics/qfufbv_ackr_tactic.h b/src/tactic/ackr_tactics/qfufbv_ackr_tactic.h index 699500be6..57b32bf75 100644 --- a/src/tactic/ackr_tactics/qfufbv_ackr_tactic.h +++ b/src/tactic/ackr_tactics/qfufbv_ackr_tactic.h @@ -21,7 +21,7 @@ Revision History: tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p); /* -ADD_TACTIC("qfufbv_ackr", "A tactic for solving QF_UFBV based on Ackermannization.", "mk_qfufbv_ackr_tactic(m, p)") + ADD_TACTIC("qfufbv_ackr", "A tactic for solving QF_UFBV based on Ackermannization.", "mk_qfufbv_ackr_tactic(m, p)") */ #endif diff --git a/src/tactic/smtlogics/qfbv_tactic.cpp b/src/tactic/smtlogics/qfbv_tactic.cpp index fa188dffd..207115455 100644 --- a/src/tactic/smtlogics/qfbv_tactic.cpp +++ b/src/tactic/smtlogics/qfbv_tactic.cpp @@ -29,8 +29,6 @@ Notes: #include"aig_tactic.h" #include"sat_tactic.h" #include"ackermannize_tactic.h" -#include"ackr_bound_probe.h" -#include"qfbv_tactic_params.hpp" #define MEMLIMIT 300 @@ -49,21 +47,10 @@ tactic * mk_qfbv_preamble(ast_manager& m, params_ref const& p) { simp2_p.set_bool("flat", true); // required by som simp2_p.set_bool("hoist_mul", false); // required by som - - qfbv_tactic_params my_params(p); - params_ref hoist_p; hoist_p.set_bool("hoist_mul", true); hoist_p.set_bool("som", false); - const double should_ackermannize = static_cast(my_params.div0ackermann()); - const double ackermannize_limit = static_cast(my_params.div0_ackermann_limit()); - probe * const should_ackermann_p = mk_and( - mk_const_probe(should_ackermannize), - mk_lt(mk_ackr_bound_probe(), mk_const_probe(ackermannize_limit)) - ); - - return and_then( mk_simplify_tactic(m), @@ -80,7 +67,7 @@ tactic * mk_qfbv_preamble(ast_manager& m, params_ref const& p) { // using_params(mk_simplify_tactic(m), hoist_p), mk_max_bv_sharing_tactic(m), - when(should_ackermann_p, mk_ackermannize_tactic(m,p)) + mk_ackermannize_bounded_tactic(m,p) ); } diff --git a/src/tactic/smtlogics/qfbv_tactic_params.pyg b/src/tactic/smtlogics/qfbv_tactic_params.pyg deleted file mode 100644 index 8dccfc650..000000000 --- a/src/tactic/smtlogics/qfbv_tactic_params.pyg +++ /dev/null @@ -1,6 +0,0 @@ -def_module_params(class_name='qfbv_tactic_params', - module_name="smtlogic_tactic", - export=True, - params=(('div0ackermann', BOOL, False, "if true, then try to Ackermannize div etc functions."), - ("div0_ackermann_limit", UINT, 1000, "a bound for number of congregants Ackerman lemmas") - )) From 0f0d3e55dc8f824fc31b8549537e8369d07d5e7c Mon Sep 17 00:00:00 2001 From: mikolas Date: Tue, 2 Feb 2016 17:58:23 +0000 Subject: [PATCH 25/31] refactoring --- .../ackr/ackermannize_bv_model_converter.cpp | 22 ++++++++ .../ackr/ackermannize_bv_model_converter.h | 25 +++++++++ ..._tactic.cpp => ackermannize_bv_tactic.cpp} | 51 ++++++++++--------- src/tactic/ackr/ackermannize_bv_tactic.h | 28 ++++++++++ .../ackr/ackermannize_bv_tactic_params.pyg | 8 +++ src/tactic/ackr/ackermannize_tactic.h | 30 ----------- src/tactic/ackr/ackr_bound_probe.cpp | 30 ++++------- src/tactic/ackr/ackr_bound_probe.h | 11 ++-- src/tactic/ackr/ackr_helper.cpp | 30 +++++++++++ src/tactic/ackr/ackr_helper.h | 31 +++++++++-- src/tactic/ackr/ackr_info.h | 8 +-- src/tactic/ackr/ackr_model_converter.h | 7 +-- src/tactic/ackr/lackr.cpp | 13 +++-- src/tactic/ackr/lackr.h | 22 ++++---- src/tactic/ackr/lackr_model_constructor.h | 7 +-- src/tactic/ackr/lackr_model_converter_lazy.h | 8 +-- src/tactic/ackr_tactics/ackr_tactics.pyg | 2 - src/tactic/smtlogics/qfbv_tactic.cpp | 4 +- 18 files changed, 228 insertions(+), 109 deletions(-) create mode 100644 src/tactic/ackr/ackermannize_bv_model_converter.cpp create mode 100644 src/tactic/ackr/ackermannize_bv_model_converter.h rename src/tactic/ackr/{ackermannize_tactic.cpp => ackermannize_bv_tactic.cpp} (59%) create mode 100644 src/tactic/ackr/ackermannize_bv_tactic.h create mode 100644 src/tactic/ackr/ackermannize_bv_tactic_params.pyg delete mode 100644 src/tactic/ackr/ackermannize_tactic.h create mode 100644 src/tactic/ackr/ackr_helper.cpp diff --git a/src/tactic/ackr/ackermannize_bv_model_converter.cpp b/src/tactic/ackr/ackermannize_bv_model_converter.cpp new file mode 100644 index 000000000..d47735a56 --- /dev/null +++ b/src/tactic/ackr/ackermannize_bv_model_converter.cpp @@ -0,0 +1,22 @@ +/*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackermannize_bv_model_converter.cpp + + Abstract: + + + Author: + + Mikolas Janota (MikolasJanota) + + Revision History: +--*/ +#include"ackr_model_converter.h" +#include"ackermannize_bv_model_converter.h" + +model_converter * mk_ackermannize_bv_model_converter(ast_manager & m, const ackr_info_ref& info) { + return mk_ackr_model_converter(m, info); +} diff --git a/src/tactic/ackr/ackermannize_bv_model_converter.h b/src/tactic/ackr/ackermannize_bv_model_converter.h new file mode 100644 index 000000000..e51792bad --- /dev/null +++ b/src/tactic/ackr/ackermannize_bv_model_converter.h @@ -0,0 +1,25 @@ + /*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackermannize_bv_model_converter.h + + Abstract: + + + Author: + + Mikolas Janota (MikolasJanota) + + Revision History: + --*/ +#ifndef ACKERMANNIZE_BV_MODEL_CONVERTER_H_ +#define ACKERMANNIZE_BV_MODEL_CONVERTER_H_ + +#include"model_converter.h" +#include"ackr_info.h" + +model_converter * mk_ackermannize_bv_model_converter(ast_manager & m, const ackr_info_ref& info); + +#endif /* ACKERMANNIZE_BV_MODEL_CONVERTER_H_ */ diff --git a/src/tactic/ackr/ackermannize_tactic.cpp b/src/tactic/ackr/ackermannize_bv_tactic.cpp similarity index 59% rename from src/tactic/ackr/ackermannize_tactic.cpp rename to src/tactic/ackr/ackermannize_bv_tactic.cpp index 6f67ff40b..30547d7fc 100644 --- a/src/tactic/ackr/ackermannize_tactic.cpp +++ b/src/tactic/ackr/ackermannize_bv_tactic.cpp @@ -3,7 +3,7 @@ Copyright (c) 2016 Microsoft Corporation Module Name: -ackermannize_tactic.cpp +ackermannize_bv_tactic.cpp Abstract: @@ -13,22 +13,24 @@ Mikolas Janota Revision History: --*/ +#include"ackermannize_bv_tactic.h" #include"tactical.h" #include"lackr.h" #include"ackr_params.hpp" -#include"ackr_model_converter.h" #include"model_smt2_pp.h" #include"ackr_bound_probe.h" -#include"ackr_tactics_params.hpp" +#include"ackermannize_bv_tactic_params.hpp" +#include"ackermannize_bv_model_converter.h" -class ackermannize_tactic : public tactic { + +class ackermannize_bv_tactic : public tactic { public: - ackermannize_tactic(ast_manager& m, params_ref const& p) + ackermannize_bv_tactic(ast_manager& m, params_ref const& p) : m_m(m) , m_p(p) {} - virtual ~ackermannize_tactic() { } + virtual ~ackermannize_bv_tactic() { } virtual void operator()(goal_ref const & g, goal_ref_buffer & result, @@ -48,11 +50,19 @@ public: flas.reset(); // mk result goal_ref resg(alloc(goal, *g, true)); - imp->mk_ackermann(resg); + const bool success = imp->mk_ackermann(resg, m_lemma_limit); + if (!success) { // Just pass on the input unchanged + result.reset(); + result.push_back(g.get()); + mc = 0; + pc = 0; + core = 0; + return; + } result.push_back(resg.get()); // report model if (g->models_enabled()) { - mc = mk_ackr_model_converter(m, imp->get_info()); + mc = mk_ackermannize_bv_model_converter(m, imp->get_info()); } resg->inc_depth(); @@ -60,6 +70,12 @@ public: SASSERT(resg->is_well_sorted()); } + + void updt_params(params_ref const & _p) { + ackermannize_bv_tactic_params p(_p); + m_lemma_limit = p.div0_ackermann_limit(); + } + virtual void collect_statistics(statistics & st) const { st.update("ackr-constraints", m_st.m_ackrs_sz); } @@ -69,26 +85,15 @@ public: virtual void cleanup() { } virtual tactic* translate(ast_manager& m) { - return alloc(ackermannize_tactic, m, m_p); + return alloc(ackermannize_bv_tactic, m, m_p); } private: ast_manager& m_m; params_ref m_p; lackr_stats m_st; + double m_lemma_limit; }; -tactic * mk_ackermannize_tactic(ast_manager & m, params_ref const & p) { - return alloc(ackermannize_tactic, m, p); +tactic * mk_ackermannize_bv_tactic(ast_manager & m, params_ref const & p) { + return alloc(ackermannize_bv_tactic, m, p); } - -tactic * mk_ackermannize_bounded_tactic(ast_manager & m, params_ref const & p) { - ackr_tactics_params my_params(p); - const double should_ackermannize = static_cast(my_params.div0ackermann()); - const double ackermannize_limit = static_cast(my_params.div0_ackermann_limit()); - probe * const should_ackermann_p = mk_and( - mk_const_probe(should_ackermannize), - mk_lt(mk_ackr_bound_probe(), mk_const_probe(ackermannize_limit)) - ); - tactic * const actual_tactic = mk_ackermannize_tactic(m, p); - return when(should_ackermann_p, actual_tactic); -} \ No newline at end of file diff --git a/src/tactic/ackr/ackermannize_bv_tactic.h b/src/tactic/ackr/ackermannize_bv_tactic.h new file mode 100644 index 000000000..3e3d5636c --- /dev/null +++ b/src/tactic/ackr/ackermannize_bv_tactic.h @@ -0,0 +1,28 @@ +/*++ +Copyright (c) 2016 Microsoft Corporation + +Module Name: + +ackermannize_bv_tactic.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ + +#ifndef _ACKERMANNIZE_TACTIC_H_ +#define _ACKERMANNIZE_TACTIC_H_ +#include"tactical.h" + +tactic * mk_ackermannize_bv_tactic(ast_manager & m, params_ref const & p); + +/* + ADD_TACTIC("ackermannize_bv", "A tactic for performing full Ackermannization on bv instances.", "mk_ackermannize_bv_tactic(m, p)") +*/ + +#endif + diff --git a/src/tactic/ackr/ackermannize_bv_tactic_params.pyg b/src/tactic/ackr/ackermannize_bv_tactic_params.pyg new file mode 100644 index 000000000..bd17dcaff --- /dev/null +++ b/src/tactic/ackr/ackermannize_bv_tactic_params.pyg @@ -0,0 +1,8 @@ +def_module_params(module_name='rewriter', + class_name='ackermannize_bv_tactic_params', + export=True, + params=( + ("div0_ackermann_limit", UINT, 1000, "a bound for number of congruence Ackermann lemmas for div0 modelling"), + ) + ) + diff --git a/src/tactic/ackr/ackermannize_tactic.h b/src/tactic/ackr/ackermannize_tactic.h deleted file mode 100644 index 18ea20683..000000000 --- a/src/tactic/ackr/ackermannize_tactic.h +++ /dev/null @@ -1,30 +0,0 @@ -/*++ -Copyright (c) 2016 Microsoft Corporation - -Module Name: - -ackermannize_tactic.h - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ - -#ifndef _ACKERMANNIZE_TACTIC_H_ -#define _ACKERMANNIZE_TACTIC_H -#include"tactical.h" - -tactic * mk_ackermannize_bounded_tactic(ast_manager & m, params_ref const & p); -tactic * mk_ackermannize_tactic(ast_manager & m, params_ref const & p); - -/* - ADD_TACTIC("ackermannize", "A tactic for performing full Ackermannization.", "mk_ackermannize_tactic(m, p)") - ADD_TACTIC("ackermannize_bounded", "A tactic for performing full Ackermannization where Ackermannization is invoked only if bounds given by the parameters of the tactic are not exceeded.", "mk_ackermannize_bounded_tactic(m, p)") -*/ - -#endif - diff --git a/src/tactic/ackr/ackr_bound_probe.cpp b/src/tactic/ackr/ackr_bound_probe.cpp index cca2d6a90..5cb8e9448 100644 --- a/src/tactic/ackr/ackr_bound_probe.cpp +++ b/src/tactic/ackr/ackr_bound_probe.cpp @@ -18,18 +18,18 @@ #include"ackr_bound_probe.h" #include"ast_smt2_pp.h" -/** \brief - * For each function f, calculate the number of its occurrences o_f and compute "o_f choose 2". - * The probe then sums up these for all functions. - * This upper bound might be crude because some congruence lemmas trivially simplify to true. - */ +/* + For each function f, calculate the number of its occurrences o_f and compute "o_f choose 2". + The probe then sums up these for all functions. + This upper bound might be crude because some congruence lemmas trivially simplify to true. +*/ class ackr_bound_probe : public probe { struct proc { - typedef obj_hashtable app_set; - typedef obj_map fun2terms_map; - ast_manager & m_m; - fun2terms_map m_fun2terms; // a map from functions to occurrences - ackr_helper m_ackr_helper; + typedef ackr_helper::fun2terms_map fun2terms_map; + typedef ackr_helper::app_set app_set; + ast_manager& m_m; + fun2terms_map m_fun2terms; // a map from functions to occurrences + ackr_helper m_ackr_helper; proc(ast_manager & m) : m_m(m), m_ackr_helper(m) { } @@ -64,15 +64,7 @@ public: for (unsigned i = 0; i < sz; i++) { for_each_expr_core(p, visited, g.form(i)); } - proc::fun2terms_map::iterator it = p.m_fun2terms.begin(); - proc::fun2terms_map::iterator end = p.m_fun2terms.end(); - unsigned total = 0; - for (; it != end; ++it) { - const unsigned fsz = it->m_value->size(); - const unsigned n2 = n_choose_2(fsz); - TRACE("ackr_bound_probe", tout << mk_ismt2_pp(it->m_key, g.m(), 0) << " #" << fsz << " n_choose_2=" << n2 << std::endl;); - total += n2; - } + const double total = ackr_helper::calculate_lemma_bound(p.m_fun2terms); TRACE("ackr_bound_probe", tout << "total=" << total << std::endl;); return result(total); } diff --git a/src/tactic/ackr/ackr_bound_probe.h b/src/tactic/ackr/ackr_bound_probe.h index 9c966d46e..6c0a66605 100644 --- a/src/tactic/ackr/ackr_bound_probe.h +++ b/src/tactic/ackr/ackr_bound_probe.h @@ -15,12 +15,15 @@ Revision History: --*/ -#ifndef ACKR_BOUND_PROBE_H_15037 -#define ACKR_BOUND_PROBE_H_15037 +#ifndef ACKR_BOUND_PROBE_H_ +#define ACKR_BOUND_PROBE_H_ + #include"probe.h" + probe * mk_ackr_bound_probe(); /* -ADD_PROBE("ackr-bound-probe", "A probe to give an upper bound of Ackermann congruence lemmas that a formula might generate.", "mk_ackr_bound_probe()") + ADD_PROBE("ackr-bound-probe", "A probe to give an upper bound of Ackermann congruence lemmas that a formula might generate.", "mk_ackr_bound_probe()") */ -#endif /* ACKR_BOUND_PROBE_H_15037 */ + +#endif /* ACKR_BOUND_PROBE_H_ */ diff --git a/src/tactic/ackr/ackr_helper.cpp b/src/tactic/ackr/ackr_helper.cpp new file mode 100644 index 000000000..b3a6a07d1 --- /dev/null +++ b/src/tactic/ackr/ackr_helper.cpp @@ -0,0 +1,30 @@ +/*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackr_helper.cpp + + Abstract: + + + Author: + + Mikolas Janota (MikolasJanota) + + Revision History: +--*/ +#include"ackr_helper.h" + + +double ackr_helper::calculate_lemma_bound(ackr_helper::fun2terms_map& occurrences) { + fun2terms_map::iterator it = occurrences.begin(); + const fun2terms_map::iterator end = occurrences.end(); + double total = 0; + for (; it != end; ++it) { + const unsigned fsz = it->m_value->size(); + const double n2 = n_choose_2_chk(fsz); + total += n2; + } + return total; +} diff --git a/src/tactic/ackr/ackr_helper.h b/src/tactic/ackr/ackr_helper.h index 60f9639a4..c89ebc09e 100644 --- a/src/tactic/ackr/ackr_helper.h +++ b/src/tactic/ackr/ackr_helper.h @@ -14,12 +14,24 @@ Revision History: --*/ -#ifndef ACKR_HELPER_H_6475 -#define ACKR_HELPER_H_6475 +#ifndef ACKR_HELPER_H_ +#define ACKR_HELPER_H_ + #include"bv_decl_plugin.h" + class ackr_helper { public: + typedef obj_hashtable app_set; + typedef obj_map fun2terms_map; + ackr_helper(ast_manager& m) : m_bvutil(m) {} + + /** + \brief Determines if a given function should be Ackermannized. + + This includes all uninterpreted functions but also "special" functions, e.g. OP_BSMOD0, + which are not marked as uninterpreted but effectively are. + */ inline bool should_ackermannize(app const * a) const { if (a->get_family_id() == m_bvutil.get_family_id()) { switch (a->get_decl_kind()) { @@ -36,7 +48,20 @@ class ackr_helper { return (is_uninterp(a)); } - bv_util& bvutil() { return m_bvutil; } + inline bv_util& bvutil() { return m_bvutil; } + + /** + \brief Calculates an upper bound for congruence lemmas given a map of function of occurrences. + */ + static double calculate_lemma_bound(fun2terms_map& occurrences); + + /** \brief Calculate n choose 2. **/ + inline static unsigned n_choose_2(unsigned n) { return n & 1 ? (n * (n >> 1)) : (n >> 1) * (n - 1); } + + inline static double n_choose_2_chk(unsigned n) { + SASSERT(std::numeric_limits().max() & 32); + return n & (1 << 16) ? n_choose_2(n) : std::numeric_limits().infinity();; + } private: bv_util m_bvutil; }; diff --git a/src/tactic/ackr/ackr_info.h b/src/tactic/ackr/ackr_info.h index 1b21f0c91..703f1f3d5 100644 --- a/src/tactic/ackr/ackr_info.h +++ b/src/tactic/ackr/ackr_info.h @@ -13,8 +13,9 @@ Mikolas Janota Revision History: --*/ -#ifndef ACKR_INFO_H_12278 -#define ACKR_INFO_H_12278 +#ifndef ACKR_INFO_H_ +#define ACKR_INFO_H_ + #include"obj_hashtable.h" #include"ast.h" #include"ref.h" @@ -104,4 +105,5 @@ class ackr_info { }; typedef ref ackr_info_ref; -#endif /* ACKR_INFO_H_12278 */ + +#endif /* ACKR_INFO_H_ */ diff --git a/src/tactic/ackr/ackr_model_converter.h b/src/tactic/ackr/ackr_model_converter.h index 611203242..98c03f381 100644 --- a/src/tactic/ackr/ackr_model_converter.h +++ b/src/tactic/ackr/ackr_model_converter.h @@ -13,11 +13,12 @@ Mikolas Janota Revision History: --*/ -#ifndef ACKR_MODEL_CONVERTER_H_5814 -#define ACKR_MODEL_CONVERTER_H_5814 +#ifndef ACKR_MODEL_CONVERTER_H_ +#define ACKR_MODEL_CONVERTER_H_ + #include"model_converter.h" #include"ackr_info.h" model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info); -#endif /* LACKR_MODEL_CONVERTER_H_5814 */ +#endif /* LACKR_MODEL_CONVERTER_H_ */ diff --git a/src/tactic/ackr/lackr.cpp b/src/tactic/ackr/lackr.cpp index 05c87bb58..0a526f7ec 100644 --- a/src/tactic/ackr/lackr.cpp +++ b/src/tactic/ackr/lackr.cpp @@ -14,16 +14,15 @@ Revision History: --*/ -/////////////// + #include"lackr.h" #include"ackr_params.hpp" #include"tactic.h" #include"lackr_model_constructor.h" #include"ackr_info.h" #include"for_each_expr.h" -/////////////// #include"model_smt2_pp.h" -/////////////// + lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas, solver * uffree_solver) : m_m(m) @@ -57,11 +56,17 @@ lbool lackr::operator() () { return rv; } -void lackr::mk_ackermann(/*out*/goal_ref& g) { +bool lackr::mk_ackermann(/*out*/goal_ref& g, double lemmas_upper_bound) { + if (lemmas_upper_bound <= 0) return false; init(); + if (lemmas_upper_bound != std::numeric_limits::infinity()) { + const double lemmas_bound = ackr_helper::calculate_lemma_bound(m_fun2terms); + if (lemmas_bound > lemmas_upper_bound) return false; + } eager_enc(); for (unsigned i = 0; i < m_abstr.size(); ++i) g->assert_expr(m_abstr.get(i)); for (unsigned i = 0; i < m_ackrs.size(); ++i) g->assert_expr(m_ackrs.get(i)); + return true; } void lackr::init() { diff --git a/src/tactic/ackr/lackr.h b/src/tactic/ackr/lackr.h index f684b4b6a..e6612a578 100644 --- a/src/tactic/ackr/lackr.h +++ b/src/tactic/ackr/lackr.h @@ -14,9 +14,9 @@ Revision History: --*/ -#ifndef LACKR_H_15079 -#define LACKR_H_15079 -/////////////// +#ifndef LACKR_H_ +#define LACKR_H_ + #include"ackr_info.h" #include"ackr_helper.h" #include"ackr_params.hpp" @@ -58,11 +58,13 @@ class lackr { /** \brief - * Converts function occurrences to constants and encodes all congruence ackermann lemmas. - * This guarantees a equisatisfiability with the input formula. It has a worst-case quadratic blowup. - **/ - void mk_ackermann(/*out*/goal_ref& g); + Converts function occurrences to constants and encodes all congruence ackermann lemmas. + This procedure guarantees a equisatisfiability with the input formula and it has a worst-case quadratic blowup. + Before ackermannization an upper bound on congruence lemmas is computed and tested against \p lemmas_upper_bound. + If this bound is exceeded, the function returns false, it returns true otherwise. + **/ + bool mk_ackermann(/*out*/goal_ref& g, double lemmas_upper_bound); // // getters @@ -80,8 +82,8 @@ class lackr { cooperate("lackr"); } private: - typedef obj_hashtable app_set; - typedef obj_map fun2terms_map; + typedef ackr_helper::fun2terms_map fun2terms_map; + typedef ackr_helper::app_set app_set; ast_manager& m_m; params_ref m_p; expr_ref_vector m_formulas; @@ -122,4 +124,4 @@ class lackr { // void collect_terms(); }; -#endif /* LACKR_H_15079 */ +#endif /* LACKR_H_ */ diff --git a/src/tactic/ackr/lackr_model_constructor.h b/src/tactic/ackr/lackr_model_constructor.h index f74c3c90d..ce9edba26 100644 --- a/src/tactic/ackr/lackr_model_constructor.h +++ b/src/tactic/ackr/lackr_model_constructor.h @@ -15,8 +15,9 @@ Revision History: --*/ -#ifndef LACKR_MODEL_CONSTRUCTOR_H_626 -#define LACKR_MODEL_CONSTRUCTOR_H_626 +#ifndef LACKR_MODEL_CONSTRUCTOR_H_ +#define LACKR_MODEL_CONSTRUCTOR_H_ + #include"ast.h" #include"ackr_info.h" #include"ackr_helper.h" @@ -57,4 +58,4 @@ class lackr_model_constructor { }; typedef ref lackr_model_constructor_ref; -#endif /* MODEL_CONSTRUCTOR_H_626 */ +#endif /* MODEL_CONSTRUCTOR_H_ */ diff --git a/src/tactic/ackr/lackr_model_converter_lazy.h b/src/tactic/ackr/lackr_model_converter_lazy.h index 354e1c40b..c9b3eac80 100644 --- a/src/tactic/ackr/lackr_model_converter_lazy.h +++ b/src/tactic/ackr/lackr_model_converter_lazy.h @@ -14,10 +14,12 @@ Revision History: --*/ -#ifndef LACKR_MODEL_CONVERTER_LAZY_H_14201 -#define LACKR_MODEL_CONVERTER_LAZY_H_14201 +#ifndef LACKR_MODEL_CONVERTER_LAZY_H_ +#define LACKR_MODEL_CONVERTER_LAZY_H_ + #include"model_converter.h" #include"ackr_info.h" model_converter * mk_lackr_model_converter_lazy(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); -#endif /* LACKR_MODEL_CONVERTER_LAZY_H_14201 */ + +#endif /* LACKR_MODEL_CONVERTER_LAZY_H_ */ diff --git a/src/tactic/ackr_tactics/ackr_tactics.pyg b/src/tactic/ackr_tactics/ackr_tactics.pyg index 1fccd1ad4..8f391bc91 100644 --- a/src/tactic/ackr_tactics/ackr_tactics.pyg +++ b/src/tactic/ackr_tactics/ackr_tactics.pyg @@ -3,7 +3,5 @@ def_module_params('ackr_tactics', export=True, params=( ('sat_backend', BOOL, False, 'use SAT rather than SMT in qfufbv_ackr_tactic'), - ('div0ackermann', BOOL, False, "if true, then try to Ackermannize div etc functions."), - ("div0_ackermann_limit", UINT, 1000, "a bound for number of congruence Ackermann lemmas") )) diff --git a/src/tactic/smtlogics/qfbv_tactic.cpp b/src/tactic/smtlogics/qfbv_tactic.cpp index e8c85ab5a..cd446852e 100644 --- a/src/tactic/smtlogics/qfbv_tactic.cpp +++ b/src/tactic/smtlogics/qfbv_tactic.cpp @@ -28,7 +28,7 @@ Notes: #include"bv_size_reduction_tactic.h" #include"aig_tactic.h" #include"sat_tactic.h" -#include"ackermannize_tactic.h" +#include"ackermannize_bv_tactic.h" #define MEMLIMIT 300 @@ -67,7 +67,7 @@ tactic * mk_qfbv_preamble(ast_manager& m, params_ref const& p) { // using_params(mk_simplify_tactic(m), hoist_p), mk_max_bv_sharing_tactic(m), - mk_ackermannize_bounded_tactic(m,p) + mk_ackermannize_bv_tactic(m,p) ); } From 6f12c0e6f95d528738142bb510acdbd30ed20004 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Wed, 3 Feb 2016 11:52:11 +0000 Subject: [PATCH 26/31] bugfix in refactoring --- src/tactic/ackr/ackr_helper.cpp | 1 - src/tactic/ackr/ackr_helper.h | 3 ++- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/tactic/ackr/ackr_helper.cpp b/src/tactic/ackr/ackr_helper.cpp index b3a6a07d1..803fc3c54 100644 --- a/src/tactic/ackr/ackr_helper.cpp +++ b/src/tactic/ackr/ackr_helper.cpp @@ -16,7 +16,6 @@ --*/ #include"ackr_helper.h" - double ackr_helper::calculate_lemma_bound(ackr_helper::fun2terms_map& occurrences) { fun2terms_map::iterator it = occurrences.begin(); const fun2terms_map::iterator end = occurrences.end(); diff --git a/src/tactic/ackr/ackr_helper.h b/src/tactic/ackr/ackr_helper.h index c89ebc09e..5c572907e 100644 --- a/src/tactic/ackr/ackr_helper.h +++ b/src/tactic/ackr/ackr_helper.h @@ -58,9 +58,10 @@ class ackr_helper { /** \brief Calculate n choose 2. **/ inline static unsigned n_choose_2(unsigned n) { return n & 1 ? (n * (n >> 1)) : (n >> 1) * (n - 1); } + /** \brief Calculate n choose 2 guarded for overflow. Returns infinity if unsafe. **/ inline static double n_choose_2_chk(unsigned n) { SASSERT(std::numeric_limits().max() & 32); - return n & (1 << 16) ? n_choose_2(n) : std::numeric_limits().infinity();; + return n & (1 << 16) ? std::numeric_limits().infinity() : n_choose_2(n); } private: bv_util m_bvutil; From 2679b7454330b20d729413e5b3f050f4f2179b97 Mon Sep 17 00:00:00 2001 From: mikolas Date: Wed, 3 Feb 2016 13:53:52 +0000 Subject: [PATCH 27/31] refactoring --- scripts/mk_project.py | 7 +- src/tactic/ackr/ackr_model_converter.h | 1 + .../ackr_tactics/qfufbv_ackr_tactic.cpp | 153 ----------------- src/tactic/ackr_tactics/qfufbv_ackr_tactic.h | 28 --- .../smtlogics/qfufbv_ackr_model_converter.cpp | 22 +++ .../smtlogics/qfufbv_ackr_model_converter.h | 25 +++ src/tactic/smtlogics/qfufbv_tactic.cpp | 162 ++++++++++++++++-- src/tactic/smtlogics/qfufbv_tactic.h | 3 + .../qfufbv_tactic_params.pyg} | 3 +- 9 files changed, 206 insertions(+), 198 deletions(-) delete mode 100644 src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp delete mode 100644 src/tactic/ackr_tactics/qfufbv_ackr_tactic.h create mode 100644 src/tactic/smtlogics/qfufbv_ackr_model_converter.cpp create mode 100644 src/tactic/smtlogics/qfufbv_ackr_model_converter.h rename src/tactic/{ackr_tactics/ackr_tactics.pyg => smtlogics/qfufbv_tactic_params.pyg} (77%) diff --git a/scripts/mk_project.py b/scripts/mk_project.py index d99956652..82cdc04d0 100644 --- a/scripts/mk_project.py +++ b/scripts/mk_project.py @@ -71,18 +71,17 @@ def init_project_def(): add_lib('duality_intf', ['muz', 'transforms', 'duality'], 'muz/duality') add_lib('fp', ['muz', 'pdr', 'clp', 'tab', 'rel', 'bmc', 'duality_intf', 'ddnf'], 'muz/fp') add_lib('nlsat_smt_tactic', ['nlsat_tactic', 'smt_tactic'], 'tactic/nlsat_smt') - add_lib('smtlogic_tactics', ['ackr', 'arith_tactics', 'bv_tactics', 'nlsat_tactic', 'smt_tactic', 'aig_tactic', 'fp', 'muz','qe','nlsat_smt_tactic'], 'tactic/smtlogics') - add_lib('fpa_tactics', ['fpa', 'core_tactics', 'bv_tactics', 'sat_tactic', 'smt_tactic', 'arith_tactics', 'smtlogic_tactics'], 'tactic/fpa') add_lib('ufbv_tactic', ['normal_forms', 'core_tactics', 'macros', 'smt_tactic', 'rewriter'], 'tactic/ufbv') add_lib('sat_solver', ['solver', 'core_tactics', 'aig_tactic', 'bv_tactics', 'arith_tactics', 'sat_tactic'], 'sat/sat_solver') - add_lib('ackr_tactics', ['bv_tactics', 'smt_tactic', 'aig_tactic', 'sat_solver', 'ackr', 'smtlogic_tactics'], 'tactic/ackr_tactics') + add_lib('smtlogic_tactics', ['ackr', 'sat_solver', 'arith_tactics', 'bv_tactics', 'nlsat_tactic', 'smt_tactic', 'aig_tactic', 'fp', 'muz','qe','nlsat_smt_tactic'], 'tactic/smtlogics') + add_lib('fpa_tactics', ['fpa', 'core_tactics', 'bv_tactics', 'sat_tactic', 'smt_tactic', 'arith_tactics', 'smtlogic_tactics'], 'tactic/fpa') add_lib('portfolio', ['smtlogic_tactics', 'sat_solver', 'ufbv_tactic', 'fpa_tactics', 'aig_tactic', 'fp', 'qe','sls_tactic', 'subpaving_tactic'], 'tactic/portfolio') add_lib('smtparser', ['portfolio'], 'parsers/smt') add_lib('opt', ['smt', 'smtlogic_tactics', 'sls_tactic', 'sat_solver'], 'opt') API_files = ['z3_api.h', 'z3_ast_containers.h', 'z3_algebraic.h', 'z3_polynomial.h', 'z3_rcf.h', 'z3_fixedpoint.h', 'z3_optimization.h', 'z3_interp.h', 'z3_fpa.h'] add_lib('api', ['portfolio', 'smtparser', 'realclosure', 'interp', 'opt'], includes2install=['z3.h', 'z3_v1.h', 'z3_macros.h'] + API_files) - add_exe('shell', ['api', 'sat', 'extra_cmds','opt','ackr_tactics'], exe_name='z3') + add_exe('shell', ['api', 'sat', 'extra_cmds','opt'], exe_name='z3') add_exe('test', ['api', 'fuzzing', 'simplex'], exe_name='test-z3', install=False) _libz3Component = add_dll('api_dll', ['api', 'sat', 'extra_cmds'], 'api/dll', reexports=['api'], diff --git a/src/tactic/ackr/ackr_model_converter.h b/src/tactic/ackr/ackr_model_converter.h index 98c03f381..2c1f0c78c 100644 --- a/src/tactic/ackr/ackr_model_converter.h +++ b/src/tactic/ackr/ackr_model_converter.h @@ -21,4 +21,5 @@ Revision History: model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info); + #endif /* LACKR_MODEL_CONVERTER_H_ */ diff --git a/src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp b/src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp deleted file mode 100644 index a8a76e698..000000000 --- a/src/tactic/ackr_tactics/qfufbv_ackr_tactic.cpp +++ /dev/null @@ -1,153 +0,0 @@ -/*++ -Copyright (c) 2015 Microsoft Corporation - -Module Name: - -qfufbv_ackr_tactic.cpp - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ -#include"tactical.h" -/////////////// -#include"solve_eqs_tactic.h" -#include"simplify_tactic.h" -#include"propagate_values_tactic.h" -#include"bit_blaster_tactic.h" -#include"elim_uncnstr_tactic.h" -#include"max_bv_sharing_tactic.h" -#include"bv_size_reduction_tactic.h" -#include"ctx_simplify_tactic.h" -#include"smt_tactic.h" -/////////////// -#include"model_smt2_pp.h" -#include"cooperate.h" -#include"lackr.h" -#include"ackr_tactics_params.hpp" -#include"ackr_model_converter.h" -/////////////// -#include"inc_sat_solver.h" -#include"qfaufbv_tactic.h" -#include"qfbv_tactic.h" -#include"tactic2solver.h" -/////////////// - - -class qfufbv_ackr_tactic : public tactic { -public: - qfufbv_ackr_tactic(ast_manager& m, params_ref const& p) - : m_m(m) - , m_p(p) - , m_use_sat(false) - {} - - virtual ~qfufbv_ackr_tactic() { } - - virtual void operator()(goal_ref const & g, - goal_ref_buffer & result, - model_converter_ref & mc, - proof_converter_ref & pc, - expr_dependency_ref & core) { - mc = 0; - ast_manager& m(g->m()); - tactic_report report("qfufbv_ackr", *g); - fail_if_unsat_core_generation("qfufbv_ackr", g); - fail_if_proof_generation("qfufbv_ackr", g); - - TRACE("qfufbv_ackr_tactic", g->display(tout << "goal:\n");); - // running implementation - expr_ref_vector flas(m); - const unsigned sz = g->size(); - for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); - scoped_ptr uffree_solver = setup_sat(); - scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas, uffree_solver.get()); - const lbool o = imp->operator()(); - flas.reset(); - // report result - goal_ref resg(alloc(goal, *g, true)); - if (o == l_false) resg->assert_expr(m.mk_false()); - if (o != l_undef) result.push_back(resg.get()); - // report model - if (g->models_enabled() && (o == l_true)) { - model_ref abstr_model = imp->get_model(); - mc = mk_ackr_model_converter(m, imp->get_info(), abstr_model); - } - } - - void updt_params(params_ref const & _p) { - ackr_tactics_params p(_p); - m_use_sat = p.sat_backend(); - } - - virtual void collect_statistics(statistics & st) const { - ackr_params p(m_p); - if (!p.eager()) st.update("lackr-its", m_st.m_it); - st.update("ackr-constraints", m_st.m_ackrs_sz); - } - - virtual void reset_statistics() { m_st.reset(); } - - virtual void cleanup() { } - - virtual tactic* translate(ast_manager& m) { - return alloc(qfufbv_ackr_tactic, m, m_p); - } -private: - ast_manager& m_m; - params_ref m_p; - lackr_stats m_st; - bool m_use_sat; - - solver* setup_sat() { - solver * sat(NULL); - if (m_use_sat) { - tactic_ref t = mk_qfbv_tactic(m_m, m_p); - sat = mk_tactic2solver(m_m, t.get(), m_p); - } - else { - tactic_ref t = mk_qfaufbv_tactic(m_m, m_p); - sat = mk_tactic2solver(m_m, t.get(), m_p); - } - SASSERT(sat != NULL); - sat->set_produce_models(true); - return sat; - } - - -}; - -tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p) { - params_ref simp2_p = p; - simp2_p.set_bool("som", true); - simp2_p.set_bool("pull_cheap_ite", true); - simp2_p.set_bool("push_ite_bv", false); - simp2_p.set_bool("local_ctx", true); - simp2_p.set_uint("local_ctx_limit", 10000000); - - simp2_p.set_bool("ite_extra_rules", true); - simp2_p.set_bool("mul2concat", true); - - params_ref ctx_simp_p; - ctx_simp_p.set_uint("max_depth", 32); - ctx_simp_p.set_uint("max_steps", 5000000); - - tactic * const preamble_t = and_then( - mk_simplify_tactic(m), - mk_propagate_values_tactic(m), - //using_params(mk_ctx_simplify_tactic(m_m), ctx_simp_p), - mk_solve_eqs_tactic(m), - mk_elim_uncnstr_tactic(m), - if_no_proofs(if_no_unsat_cores(mk_bv_size_reduction_tactic(m))), - mk_max_bv_sharing_tactic(m), - using_params(mk_simplify_tactic(m), simp2_p) - ); - - tactic * const actual_tactic = alloc(qfufbv_ackr_tactic, m, p); - return and_then(preamble_t, - cond(mk_is_qfufbv_probe(), actual_tactic, mk_smt_tactic())); -} diff --git a/src/tactic/ackr_tactics/qfufbv_ackr_tactic.h b/src/tactic/ackr_tactics/qfufbv_ackr_tactic.h deleted file mode 100644 index 57b32bf75..000000000 --- a/src/tactic/ackr_tactics/qfufbv_ackr_tactic.h +++ /dev/null @@ -1,28 +0,0 @@ -/*++ -Copyright (c) 2015 Microsoft Corporation - -Module Name: - -qfufbv_ackr_tactic.h - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ - -#ifndef _QFUFBV_ACKR_TACTIC_H_ -#define _QFUFBV_ACKR_TACTIC_H_ -#include"tactical.h" - -tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p); - -/* - ADD_TACTIC("qfufbv_ackr", "A tactic for solving QF_UFBV based on Ackermannization.", "mk_qfufbv_ackr_tactic(m, p)") -*/ - -#endif - diff --git a/src/tactic/smtlogics/qfufbv_ackr_model_converter.cpp b/src/tactic/smtlogics/qfufbv_ackr_model_converter.cpp new file mode 100644 index 000000000..4d20631b1 --- /dev/null +++ b/src/tactic/smtlogics/qfufbv_ackr_model_converter.cpp @@ -0,0 +1,22 @@ +/*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + qfufbv_ackr_model_converter.cpp + + Abstract: + + + Author: + + Mikolas Janota (MikolasJanota) + + Revision History: +--*/ +#include"qfufbv_ackr_model_converter.h" +#include"ackr_model_converter.h" + +model_converter * mk_qfufbv_ackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model) { + return mk_ackr_model_converter(m, info, abstr_model); +} diff --git a/src/tactic/smtlogics/qfufbv_ackr_model_converter.h b/src/tactic/smtlogics/qfufbv_ackr_model_converter.h new file mode 100644 index 000000000..a361a78e0 --- /dev/null +++ b/src/tactic/smtlogics/qfufbv_ackr_model_converter.h @@ -0,0 +1,25 @@ + /*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + qfufbv_ackr_model_converter.h + + Abstract: + + + Author: + + Mikolas Janota (MikolasJanota) + + Revision History: + --*/ +#ifndef QFUFBV_ACKR_MODEL_CONVERTER_H_ +#define QFUFBV_ACKR_MODEL_CONVERTER_H_ + +#include"model_converter.h" +#include"ackr_info.h" + +model_converter * mk_qfufbv_ackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); + +#endif /* QFUFBV_ACKR_MODEL_CONVERTER_H_ */ diff --git a/src/tactic/smtlogics/qfufbv_tactic.cpp b/src/tactic/smtlogics/qfufbv_tactic.cpp index 816d69b1f..526d72faa 100644 --- a/src/tactic/smtlogics/qfufbv_tactic.cpp +++ b/src/tactic/smtlogics/qfufbv_tactic.cpp @@ -12,6 +12,7 @@ Abstract: Author: Leonardo (leonardo) 2012-02-27 + Mikolas Janota Notes: @@ -26,27 +27,164 @@ Notes: #include"bv_size_reduction_tactic.h" #include"reduce_args_tactic.h" #include"qfbv_tactic.h" +#include"qfufbv_tactic_params.hpp" +/////////////// +#include"model_smt2_pp.h" +#include"cooperate.h" +#include"lackr.h" +#include"qfufbv_ackr_model_converter.h" +/////////////// +#include"inc_sat_solver.h" +#include"qfaufbv_tactic.h" +#include"qfbv_tactic.h" +#include"tactic2solver.h" +/////////////// + +class qfufbv_ackr_tactic : public tactic { +public: + qfufbv_ackr_tactic(ast_manager& m, params_ref const& p) + : m_m(m) + , m_p(p) + , m_use_sat(false) + {} + + virtual ~qfufbv_ackr_tactic() { } + + virtual void operator()(goal_ref const & g, + goal_ref_buffer & result, + model_converter_ref & mc, + proof_converter_ref & pc, + expr_dependency_ref & core) { + mc = 0; + ast_manager& m(g->m()); + tactic_report report("qfufbv_ackr", *g); + fail_if_unsat_core_generation("qfufbv_ackr", g); + fail_if_proof_generation("qfufbv_ackr", g); + + TRACE("qfufbv_ackr_tactic", g->display(tout << "goal:\n");); + // running implementation + expr_ref_vector flas(m); + const unsigned sz = g->size(); + for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); + scoped_ptr uffree_solver = setup_sat(); + scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas, uffree_solver.get()); + const lbool o = imp->operator()(); + flas.reset(); + // report result + goal_ref resg(alloc(goal, *g, true)); + if (o == l_false) resg->assert_expr(m.mk_false()); + if (o != l_undef) result.push_back(resg.get()); + // report model + if (g->models_enabled() && (o == l_true)) { + model_ref abstr_model = imp->get_model(); + mc = mk_qfufbv_ackr_model_converter(m, imp->get_info(), abstr_model); + } + } + + void updt_params(params_ref const & _p) { + qfufbv_tactic_params p(_p); + m_use_sat = p.sat_backend(); + } + + virtual void collect_statistics(statistics & st) const { + ackr_params p(m_p); + if (!p.eager()) st.update("lackr-its", m_st.m_it); + st.update("ackr-constraints", m_st.m_ackrs_sz); + } + + virtual void reset_statistics() { m_st.reset(); } + + virtual void cleanup() { } + + virtual tactic* translate(ast_manager& m) { + return alloc(qfufbv_ackr_tactic, m, m_p); + } +private: + ast_manager& m_m; + params_ref m_p; + lackr_stats m_st; + bool m_use_sat; + + solver* setup_sat() { + solver * sat(NULL); + if (m_use_sat) { + tactic_ref t = mk_qfbv_tactic(m_m, m_p); + sat = mk_tactic2solver(m_m, t.get(), m_p); + } + else { + tactic_ref t = mk_qfaufbv_tactic(m_m, m_p); + sat = mk_tactic2solver(m_m, t.get(), m_p); + } + SASSERT(sat != NULL); + sat->set_produce_models(true); + return sat; + } + + +}; + +tactic * mk_qfufbv_preamble1(ast_manager & m, params_ref const & p) { + params_ref simp2_p = p; + simp2_p.set_bool("pull_cheap_ite", true); + simp2_p.set_bool("push_ite_bv", false); + simp2_p.set_bool("local_ctx", true); + simp2_p.set_uint("local_ctx_limit", 10000000); + + simp2_p.set_bool("ite_extra_rules", true); + simp2_p.set_bool("mul2concat", true); + + params_ref ctx_simp_p; + ctx_simp_p.set_uint("max_depth", 32); + ctx_simp_p.set_uint("max_steps", 5000000); + + return and_then( + mk_simplify_tactic(m), + mk_propagate_values_tactic(m), + //using_params(mk_ctx_simplify_tactic(m_m), ctx_simp_p), + mk_solve_eqs_tactic(m), + mk_elim_uncnstr_tactic(m), + if_no_proofs(if_no_unsat_cores(mk_bv_size_reduction_tactic(m))), + mk_max_bv_sharing_tactic(m), + using_params(mk_simplify_tactic(m), simp2_p) + ); +} + +tactic * mk_qfufbv_preamble(ast_manager & m, params_ref const & p) { + params_ref main_p; + main_p.set_bool("elim_and", true); + main_p.set_bool("blast_distinct", true); + + return and_then(mk_simplify_tactic(m), + mk_propagate_values_tactic(m), + mk_solve_eqs_tactic(m), + mk_elim_uncnstr_tactic(m), + if_no_proofs(if_no_unsat_cores(mk_reduce_args_tactic(m))), + if_no_proofs(if_no_unsat_cores(mk_bv_size_reduction_tactic(m))), + mk_max_bv_sharing_tactic(m) + ); +} tactic * mk_qfufbv_tactic(ast_manager & m, params_ref const & p) { params_ref main_p; main_p.set_bool("elim_and", true); main_p.set_bool("blast_distinct", true); - tactic * preamble_st = and_then(mk_simplify_tactic(m), - mk_propagate_values_tactic(m), - mk_solve_eqs_tactic(m), - mk_elim_uncnstr_tactic(m), - if_no_proofs(if_no_unsat_cores(mk_reduce_args_tactic(m))), - if_no_proofs(if_no_unsat_cores(mk_bv_size_reduction_tactic(m))), - mk_max_bv_sharing_tactic(m) - ); + tactic * const preamble_st = mk_qfufbv_preamble(m, p); tactic * st = using_params(and_then(preamble_st, - cond(mk_is_qfbv_probe(), - mk_qfbv_tactic(m), - mk_smt_tactic())), - main_p); + cond(mk_is_qfbv_probe(), + mk_qfbv_tactic(m), + mk_smt_tactic())), + main_p); st->updt_params(p); return st; } + +tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p) { + tactic * const preamble_t = mk_qfufbv_preamble(m, p); + + tactic * const actual_tactic = alloc(qfufbv_ackr_tactic, m, p); + return and_then(preamble_t, + cond(mk_is_qfufbv_probe(), actual_tactic, mk_smt_tactic())); +} \ No newline at end of file diff --git a/src/tactic/smtlogics/qfufbv_tactic.h b/src/tactic/smtlogics/qfufbv_tactic.h index ceb125517..1f1431d70 100644 --- a/src/tactic/smtlogics/qfufbv_tactic.h +++ b/src/tactic/smtlogics/qfufbv_tactic.h @@ -25,8 +25,11 @@ class tactic; tactic * mk_qfufbv_tactic(ast_manager & m, params_ref const & p = params_ref()); +tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p); + /* ADD_TACTIC("qfufbv", "builtin strategy for solving QF_UFBV problems.", "mk_qfufbv_tactic(m, p)") + ADD_TACTIC("qfufbv_ackr", "A tactic for solving QF_UFBV based on Ackermannization.", "mk_qfufbv_ackr_tactic(m, p)") */ #endif diff --git a/src/tactic/ackr_tactics/ackr_tactics.pyg b/src/tactic/smtlogics/qfufbv_tactic_params.pyg similarity index 77% rename from src/tactic/ackr_tactics/ackr_tactics.pyg rename to src/tactic/smtlogics/qfufbv_tactic_params.pyg index 8f391bc91..6c4bd5b8e 100644 --- a/src/tactic/ackr_tactics/ackr_tactics.pyg +++ b/src/tactic/smtlogics/qfufbv_tactic_params.pyg @@ -1,5 +1,6 @@ -def_module_params('ackr_tactics', +def_module_params('ackermannization', description='tactics based on solving UF-theories via ackermannization (see also ackr module)', + class_name='qfufbv_tactic_params', export=True, params=( ('sat_backend', BOOL, False, 'use SAT rather than SMT in qfufbv_ackr_tactic'), From f3240024e7ba129e89e390e7a2e46aae411f9cd0 Mon Sep 17 00:00:00 2001 From: mikolas Date: Wed, 3 Feb 2016 17:26:58 +0000 Subject: [PATCH 28/31] Further refactoring ackermannization. --- src/ackermannization/ackermannization.pyg | 7 + .../ackermannization_params.pyg | 7 + .../ackermannize_bv_model_converter.cpp | 22 + .../ackermannize_bv_model_converter.h | 25 ++ .../ackermannize_bv_tactic.cpp | 97 +++++ src/ackermannization/ackermannize_bv_tactic.h | 28 ++ .../ackermannize_bv_tactic_params.pyg | 8 + src/ackermannization/ackr_bound_probe.cpp | 77 ++++ src/ackermannization/ackr_bound_probe.h | 29 ++ src/ackermannization/ackr_helper.cpp | 29 ++ src/ackermannization/ackr_helper.h | 69 ++++ src/ackermannization/ackr_info.h | 109 +++++ src/ackermannization/ackr_model_converter.cpp | 153 +++++++ src/ackermannization/ackr_model_converter.h | 25 ++ src/ackermannization/lackr.cpp | 290 ++++++++++++++ src/ackermannization/lackr.h | 123 ++++++ .../lackr_model_constructor.cpp | 378 ++++++++++++++++++ .../lackr_model_constructor.h | 61 +++ .../lackr_model_converter_lazy.cpp | 59 +++ .../lackr_model_converter_lazy.h | 25 ++ 20 files changed, 1621 insertions(+) create mode 100644 src/ackermannization/ackermannization.pyg create mode 100644 src/ackermannization/ackermannization_params.pyg create mode 100644 src/ackermannization/ackermannize_bv_model_converter.cpp create mode 100644 src/ackermannization/ackermannize_bv_model_converter.h create mode 100644 src/ackermannization/ackermannize_bv_tactic.cpp create mode 100644 src/ackermannization/ackermannize_bv_tactic.h create mode 100644 src/ackermannization/ackermannize_bv_tactic_params.pyg create mode 100644 src/ackermannization/ackr_bound_probe.cpp create mode 100644 src/ackermannization/ackr_bound_probe.h create mode 100644 src/ackermannization/ackr_helper.cpp create mode 100644 src/ackermannization/ackr_helper.h create mode 100644 src/ackermannization/ackr_info.h create mode 100644 src/ackermannization/ackr_model_converter.cpp create mode 100644 src/ackermannization/ackr_model_converter.h create mode 100644 src/ackermannization/lackr.cpp create mode 100644 src/ackermannization/lackr.h create mode 100644 src/ackermannization/lackr_model_constructor.cpp create mode 100644 src/ackermannization/lackr_model_constructor.h create mode 100644 src/ackermannization/lackr_model_converter_lazy.cpp create mode 100644 src/ackermannization/lackr_model_converter_lazy.h diff --git a/src/ackermannization/ackermannization.pyg b/src/ackermannization/ackermannization.pyg new file mode 100644 index 000000000..fe7ac7b63 --- /dev/null +++ b/src/ackermannization/ackermannization.pyg @@ -0,0 +1,7 @@ +def_module_params('ackermannization', + description='solving UF via ackermannization', + export=True, + params=( + ('eager', BOOL, True, 'eagerly instantiate all congruence rules'), + )) + diff --git a/src/ackermannization/ackermannization_params.pyg b/src/ackermannization/ackermannization_params.pyg new file mode 100644 index 000000000..fe7ac7b63 --- /dev/null +++ b/src/ackermannization/ackermannization_params.pyg @@ -0,0 +1,7 @@ +def_module_params('ackermannization', + description='solving UF via ackermannization', + export=True, + params=( + ('eager', BOOL, True, 'eagerly instantiate all congruence rules'), + )) + diff --git a/src/ackermannization/ackermannize_bv_model_converter.cpp b/src/ackermannization/ackermannize_bv_model_converter.cpp new file mode 100644 index 000000000..d47735a56 --- /dev/null +++ b/src/ackermannization/ackermannize_bv_model_converter.cpp @@ -0,0 +1,22 @@ +/*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackermannize_bv_model_converter.cpp + + Abstract: + + + Author: + + Mikolas Janota (MikolasJanota) + + Revision History: +--*/ +#include"ackr_model_converter.h" +#include"ackermannize_bv_model_converter.h" + +model_converter * mk_ackermannize_bv_model_converter(ast_manager & m, const ackr_info_ref& info) { + return mk_ackr_model_converter(m, info); +} diff --git a/src/ackermannization/ackermannize_bv_model_converter.h b/src/ackermannization/ackermannize_bv_model_converter.h new file mode 100644 index 000000000..e51792bad --- /dev/null +++ b/src/ackermannization/ackermannize_bv_model_converter.h @@ -0,0 +1,25 @@ + /*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackermannize_bv_model_converter.h + + Abstract: + + + Author: + + Mikolas Janota (MikolasJanota) + + Revision History: + --*/ +#ifndef ACKERMANNIZE_BV_MODEL_CONVERTER_H_ +#define ACKERMANNIZE_BV_MODEL_CONVERTER_H_ + +#include"model_converter.h" +#include"ackr_info.h" + +model_converter * mk_ackermannize_bv_model_converter(ast_manager & m, const ackr_info_ref& info); + +#endif /* ACKERMANNIZE_BV_MODEL_CONVERTER_H_ */ diff --git a/src/ackermannization/ackermannize_bv_tactic.cpp b/src/ackermannization/ackermannize_bv_tactic.cpp new file mode 100644 index 000000000..310269fec --- /dev/null +++ b/src/ackermannization/ackermannize_bv_tactic.cpp @@ -0,0 +1,97 @@ +/*++ +Copyright (c) 2016 Microsoft Corporation + +Module Name: + +ackermannize_bv_tactic.cpp + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#include"ackermannize_bv_tactic.h" +#include"tactical.h" +#include"lackr.h" +#include"model_smt2_pp.h" +#include"ackermannize_bv_tactic_params.hpp" +#include"ackermannize_bv_model_converter.h" + + +class ackermannize_bv_tactic : public tactic { +public: + ackermannize_bv_tactic(ast_manager& m, params_ref const& p) + : m_m(m) + , m_p(p) + {} + + virtual ~ackermannize_bv_tactic() { } + + virtual void operator()(goal_ref const & g, + goal_ref_buffer & result, + model_converter_ref & mc, + proof_converter_ref & pc, + expr_dependency_ref & core) { + mc = 0; + ast_manager& m(g->m()); + tactic_report report("ackermannize", *g); + fail_if_unsat_core_generation("ackermannize", g); + fail_if_proof_generation("ackermannize", g); + + expr_ref_vector flas(m); + const unsigned sz = g->size(); + for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); + scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas, NULL); + flas.reset(); + // mk result + goal_ref resg(alloc(goal, *g, true)); + const bool success = imp->mk_ackermann(resg, m_lemma_limit); + if (!success) { // Just pass on the input unchanged + result.reset(); + result.push_back(g.get()); + mc = 0; + pc = 0; + core = 0; + return; + } + result.push_back(resg.get()); + // report model + if (g->models_enabled()) { + mc = mk_ackermannize_bv_model_converter(m, imp->get_info()); + } + + resg->inc_depth(); + TRACE("ackermannize", resg->display(tout);); + SASSERT(resg->is_well_sorted()); + } + + + void updt_params(params_ref const & _p) { + ackermannize_bv_tactic_params p(_p); + m_lemma_limit = p.div0_ackermann_limit(); + } + + virtual void collect_statistics(statistics & st) const { + st.update("ackr-constraints", m_st.m_ackrs_sz); + } + + virtual void reset_statistics() { m_st.reset(); } + + virtual void cleanup() { } + + virtual tactic* translate(ast_manager& m) { + return alloc(ackermannize_bv_tactic, m, m_p); + } +private: + ast_manager& m_m; + params_ref m_p; + lackr_stats m_st; + double m_lemma_limit; +}; + +tactic * mk_ackermannize_bv_tactic(ast_manager & m, params_ref const & p) { + return alloc(ackermannize_bv_tactic, m, p); +} diff --git a/src/ackermannization/ackermannize_bv_tactic.h b/src/ackermannization/ackermannize_bv_tactic.h new file mode 100644 index 000000000..3e3d5636c --- /dev/null +++ b/src/ackermannization/ackermannize_bv_tactic.h @@ -0,0 +1,28 @@ +/*++ +Copyright (c) 2016 Microsoft Corporation + +Module Name: + +ackermannize_bv_tactic.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ + +#ifndef _ACKERMANNIZE_TACTIC_H_ +#define _ACKERMANNIZE_TACTIC_H_ +#include"tactical.h" + +tactic * mk_ackermannize_bv_tactic(ast_manager & m, params_ref const & p); + +/* + ADD_TACTIC("ackermannize_bv", "A tactic for performing full Ackermannization on bv instances.", "mk_ackermannize_bv_tactic(m, p)") +*/ + +#endif + diff --git a/src/ackermannization/ackermannize_bv_tactic_params.pyg b/src/ackermannization/ackermannize_bv_tactic_params.pyg new file mode 100644 index 000000000..bd17dcaff --- /dev/null +++ b/src/ackermannization/ackermannize_bv_tactic_params.pyg @@ -0,0 +1,8 @@ +def_module_params(module_name='rewriter', + class_name='ackermannize_bv_tactic_params', + export=True, + params=( + ("div0_ackermann_limit", UINT, 1000, "a bound for number of congruence Ackermann lemmas for div0 modelling"), + ) + ) + diff --git a/src/ackermannization/ackr_bound_probe.cpp b/src/ackermannization/ackr_bound_probe.cpp new file mode 100644 index 000000000..5cb8e9448 --- /dev/null +++ b/src/ackermannization/ackr_bound_probe.cpp @@ -0,0 +1,77 @@ +/*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackr_bound_probe.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ +#include"ackr_helper.h" +#include"ackr_bound_probe.h" +#include"ast_smt2_pp.h" + +/* + For each function f, calculate the number of its occurrences o_f and compute "o_f choose 2". + The probe then sums up these for all functions. + This upper bound might be crude because some congruence lemmas trivially simplify to true. +*/ +class ackr_bound_probe : public probe { + struct proc { + typedef ackr_helper::fun2terms_map fun2terms_map; + typedef ackr_helper::app_set app_set; + ast_manager& m_m; + fun2terms_map m_fun2terms; // a map from functions to occurrences + ackr_helper m_ackr_helper; + + proc(ast_manager & m) : m_m(m), m_ackr_helper(m) { } + + ~proc() { + fun2terms_map::iterator it = m_fun2terms.begin(); + const fun2terms_map::iterator end = m_fun2terms.end(); + for (; it != end; ++it) dealloc(it->get_value()); + } + + void operator()(quantifier *) {} + void operator()(var *) {} + void operator()(app * a) { + if (a->get_num_args() == 0) return; + if (!m_ackr_helper.should_ackermannize(a)) return; + func_decl* const fd = a->get_decl(); + app_set* ts = 0; + if (!m_fun2terms.find(fd, ts)) { + ts = alloc(app_set); + m_fun2terms.insert(fd, ts); + } + ts->insert(a); + } + }; + +public: + ackr_bound_probe() {} + + virtual result operator()(goal const & g) { + proc p(g.m()); + unsigned sz = g.size(); + expr_fast_mark1 visited; + for (unsigned i = 0; i < sz; i++) { + for_each_expr_core(p, visited, g.form(i)); + } + const double total = ackr_helper::calculate_lemma_bound(p.m_fun2terms); + TRACE("ackr_bound_probe", tout << "total=" << total << std::endl;); + return result(total); + } + + inline static unsigned n_choose_2(unsigned n) { return n & 1 ? (n * (n >> 1)) : (n >> 1) * (n - 1); } +}; + +probe * mk_ackr_bound_probe() { + return alloc(ackr_bound_probe); +} diff --git a/src/ackermannization/ackr_bound_probe.h b/src/ackermannization/ackr_bound_probe.h new file mode 100644 index 000000000..6c0a66605 --- /dev/null +++ b/src/ackermannization/ackr_bound_probe.h @@ -0,0 +1,29 @@ + /*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackr_bound_probe.h + + Abstract: + + A probe to give an upper bound of Ackermann congruence lemmas that a formula might generate. + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef ACKR_BOUND_PROBE_H_ +#define ACKR_BOUND_PROBE_H_ + +#include"probe.h" + +probe * mk_ackr_bound_probe(); + +/* + ADD_PROBE("ackr-bound-probe", "A probe to give an upper bound of Ackermann congruence lemmas that a formula might generate.", "mk_ackr_bound_probe()") +*/ + +#endif /* ACKR_BOUND_PROBE_H_ */ diff --git a/src/ackermannization/ackr_helper.cpp b/src/ackermannization/ackr_helper.cpp new file mode 100644 index 000000000..803fc3c54 --- /dev/null +++ b/src/ackermannization/ackr_helper.cpp @@ -0,0 +1,29 @@ +/*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackr_helper.cpp + + Abstract: + + + Author: + + Mikolas Janota (MikolasJanota) + + Revision History: +--*/ +#include"ackr_helper.h" + +double ackr_helper::calculate_lemma_bound(ackr_helper::fun2terms_map& occurrences) { + fun2terms_map::iterator it = occurrences.begin(); + const fun2terms_map::iterator end = occurrences.end(); + double total = 0; + for (; it != end; ++it) { + const unsigned fsz = it->m_value->size(); + const double n2 = n_choose_2_chk(fsz); + total += n2; + } + return total; +} diff --git a/src/ackermannization/ackr_helper.h b/src/ackermannization/ackr_helper.h new file mode 100644 index 000000000..5c572907e --- /dev/null +++ b/src/ackermannization/ackr_helper.h @@ -0,0 +1,69 @@ + /*++ + Copyright (c) 2016 Microsoft Corporation + + Module Name: + + ackr_helper.h + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef ACKR_HELPER_H_ +#define ACKR_HELPER_H_ + +#include"bv_decl_plugin.h" + +class ackr_helper { + public: + typedef obj_hashtable app_set; + typedef obj_map fun2terms_map; + + ackr_helper(ast_manager& m) : m_bvutil(m) {} + + /** + \brief Determines if a given function should be Ackermannized. + + This includes all uninterpreted functions but also "special" functions, e.g. OP_BSMOD0, + which are not marked as uninterpreted but effectively are. + */ + inline bool should_ackermannize(app const * a) const { + if (a->get_family_id() == m_bvutil.get_family_id()) { + switch (a->get_decl_kind()) { + case OP_BSDIV0: + case OP_BUDIV0: + case OP_BSREM0: + case OP_BUREM0: + case OP_BSMOD0: + return true; + default: + return is_uninterp(a); + } + } + return (is_uninterp(a)); + } + + inline bv_util& bvutil() { return m_bvutil; } + + /** + \brief Calculates an upper bound for congruence lemmas given a map of function of occurrences. + */ + static double calculate_lemma_bound(fun2terms_map& occurrences); + + /** \brief Calculate n choose 2. **/ + inline static unsigned n_choose_2(unsigned n) { return n & 1 ? (n * (n >> 1)) : (n >> 1) * (n - 1); } + + /** \brief Calculate n choose 2 guarded for overflow. Returns infinity if unsafe. **/ + inline static double n_choose_2_chk(unsigned n) { + SASSERT(std::numeric_limits().max() & 32); + return n & (1 << 16) ? std::numeric_limits().infinity() : n_choose_2(n); + } + private: + bv_util m_bvutil; +}; +#endif /* ACKR_HELPER_H_6475 */ diff --git a/src/ackermannization/ackr_info.h b/src/ackermannization/ackr_info.h new file mode 100644 index 000000000..703f1f3d5 --- /dev/null +++ b/src/ackermannization/ackr_info.h @@ -0,0 +1,109 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +ackr_info.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#ifndef ACKR_INFO_H_ +#define ACKR_INFO_H_ + +#include"obj_hashtable.h" +#include"ast.h" +#include"ref.h" +#include"expr_replacer.h" + +/** \brief + Information about how a formula is being converted into + a formula without uninterpreted function symbols via ackermannization. + + The intended use is that new terms are added via set_abstr. + Once all terms are abstracted, call seal. + The function abstract may only be called when sealed. + + The class enables reference counting. +**/ +class ackr_info { + public: + ackr_info(ast_manager& m) + : m_m(m) + , m_consts(m) + , m_er(mk_default_expr_replacer(m)) + , m_subst(m_m) + , m_ref_count(0) + , m_sealed(false) + {} + + virtual ~ackr_info() { + m_consts.reset(); + } + + inline void set_abstr(app* term, app* c) { + SASSERT(!m_sealed); + SASSERT(c); + m_t2c.insert(term,c); + m_c2t.insert(c->get_decl(),term); + m_subst.insert(term, c); + m_consts.push_back(c); + } + + inline void abstract(expr * e, expr_ref& res) { + SASSERT(m_sealed); + (*m_er)(e, res); + } + + inline app* find_term(func_decl* c) const { + app * rv = 0; + m_c2t.find(c,rv); + return rv; + } + + inline app* get_abstr(app* term) const { + app * const rv = m_t2c.find(term); + SASSERT(rv); + return rv; + } + + inline void seal() { + m_sealed=true; + m_er->set_substitution(&m_subst); + } + + // + // Reference counting + // + void inc_ref() { ++m_ref_count; } + void dec_ref() { + --m_ref_count; + if (m_ref_count == 0) { + dealloc(this); + } + } + private: + typedef obj_map t2ct; + typedef obj_map c2tt; + ast_manager& m_m; + + t2ct m_t2c; // terms to constants + c2tt m_c2t; // constants to terms (inversion of m_t2c) + expr_ref_vector m_consts; // the constants introduced during abstraction + + // replacer and substitution used to compute abstractions + scoped_ptr m_er; + expr_substitution m_subst; + + unsigned m_ref_count; // reference counting + bool m_sealed; // debugging +}; + +typedef ref ackr_info_ref; + +#endif /* ACKR_INFO_H_ */ diff --git a/src/ackermannization/ackr_model_converter.cpp b/src/ackermannization/ackr_model_converter.cpp new file mode 100644 index 000000000..7967cdb63 --- /dev/null +++ b/src/ackermannization/ackr_model_converter.cpp @@ -0,0 +1,153 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +ackr_model_converter.cpp + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#include"ackr_model_converter.h" +#include"model_evaluator.h" +#include"ast_smt2_pp.h" +#include"ackr_info.h" + + +class ackr_model_converter : public model_converter { +public: + ackr_model_converter(ast_manager & m, + const ackr_info_ref& info, + model_ref& abstr_model) + : m(m) + , info(info) + , abstr_model(abstr_model) + , fixed_model(true) + { } + + ackr_model_converter(ast_manager & m, + const ackr_info_ref& info) + : m(m) + , info(info) + , fixed_model(false) + { } + + virtual ~ackr_model_converter() { } + + virtual void operator()(model_ref & md, unsigned goal_idx) { + SASSERT(goal_idx == 0); + SASSERT(!fixed_model || md.get() == 0 || (!md->get_num_constants() && !md->get_num_functions())); + model_ref& old_model = fixed_model ? abstr_model : md; + SASSERT(old_model.get()); + model * new_model = alloc(model, m); + convert(old_model.get(), new_model); + md = new_model; + } + + virtual void operator()(model_ref & md) { operator()(md, 0); } + + //void display(std::ostream & out); + + virtual model_converter * translate(ast_translation & translator) {NOT_IMPLEMENTED_YET();} +protected: + ast_manager& m; + const ackr_info_ref info; + model_ref abstr_model; + bool fixed_model; + void convert(model * source, model * destination); + void add_entry(model_evaluator & evaluator, + app* term, expr* value, + obj_map& interpretations); + void convert_sorts(model * source, model * destination); + void convert_constants(model * source, model * destination); +}; + +void ackr_model_converter::convert(model * source, model * destination) { + //SASSERT(source->get_num_functions() == 0); + for (unsigned i = 0; i < source->get_num_functions(); i++) { + func_decl * const fd = source->get_function(i); + func_interp * const fi = source->get_func_interp(fd); + destination->register_decl(fd, fi); + } + convert_constants(source,destination); + convert_sorts(source,destination); +} + +void ackr_model_converter::convert_constants(model * source, model * destination) { + TRACE("ackr_model", tout << "converting constants\n";); + obj_map interpretations; + model_evaluator evaluator(*source); + for (unsigned i = 0; i < source->get_num_constants(); i++) { + func_decl * const c = source->get_constant(i); + app * const term = info->find_term(c); + expr * value = source->get_const_interp(c); + if(!term) { + destination->register_decl(c, value); + } else { + add_entry(evaluator, term, value, interpretations); + } + } + + obj_map::iterator e = interpretations.end(); + for (obj_map::iterator i = interpretations.begin(); + i!=e; ++i) { + func_decl* const fd = i->m_key; + func_interp* const fi = i->get_value(); + fi->set_else(m.get_some_value(fd->get_range())); + destination->register_decl(fd, fi); + } +} + +void ackr_model_converter::add_entry(model_evaluator & evaluator, + app* term, expr* value, + obj_map& interpretations) { + TRACE("ackr_model", tout << "add_entry" + << mk_ismt2_pp(term, m, 2) + << "->" + << mk_ismt2_pp(value, m, 2) << "\n"; + ); + + func_interp* fi = 0; + func_decl * const declaration = term->get_decl(); + const unsigned sz = declaration->get_arity(); + SASSERT(sz == term->get_num_args()); + if (!interpretations.find(declaration, fi)) { + fi = alloc(func_interp,m,sz); + interpretations.insert(declaration, fi); + } + expr_ref_vector args(m); + for (unsigned gi = 0; gi < sz; ++gi) { + expr * const arg = term->get_arg(gi); + expr_ref aarg(m); + info->abstract(arg, aarg); + expr_ref arg_value(m); + evaluator(aarg,arg_value); + args.push_back(arg_value); + } + if (fi->get_entry(args.c_ptr()) == 0) { + fi->insert_new_entry(args.c_ptr(), value); + } else { + TRACE("ackr_model", tout << "entry already present\n";); + } +} + +void ackr_model_converter::convert_sorts(model * source, model * destination) { + for (unsigned i = 0; i < source->get_num_uninterpreted_sorts(); i++) { + sort * const s = source->get_uninterpreted_sort(i); + ptr_vector u = source->get_universe(s); + destination->register_usort(s, u.size(), u.c_ptr()); + } +} + +model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info) { + return alloc(ackr_model_converter, m, info); +} + +model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model) { + return alloc(ackr_model_converter, m, info, abstr_model); +} diff --git a/src/ackermannization/ackr_model_converter.h b/src/ackermannization/ackr_model_converter.h new file mode 100644 index 000000000..2c1f0c78c --- /dev/null +++ b/src/ackermannization/ackr_model_converter.h @@ -0,0 +1,25 @@ +/*++ +Copyright (c) 2015 Microsoft Corporation + +Module Name: + +ackr_model_converter.h + +Abstract: + +Author: + +Mikolas Janota + +Revision History: +--*/ +#ifndef ACKR_MODEL_CONVERTER_H_ +#define ACKR_MODEL_CONVERTER_H_ + +#include"model_converter.h" +#include"ackr_info.h" + +model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); +model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info); + +#endif /* LACKR_MODEL_CONVERTER_H_ */ diff --git a/src/ackermannization/lackr.cpp b/src/ackermannization/lackr.cpp new file mode 100644 index 000000000..cc98545de --- /dev/null +++ b/src/ackermannization/lackr.cpp @@ -0,0 +1,290 @@ +/*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ + +#include"lackr.h" +#include"ackermannization_params.hpp" +#include"tactic.h" +#include"lackr_model_constructor.h" +#include"ackr_info.h" +#include"for_each_expr.h" +#include"model_smt2_pp.h" + +lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas, + solver * uffree_solver) + : m_m(m) + , m_p(p) + , m_formulas(formulas) + , m_abstr(m) + , m_sat(uffree_solver) + , m_ackr_helper(m) + , m_simp(m) + , m_ackrs(m) + , m_st(st) + , m_is_init(false) +{ + updt_params(p); +} + +void lackr::updt_params(params_ref const & _p) { + ackermannization_params p(_p); + m_eager = p.eager(); +} + +lackr::~lackr() { + const fun2terms_map::iterator e = m_fun2terms.end(); + for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { + dealloc(i->get_value()); + } +} + +lbool lackr::operator() () { + SASSERT(m_sat); + init(); + const lbool rv = m_eager ? eager() : lazy(); + if (rv == l_true) m_sat->get_model(m_model); + CTRACE("lackr", rv == l_true, + model_smt2_pp(tout << "abstr_model(\n", m_m, *(m_model.get()), 2); tout << ")\n"; ); + return rv; +} + +bool lackr::mk_ackermann(/*out*/goal_ref& g, double lemmas_upper_bound) { + if (lemmas_upper_bound <= 0) return false; + init(); + if (lemmas_upper_bound != std::numeric_limits::infinity()) { + const double lemmas_bound = ackr_helper::calculate_lemma_bound(m_fun2terms); + if (lemmas_bound > lemmas_upper_bound) return false; + } + eager_enc(); + for (unsigned i = 0; i < m_abstr.size(); ++i) g->assert_expr(m_abstr.get(i)); + for (unsigned i = 0; i < m_ackrs.size(); ++i) g->assert_expr(m_ackrs.get(i)); + return true; +} + +void lackr::init() { + SASSERT(!m_is_init); + m_is_init = true; + params_ref simp_p(m_p); + m_simp.updt_params(simp_p); + m_info = alloc(ackr_info, m_m); + collect_terms(); + abstract(); +} + +// +// Introduce ackermann lemma for the two given terms. +// +bool lackr::ackr(app * const t1, app * const t2) { + TRACE("lackr", tout << "ackr " + << mk_ismt2_pp(t1, m_m, 2) << " , " << mk_ismt2_pp(t2, m_m, 2) << "\n";); + const unsigned sz = t1->get_num_args(); + SASSERT(t2->get_num_args() == sz); + expr_ref_vector eqs(m_m); + for (unsigned i = 0; i < sz; ++i) { + expr * const arg1 = t1->get_arg(i); + expr * const arg2 = t2->get_arg(i); + if (arg1 == arg2) continue; // quickly skip syntactically equal + if (m_ackr_helper.bvutil().is_numeral(arg1) && m_ackr_helper.bvutil().is_numeral(arg2)) { + // quickly abort if there are two distinct numerals + SASSERT(arg1 != arg2); + TRACE("lackr", tout << "never eq\n";); + return false; + } + eqs.push_back(m_m.mk_eq(arg1, arg2)); + } + app * const a1 = m_info->get_abstr(t1); + app * const a2 = m_info->get_abstr(t2); + SASSERT(a1 && a2); + TRACE("lackr", tout << "abstr1 " << mk_ismt2_pp(a1, m_m, 2) << "\n";); + TRACE("lackr", tout << "abstr2 " << mk_ismt2_pp(a2, m_m, 2) << "\n";); + expr_ref lhs(m_m.mk_and(eqs.size(), eqs.c_ptr()), m_m); + TRACE("lackr", tout << "ackr constr lhs" << mk_ismt2_pp(lhs, m_m, 2) << "\n";); + expr_ref rhs(m_m.mk_eq(a1, a2),m_m); + TRACE("lackr", tout << "ackr constr rhs" << mk_ismt2_pp(rhs, m_m, 2) << "\n";); + expr_ref cg(m_m.mk_implies(lhs, rhs), m_m); + TRACE("lackr", tout << "ackr constr" << mk_ismt2_pp(cg, m_m, 2) << "\n";); + expr_ref cga(m_m); + m_info->abstract(cg, cga); // constraint needs abstraction due to nested applications + m_simp(cga); + TRACE("lackr", tout << "ackr constr abs:" << mk_ismt2_pp(cga, m_m, 2) << "\n";); + if (m_m.is_true(cga)) return false; + m_st.m_ackrs_sz++; + m_ackrs.push_back(cga); + return true; +} + +// +// Introduce the ackermann lemma for each pair of terms. +// +void lackr::eager_enc() { + TRACE("lackr", tout << "#funs: " << m_fun2terms.size() << std::endl;); + const fun2terms_map::iterator e = m_fun2terms.end(); + for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { + checkpoint(); + func_decl* const fd = i->m_key; + app_set * const ts = i->get_value(); + const app_set::iterator r = ts->end(); + for (app_set::iterator j = ts->begin(); j != r; ++j) { + app_set::iterator k = j; + ++k; + for (; k != r; ++k) { + app * const t1 = *j; + app * const t2 = *k; + SASSERT(t1->get_decl() == fd); + SASSERT(t2->get_decl() == fd); + if (t1 == t2) continue; + ackr(t1,t2); + } + } + } +} + + +void lackr::abstract() { + const fun2terms_map::iterator e = m_fun2terms.end(); + for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { + func_decl* const fd = i->m_key; + app_set * const ts = i->get_value(); + sort* const s = fd->get_range(); + const app_set::iterator r = ts->end(); + for (app_set::iterator j = ts->begin(); j != r; ++j) { + app * const fc = m_m.mk_fresh_const(fd->get_name().str().c_str(), s); + app * const t = *j; + SASSERT(t->get_decl() == fd); + m_info->set_abstr(t, fc); + TRACE("lackr", tout << "abstr term " + << mk_ismt2_pp(t, m_m, 2) + << " -> " + << mk_ismt2_pp(fc, m_m, 2) + << "\n";); + } + } + m_info->seal(); + // perform abstraction of the formulas + const unsigned sz = m_formulas.size(); + for (unsigned i = 0; i < sz; ++i) { + expr_ref a(m_m); + m_info->abstract(m_formulas.get(i), a); + m_abstr.push_back(a); + } +} + +void lackr::add_term(app* a) { + if (a->get_num_args() == 0) return; + if (!m_ackr_helper.should_ackermannize(a)) return; + func_decl* const fd = a->get_decl(); + app_set* ts = 0; + if (!m_fun2terms.find(fd, ts)) { + ts = alloc(app_set); + m_fun2terms.insert(fd, ts); + } + TRACE("lackr", tout << "term(" << mk_ismt2_pp(a, m_m, 2) << ")\n";); + ts->insert(a); +} + +void lackr::push_abstraction() { + const unsigned sz = m_abstr.size(); + for (unsigned i = 0; i < sz; ++i) { + m_sat->assert_expr(m_abstr.get(i)); + } +} + +lbool lackr::eager() { + push_abstraction(); + TRACE("lackr", tout << "run sat 0\n"; ); + const lbool rv0 = m_sat->check_sat(0, 0); + if (rv0 == l_false) return l_false; + eager_enc(); + expr_ref all(m_m); + all = m_m.mk_and(m_ackrs.size(), m_ackrs.c_ptr()); + m_simp(all); + m_sat->assert_expr(all); + TRACE("lackr", tout << "run sat all\n"; ); + return m_sat->check_sat(0, 0); +} + +lbool lackr::lazy() { + lackr_model_constructor mc(m_m, m_info); + push_abstraction(); + unsigned ackr_head = 0; + while (1) { + m_st.m_it++; + checkpoint(); + TRACE("lackr", tout << "lazy check: " << m_st.m_it << "\n";); + const lbool r = m_sat->check_sat(0, 0); + if (r == l_undef) return l_undef; // give up + if (r == l_false) return l_false; // abstraction unsat + // reconstruct model + model_ref am; + m_sat->get_model(am); + const bool mc_res = mc.check(am); + if (mc_res) return l_true; // model okay + // refine abstraction + const lackr_model_constructor::conflict_list conflicts = mc.get_conflicts(); + for (lackr_model_constructor::conflict_list::const_iterator i = conflicts.begin(); + i != conflicts.end(); ++i) { + ackr(i->first, i->second); + } + while (ackr_head < m_ackrs.size()) { + m_sat->assert_expr(m_ackrs.get(ackr_head++)); + } + } +} + +// +// Collect all uninterpreted terms, skipping 0-arity. +// +void lackr::collect_terms() { + ptr_vector stack; + expr * curr; + expr_mark visited; + for(unsigned i = 0; i < m_formulas.size(); ++i) { + stack.push_back(m_formulas.get(i)); + TRACE("lackr", tout << "infla: " <get_kind()) { + case AST_VAR: + visited.mark(curr, true); + stack.pop_back(); + break; + case AST_APP: + { + app * const a = to_app(curr); + if (for_each_expr_args(stack, visited, a->get_num_args(), a->get_args())) { + visited.mark(curr, true); + stack.pop_back(); + add_term(a); + } + } + break; + case AST_QUANTIFIER: + UNREACHABLE(); + break; + default: + UNREACHABLE(); + return; + } + } + visited.reset(); +} diff --git a/src/ackermannization/lackr.h b/src/ackermannization/lackr.h new file mode 100644 index 000000000..cdac6b3e4 --- /dev/null +++ b/src/ackermannization/lackr.h @@ -0,0 +1,123 @@ + /*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr.h + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef LACKR_H_ +#define LACKR_H_ + +#include"ackr_info.h" +#include"ackr_helper.h" +#include"th_rewriter.h" +#include"cooperate.h" +#include"bv_decl_plugin.h" +#include"lbool.h" +#include"model.h" +#include"solver.h" +#include"util.h" +#include"tactic_exception.h" +#include"goal.h" + +struct lackr_stats { + lackr_stats() : m_it(0), m_ackrs_sz(0) {} + void reset() { m_it = m_ackrs_sz = 0; } + unsigned m_it; // number of lazy iterations + unsigned m_ackrs_sz; // number of congruence constraints +}; + +/** \brief + A class to encode or directly solve problems with uninterpreted functions via ackermannization. + Currently, solving is supported only for QF_UFBV. +**/ +class lackr { + public: + lackr(ast_manager& m, params_ref p, lackr_stats& st, + expr_ref_vector& formulas, solver * uffree_solver); + ~lackr(); + void updt_params(params_ref const & _p); + + /** \brief + * Solve the formula that the class was initialized with. + **/ + lbool operator() (); + + + /** \brief + Converts function occurrences to constants and encodes all congruence ackermann lemmas. + + This procedure guarantees a equisatisfiability with the input formula and it has a worst-case quadratic blowup. + Before ackermannization an upper bound on congruence lemmas is computed and tested against \p lemmas_upper_bound. + If this bound is exceeded, the function returns false, it returns true otherwise. + **/ + bool mk_ackermann(/*out*/goal_ref& g, double lemmas_upper_bound); + + // + // getters + // + inline ackr_info_ref get_info() { return m_info; } + inline model_ref get_model() { return m_model; } + + // + // timeout mechanism + // + void checkpoint() { + if (m_m.canceled()) { + throw tactic_exception(TACTIC_CANCELED_MSG); + } + cooperate("lackr"); + } + private: + typedef ackr_helper::fun2terms_map fun2terms_map; + typedef ackr_helper::app_set app_set; + ast_manager& m_m; + params_ref m_p; + expr_ref_vector m_formulas; + expr_ref_vector m_abstr; + fun2terms_map m_fun2terms; + ackr_info_ref m_info; + solver* m_sat; + ackr_helper m_ackr_helper; + th_rewriter m_simp; + expr_ref_vector m_ackrs; + model_ref m_model; + bool m_eager; + lackr_stats& m_st; + bool m_is_init; + + void init(); + lbool eager(); + lbool lazy(); + + // + // Introduce congruence ackermann lemma for the two given terms. + // + bool ackr(app * const t1, app * const t2); + + // + // Introduce the ackermann lemma for each pair of terms. + // + void eager_enc(); + + void abstract(); + + void push_abstraction(); + + void add_term(app* a); + + // + // Collect all uninterpreted terms, skipping 0-arity. + // + void collect_terms(); +}; +#endif /* LACKR_H_ */ diff --git a/src/ackermannization/lackr_model_constructor.cpp b/src/ackermannization/lackr_model_constructor.cpp new file mode 100644 index 000000000..496f65383 --- /dev/null +++ b/src/ackermannization/lackr_model_constructor.cpp @@ -0,0 +1,378 @@ +/*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + model_constructor.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ +#include"lackr_model_constructor.h" +#include"model_evaluator.h" +#include"ast_smt2_pp.h" +#include"ackr_info.h" +#include"for_each_expr.h" +#include"bv_rewriter.h" +#include"bool_rewriter.h" + +struct lackr_model_constructor::imp { + public: + imp(ast_manager& m, + ackr_info_ref info, + model_ref& abstr_model, + conflict_list& conflicts) + : m_m(m) + , m_info(info) + , m_abstr_model(abstr_model) + , m_conflicts(conflicts) + , m_b_rw(m) + , m_bv_rw(m) + , m_empty_model(m) + , m_ackr_helper(m) + {} + + ~imp() { + { + values2val_t::iterator i = m_values2val.begin(); + const values2val_t::iterator e = m_values2val.end(); + for (; i != e; ++i) { + m_m.dec_ref(i->m_key); + m_m.dec_ref(i->m_value.value); + m_m.dec_ref(i->m_value.source_term); + } + } + { + app2val_t::iterator i = m_app2val.begin(); + const app2val_t::iterator e = m_app2val.end(); + for (; i != e; ++i) { + m_m.dec_ref(i->m_value); + m_m.dec_ref(i->m_key); + } + } + } + + // + // Returns true iff model was successfully constructed. + // + bool check() { + for (unsigned i = 0; i < m_abstr_model->get_num_constants(); i++) { + func_decl * const c = m_abstr_model->get_constant(i); + app * const term = m_info->find_term(c); + if (term) m_stack.push_back(term); + else m_stack.push_back(m_m.mk_const(c)); + } + return run(); + } + + + void make_model(model_ref& destination) { + { + for (unsigned i = 0; i < m_abstr_model->get_num_uninterpreted_sorts(); i++) { + sort * const s = m_abstr_model->get_uninterpreted_sort(i); + ptr_vector u = m_abstr_model->get_universe(s); + destination->register_usort(s, u.size(), u.c_ptr()); + } + } + for (unsigned i = 0; i < m_abstr_model->get_num_functions(); i++) { + func_decl * const fd = m_abstr_model->get_function(i); + func_interp * const fi = m_abstr_model->get_func_interp(fd); + destination->register_decl(fd, fi); + } + + { + const app2val_t::iterator e = m_app2val.end(); + app2val_t::iterator i = m_app2val.end(); + for (; i != e; ++i) { + app * a = i->m_key; + if (a->get_num_args()) continue; + destination->register_decl(a->get_decl(), i->m_value); + } + } + + obj_map interpretations; + { + const values2val_t::iterator e = m_values2val.end(); + values2val_t::iterator i = m_values2val.end(); + for (; i != e; ++i) add_entry(i->m_key, i->m_value.value, interpretations); + } + + { + obj_map::iterator ie = interpretations.end(); + obj_map::iterator ii = interpretations.begin(); + for (; ii != ie; ++ii) { + func_decl* const fd = ii->m_key; + func_interp* const fi = ii->get_value(); + fi->set_else(m_m.get_some_value(fd->get_range())); + destination->register_decl(fd, fi); + } + } + } + + void add_entry(app* term, expr* value, + obj_map& interpretations) { + func_interp* fi = 0; + func_decl * const declaration = term->get_decl(); + const unsigned sz = declaration->get_arity(); + SASSERT(sz == term->get_num_args()); + if (!interpretations.find(declaration, fi)) { + fi = alloc(func_interp, m_m, sz); + interpretations.insert(declaration, fi); + } + fi->insert_new_entry(term->get_args(), value); + } + private: + ast_manager& m_m; + ackr_info_ref m_info; + model_ref& m_abstr_model; + conflict_list& m_conflicts; + bool_rewriter m_b_rw; + bv_rewriter m_bv_rw; + scoped_ptr m_evaluator; + model m_empty_model; + private: + struct val_info { expr * value; app * source_term; }; + typedef obj_map app2val_t; + typedef obj_map values2val_t; + values2val_t m_values2val; + app2val_t m_app2val; + ptr_vector m_stack; + ackr_helper m_ackr_helper; + + static inline val_info mk_val_info(expr* value, app* source_term) { + val_info rv; + rv.value = value; + rv.source_term = source_term; + return rv; + } + + // + // Performs congruence check on terms on the stack. + // (Currently stops upon the first failure). + // Returns true if and only if congruence check succeeded. + bool run() { + m_evaluator = alloc(model_evaluator, m_empty_model); + expr_mark visited; + expr * curr; + while (!m_stack.empty()) { + curr = m_stack.back(); + if (visited.is_marked(curr)) { + m_stack.pop_back(); + continue; + } + + switch (curr->get_kind()) { + case AST_VAR: + UNREACHABLE(); + return false; + case AST_APP: { + app * a = to_app(curr); + if (for_each_expr_args(m_stack, visited, a->get_num_args(), a->get_args())) { + visited.mark(a, true); + m_stack.pop_back(); + if (!mk_value(a)) return false; + } + } + break; + case AST_QUANTIFIER: + UNREACHABLE(); + return false; + default: + UNREACHABLE(); + return false; + } + } + return true; + } + + inline bool is_val(expr * e) { return m_m.is_value(e); } + + inline bool eval_cached(app * a, expr *& val) { + if (is_val(a)) { val = a; return true; } + return m_app2val.find(a, val); + } + + bool evaluate(app * const a, expr_ref& result) { + SASSERT(!is_val(a)); + const unsigned num = a->get_num_args(); + if (num == 0) { // handle constants + make_value_constant(a, result); + return true; + } + // evaluate arguments + expr_ref_vector values(m_m); + values.reserve(num); + expr * const * args = a->get_args(); + for (unsigned i = 0; i < num; ++i) { + expr * val; + const bool b = eval_cached(to_app(args[i]), val); // TODO: OK conversion to_app? + CTRACE("model_constructor", !b, tout << "fail arg val(\n" << mk_ismt2_pp(args[i], m_m, 2); ); + TRACE("model_constructor", tout << + "arg val " << i << "(\n" << mk_ismt2_pp(args[i], m_m, 2) + << " : " << mk_ismt2_pp(val, m_m, 2) << '\n'; ); + SASSERT(b); + values[i] = val; + } + // handle functions + if (m_ackr_helper.should_ackermannize(a)) { // handle uninterpreted + app_ref key(m_m.mk_app(a->get_decl(), values.c_ptr()), m_m); + if (!make_value_uninterpreted_function(a, values, key.get(), result)) { + return false; + } + } + else { // handle interpreted + make_value_interpreted_function(a, values, result); + } + return true; + } + + // + // Check and record the value for a given term, given that all arguments are already checked. + // + bool mk_value(app * a) { + if (is_val(a)) return true; // skip numerals + TRACE("model_constructor", tout << "mk_value(\n" << mk_ismt2_pp(a, m_m, 2) << ")\n";); + SASSERT(!m_app2val.contains(a)); + const unsigned num = a->get_num_args(); + expr_ref result(m_m); + if (!evaluate(a, result)) return false; + SASSERT(is_val(result)); + TRACE("model_constructor", + tout << "map term(\n" << mk_ismt2_pp(a, m_m, 2) << "\n->" + << mk_ismt2_pp(result.get(), m_m, 2)<< ")\n"; ); + CTRACE("model_constructor", + !is_val(result.get()), + tout << "eval fail\n" << mk_ismt2_pp(a, m_m, 2) << mk_ismt2_pp(result, m_m, 2) << "\n"; + ); + SASSERT(is_val(result.get())); + m_app2val.insert(a, result.get()); // memoize + m_m.inc_ref(a); + m_m.inc_ref(result.get()); + return true; + } + + // Constants from the abstract model are directly mapped to the concrete one. + void make_value_constant(app * const a, expr_ref& result) { + SASSERT(a->get_num_args() == 0); + func_decl * const fd = a->get_decl(); + expr * val = m_abstr_model->get_const_interp(fd); + if (val == 0) { // TODO: avoid model completetion? + sort * s = fd->get_range(); + val = m_abstr_model->get_some_value(s); + } + result = val; + } + + bool make_value_uninterpreted_function(app* a, + expr_ref_vector& values, + app* key, + expr_ref& result) { + // get ackermann constant + app * const ac = m_info->get_abstr(a); + func_decl * const a_fd = a->get_decl(); + SASSERT(ac->get_num_args() == 0); + SASSERT(a_fd->get_range() == ac->get_decl()->get_range()); + expr_ref value(m_m); + value = m_abstr_model->get_const_interp(ac->get_decl()); + // get ackermann constant's interpretation + if (value.get() == 0) { // TODO: avoid model completion? + sort * s = a_fd->get_range(); + value = m_abstr_model->get_some_value(s); + } + // check congruence + val_info vi; + if(m_values2val.find(key,vi)) { // already is mapped to a value + SASSERT(vi.source_term); + const bool ok = vi.value == value; + if (!ok) { + TRACE("model_constructor", + tout << "already mapped by(\n" << mk_ismt2_pp(vi.source_term, m_m, 2) << "\n->" + << mk_ismt2_pp(vi.value, m_m, 2) << ")\n"; ); + m_conflicts.push_back(std::make_pair(a, vi.source_term)); + } + result = vi.value; + return ok; + } else { // new value + result = value; + vi.value = value; + vi.source_term = a; + m_values2val.insert(key,vi); + m_m.inc_ref(vi.source_term); + m_m.inc_ref(vi.value); + m_m.inc_ref(key); + return true; + } + UNREACHABLE(); + } + + void make_value_interpreted_function(app* a, + expr_ref_vector& values, + expr_ref& result) { + const unsigned num = values.size(); + func_decl * const fd = a->get_decl(); + const family_id fid = fd->get_family_id(); + expr_ref term(m_m); + term = m_m.mk_app(a->get_decl(), num, values.c_ptr()); + m_evaluator->operator() (term, result); + TRACE("model_constructor", + tout << "eval(\n" << mk_ismt2_pp(term.get(), m_m, 2) << "\n->" + << mk_ismt2_pp(result.get(), m_m, 2) << ")\n"; ); + return; + if (fid == m_b_rw.get_fid()) { + decl_kind k = fd->get_decl_kind(); + if (k == OP_EQ) { + // theory dispatch for = + SASSERT(num == 2); + family_id s_fid = m_m.get_sort(values.get(0))->get_family_id(); + br_status st = BR_FAILED; + if (s_fid == m_bv_rw.get_fid()) + st = m_bv_rw.mk_eq_core(values.get(0), values.get(1), result); + } else { + br_status st = m_b_rw.mk_app_core(fd, num, values.c_ptr(), result); + } + } else { + br_status st = BR_FAILED; + if (fid == m_bv_rw.get_fid()) { + st = m_bv_rw.mk_app_core(fd, num, values.c_ptr(), result); + } + else { + UNREACHABLE(); + } + } + } +}; + +lackr_model_constructor::lackr_model_constructor(ast_manager& m, ackr_info_ref info) + : m_imp(0) + , m_m(m) + , m_state(UNKNOWN) + , m_info(info) + , m_ref_count(0) +{} + +lackr_model_constructor::~lackr_model_constructor() { + if (m_imp) dealloc(m_imp); +} + +bool lackr_model_constructor::check(model_ref& abstr_model) { + m_conflicts.reset(); + if (m_imp) { + dealloc(m_imp); + m_imp = 0; + } + m_imp = alloc(lackr_model_constructor::imp, m_m, m_info, abstr_model, m_conflicts); + const bool rv = m_imp->check(); + m_state = rv ? CHECKED : CONFLICT; + return rv; +} + +void lackr_model_constructor::make_model(model_ref& model) { + SASSERT(m_state == CHECKED); + m_imp->make_model(model); +} diff --git a/src/ackermannization/lackr_model_constructor.h b/src/ackermannization/lackr_model_constructor.h new file mode 100644 index 000000000..ce9edba26 --- /dev/null +++ b/src/ackermannization/lackr_model_constructor.h @@ -0,0 +1,61 @@ + /*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + model_constructor.h + + Abstract: + Given a propositional abstraction, attempt to construct a model. + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef LACKR_MODEL_CONSTRUCTOR_H_ +#define LACKR_MODEL_CONSTRUCTOR_H_ + +#include"ast.h" +#include"ackr_info.h" +#include"ackr_helper.h" +#include"model.h" + +class lackr_model_constructor { + public: + typedef std::pair app_pair; + typedef vector conflict_list; + lackr_model_constructor(ast_manager& m, ackr_info_ref info); + ~lackr_model_constructor(); + bool check(model_ref& abstr_model); + const conflict_list& get_conflicts() { + SASSERT(m_state == CONFLICT); + return m_conflicts; + } + void make_model(model_ref& model); + + // + // Reference counting + // + void inc_ref() { ++m_ref_count; } + void dec_ref() { + --m_ref_count; + if (m_ref_count == 0) { + dealloc(this); + } + } + private: + struct imp; + imp * m_imp; + enum {CHECKED, CONFLICT, UNKNOWN} m_state; + conflict_list m_conflicts; + ast_manager& m_m; + const ackr_info_ref m_info; + + unsigned m_ref_count; // reference counting +}; + +typedef ref lackr_model_constructor_ref; +#endif /* MODEL_CONSTRUCTOR_H_ */ diff --git a/src/ackermannization/lackr_model_converter_lazy.cpp b/src/ackermannization/lackr_model_converter_lazy.cpp new file mode 100644 index 000000000..5a0d1c4be --- /dev/null +++ b/src/ackermannization/lackr_model_converter_lazy.cpp @@ -0,0 +1,59 @@ +/*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr_model_converter_lazy.cpp + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: +--*/ +#include"lackr_model_converter_lazy.h" +#include"model_evaluator.h" +#include"ast_smt2_pp.h" +#include"ackr_info.h" +#include"lackr_model_constructor.h" + +class lackr_model_converter_lazy : public model_converter { +public: + lackr_model_converter_lazy(ast_manager & m, + const lackr_model_constructor_ref& lmc) + : m(m) + , model_constructor(lmc) + { } + + virtual ~lackr_model_converter_lazy() { } + + virtual void operator()(model_ref & md, unsigned goal_idx) { + SASSERT(goal_idx == 0); + SASSERT(md.get() == 0 || (!md->get_num_constants() && !md->get_num_functions())); + SASSERT(model_constructor.get()); + model * new_model = alloc(model, m); + md = new_model; + model_constructor->make_model(md); + } + + virtual void operator()(model_ref & md) { + operator()(md, 0); + } + + //void display(std::ostream & out); + + virtual model_converter * translate(ast_translation & translator) { + NOT_IMPLEMENTED_YET(); + } +protected: + ast_manager& m; + const lackr_model_constructor_ref model_constructor; +}; + +model_converter * mk_lackr_model_converter_lazy(ast_manager & m, + const lackr_model_constructor_ref& model_constructor) { + return alloc(lackr_model_converter_lazy, m, model_constructor); +} diff --git a/src/ackermannization/lackr_model_converter_lazy.h b/src/ackermannization/lackr_model_converter_lazy.h new file mode 100644 index 000000000..c9b3eac80 --- /dev/null +++ b/src/ackermannization/lackr_model_converter_lazy.h @@ -0,0 +1,25 @@ + /*++ + Copyright (c) 2015 Microsoft Corporation + + Module Name: + + lackr_model_converter_lazy.h + + Abstract: + + + Author: + + Mikolas Janota + + Revision History: + --*/ +#ifndef LACKR_MODEL_CONVERTER_LAZY_H_ +#define LACKR_MODEL_CONVERTER_LAZY_H_ + +#include"model_converter.h" +#include"ackr_info.h" + +model_converter * mk_lackr_model_converter_lazy(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); + +#endif /* LACKR_MODEL_CONVERTER_LAZY_H_ */ From faa620f673b98efe08a5f59df3ac15798a9c8516 Mon Sep 17 00:00:00 2001 From: mikolas Date: Wed, 3 Feb 2016 17:31:19 +0000 Subject: [PATCH 29/31] Further refactoring ackermannization. --- scripts/mk_project.py | 4 +- src/ackermannization/ackermannization.pyg | 7 - .../ackr/ackermannize_bv_model_converter.cpp | 22 - .../ackr/ackermannize_bv_model_converter.h | 25 -- src/tactic/ackr/ackermannize_bv_tactic.cpp | 99 ----- src/tactic/ackr/ackermannize_bv_tactic.h | 28 -- .../ackr/ackermannize_bv_tactic_params.pyg | 8 - src/tactic/ackr/ackr.pyg | 7 - src/tactic/ackr/ackr_bound_probe.cpp | 77 ---- src/tactic/ackr/ackr_bound_probe.h | 29 -- src/tactic/ackr/ackr_helper.cpp | 29 -- src/tactic/ackr/ackr_helper.h | 69 ---- src/tactic/ackr/ackr_info.h | 109 ----- src/tactic/ackr/ackr_model_converter.cpp | 153 ------- src/tactic/ackr/ackr_model_converter.h | 25 -- src/tactic/ackr/lackr.cpp | 285 ------------- src/tactic/ackr/lackr.h | 127 ------ src/tactic/ackr/lackr_model_constructor.cpp | 377 ------------------ src/tactic/ackr/lackr_model_constructor.h | 61 --- .../ackr/lackr_model_converter_lazy.cpp | 60 --- src/tactic/ackr/lackr_model_converter_lazy.h | 25 -- src/tactic/smtlogics/qfufbv_tactic.cpp | 9 +- 22 files changed, 6 insertions(+), 1629 deletions(-) delete mode 100644 src/ackermannization/ackermannization.pyg delete mode 100644 src/tactic/ackr/ackermannize_bv_model_converter.cpp delete mode 100644 src/tactic/ackr/ackermannize_bv_model_converter.h delete mode 100644 src/tactic/ackr/ackermannize_bv_tactic.cpp delete mode 100644 src/tactic/ackr/ackermannize_bv_tactic.h delete mode 100644 src/tactic/ackr/ackermannize_bv_tactic_params.pyg delete mode 100644 src/tactic/ackr/ackr.pyg delete mode 100644 src/tactic/ackr/ackr_bound_probe.cpp delete mode 100644 src/tactic/ackr/ackr_bound_probe.h delete mode 100644 src/tactic/ackr/ackr_helper.cpp delete mode 100644 src/tactic/ackr/ackr_helper.h delete mode 100644 src/tactic/ackr/ackr_info.h delete mode 100644 src/tactic/ackr/ackr_model_converter.cpp delete mode 100644 src/tactic/ackr/ackr_model_converter.h delete mode 100644 src/tactic/ackr/lackr.cpp delete mode 100644 src/tactic/ackr/lackr.h delete mode 100644 src/tactic/ackr/lackr_model_constructor.cpp delete mode 100644 src/tactic/ackr/lackr_model_constructor.h delete mode 100644 src/tactic/ackr/lackr_model_converter_lazy.cpp delete mode 100644 src/tactic/ackr/lackr_model_converter_lazy.h diff --git a/scripts/mk_project.py b/scripts/mk_project.py index 82cdc04d0..a22d34209 100644 --- a/scripts/mk_project.py +++ b/scripts/mk_project.py @@ -36,6 +36,7 @@ def init_project_def(): add_lib('subpaving_tactic', ['core_tactics', 'subpaving'], 'math/subpaving/tactic') add_lib('aig_tactic', ['tactic'], 'tactic/aig') add_lib('solver', ['model', 'tactic']) + add_lib('ackermannization', ['model', 'rewriter', 'ast', 'solver', 'tactic'], 'ackermannization') add_lib('interp', ['solver']) add_lib('cmd_context', ['solver', 'rewriter', 'interp']) add_lib('extra_cmds', ['cmd_context', 'subpaving_tactic', 'arith_tactics'], 'cmd_context/extra_cmds') @@ -52,7 +53,6 @@ def init_project_def(): add_lib('proto_model', ['model', 'simplifier', 'smt_params'], 'smt/proto_model') add_lib('smt', ['bit_blaster', 'macros', 'normal_forms', 'cmd_context', 'proto_model', 'substitution', 'grobner', 'euclid', 'simplex', 'proof_checker', 'pattern', 'parser_util', 'fpa']) - add_lib('ackr', ['smt'], 'tactic/ackr') add_lib('bv_tactics', ['tactic', 'bit_blaster'], 'tactic/bv') add_lib('fuzzing', ['ast'], 'test/fuzzing') add_lib('smt_tactic', ['smt'], 'smt/tactic') @@ -73,7 +73,7 @@ def init_project_def(): add_lib('nlsat_smt_tactic', ['nlsat_tactic', 'smt_tactic'], 'tactic/nlsat_smt') add_lib('ufbv_tactic', ['normal_forms', 'core_tactics', 'macros', 'smt_tactic', 'rewriter'], 'tactic/ufbv') add_lib('sat_solver', ['solver', 'core_tactics', 'aig_tactic', 'bv_tactics', 'arith_tactics', 'sat_tactic'], 'sat/sat_solver') - add_lib('smtlogic_tactics', ['ackr', 'sat_solver', 'arith_tactics', 'bv_tactics', 'nlsat_tactic', 'smt_tactic', 'aig_tactic', 'fp', 'muz','qe','nlsat_smt_tactic'], 'tactic/smtlogics') + add_lib('smtlogic_tactics', ['ackermannization', 'sat_solver', 'arith_tactics', 'bv_tactics', 'nlsat_tactic', 'smt_tactic', 'aig_tactic', 'fp', 'muz','qe','nlsat_smt_tactic'], 'tactic/smtlogics') add_lib('fpa_tactics', ['fpa', 'core_tactics', 'bv_tactics', 'sat_tactic', 'smt_tactic', 'arith_tactics', 'smtlogic_tactics'], 'tactic/fpa') add_lib('portfolio', ['smtlogic_tactics', 'sat_solver', 'ufbv_tactic', 'fpa_tactics', 'aig_tactic', 'fp', 'qe','sls_tactic', 'subpaving_tactic'], 'tactic/portfolio') add_lib('smtparser', ['portfolio'], 'parsers/smt') diff --git a/src/ackermannization/ackermannization.pyg b/src/ackermannization/ackermannization.pyg deleted file mode 100644 index fe7ac7b63..000000000 --- a/src/ackermannization/ackermannization.pyg +++ /dev/null @@ -1,7 +0,0 @@ -def_module_params('ackermannization', - description='solving UF via ackermannization', - export=True, - params=( - ('eager', BOOL, True, 'eagerly instantiate all congruence rules'), - )) - diff --git a/src/tactic/ackr/ackermannize_bv_model_converter.cpp b/src/tactic/ackr/ackermannize_bv_model_converter.cpp deleted file mode 100644 index d47735a56..000000000 --- a/src/tactic/ackr/ackermannize_bv_model_converter.cpp +++ /dev/null @@ -1,22 +0,0 @@ -/*++ - Copyright (c) 2016 Microsoft Corporation - - Module Name: - - ackermannize_bv_model_converter.cpp - - Abstract: - - - Author: - - Mikolas Janota (MikolasJanota) - - Revision History: ---*/ -#include"ackr_model_converter.h" -#include"ackermannize_bv_model_converter.h" - -model_converter * mk_ackermannize_bv_model_converter(ast_manager & m, const ackr_info_ref& info) { - return mk_ackr_model_converter(m, info); -} diff --git a/src/tactic/ackr/ackermannize_bv_model_converter.h b/src/tactic/ackr/ackermannize_bv_model_converter.h deleted file mode 100644 index e51792bad..000000000 --- a/src/tactic/ackr/ackermannize_bv_model_converter.h +++ /dev/null @@ -1,25 +0,0 @@ - /*++ - Copyright (c) 2016 Microsoft Corporation - - Module Name: - - ackermannize_bv_model_converter.h - - Abstract: - - - Author: - - Mikolas Janota (MikolasJanota) - - Revision History: - --*/ -#ifndef ACKERMANNIZE_BV_MODEL_CONVERTER_H_ -#define ACKERMANNIZE_BV_MODEL_CONVERTER_H_ - -#include"model_converter.h" -#include"ackr_info.h" - -model_converter * mk_ackermannize_bv_model_converter(ast_manager & m, const ackr_info_ref& info); - -#endif /* ACKERMANNIZE_BV_MODEL_CONVERTER_H_ */ diff --git a/src/tactic/ackr/ackermannize_bv_tactic.cpp b/src/tactic/ackr/ackermannize_bv_tactic.cpp deleted file mode 100644 index 30547d7fc..000000000 --- a/src/tactic/ackr/ackermannize_bv_tactic.cpp +++ /dev/null @@ -1,99 +0,0 @@ -/*++ -Copyright (c) 2016 Microsoft Corporation - -Module Name: - -ackermannize_bv_tactic.cpp - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ -#include"ackermannize_bv_tactic.h" -#include"tactical.h" -#include"lackr.h" -#include"ackr_params.hpp" -#include"model_smt2_pp.h" -#include"ackr_bound_probe.h" -#include"ackermannize_bv_tactic_params.hpp" -#include"ackermannize_bv_model_converter.h" - - -class ackermannize_bv_tactic : public tactic { -public: - ackermannize_bv_tactic(ast_manager& m, params_ref const& p) - : m_m(m) - , m_p(p) - {} - - virtual ~ackermannize_bv_tactic() { } - - virtual void operator()(goal_ref const & g, - goal_ref_buffer & result, - model_converter_ref & mc, - proof_converter_ref & pc, - expr_dependency_ref & core) { - mc = 0; - ast_manager& m(g->m()); - tactic_report report("ackermannize", *g); - fail_if_unsat_core_generation("ackermannize", g); - fail_if_proof_generation("ackermannize", g); - - expr_ref_vector flas(m); - const unsigned sz = g->size(); - for (unsigned i = 0; i < sz; i++) flas.push_back(g->form(i)); - scoped_ptr imp = alloc(lackr, m, m_p, m_st, flas, NULL); - flas.reset(); - // mk result - goal_ref resg(alloc(goal, *g, true)); - const bool success = imp->mk_ackermann(resg, m_lemma_limit); - if (!success) { // Just pass on the input unchanged - result.reset(); - result.push_back(g.get()); - mc = 0; - pc = 0; - core = 0; - return; - } - result.push_back(resg.get()); - // report model - if (g->models_enabled()) { - mc = mk_ackermannize_bv_model_converter(m, imp->get_info()); - } - - resg->inc_depth(); - TRACE("ackermannize", resg->display(tout);); - SASSERT(resg->is_well_sorted()); - } - - - void updt_params(params_ref const & _p) { - ackermannize_bv_tactic_params p(_p); - m_lemma_limit = p.div0_ackermann_limit(); - } - - virtual void collect_statistics(statistics & st) const { - st.update("ackr-constraints", m_st.m_ackrs_sz); - } - - virtual void reset_statistics() { m_st.reset(); } - - virtual void cleanup() { } - - virtual tactic* translate(ast_manager& m) { - return alloc(ackermannize_bv_tactic, m, m_p); - } -private: - ast_manager& m_m; - params_ref m_p; - lackr_stats m_st; - double m_lemma_limit; -}; - -tactic * mk_ackermannize_bv_tactic(ast_manager & m, params_ref const & p) { - return alloc(ackermannize_bv_tactic, m, p); -} diff --git a/src/tactic/ackr/ackermannize_bv_tactic.h b/src/tactic/ackr/ackermannize_bv_tactic.h deleted file mode 100644 index 3e3d5636c..000000000 --- a/src/tactic/ackr/ackermannize_bv_tactic.h +++ /dev/null @@ -1,28 +0,0 @@ -/*++ -Copyright (c) 2016 Microsoft Corporation - -Module Name: - -ackermannize_bv_tactic.h - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ - -#ifndef _ACKERMANNIZE_TACTIC_H_ -#define _ACKERMANNIZE_TACTIC_H_ -#include"tactical.h" - -tactic * mk_ackermannize_bv_tactic(ast_manager & m, params_ref const & p); - -/* - ADD_TACTIC("ackermannize_bv", "A tactic for performing full Ackermannization on bv instances.", "mk_ackermannize_bv_tactic(m, p)") -*/ - -#endif - diff --git a/src/tactic/ackr/ackermannize_bv_tactic_params.pyg b/src/tactic/ackr/ackermannize_bv_tactic_params.pyg deleted file mode 100644 index bd17dcaff..000000000 --- a/src/tactic/ackr/ackermannize_bv_tactic_params.pyg +++ /dev/null @@ -1,8 +0,0 @@ -def_module_params(module_name='rewriter', - class_name='ackermannize_bv_tactic_params', - export=True, - params=( - ("div0_ackermann_limit", UINT, 1000, "a bound for number of congruence Ackermann lemmas for div0 modelling"), - ) - ) - diff --git a/src/tactic/ackr/ackr.pyg b/src/tactic/ackr/ackr.pyg deleted file mode 100644 index f2e244cfd..000000000 --- a/src/tactic/ackr/ackr.pyg +++ /dev/null @@ -1,7 +0,0 @@ -def_module_params('ackr', - description='solving UF via ackermannization', - export=True, - params=( - ('eager', BOOL, True, 'eagerly instantiate all congruence rules'), - )) - diff --git a/src/tactic/ackr/ackr_bound_probe.cpp b/src/tactic/ackr/ackr_bound_probe.cpp deleted file mode 100644 index 5cb8e9448..000000000 --- a/src/tactic/ackr/ackr_bound_probe.cpp +++ /dev/null @@ -1,77 +0,0 @@ -/*++ - Copyright (c) 2016 Microsoft Corporation - - Module Name: - - ackr_bound_probe.cpp - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: ---*/ -#include"ackr_helper.h" -#include"ackr_bound_probe.h" -#include"ast_smt2_pp.h" - -/* - For each function f, calculate the number of its occurrences o_f and compute "o_f choose 2". - The probe then sums up these for all functions. - This upper bound might be crude because some congruence lemmas trivially simplify to true. -*/ -class ackr_bound_probe : public probe { - struct proc { - typedef ackr_helper::fun2terms_map fun2terms_map; - typedef ackr_helper::app_set app_set; - ast_manager& m_m; - fun2terms_map m_fun2terms; // a map from functions to occurrences - ackr_helper m_ackr_helper; - - proc(ast_manager & m) : m_m(m), m_ackr_helper(m) { } - - ~proc() { - fun2terms_map::iterator it = m_fun2terms.begin(); - const fun2terms_map::iterator end = m_fun2terms.end(); - for (; it != end; ++it) dealloc(it->get_value()); - } - - void operator()(quantifier *) {} - void operator()(var *) {} - void operator()(app * a) { - if (a->get_num_args() == 0) return; - if (!m_ackr_helper.should_ackermannize(a)) return; - func_decl* const fd = a->get_decl(); - app_set* ts = 0; - if (!m_fun2terms.find(fd, ts)) { - ts = alloc(app_set); - m_fun2terms.insert(fd, ts); - } - ts->insert(a); - } - }; - -public: - ackr_bound_probe() {} - - virtual result operator()(goal const & g) { - proc p(g.m()); - unsigned sz = g.size(); - expr_fast_mark1 visited; - for (unsigned i = 0; i < sz; i++) { - for_each_expr_core(p, visited, g.form(i)); - } - const double total = ackr_helper::calculate_lemma_bound(p.m_fun2terms); - TRACE("ackr_bound_probe", tout << "total=" << total << std::endl;); - return result(total); - } - - inline static unsigned n_choose_2(unsigned n) { return n & 1 ? (n * (n >> 1)) : (n >> 1) * (n - 1); } -}; - -probe * mk_ackr_bound_probe() { - return alloc(ackr_bound_probe); -} diff --git a/src/tactic/ackr/ackr_bound_probe.h b/src/tactic/ackr/ackr_bound_probe.h deleted file mode 100644 index 6c0a66605..000000000 --- a/src/tactic/ackr/ackr_bound_probe.h +++ /dev/null @@ -1,29 +0,0 @@ - /*++ - Copyright (c) 2016 Microsoft Corporation - - Module Name: - - ackr_bound_probe.h - - Abstract: - - A probe to give an upper bound of Ackermann congruence lemmas that a formula might generate. - - Author: - - Mikolas Janota - - Revision History: - --*/ -#ifndef ACKR_BOUND_PROBE_H_ -#define ACKR_BOUND_PROBE_H_ - -#include"probe.h" - -probe * mk_ackr_bound_probe(); - -/* - ADD_PROBE("ackr-bound-probe", "A probe to give an upper bound of Ackermann congruence lemmas that a formula might generate.", "mk_ackr_bound_probe()") -*/ - -#endif /* ACKR_BOUND_PROBE_H_ */ diff --git a/src/tactic/ackr/ackr_helper.cpp b/src/tactic/ackr/ackr_helper.cpp deleted file mode 100644 index 803fc3c54..000000000 --- a/src/tactic/ackr/ackr_helper.cpp +++ /dev/null @@ -1,29 +0,0 @@ -/*++ - Copyright (c) 2016 Microsoft Corporation - - Module Name: - - ackr_helper.cpp - - Abstract: - - - Author: - - Mikolas Janota (MikolasJanota) - - Revision History: ---*/ -#include"ackr_helper.h" - -double ackr_helper::calculate_lemma_bound(ackr_helper::fun2terms_map& occurrences) { - fun2terms_map::iterator it = occurrences.begin(); - const fun2terms_map::iterator end = occurrences.end(); - double total = 0; - for (; it != end; ++it) { - const unsigned fsz = it->m_value->size(); - const double n2 = n_choose_2_chk(fsz); - total += n2; - } - return total; -} diff --git a/src/tactic/ackr/ackr_helper.h b/src/tactic/ackr/ackr_helper.h deleted file mode 100644 index 5c572907e..000000000 --- a/src/tactic/ackr/ackr_helper.h +++ /dev/null @@ -1,69 +0,0 @@ - /*++ - Copyright (c) 2016 Microsoft Corporation - - Module Name: - - ackr_helper.h - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: - --*/ -#ifndef ACKR_HELPER_H_ -#define ACKR_HELPER_H_ - -#include"bv_decl_plugin.h" - -class ackr_helper { - public: - typedef obj_hashtable app_set; - typedef obj_map fun2terms_map; - - ackr_helper(ast_manager& m) : m_bvutil(m) {} - - /** - \brief Determines if a given function should be Ackermannized. - - This includes all uninterpreted functions but also "special" functions, e.g. OP_BSMOD0, - which are not marked as uninterpreted but effectively are. - */ - inline bool should_ackermannize(app const * a) const { - if (a->get_family_id() == m_bvutil.get_family_id()) { - switch (a->get_decl_kind()) { - case OP_BSDIV0: - case OP_BUDIV0: - case OP_BSREM0: - case OP_BUREM0: - case OP_BSMOD0: - return true; - default: - return is_uninterp(a); - } - } - return (is_uninterp(a)); - } - - inline bv_util& bvutil() { return m_bvutil; } - - /** - \brief Calculates an upper bound for congruence lemmas given a map of function of occurrences. - */ - static double calculate_lemma_bound(fun2terms_map& occurrences); - - /** \brief Calculate n choose 2. **/ - inline static unsigned n_choose_2(unsigned n) { return n & 1 ? (n * (n >> 1)) : (n >> 1) * (n - 1); } - - /** \brief Calculate n choose 2 guarded for overflow. Returns infinity if unsafe. **/ - inline static double n_choose_2_chk(unsigned n) { - SASSERT(std::numeric_limits().max() & 32); - return n & (1 << 16) ? std::numeric_limits().infinity() : n_choose_2(n); - } - private: - bv_util m_bvutil; -}; -#endif /* ACKR_HELPER_H_6475 */ diff --git a/src/tactic/ackr/ackr_info.h b/src/tactic/ackr/ackr_info.h deleted file mode 100644 index 703f1f3d5..000000000 --- a/src/tactic/ackr/ackr_info.h +++ /dev/null @@ -1,109 +0,0 @@ -/*++ -Copyright (c) 2015 Microsoft Corporation - -Module Name: - -ackr_info.h - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ -#ifndef ACKR_INFO_H_ -#define ACKR_INFO_H_ - -#include"obj_hashtable.h" -#include"ast.h" -#include"ref.h" -#include"expr_replacer.h" - -/** \brief - Information about how a formula is being converted into - a formula without uninterpreted function symbols via ackermannization. - - The intended use is that new terms are added via set_abstr. - Once all terms are abstracted, call seal. - The function abstract may only be called when sealed. - - The class enables reference counting. -**/ -class ackr_info { - public: - ackr_info(ast_manager& m) - : m_m(m) - , m_consts(m) - , m_er(mk_default_expr_replacer(m)) - , m_subst(m_m) - , m_ref_count(0) - , m_sealed(false) - {} - - virtual ~ackr_info() { - m_consts.reset(); - } - - inline void set_abstr(app* term, app* c) { - SASSERT(!m_sealed); - SASSERT(c); - m_t2c.insert(term,c); - m_c2t.insert(c->get_decl(),term); - m_subst.insert(term, c); - m_consts.push_back(c); - } - - inline void abstract(expr * e, expr_ref& res) { - SASSERT(m_sealed); - (*m_er)(e, res); - } - - inline app* find_term(func_decl* c) const { - app * rv = 0; - m_c2t.find(c,rv); - return rv; - } - - inline app* get_abstr(app* term) const { - app * const rv = m_t2c.find(term); - SASSERT(rv); - return rv; - } - - inline void seal() { - m_sealed=true; - m_er->set_substitution(&m_subst); - } - - // - // Reference counting - // - void inc_ref() { ++m_ref_count; } - void dec_ref() { - --m_ref_count; - if (m_ref_count == 0) { - dealloc(this); - } - } - private: - typedef obj_map t2ct; - typedef obj_map c2tt; - ast_manager& m_m; - - t2ct m_t2c; // terms to constants - c2tt m_c2t; // constants to terms (inversion of m_t2c) - expr_ref_vector m_consts; // the constants introduced during abstraction - - // replacer and substitution used to compute abstractions - scoped_ptr m_er; - expr_substitution m_subst; - - unsigned m_ref_count; // reference counting - bool m_sealed; // debugging -}; - -typedef ref ackr_info_ref; - -#endif /* ACKR_INFO_H_ */ diff --git a/src/tactic/ackr/ackr_model_converter.cpp b/src/tactic/ackr/ackr_model_converter.cpp deleted file mode 100644 index 7967cdb63..000000000 --- a/src/tactic/ackr/ackr_model_converter.cpp +++ /dev/null @@ -1,153 +0,0 @@ -/*++ -Copyright (c) 2015 Microsoft Corporation - -Module Name: - -ackr_model_converter.cpp - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ -#include"ackr_model_converter.h" -#include"model_evaluator.h" -#include"ast_smt2_pp.h" -#include"ackr_info.h" - - -class ackr_model_converter : public model_converter { -public: - ackr_model_converter(ast_manager & m, - const ackr_info_ref& info, - model_ref& abstr_model) - : m(m) - , info(info) - , abstr_model(abstr_model) - , fixed_model(true) - { } - - ackr_model_converter(ast_manager & m, - const ackr_info_ref& info) - : m(m) - , info(info) - , fixed_model(false) - { } - - virtual ~ackr_model_converter() { } - - virtual void operator()(model_ref & md, unsigned goal_idx) { - SASSERT(goal_idx == 0); - SASSERT(!fixed_model || md.get() == 0 || (!md->get_num_constants() && !md->get_num_functions())); - model_ref& old_model = fixed_model ? abstr_model : md; - SASSERT(old_model.get()); - model * new_model = alloc(model, m); - convert(old_model.get(), new_model); - md = new_model; - } - - virtual void operator()(model_ref & md) { operator()(md, 0); } - - //void display(std::ostream & out); - - virtual model_converter * translate(ast_translation & translator) {NOT_IMPLEMENTED_YET();} -protected: - ast_manager& m; - const ackr_info_ref info; - model_ref abstr_model; - bool fixed_model; - void convert(model * source, model * destination); - void add_entry(model_evaluator & evaluator, - app* term, expr* value, - obj_map& interpretations); - void convert_sorts(model * source, model * destination); - void convert_constants(model * source, model * destination); -}; - -void ackr_model_converter::convert(model * source, model * destination) { - //SASSERT(source->get_num_functions() == 0); - for (unsigned i = 0; i < source->get_num_functions(); i++) { - func_decl * const fd = source->get_function(i); - func_interp * const fi = source->get_func_interp(fd); - destination->register_decl(fd, fi); - } - convert_constants(source,destination); - convert_sorts(source,destination); -} - -void ackr_model_converter::convert_constants(model * source, model * destination) { - TRACE("ackr_model", tout << "converting constants\n";); - obj_map interpretations; - model_evaluator evaluator(*source); - for (unsigned i = 0; i < source->get_num_constants(); i++) { - func_decl * const c = source->get_constant(i); - app * const term = info->find_term(c); - expr * value = source->get_const_interp(c); - if(!term) { - destination->register_decl(c, value); - } else { - add_entry(evaluator, term, value, interpretations); - } - } - - obj_map::iterator e = interpretations.end(); - for (obj_map::iterator i = interpretations.begin(); - i!=e; ++i) { - func_decl* const fd = i->m_key; - func_interp* const fi = i->get_value(); - fi->set_else(m.get_some_value(fd->get_range())); - destination->register_decl(fd, fi); - } -} - -void ackr_model_converter::add_entry(model_evaluator & evaluator, - app* term, expr* value, - obj_map& interpretations) { - TRACE("ackr_model", tout << "add_entry" - << mk_ismt2_pp(term, m, 2) - << "->" - << mk_ismt2_pp(value, m, 2) << "\n"; - ); - - func_interp* fi = 0; - func_decl * const declaration = term->get_decl(); - const unsigned sz = declaration->get_arity(); - SASSERT(sz == term->get_num_args()); - if (!interpretations.find(declaration, fi)) { - fi = alloc(func_interp,m,sz); - interpretations.insert(declaration, fi); - } - expr_ref_vector args(m); - for (unsigned gi = 0; gi < sz; ++gi) { - expr * const arg = term->get_arg(gi); - expr_ref aarg(m); - info->abstract(arg, aarg); - expr_ref arg_value(m); - evaluator(aarg,arg_value); - args.push_back(arg_value); - } - if (fi->get_entry(args.c_ptr()) == 0) { - fi->insert_new_entry(args.c_ptr(), value); - } else { - TRACE("ackr_model", tout << "entry already present\n";); - } -} - -void ackr_model_converter::convert_sorts(model * source, model * destination) { - for (unsigned i = 0; i < source->get_num_uninterpreted_sorts(); i++) { - sort * const s = source->get_uninterpreted_sort(i); - ptr_vector u = source->get_universe(s); - destination->register_usort(s, u.size(), u.c_ptr()); - } -} - -model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info) { - return alloc(ackr_model_converter, m, info); -} - -model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model) { - return alloc(ackr_model_converter, m, info, abstr_model); -} diff --git a/src/tactic/ackr/ackr_model_converter.h b/src/tactic/ackr/ackr_model_converter.h deleted file mode 100644 index 2c1f0c78c..000000000 --- a/src/tactic/ackr/ackr_model_converter.h +++ /dev/null @@ -1,25 +0,0 @@ -/*++ -Copyright (c) 2015 Microsoft Corporation - -Module Name: - -ackr_model_converter.h - -Abstract: - -Author: - -Mikolas Janota - -Revision History: ---*/ -#ifndef ACKR_MODEL_CONVERTER_H_ -#define ACKR_MODEL_CONVERTER_H_ - -#include"model_converter.h" -#include"ackr_info.h" - -model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); -model_converter * mk_ackr_model_converter(ast_manager & m, const ackr_info_ref& info); - -#endif /* LACKR_MODEL_CONVERTER_H_ */ diff --git a/src/tactic/ackr/lackr.cpp b/src/tactic/ackr/lackr.cpp deleted file mode 100644 index 0a526f7ec..000000000 --- a/src/tactic/ackr/lackr.cpp +++ /dev/null @@ -1,285 +0,0 @@ -/*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - lackr.cpp - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: ---*/ - -#include"lackr.h" -#include"ackr_params.hpp" -#include"tactic.h" -#include"lackr_model_constructor.h" -#include"ackr_info.h" -#include"for_each_expr.h" -#include"model_smt2_pp.h" - -lackr::lackr(ast_manager& m, params_ref p, lackr_stats& st, expr_ref_vector& formulas, - solver * uffree_solver) - : m_m(m) - , m_p(p) - , m_formulas(formulas) - , m_abstr(m) - , m_sat(uffree_solver) - , m_ackr_helper(m) - , m_simp(m) - , m_ackrs(m) - , m_st(st) - , m_is_init(false) -{ - updt_params(p); -} - -lackr::~lackr() { - const fun2terms_map::iterator e = m_fun2terms.end(); - for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { - dealloc(i->get_value()); - } -} - -lbool lackr::operator() () { - SASSERT(m_sat); - init(); - const lbool rv = m_eager ? eager() : lazy(); - if (rv == l_true) m_sat->get_model(m_model); - CTRACE("lackr", rv == l_true, - model_smt2_pp(tout << "abstr_model(\n", m_m, *(m_model.get()), 2); tout << ")\n"; ); - return rv; -} - -bool lackr::mk_ackermann(/*out*/goal_ref& g, double lemmas_upper_bound) { - if (lemmas_upper_bound <= 0) return false; - init(); - if (lemmas_upper_bound != std::numeric_limits::infinity()) { - const double lemmas_bound = ackr_helper::calculate_lemma_bound(m_fun2terms); - if (lemmas_bound > lemmas_upper_bound) return false; - } - eager_enc(); - for (unsigned i = 0; i < m_abstr.size(); ++i) g->assert_expr(m_abstr.get(i)); - for (unsigned i = 0; i < m_ackrs.size(); ++i) g->assert_expr(m_ackrs.get(i)); - return true; -} - -void lackr::init() { - SASSERT(!m_is_init); - m_is_init = true; - params_ref simp_p(m_p); - m_simp.updt_params(simp_p); - m_info = alloc(ackr_info, m_m); - collect_terms(); - abstract(); -} - -// -// Introduce ackermann lemma for the two given terms. -// -bool lackr::ackr(app * const t1, app * const t2) { - TRACE("lackr", tout << "ackr " - << mk_ismt2_pp(t1, m_m, 2) << " , " << mk_ismt2_pp(t2, m_m, 2) << "\n";); - const unsigned sz = t1->get_num_args(); - SASSERT(t2->get_num_args() == sz); - expr_ref_vector eqs(m_m); - for (unsigned i = 0; i < sz; ++i) { - expr * const arg1 = t1->get_arg(i); - expr * const arg2 = t2->get_arg(i); - if (arg1 == arg2) continue; // quickly skip syntactically equal - if (m_ackr_helper.bvutil().is_numeral(arg1) && m_ackr_helper.bvutil().is_numeral(arg2)) { - // quickly abort if there are two distinct numerals - SASSERT(arg1 != arg2); - TRACE("lackr", tout << "never eq\n";); - return false; - } - eqs.push_back(m_m.mk_eq(arg1, arg2)); - } - app * const a1 = m_info->get_abstr(t1); - app * const a2 = m_info->get_abstr(t2); - SASSERT(a1 && a2); - TRACE("lackr", tout << "abstr1 " << mk_ismt2_pp(a1, m_m, 2) << "\n";); - TRACE("lackr", tout << "abstr2 " << mk_ismt2_pp(a2, m_m, 2) << "\n";); - expr_ref lhs(m_m.mk_and(eqs.size(), eqs.c_ptr()), m_m); - TRACE("lackr", tout << "ackr constr lhs" << mk_ismt2_pp(lhs, m_m, 2) << "\n";); - expr_ref rhs(m_m.mk_eq(a1, a2),m_m); - TRACE("lackr", tout << "ackr constr rhs" << mk_ismt2_pp(rhs, m_m, 2) << "\n";); - expr_ref cg(m_m.mk_implies(lhs, rhs), m_m); - TRACE("lackr", tout << "ackr constr" << mk_ismt2_pp(cg, m_m, 2) << "\n";); - expr_ref cga(m_m); - m_info->abstract(cg, cga); // constraint needs abstraction due to nested applications - m_simp(cga); - TRACE("lackr", tout << "ackr constr abs:" << mk_ismt2_pp(cga, m_m, 2) << "\n";); - if (m_m.is_true(cga)) return false; - m_st.m_ackrs_sz++; - m_ackrs.push_back(cga); - return true; -} - -// -// Introduce the ackermann lemma for each pair of terms. -// -void lackr::eager_enc() { - TRACE("lackr", tout << "#funs: " << m_fun2terms.size() << std::endl;); - const fun2terms_map::iterator e = m_fun2terms.end(); - for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { - checkpoint(); - func_decl* const fd = i->m_key; - app_set * const ts = i->get_value(); - const app_set::iterator r = ts->end(); - for (app_set::iterator j = ts->begin(); j != r; ++j) { - app_set::iterator k = j; - ++k; - for (; k != r; ++k) { - app * const t1 = *j; - app * const t2 = *k; - SASSERT(t1->get_decl() == fd); - SASSERT(t2->get_decl() == fd); - if (t1 == t2) continue; - ackr(t1,t2); - } - } - } -} - - -void lackr::abstract() { - const fun2terms_map::iterator e = m_fun2terms.end(); - for (fun2terms_map::iterator i = m_fun2terms.begin(); i != e; ++i) { - func_decl* const fd = i->m_key; - app_set * const ts = i->get_value(); - sort* const s = fd->get_range(); - const app_set::iterator r = ts->end(); - for (app_set::iterator j = ts->begin(); j != r; ++j) { - app * const fc = m_m.mk_fresh_const(fd->get_name().str().c_str(), s); - app * const t = *j; - SASSERT(t->get_decl() == fd); - m_info->set_abstr(t, fc); - TRACE("lackr", tout << "abstr term " - << mk_ismt2_pp(t, m_m, 2) - << " -> " - << mk_ismt2_pp(fc, m_m, 2) - << "\n";); - } - } - m_info->seal(); - // perform abstraction of the formulas - const unsigned sz = m_formulas.size(); - for (unsigned i = 0; i < sz; ++i) { - expr_ref a(m_m); - m_info->abstract(m_formulas.get(i), a); - m_abstr.push_back(a); - } -} - -void lackr::add_term(app* a) { - if (a->get_num_args() == 0) return; - if (!m_ackr_helper.should_ackermannize(a)) return; - func_decl* const fd = a->get_decl(); - app_set* ts = 0; - if (!m_fun2terms.find(fd, ts)) { - ts = alloc(app_set); - m_fun2terms.insert(fd, ts); - } - TRACE("lackr", tout << "term(" << mk_ismt2_pp(a, m_m, 2) << ")\n";); - ts->insert(a); -} - -void lackr::push_abstraction() { - const unsigned sz = m_abstr.size(); - for (unsigned i = 0; i < sz; ++i) { - m_sat->assert_expr(m_abstr.get(i)); - } -} - -lbool lackr::eager() { - push_abstraction(); - TRACE("lackr", tout << "run sat 0\n"; ); - const lbool rv0 = m_sat->check_sat(0, 0); - if (rv0 == l_false) return l_false; - eager_enc(); - expr_ref all(m_m); - all = m_m.mk_and(m_ackrs.size(), m_ackrs.c_ptr()); - m_simp(all); - m_sat->assert_expr(all); - TRACE("lackr", tout << "run sat all\n"; ); - return m_sat->check_sat(0, 0); -} - -lbool lackr::lazy() { - lackr_model_constructor mc(m_m, m_info); - push_abstraction(); - unsigned ackr_head = 0; - while (1) { - m_st.m_it++; - checkpoint(); - TRACE("lackr", tout << "lazy check: " << m_st.m_it << "\n";); - const lbool r = m_sat->check_sat(0, 0); - if (r == l_undef) return l_undef; // give up - if (r == l_false) return l_false; // abstraction unsat - // reconstruct model - model_ref am; - m_sat->get_model(am); - const bool mc_res = mc.check(am); - if (mc_res) return l_true; // model okay - // refine abstraction - const lackr_model_constructor::conflict_list conflicts = mc.get_conflicts(); - for (lackr_model_constructor::conflict_list::const_iterator i = conflicts.begin(); - i != conflicts.end(); ++i) { - ackr(i->first, i->second); - } - while (ackr_head < m_ackrs.size()) { - m_sat->assert_expr(m_ackrs.get(ackr_head++)); - } - } -} - -// -// Collect all uninterpreted terms, skipping 0-arity. -// -void lackr::collect_terms() { - ptr_vector stack; - expr * curr; - expr_mark visited; - for(unsigned i = 0; i < m_formulas.size(); ++i) { - stack.push_back(m_formulas.get(i)); - TRACE("lackr", tout << "infla: " <get_kind()) { - case AST_VAR: - visited.mark(curr, true); - stack.pop_back(); - break; - case AST_APP: - { - app * const a = to_app(curr); - if (for_each_expr_args(stack, visited, a->get_num_args(), a->get_args())) { - visited.mark(curr, true); - stack.pop_back(); - add_term(a); - } - } - break; - case AST_QUANTIFIER: - UNREACHABLE(); - break; - default: - UNREACHABLE(); - return; - } - } - visited.reset(); -} diff --git a/src/tactic/ackr/lackr.h b/src/tactic/ackr/lackr.h deleted file mode 100644 index e6612a578..000000000 --- a/src/tactic/ackr/lackr.h +++ /dev/null @@ -1,127 +0,0 @@ - /*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - lackr.h - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: - --*/ -#ifndef LACKR_H_ -#define LACKR_H_ - -#include"ackr_info.h" -#include"ackr_helper.h" -#include"ackr_params.hpp" -#include"th_rewriter.h" -#include"cooperate.h" -#include"bv_decl_plugin.h" -#include"lbool.h" -#include"model.h" -#include"solver.h" -#include"util.h" -#include"tactic_exception.h" -#include"goal.h" - -struct lackr_stats { - lackr_stats() : m_it(0), m_ackrs_sz(0) {} - void reset() { m_it = m_ackrs_sz = 0; } - unsigned m_it; // number of lazy iterations - unsigned m_ackrs_sz; // number of congruence constraints -}; - -/** \brief - A class to encode or directly solve problems with uninterpreted functions via ackermannization. - Currently, solving is supported only for QF_UFBV. -**/ -class lackr { - public: - lackr(ast_manager& m, params_ref p, lackr_stats& st, - expr_ref_vector& formulas, solver * uffree_solver); - ~lackr(); - void updt_params(params_ref const & _p) { - ackr_params p(_p); - m_eager = p.eager(); - } - - /** \brief - * Solve the formula that the class was initialized with. - **/ - lbool operator() (); - - - /** \brief - Converts function occurrences to constants and encodes all congruence ackermann lemmas. - - This procedure guarantees a equisatisfiability with the input formula and it has a worst-case quadratic blowup. - Before ackermannization an upper bound on congruence lemmas is computed and tested against \p lemmas_upper_bound. - If this bound is exceeded, the function returns false, it returns true otherwise. - **/ - bool mk_ackermann(/*out*/goal_ref& g, double lemmas_upper_bound); - - // - // getters - // - inline ackr_info_ref get_info() { return m_info; } - inline model_ref get_model() { return m_model; } - - // - // timeout mechanism - // - void checkpoint() { - if (m_m.canceled()) { - throw tactic_exception(TACTIC_CANCELED_MSG); - } - cooperate("lackr"); - } - private: - typedef ackr_helper::fun2terms_map fun2terms_map; - typedef ackr_helper::app_set app_set; - ast_manager& m_m; - params_ref m_p; - expr_ref_vector m_formulas; - expr_ref_vector m_abstr; - fun2terms_map m_fun2terms; - ackr_info_ref m_info; - solver* m_sat; - ackr_helper m_ackr_helper; - th_rewriter m_simp; - expr_ref_vector m_ackrs; - model_ref m_model; - bool m_eager; - lackr_stats& m_st; - bool m_is_init; - - void init(); - lbool eager(); - lbool lazy(); - - // - // Introduce congruence ackermann lemma for the two given terms. - // - bool ackr(app * const t1, app * const t2); - - // - // Introduce the ackermann lemma for each pair of terms. - // - void eager_enc(); - - void abstract(); - - void push_abstraction(); - - void add_term(app* a); - - // - // Collect all uninterpreted terms, skipping 0-arity. - // - void collect_terms(); -}; -#endif /* LACKR_H_ */ diff --git a/src/tactic/ackr/lackr_model_constructor.cpp b/src/tactic/ackr/lackr_model_constructor.cpp deleted file mode 100644 index f9eded93b..000000000 --- a/src/tactic/ackr/lackr_model_constructor.cpp +++ /dev/null @@ -1,377 +0,0 @@ -/*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - model_constructor.cpp - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: ---*/ -#include"lackr_model_constructor.h" -#include"model_evaluator.h" -#include"ast_smt2_pp.h" -#include"ackr_info.h" -#include"for_each_expr.h" -#include"bv_rewriter.h" -#include"bool_rewriter.h" -struct lackr_model_constructor::imp { - public: - imp(ast_manager& m, - ackr_info_ref info, - model_ref& abstr_model, - conflict_list& conflicts) - : m_m(m) - , m_info(info) - , m_abstr_model(abstr_model) - , m_conflicts(conflicts) - , m_b_rw(m) - , m_bv_rw(m) - , m_empty_model(m) - , m_ackr_helper(m) - {} - - ~imp() { - { - values2val_t::iterator i = m_values2val.begin(); - const values2val_t::iterator e = m_values2val.end(); - for (; i != e; ++i) { - m_m.dec_ref(i->m_key); - m_m.dec_ref(i->m_value.value); - m_m.dec_ref(i->m_value.source_term); - } - } - { - app2val_t::iterator i = m_app2val.begin(); - const app2val_t::iterator e = m_app2val.end(); - for (; i != e; ++i) { - m_m.dec_ref(i->m_value); - m_m.dec_ref(i->m_key); - } - } - } - - // - // Returns true iff model was successfully constructed. - // - bool check() { - for (unsigned i = 0; i < m_abstr_model->get_num_constants(); i++) { - func_decl * const c = m_abstr_model->get_constant(i); - app * const term = m_info->find_term(c); - if (term) m_stack.push_back(term); - else m_stack.push_back(m_m.mk_const(c)); - } - return run(); - } - - - void make_model(model_ref& destination) { - { - for (unsigned i = 0; i < m_abstr_model->get_num_uninterpreted_sorts(); i++) { - sort * const s = m_abstr_model->get_uninterpreted_sort(i); - ptr_vector u = m_abstr_model->get_universe(s); - destination->register_usort(s, u.size(), u.c_ptr()); - } - } - for (unsigned i = 0; i < m_abstr_model->get_num_functions(); i++) { - func_decl * const fd = m_abstr_model->get_function(i); - func_interp * const fi = m_abstr_model->get_func_interp(fd); - destination->register_decl(fd, fi); - } - - { - const app2val_t::iterator e = m_app2val.end(); - app2val_t::iterator i = m_app2val.end(); - for (; i != e; ++i) { - app * a = i->m_key; - if (a->get_num_args()) continue; - destination->register_decl(a->get_decl(), i->m_value); - } - } - - obj_map interpretations; - { - const values2val_t::iterator e = m_values2val.end(); - values2val_t::iterator i = m_values2val.end(); - for (; i != e; ++i) add_entry(i->m_key, i->m_value.value, interpretations); - } - - { - obj_map::iterator ie = interpretations.end(); - obj_map::iterator ii = interpretations.begin(); - for (; ii != ie; ++ii) { - func_decl* const fd = ii->m_key; - func_interp* const fi = ii->get_value(); - fi->set_else(m_m.get_some_value(fd->get_range())); - destination->register_decl(fd, fi); - } - } - } - - void add_entry(app* term, expr* value, - obj_map& interpretations) { - func_interp* fi = 0; - func_decl * const declaration = term->get_decl(); - const unsigned sz = declaration->get_arity(); - SASSERT(sz == term->get_num_args()); - if (!interpretations.find(declaration, fi)) { - fi = alloc(func_interp, m_m, sz); - interpretations.insert(declaration, fi); - } - fi->insert_new_entry(term->get_args(), value); - } - private: - ast_manager& m_m; - ackr_info_ref m_info; - model_ref& m_abstr_model; - conflict_list& m_conflicts; - bool_rewriter m_b_rw; - bv_rewriter m_bv_rw; - scoped_ptr m_evaluator; - model m_empty_model; - private: - struct val_info { expr * value; app * source_term; }; - typedef obj_map app2val_t; - typedef obj_map values2val_t; - values2val_t m_values2val; - app2val_t m_app2val; - ptr_vector m_stack; - ackr_helper m_ackr_helper; - - static inline val_info mk_val_info(expr* value, app* source_term) { - val_info rv; - rv.value = value; - rv.source_term = source_term; - return rv; - } - - // - // Performs congruence check on terms on the stack. - // (Currently stops upon the first failure). - // Returns true if and only if congruence check succeeded. - bool run() { - m_evaluator = alloc(model_evaluator, m_empty_model); - expr_mark visited; - expr * curr; - while (!m_stack.empty()) { - curr = m_stack.back(); - if (visited.is_marked(curr)) { - m_stack.pop_back(); - continue; - } - - switch (curr->get_kind()) { - case AST_VAR: - UNREACHABLE(); - return false; - case AST_APP: { - app * a = to_app(curr); - if (for_each_expr_args(m_stack, visited, a->get_num_args(), a->get_args())) { - visited.mark(a, true); - m_stack.pop_back(); - if (!mk_value(a)) return false; - } - } - break; - case AST_QUANTIFIER: - UNREACHABLE(); - return false; - default: - UNREACHABLE(); - return false; - } - } - return true; - } - - inline bool is_val(expr * e) { return m_m.is_value(e); } - - inline bool eval_cached(app * a, expr *& val) { - if (is_val(a)) { val = a; return true; } - return m_app2val.find(a, val); - } - - bool evaluate(app * const a, expr_ref& result) { - SASSERT(!is_val(a)); - const unsigned num = a->get_num_args(); - if (num == 0) { // handle constants - make_value_constant(a, result); - return true; - } - // evaluate arguments - expr_ref_vector values(m_m); - values.reserve(num); - expr * const * args = a->get_args(); - for (unsigned i = 0; i < num; ++i) { - expr * val; - const bool b = eval_cached(to_app(args[i]), val); // TODO: OK conversion to_app? - CTRACE("model_constructor", !b, tout << "fail arg val(\n" << mk_ismt2_pp(args[i], m_m, 2); ); - TRACE("model_constructor", tout << - "arg val " << i << "(\n" << mk_ismt2_pp(args[i], m_m, 2) - << " : " << mk_ismt2_pp(val, m_m, 2) << '\n'; ); - SASSERT(b); - values[i] = val; - } - // handle functions - if (m_ackr_helper.should_ackermannize(a)) { // handle uninterpreted - app_ref key(m_m.mk_app(a->get_decl(), values.c_ptr()), m_m); - if (!make_value_uninterpreted_function(a, values, key.get(), result)) { - return false; - } - } - else { // handle interpreted - make_value_interpreted_function(a, values, result); - } - return true; - } - - // - // Check and record the value for a given term, given that all arguments are already checked. - // - bool mk_value(app * a) { - if (is_val(a)) return true; // skip numerals - TRACE("model_constructor", tout << "mk_value(\n" << mk_ismt2_pp(a, m_m, 2) << ")\n";); - SASSERT(!m_app2val.contains(a)); - const unsigned num = a->get_num_args(); - expr_ref result(m_m); - if (!evaluate(a, result)) return false; - SASSERT(is_val(result)); - TRACE("model_constructor", - tout << "map term(\n" << mk_ismt2_pp(a, m_m, 2) << "\n->" - << mk_ismt2_pp(result.get(), m_m, 2)<< ")\n"; ); - CTRACE("model_constructor", - !is_val(result.get()), - tout << "eval fail\n" << mk_ismt2_pp(a, m_m, 2) << mk_ismt2_pp(result, m_m, 2) << "\n"; - ); - SASSERT(is_val(result.get())); - m_app2val.insert(a, result.get()); // memoize - m_m.inc_ref(a); - m_m.inc_ref(result.get()); - return true; - } - - // Constants from the abstract model are directly mapped to the concrete one. - void make_value_constant(app * const a, expr_ref& result) { - SASSERT(a->get_num_args() == 0); - func_decl * const fd = a->get_decl(); - expr * val = m_abstr_model->get_const_interp(fd); - if (val == 0) { // TODO: avoid model completetion? - sort * s = fd->get_range(); - val = m_abstr_model->get_some_value(s); - } - result = val; - } - - bool make_value_uninterpreted_function(app* a, - expr_ref_vector& values, - app* key, - expr_ref& result) { - // get ackermann constant - app * const ac = m_info->get_abstr(a); - func_decl * const a_fd = a->get_decl(); - SASSERT(ac->get_num_args() == 0); - SASSERT(a_fd->get_range() == ac->get_decl()->get_range()); - expr_ref value(m_m); - value = m_abstr_model->get_const_interp(ac->get_decl()); - // get ackermann constant's interpretation - if (value.get() == 0) { // TODO: avoid model completion? - sort * s = a_fd->get_range(); - value = m_abstr_model->get_some_value(s); - } - // check congruence - val_info vi; - if(m_values2val.find(key,vi)) { // already is mapped to a value - SASSERT(vi.source_term); - const bool ok = vi.value == value; - if (!ok) { - TRACE("model_constructor", - tout << "already mapped by(\n" << mk_ismt2_pp(vi.source_term, m_m, 2) << "\n->" - << mk_ismt2_pp(vi.value, m_m, 2) << ")\n"; ); - m_conflicts.push_back(std::make_pair(a, vi.source_term)); - } - result = vi.value; - return ok; - } else { // new value - result = value; - vi.value = value; - vi.source_term = a; - m_values2val.insert(key,vi); - m_m.inc_ref(vi.source_term); - m_m.inc_ref(vi.value); - m_m.inc_ref(key); - return true; - } - UNREACHABLE(); - } - - void make_value_interpreted_function(app* a, - expr_ref_vector& values, - expr_ref& result) { - const unsigned num = values.size(); - func_decl * const fd = a->get_decl(); - const family_id fid = fd->get_family_id(); - expr_ref term(m_m); - term = m_m.mk_app(a->get_decl(), num, values.c_ptr()); - m_evaluator->operator() (term, result); - TRACE("model_constructor", - tout << "eval(\n" << mk_ismt2_pp(term.get(), m_m, 2) << "\n->" - << mk_ismt2_pp(result.get(), m_m, 2) << ")\n"; ); - return; - if (fid == m_b_rw.get_fid()) { - decl_kind k = fd->get_decl_kind(); - if (k == OP_EQ) { - // theory dispatch for = - SASSERT(num == 2); - family_id s_fid = m_m.get_sort(values.get(0))->get_family_id(); - br_status st = BR_FAILED; - if (s_fid == m_bv_rw.get_fid()) - st = m_bv_rw.mk_eq_core(values.get(0), values.get(1), result); - } else { - br_status st = m_b_rw.mk_app_core(fd, num, values.c_ptr(), result); - } - } else { - br_status st = BR_FAILED; - if (fid == m_bv_rw.get_fid()) { - st = m_bv_rw.mk_app_core(fd, num, values.c_ptr(), result); - } - else { - UNREACHABLE(); - } - } - } -}; - -lackr_model_constructor::lackr_model_constructor(ast_manager& m, ackr_info_ref info) - : m_imp(0) - , m_m(m) - , m_state(UNKNOWN) - , m_info(info) - , m_ref_count(0) -{} - -lackr_model_constructor::~lackr_model_constructor() { - if (m_imp) dealloc(m_imp); -} - -bool lackr_model_constructor::check(model_ref& abstr_model) { - m_conflicts.reset(); - if (m_imp) { - dealloc(m_imp); - m_imp = 0; - } - m_imp = alloc(lackr_model_constructor::imp, m_m, m_info, abstr_model, m_conflicts); - const bool rv = m_imp->check(); - m_state = rv ? CHECKED : CONFLICT; - return rv; -} - -void lackr_model_constructor::make_model(model_ref& model) { - SASSERT(m_state == CHECKED); - m_imp->make_model(model); -} diff --git a/src/tactic/ackr/lackr_model_constructor.h b/src/tactic/ackr/lackr_model_constructor.h deleted file mode 100644 index ce9edba26..000000000 --- a/src/tactic/ackr/lackr_model_constructor.h +++ /dev/null @@ -1,61 +0,0 @@ - /*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - model_constructor.h - - Abstract: - Given a propositional abstraction, attempt to construct a model. - - - Author: - - Mikolas Janota - - Revision History: - --*/ -#ifndef LACKR_MODEL_CONSTRUCTOR_H_ -#define LACKR_MODEL_CONSTRUCTOR_H_ - -#include"ast.h" -#include"ackr_info.h" -#include"ackr_helper.h" -#include"model.h" - -class lackr_model_constructor { - public: - typedef std::pair app_pair; - typedef vector conflict_list; - lackr_model_constructor(ast_manager& m, ackr_info_ref info); - ~lackr_model_constructor(); - bool check(model_ref& abstr_model); - const conflict_list& get_conflicts() { - SASSERT(m_state == CONFLICT); - return m_conflicts; - } - void make_model(model_ref& model); - - // - // Reference counting - // - void inc_ref() { ++m_ref_count; } - void dec_ref() { - --m_ref_count; - if (m_ref_count == 0) { - dealloc(this); - } - } - private: - struct imp; - imp * m_imp; - enum {CHECKED, CONFLICT, UNKNOWN} m_state; - conflict_list m_conflicts; - ast_manager& m_m; - const ackr_info_ref m_info; - - unsigned m_ref_count; // reference counting -}; - -typedef ref lackr_model_constructor_ref; -#endif /* MODEL_CONSTRUCTOR_H_ */ diff --git a/src/tactic/ackr/lackr_model_converter_lazy.cpp b/src/tactic/ackr/lackr_model_converter_lazy.cpp deleted file mode 100644 index 30dba8a04..000000000 --- a/src/tactic/ackr/lackr_model_converter_lazy.cpp +++ /dev/null @@ -1,60 +0,0 @@ -/*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - lackr_model_converter_lazy.cpp - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: ---*/ -#include"lackr_model_converter_lazy.h" -#include"model_evaluator.h" -#include"ast_smt2_pp.h" -#include"ackr_info.h" -#include"lackr_model_constructor.h" - - -class lackr_model_converter_lazy : public model_converter { -public: - lackr_model_converter_lazy(ast_manager & m, - const lackr_model_constructor_ref& lmc) - : m(m) - , model_constructor(lmc) - { } - - virtual ~lackr_model_converter_lazy() { } - - virtual void operator()(model_ref & md, unsigned goal_idx) { - SASSERT(goal_idx == 0); - SASSERT(md.get() == 0 || (!md->get_num_constants() && !md->get_num_functions())); - SASSERT(model_constructor.get()); - model * new_model = alloc(model, m); - md = new_model; - model_constructor->make_model(md); - } - - virtual void operator()(model_ref & md) { - operator()(md, 0); - } - - //void display(std::ostream & out); - - virtual model_converter * translate(ast_translation & translator) { - NOT_IMPLEMENTED_YET(); - } -protected: - ast_manager& m; - const lackr_model_constructor_ref model_constructor; -}; - -model_converter * mk_lackr_model_converter_lazy(ast_manager & m, - const lackr_model_constructor_ref& model_constructor) { - return alloc(lackr_model_converter_lazy, m, model_constructor); -} diff --git a/src/tactic/ackr/lackr_model_converter_lazy.h b/src/tactic/ackr/lackr_model_converter_lazy.h deleted file mode 100644 index c9b3eac80..000000000 --- a/src/tactic/ackr/lackr_model_converter_lazy.h +++ /dev/null @@ -1,25 +0,0 @@ - /*++ - Copyright (c) 2015 Microsoft Corporation - - Module Name: - - lackr_model_converter_lazy.h - - Abstract: - - - Author: - - Mikolas Janota - - Revision History: - --*/ -#ifndef LACKR_MODEL_CONVERTER_LAZY_H_ -#define LACKR_MODEL_CONVERTER_LAZY_H_ - -#include"model_converter.h" -#include"ackr_info.h" - -model_converter * mk_lackr_model_converter_lazy(ast_manager & m, const ackr_info_ref& info, model_ref& abstr_model); - -#endif /* LACKR_MODEL_CONVERTER_LAZY_H_ */ diff --git a/src/tactic/smtlogics/qfufbv_tactic.cpp b/src/tactic/smtlogics/qfufbv_tactic.cpp index 526d72faa..c3ebb4f48 100644 --- a/src/tactic/smtlogics/qfufbv_tactic.cpp +++ b/src/tactic/smtlogics/qfufbv_tactic.cpp @@ -32,6 +32,7 @@ Notes: #include"model_smt2_pp.h" #include"cooperate.h" #include"lackr.h" +#include"ackermannization_params.hpp" #include"qfufbv_ackr_model_converter.h" /////////////// #include"inc_sat_solver.h" @@ -87,7 +88,7 @@ public: } virtual void collect_statistics(statistics & st) const { - ackr_params p(m_p); + ackermannization_params p(m_p); if (!p.eager()) st.update("lackr-its", m_st.m_it); st.update("ackr-constraints", m_st.m_ackrs_sz); } @@ -172,9 +173,7 @@ tactic * mk_qfufbv_tactic(ast_manager & m, params_ref const & p) { tactic * const preamble_st = mk_qfufbv_preamble(m, p); tactic * st = using_params(and_then(preamble_st, - cond(mk_is_qfbv_probe(), - mk_qfbv_tactic(m), - mk_smt_tactic())), + cond(mk_is_qfbv_probe(), mk_qfbv_tactic(m), mk_smt_tactic())), main_p); st->updt_params(p); @@ -187,4 +186,4 @@ tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p) { tactic * const actual_tactic = alloc(qfufbv_ackr_tactic, m, p); return and_then(preamble_t, cond(mk_is_qfufbv_probe(), actual_tactic, mk_smt_tactic())); -} \ No newline at end of file +} From 8547a965ab37309dd439ac18101e4dca9d993f94 Mon Sep 17 00:00:00 2001 From: Mikolas Janota Date: Thu, 4 Feb 2016 14:05:40 +0000 Subject: [PATCH 30/31] changing preamble for qfufbv_ackr_tactic. --- src/tactic/smtlogics/qfufbv_tactic.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/tactic/smtlogics/qfufbv_tactic.cpp b/src/tactic/smtlogics/qfufbv_tactic.cpp index c3ebb4f48..f943b68c9 100644 --- a/src/tactic/smtlogics/qfufbv_tactic.cpp +++ b/src/tactic/smtlogics/qfufbv_tactic.cpp @@ -181,7 +181,7 @@ tactic * mk_qfufbv_tactic(ast_manager & m, params_ref const & p) { } tactic * mk_qfufbv_ackr_tactic(ast_manager & m, params_ref const & p) { - tactic * const preamble_t = mk_qfufbv_preamble(m, p); + tactic * const preamble_t = mk_qfufbv_preamble1(m, p); tactic * const actual_tactic = alloc(qfufbv_ackr_tactic, m, p); return and_then(preamble_t, From fa68b005634db48aa45a0070c6600c6c102bbe02 Mon Sep 17 00:00:00 2001 From: "Christoph M. Wintersteiger" Date: Wed, 10 Feb 2016 14:39:33 +0000 Subject: [PATCH 31/31] Cleanliness --- .../lackr_model_constructor.cpp | 18 ++++++++---------- src/ackermannization/lackr_model_constructor.h | 2 +- 2 files changed, 9 insertions(+), 11 deletions(-) diff --git a/src/ackermannization/lackr_model_constructor.cpp b/src/ackermannization/lackr_model_constructor.cpp index 496f65383..09034fa36 100644 --- a/src/ackermannization/lackr_model_constructor.cpp +++ b/src/ackermannization/lackr_model_constructor.cpp @@ -24,10 +24,10 @@ struct lackr_model_constructor::imp { public: - imp(ast_manager& m, + imp(ast_manager & m, ackr_info_ref info, - model_ref& abstr_model, - conflict_list& conflicts) + model_ref & abstr_model, + conflict_list & conflicts) : m_m(m) , m_info(info) , m_abstr_model(abstr_model) @@ -239,7 +239,6 @@ struct lackr_model_constructor::imp { if (is_val(a)) return true; // skip numerals TRACE("model_constructor", tout << "mk_value(\n" << mk_ismt2_pp(a, m_m, 2) << ")\n";); SASSERT(!m_app2val.contains(a)); - const unsigned num = a->get_num_args(); expr_ref result(m_m); if (!evaluate(a, result)) return false; SASSERT(is_val(result)); @@ -309,8 +308,9 @@ struct lackr_model_constructor::imp { return true; } UNREACHABLE(); + return true; } - + void make_value_interpreted_function(app* a, expr_ref_vector& values, expr_ref& result) { @@ -330,16 +330,14 @@ struct lackr_model_constructor::imp { // theory dispatch for = SASSERT(num == 2); family_id s_fid = m_m.get_sort(values.get(0))->get_family_id(); - br_status st = BR_FAILED; if (s_fid == m_bv_rw.get_fid()) - st = m_bv_rw.mk_eq_core(values.get(0), values.get(1), result); + m_bv_rw.mk_eq_core(values.get(0), values.get(1), result); } else { - br_status st = m_b_rw.mk_app_core(fd, num, values.c_ptr(), result); + m_b_rw.mk_app_core(fd, num, values.c_ptr(), result); } } else { - br_status st = BR_FAILED; if (fid == m_bv_rw.get_fid()) { - st = m_bv_rw.mk_app_core(fd, num, values.c_ptr(), result); + m_bv_rw.mk_app_core(fd, num, values.c_ptr(), result); } else { UNREACHABLE(); diff --git a/src/ackermannization/lackr_model_constructor.h b/src/ackermannization/lackr_model_constructor.h index ce9edba26..71c9c0710 100644 --- a/src/ackermannization/lackr_model_constructor.h +++ b/src/ackermannization/lackr_model_constructor.h @@ -49,9 +49,9 @@ class lackr_model_constructor { private: struct imp; imp * m_imp; + ast_manager & m_m; enum {CHECKED, CONFLICT, UNKNOWN} m_state; conflict_list m_conflicts; - ast_manager& m_m; const ackr_info_ref m_info; unsigned m_ref_count; // reference counting