mirror of
https://github.com/Z3Prover/z3
synced 2025-10-24 08:24:34 +00:00
fix #5065 - regression solving str.from_int equations now that it isn't injective any longer
Signed-off-by: Nikolaj Bjorner <nbjorner@microsoft.com>
This commit is contained in:
parent
4c9fed21e2
commit
0ce1c34d81
29 changed files with 218 additions and 178 deletions
|
@ -33,6 +33,7 @@ z3_add_component(rewriter
|
|||
recfun_rewriter.cpp
|
||||
rewriter.cpp
|
||||
seq_axioms.cpp
|
||||
seq_eq_solver.cpp
|
||||
seq_rewriter.cpp
|
||||
seq_skolem.cpp
|
||||
th_rewriter.cpp
|
||||
|
|
|
@ -48,11 +48,6 @@ namespace seq {
|
|||
expr_ref mk_seq_eq(expr* a, expr* b);
|
||||
expr_ref mk_eq_empty(expr* e);
|
||||
|
||||
expr_ref mk_ge(expr* x, int n) { return mk_ge_e(x, a.mk_int(n)); }
|
||||
expr_ref mk_le(expr* x, int n) { return mk_le_e(x, a.mk_int(n)); }
|
||||
expr_ref mk_ge(expr* x, rational const& n) { return mk_ge_e(x, a.mk_int(n)); }
|
||||
expr_ref mk_le(expr* x, rational const& n) { return mk_le_e(x, a.mk_int(n)); }
|
||||
|
||||
expr_ref mk_ge_e(expr* x, expr* y);
|
||||
expr_ref mk_le_e(expr* x, expr* y);
|
||||
|
||||
|
@ -111,6 +106,11 @@ namespace seq {
|
|||
expr_ref length_limit(expr* s, unsigned k);
|
||||
expr_ref is_digit(expr* ch);
|
||||
|
||||
expr_ref mk_ge(expr* x, int n) { return mk_ge_e(x, a.mk_int(n)); }
|
||||
expr_ref mk_le(expr* x, int n) { return mk_le_e(x, a.mk_int(n)); }
|
||||
expr_ref mk_ge(expr* x, rational const& n) { return mk_ge_e(x, a.mk_int(n)); }
|
||||
expr_ref mk_le(expr* x, rational const& n) { return mk_le_e(x, a.mk_int(n)); }
|
||||
|
||||
};
|
||||
|
||||
};
|
||||
|
|
|
@ -97,6 +97,8 @@ namespace smt {
|
|||
literal mk_ge(expr* e, rational const& k) { return mk_ge_e(e, a.mk_int(k)); }
|
||||
literal mk_le(expr* e, rational const& k) { return mk_le_e(e, a.mk_int(k)); }
|
||||
|
||||
seq::axioms& ax() { return m_ax; }
|
||||
|
||||
};
|
||||
|
||||
};
|
||||
|
|
|
@ -41,18 +41,18 @@ bool theory_seq::solve_eqs(unsigned i) {
|
|||
}
|
||||
|
||||
bool theory_seq::solve_eq(unsigned idx) {
|
||||
const eq& e = m_eqs[idx];
|
||||
const depeq& e = m_eqs[idx];
|
||||
expr_ref_vector& ls = m_ls;
|
||||
expr_ref_vector& rs = m_rs;
|
||||
m_ls.reset();
|
||||
m_rs.reset();
|
||||
dependency* dep2 = nullptr;
|
||||
bool change = false;
|
||||
if (!canonize(e.ls(), ls, dep2, change)) return false;
|
||||
if (!canonize(e.rs(), rs, dep2, change)) return false;
|
||||
if (!canonize(e.ls, ls, dep2, change)) return false;
|
||||
if (!canonize(e.rs, rs, dep2, change)) return false;
|
||||
dependency* deps = m_dm.mk_join(dep2, e.dep());
|
||||
TRACE("seq_verbose",
|
||||
tout << e.ls() << " = " << e.rs() << " ==> ";
|
||||
tout << e.ls << " = " << e.rs << " ==> ";
|
||||
tout << ls << " = " << rs << "\n";
|
||||
display_deps(tout, deps););
|
||||
|
||||
|
@ -81,6 +81,14 @@ bool theory_seq::solve_eq(unsigned idx) {
|
|||
if (!ctx.inconsistent() && solve_nth_eq1(rs, ls, deps)) {
|
||||
return true;
|
||||
}
|
||||
seq::eq_ptr r;
|
||||
m_eq_deps = deps;
|
||||
if (!ctx.inconsistent() && m_eq.solve(e, r)) {
|
||||
if (!r)
|
||||
return true;
|
||||
m_eqs.set(idx, depeq(m_eq_id++, r->ls, r->rs, deps));
|
||||
return false;
|
||||
}
|
||||
if (!ctx.inconsistent() && solve_itos(rs, ls, deps)) {
|
||||
return true;
|
||||
}
|
||||
|
@ -90,16 +98,34 @@ bool theory_seq::solve_eq(unsigned idx) {
|
|||
expr_ref_vector new_ls(m);
|
||||
if (!m_offset_eq.empty() && find_better_rep(ls, rs, idx, deps, new_ls)) {
|
||||
// Find a better equivalent term for lhs of an equation based on length constraints
|
||||
m_eqs.push_back(eq(m_eq_id++, new_ls, rs, deps));
|
||||
m_eqs.push_back(depeq(m_eq_id++, new_ls, rs, deps));
|
||||
return true;
|
||||
}
|
||||
else {
|
||||
m_eqs.set(idx, eq(m_eq_id++, ls, rs, deps));
|
||||
m_eqs.set(idx, depeq(m_eq_id++, ls, rs, deps));
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
void theory_seq::add_consequence(bool uses_eq, expr_ref_vector const& clause) {
|
||||
dependency* dep = uses_eq ? m_eq_deps : nullptr;
|
||||
if (clause.size() == 1) {
|
||||
propagate_lit(dep, 0, nullptr, mk_literal(clause[0]));
|
||||
return;
|
||||
}
|
||||
enode_pair_vector eqs;
|
||||
literal_vector lits;
|
||||
linearize(dep, eqs, lits);
|
||||
for (auto& lit : lits)
|
||||
lit.neg();
|
||||
for (auto eq : eqs)
|
||||
lits.push_back(~mk_eq(eq.first->get_expr(), eq.second->get_expr(), false));
|
||||
for (auto f : clause)
|
||||
lits.push_back(mk_literal(f));
|
||||
add_axiom(lits);
|
||||
}
|
||||
|
||||
bool theory_seq::solve_unit_eq(expr* l, expr* r, dependency* deps) {
|
||||
if (l == r) {
|
||||
return true;
|
||||
|
@ -169,7 +195,6 @@ bool theory_seq::solve_binary_eq(expr_ref_vector const& ls, expr_ref_vector cons
|
|||
case l_true:
|
||||
break;
|
||||
case l_undef:
|
||||
ctx.mark_as_relevant(eq);
|
||||
propagate_lit(dep, 0, nullptr, eq);
|
||||
m_new_propagation = true;
|
||||
break;
|
||||
|
@ -277,14 +302,14 @@ bool theory_seq::find_better_rep(expr_ref_vector const& ls, expr_ref_vector cons
|
|||
// Offset = 0, Changed
|
||||
|
||||
for (unsigned i = 0; i < idx; ++i) {
|
||||
eq const& e = m_eqs[i];
|
||||
if (e.ls() != ls) continue;
|
||||
depeq const& e = m_eqs[i];
|
||||
if (e.ls != ls) continue;
|
||||
expr* nl_fst = nullptr;
|
||||
if (e.rs().size() > 1 && is_var(e.rs().get(0)))
|
||||
nl_fst = e.rs().get(0);
|
||||
if (e.rs.size() > 1 && is_var(e.rs.get(0)))
|
||||
nl_fst = e.rs.get(0);
|
||||
if (nl_fst && nl_fst != r_fst && root2 == get_root(mk_len(nl_fst))) {
|
||||
res.reset();
|
||||
res.append(e.rs());
|
||||
res.append(e.rs);
|
||||
deps = m_dm.mk_join(e.dep(), deps);
|
||||
return true;
|
||||
}
|
||||
|
@ -300,18 +325,18 @@ bool theory_seq::find_better_rep(expr_ref_vector const& ls, expr_ref_vector cons
|
|||
// Offset != 0, Changed
|
||||
if (m_offset_eq.contains(root2)) {
|
||||
for (unsigned i = 0; i < idx; ++i) {
|
||||
eq const& e = m_eqs[i];
|
||||
if (e.ls() != ls) continue;
|
||||
depeq const& e = m_eqs[i];
|
||||
if (e.ls != ls) continue;
|
||||
expr* nl_fst = nullptr;
|
||||
if (e.rs().size() > 1 && is_var(e.rs().get(0)))
|
||||
nl_fst = e.rs().get(0);
|
||||
if (e.rs.size() > 1 && is_var(e.rs.get(0)))
|
||||
nl_fst = e.rs.get(0);
|
||||
if (nl_fst && nl_fst != r_fst) {
|
||||
expr_ref len_nl_fst = mk_len(nl_fst);
|
||||
if (ctx.e_internalized(len_nl_fst)) {
|
||||
enode * root1 = get_root(len_nl_fst);
|
||||
if (m_offset_eq.contains(root2, root1)) {
|
||||
res.reset();
|
||||
res.append(e.rs());
|
||||
res.append(e.rs);
|
||||
deps = m_dm.mk_join(e.dep(), deps);
|
||||
return true;
|
||||
}
|
||||
|
@ -363,9 +388,9 @@ bool theory_seq::len_based_split() {
|
|||
eq const& e = m_eqs[i];
|
||||
expr_ref_vector new_ls(m);
|
||||
dependency *deps = e.dep();
|
||||
if (find_better_rep(e.ls(), e.rs(), i, deps, new_ls)) {
|
||||
if (find_better_rep(e.ls, e.rs, i, deps, new_ls)) {
|
||||
expr_ref_vector rs(m);
|
||||
rs.append(e.rs());
|
||||
rs.append(e.rs);
|
||||
m_eqs.set(i, eq(m_eq_id++, new_ls, rs, deps));
|
||||
TRACE("seq", display_equation(tout, m_eqs[i]););
|
||||
}
|
||||
|
@ -400,9 +425,9 @@ bool theory_seq::len_based_split() {
|
|||
|
||||
*/
|
||||
|
||||
bool theory_seq::len_based_split(eq const& e) {
|
||||
expr_ref_vector const& ls = e.ls();
|
||||
expr_ref_vector const& rs = e.rs();
|
||||
bool theory_seq::len_based_split(depeq const& e) {
|
||||
expr_ref_vector const& ls = e.ls;
|
||||
expr_ref_vector const& rs = e.rs;
|
||||
|
||||
int offset = 0;
|
||||
if (!has_len_offset(ls, rs, offset))
|
||||
|
@ -487,17 +512,17 @@ bool theory_seq::branch_variable_mb() {
|
|||
int start = ctx.get_random_value();
|
||||
for (unsigned i = 0; i < sz; ++i) {
|
||||
unsigned k = (i + start) % sz;
|
||||
eq const& e = m_eqs[k];
|
||||
depeq const& e = m_eqs[k];
|
||||
vector<rational> len1, len2;
|
||||
if (!is_complex(e)) {
|
||||
continue;
|
||||
}
|
||||
if (e.ls().empty() || e.rs().empty() ||
|
||||
(!is_var(e.ls()[0]) && !is_var(e.rs()[0]))) {
|
||||
if (e.ls.empty() || e.rs.empty() ||
|
||||
(!is_var(e.ls[0]) && !is_var(e.rs[0]))) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (!enforce_length(e.ls(), len1) || !enforce_length(e.rs(), len2)) {
|
||||
if (!enforce_length(e.ls, len1) || !enforce_length(e.rs, len2)) {
|
||||
// change = true;
|
||||
continue;
|
||||
}
|
||||
|
@ -505,15 +530,15 @@ bool theory_seq::branch_variable_mb() {
|
|||
for (const auto& elem : len1) l1 += elem;
|
||||
for (const auto& elem : len2) l2 += elem;
|
||||
if (l1 != l2) {
|
||||
expr_ref l = mk_concat(e.ls());
|
||||
expr_ref r = mk_concat(e.rs());
|
||||
expr_ref l = mk_concat(e.ls);
|
||||
expr_ref r = mk_concat(e.rs);
|
||||
expr_ref lnl = mk_len(l), lnr = mk_len(r);
|
||||
if (propagate_eq(e.dep(), lnl, lnr, false)) {
|
||||
change = true;
|
||||
}
|
||||
continue;
|
||||
}
|
||||
if (split_lengths(e.dep(), e.ls(), e.rs(), len1, len2)) {
|
||||
if (split_lengths(e.dep(), e.ls, e.rs, len1, len2)) {
|
||||
TRACE("seq", tout << "split lengths\n";);
|
||||
change = true;
|
||||
break;
|
||||
|
@ -523,12 +548,12 @@ bool theory_seq::branch_variable_mb() {
|
|||
}
|
||||
|
||||
|
||||
bool theory_seq::is_complex(eq const& e) {
|
||||
bool theory_seq::is_complex(depeq const& e) {
|
||||
unsigned num_vars1 = 0, num_vars2 = 0;
|
||||
for (auto const& elem : e.ls()) {
|
||||
for (auto const& elem : e.ls) {
|
||||
if (is_var(elem)) ++num_vars1;
|
||||
}
|
||||
for (auto const& elem : e.rs()) {
|
||||
for (auto const& elem : e.rs) {
|
||||
if (is_var(elem)) ++num_vars2;
|
||||
}
|
||||
return num_vars1 > 0 && num_vars2 > 0 && num_vars1 + num_vars2 > 2;
|
||||
|
@ -639,14 +664,14 @@ bool theory_seq::branch_binary_variable() {
|
|||
return false;
|
||||
}
|
||||
|
||||
bool theory_seq::branch_binary_variable(eq const& e) {
|
||||
bool theory_seq::branch_binary_variable(depeq const& e) {
|
||||
if (is_complex(e)) {
|
||||
return false;
|
||||
}
|
||||
ptr_vector<expr> xs, ys;
|
||||
expr_ref x(m), y(m);
|
||||
if (!is_binary_eq(e.ls(), e.rs(), x, xs, ys, y) &&
|
||||
!is_binary_eq(e.rs(), e.ls(), x, xs, ys, y))
|
||||
if (!is_binary_eq(e.ls, e.rs, x, xs, ys, y) &&
|
||||
!is_binary_eq(e.rs, e.ls, x, xs, ys, y))
|
||||
return false;
|
||||
if (x == y) {
|
||||
return false;
|
||||
|
@ -706,13 +731,13 @@ bool theory_seq::branch_binary_variable(eq const& e) {
|
|||
bool theory_seq::branch_unit_variable() {
|
||||
bool result = false;
|
||||
for (auto const& e : m_eqs) {
|
||||
if (is_unit_eq(e.ls(), e.rs()) &&
|
||||
branch_unit_variable(e.dep(), e.ls()[0], e.rs())) {
|
||||
if (is_unit_eq(e.ls, e.rs) &&
|
||||
branch_unit_variable(e.dep(), e.ls[0], e.rs)) {
|
||||
result = true;
|
||||
break;
|
||||
}
|
||||
if (is_unit_eq(e.rs(), e.ls()) &&
|
||||
branch_unit_variable(e.dep(), e.rs()[0], e.ls())) {
|
||||
if (is_unit_eq(e.rs, e.ls) &&
|
||||
branch_unit_variable(e.dep(), e.rs[0], e.ls)) {
|
||||
result = true;
|
||||
break;
|
||||
}
|
||||
|
@ -861,11 +886,11 @@ bool theory_seq::can_align_from_rhs(expr_ref_vector const& ls, expr_ref_vector c
|
|||
// Equation is of the form x ++ xs = y1 ++ ys ++ y2 where xs, ys are units.
|
||||
// If xs and ys cannot align then
|
||||
// x ++ xs = y1 ++ ys ++ y2 => x = y1 ++ ys ++ z, z ++ xs = y2
|
||||
bool theory_seq::branch_ternary_variable_rhs(eq const& e) {
|
||||
bool theory_seq::branch_ternary_variable_rhs(depeq const& e) {
|
||||
expr_ref_vector xs(m), ys(m);
|
||||
expr_ref x(m), y1(m), y2(m);
|
||||
if (!is_ternary_eq_rhs(e.ls(), e.rs(), x, xs, y1, ys, y2) &&
|
||||
!is_ternary_eq_rhs(e.rs(), e.ls(), x, xs, y1, ys, y2)) {
|
||||
if (!is_ternary_eq_rhs(e.ls, e.rs, x, xs, y1, ys, y2) &&
|
||||
!is_ternary_eq_rhs(e.rs, e.ls, x, xs, y1, ys, y2)) {
|
||||
return false;
|
||||
}
|
||||
if (m_sk.is_align_l(y1) || m_sk.is_align_r(y1))
|
||||
|
@ -905,11 +930,11 @@ bool theory_seq::branch_ternary_variable_rhs(eq const& e) {
|
|||
// Equation is of the form xs ++ x = y1 ++ ys ++ y2 where xs, ys are units.
|
||||
// If xs and ys cannot align then
|
||||
// xs ++ x = y1 ++ ys ++ y2 => xs ++ z = y1, x = z ++ ys ++ y2
|
||||
bool theory_seq::branch_ternary_variable_lhs(eq const& e) {
|
||||
bool theory_seq::branch_ternary_variable_lhs(depeq const& e) {
|
||||
expr_ref_vector xs(m), ys(m);
|
||||
expr_ref x(m), y1(m), y2(m);
|
||||
if (!is_ternary_eq_lhs(e.ls(), e.rs(), xs, x, y1, ys, y2) &&
|
||||
!is_ternary_eq_lhs(e.rs(), e.ls(), xs, x, y1, ys, y2))
|
||||
if (!is_ternary_eq_lhs(e.ls, e.rs, xs, x, y1, ys, y2) &&
|
||||
!is_ternary_eq_lhs(e.rs, e.ls, xs, x, y1, ys, y2))
|
||||
return false;
|
||||
if (m_sk.is_align_l(y1) || m_sk.is_align_r(y1))
|
||||
return false;
|
||||
|
@ -970,10 +995,10 @@ literal theory_seq::mk_alignment(expr* e1, expr* e2) {
|
|||
// Equation is of the form x1 ++ xs ++ x2 = y1 ++ ys ++ y2 where xs, ys are units.
|
||||
// If xs and ys cannot align then
|
||||
// x1 ++ xs ++ x2 = y1 ++ ys ++ y2 => |x1| >= |y1 ++ ys| V |x1 ++ xs| <= |y1|
|
||||
bool theory_seq::branch_quat_variable(eq const& e) {
|
||||
bool theory_seq::branch_quat_variable(depeq const& e) {
|
||||
expr_ref_vector xs(m), ys(m);
|
||||
expr_ref x1(m), x2(m), y1(m), y2(m);
|
||||
if (!is_quat_eq(e.ls(), e.rs(), x1, xs, x2, y1, ys, y2))
|
||||
if (!is_quat_eq(e.ls, e.rs, x1, xs, x2, y1, ys, y2))
|
||||
return false;
|
||||
dependency* dep = e.dep();
|
||||
|
||||
|
@ -1132,7 +1157,7 @@ bool theory_seq::branch_variable_eq() {
|
|||
|
||||
for (unsigned i = 0; i < sz; ++i) {
|
||||
unsigned k = (i + start) % sz;
|
||||
eq const& e = m_eqs[k];
|
||||
depeq const& e = m_eqs[k];
|
||||
|
||||
if (branch_variable_eq(e)) {
|
||||
TRACE("seq", tout << "branch variable\n";);
|
||||
|
@ -1142,15 +1167,15 @@ bool theory_seq::branch_variable_eq() {
|
|||
return ctx.inconsistent();
|
||||
}
|
||||
|
||||
bool theory_seq::branch_variable_eq(eq const& e) {
|
||||
bool theory_seq::branch_variable_eq(depeq const& e) {
|
||||
unsigned id = e.id();
|
||||
unsigned s = find_branch_start(2*id);
|
||||
TRACE("seq", tout << s << " " << id << ": " << e.ls() << " = " << e.rs() << "\n";);
|
||||
bool found = find_branch_candidate(s, e.dep(), e.ls(), e.rs());
|
||||
TRACE("seq", tout << s << " " << id << ": " << e.ls << " = " << e.rs << "\n";);
|
||||
bool found = find_branch_candidate(s, e.dep(), e.ls, e.rs);
|
||||
insert_branch_start(2*id, s);
|
||||
if (!found) {
|
||||
s = find_branch_start(2*id + 1);
|
||||
found = find_branch_candidate(s, e.dep(), e.rs(), e.ls());
|
||||
found = find_branch_candidate(s, e.dep(), e.rs, e.ls);
|
||||
insert_branch_start(2*id + 1, s);
|
||||
}
|
||||
return found;
|
||||
|
@ -1428,8 +1453,8 @@ bool theory_seq::reduce_length_eq() {
|
|||
int start = ctx.get_random_value();
|
||||
|
||||
for (unsigned i = 0; !ctx.inconsistent() && i < m_eqs.size(); ++i) {
|
||||
eq const& e = m_eqs[(i + start) % m_eqs.size()];
|
||||
if (reduce_length_eq(e.ls(), e.rs(), e.dep())) {
|
||||
depeq const& e = m_eqs[(i + start) % m_eqs.size()];
|
||||
if (reduce_length_eq(e.ls, e.rs, e.dep())) {
|
||||
TRACE("seq", tout << "reduce length eq\n";);
|
||||
return true;
|
||||
}
|
||||
|
@ -1631,7 +1656,7 @@ bool theory_seq::solve_nth_eq2(expr_ref_vector const& ls, expr_ref_vector const&
|
|||
rs1.push_back(m_util.str.mk_unit(rhs));
|
||||
rs1.push_back(m_sk.mk_post(s, idx1));
|
||||
TRACE("seq", tout << ls1 << "\n"; tout << rs1 << "\n";);
|
||||
m_eqs.push_back(eq(m_eq_id++, ls1, rs1, deps));
|
||||
m_eqs.push_back(depeq(m_eq_id++, ls1, rs1, deps));
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
|
|
|
@ -1,12 +1,12 @@
|
|||
z3_add_component(smt_tactic
|
||||
SOURCES
|
||||
ctx_solver_simplify_tactic.cpp
|
||||
smt_tactic.cpp
|
||||
smt_tactic_core.cpp
|
||||
unit_subsumption_tactic.cpp
|
||||
COMPONENT_DEPENDENCIES
|
||||
smt
|
||||
TACTIC_HEADERS
|
||||
ctx_solver_simplify_tactic.h
|
||||
smt_tactic.h
|
||||
smt_tactic_core.h
|
||||
unit_subsumption_tactic.h
|
||||
)
|
||||
|
|
|
@ -317,12 +317,12 @@ tactic * mk_parallel_smt_tactic(ast_manager& m, params_ref const& p) {
|
|||
return mk_parallel_tactic(mk_smt_solver(m, p, symbol::null), p);
|
||||
}
|
||||
|
||||
tactic * mk_smt_tactic(ast_manager& m, params_ref const& p, symbol const& logic) {
|
||||
tactic * mk_smt_tactic_core(ast_manager& m, params_ref const& p, symbol const& logic) {
|
||||
parallel_params pp(p);
|
||||
return pp.enable() ? mk_parallel_tactic(mk_smt_solver(m, p, logic), p) : mk_seq_smt_tactic(p);
|
||||
}
|
||||
|
||||
tactic * mk_smt_tactic_using(ast_manager& m, bool auto_config, params_ref const& _p) {
|
||||
tactic * mk_smt_tactic_core_using(ast_manager& m, bool auto_config, params_ref const& _p) {
|
||||
parallel_params pp(_p);
|
||||
params_ref p = _p;
|
||||
p.set_bool("auto_config", auto_config);
|
|
@ -26,14 +26,13 @@ Notes:
|
|||
class tactic;
|
||||
class filter_model_converter;
|
||||
|
||||
tactic * mk_smt_tactic(ast_manager& m, params_ref const & p = params_ref(), symbol const& logic = symbol::null);
|
||||
tactic * mk_smt_tactic_core(ast_manager& m, params_ref const & p = params_ref(), symbol const& logic = symbol::null);
|
||||
// syntax sugar for using_params(mk_smt_tactic(), p) where p = (:auto_config, auto_config)
|
||||
tactic * mk_smt_tactic_using(ast_manager& m, bool auto_config = true, params_ref const & p = params_ref());
|
||||
tactic * mk_smt_tactic_core_using(ast_manager& m, bool auto_config = true, params_ref const & p = params_ref());
|
||||
|
||||
tactic * mk_parallel_smt_tactic(ast_manager& m, params_ref const& p);
|
||||
|
||||
|
||||
/*
|
||||
ADD_TACTIC("smt", "apply a SAT based SMT solver.", "mk_smt_tactic(m, p)")
|
||||
ADD_TACTIC("psmt", "builtin strategy for SMT tactic in parallel.", "mk_parallel_smt_tactic(m, p)")
|
||||
*/
|
|
@ -272,6 +272,7 @@ theory_seq::theory_seq(context& ctx):
|
|||
m_autil(m),
|
||||
m_sk(m, m_rewrite),
|
||||
m_ax(*this, m_rewrite),
|
||||
m_eq(m, m_ax.ax()),
|
||||
m_regex(*this),
|
||||
m_arith_value(m),
|
||||
m_trail_stack(*this),
|
||||
|
@ -285,6 +286,12 @@ theory_seq::theory_seq(context& ctx):
|
|||
m_new_solution(false),
|
||||
m_new_propagation(false) {
|
||||
|
||||
std::function<void(bool, expr_ref_vector const&)> _add_consequence =
|
||||
[&](bool uses_eq, expr_ref_vector const& clause) {
|
||||
add_consequence(uses_eq, clause);
|
||||
};
|
||||
|
||||
m_eq.set_add_consequence(_add_consequence);
|
||||
}
|
||||
|
||||
theory_seq::~theory_seq() {
|
||||
|
@ -674,8 +681,8 @@ bool theory_seq::check_lts() {
|
|||
|
||||
bool theory_seq::is_solved() {
|
||||
if (!m_eqs.empty()) {
|
||||
TRACE("seq", tout << "(seq.giveup " << m_eqs[0].ls() << " = " << m_eqs[0].rs() << " is unsolved)\n";);
|
||||
IF_VERBOSE(10, verbose_stream() << "(seq.giveup " << m_eqs[0].ls() << " = " << m_eqs[0].rs() << " is unsolved)\n";);
|
||||
TRACE("seq", tout << "(seq.giveup " << m_eqs[0].ls << " = " << m_eqs[0].rs << " is unsolved)\n";);
|
||||
IF_VERBOSE(10, verbose_stream() << "(seq.giveup " << m_eqs[0].ls << " = " << m_eqs[0].rs << " is unsolved)\n";);
|
||||
return false;
|
||||
}
|
||||
if (!m_ncs.empty()) {
|
||||
|
@ -1053,7 +1060,7 @@ bool theory_seq::reduce_length_eq(expr_ref_vector const& ls, expr_ref_vector con
|
|||
rhs.append(rs.size()-1, rs.c_ptr() + 1);
|
||||
SASSERT(!lhs.empty() || !rhs.empty());
|
||||
deps = mk_join(deps, lits);
|
||||
m_eqs.push_back(eq(m_eq_id++, lhs, rhs, deps));
|
||||
m_eqs.push_back(depeq(m_eq_id++, lhs, rhs, deps));
|
||||
TRACE("seq", tout << "Propagate equal lengths " << l << " " << r << "\n";);
|
||||
propagate_eq(deps, lits, l, r, true);
|
||||
return true;
|
||||
|
@ -1067,7 +1074,7 @@ bool theory_seq::reduce_length_eq(expr_ref_vector const& ls, expr_ref_vector con
|
|||
SASSERT(!lhs.empty() || !rhs.empty());
|
||||
deps = mk_join(deps, lits);
|
||||
TRACE("seq", tout << "Propagate equal lengths " << l << " " << r << "\n" << "ls: " << ls << "\nrs: " << rs << "\n";);
|
||||
m_eqs.push_back(eq(m_eq_id++, lhs, rhs, deps));
|
||||
m_eqs.push_back(depeq(m_eq_id++, lhs, rhs, deps));
|
||||
propagate_eq(deps, lits, l, r, true);
|
||||
return true;
|
||||
}
|
||||
|
@ -1145,12 +1152,12 @@ bool theory_seq::reduce_length(unsigned i, unsigned j, bool front, expr_ref_vect
|
|||
lhs.append(l2, ls2);
|
||||
rhs.append(r2, rs2);
|
||||
for (auto const& e : m_eqs) {
|
||||
if (e.ls() == lhs && e.rs() == rhs) {
|
||||
if (e.ls == lhs && e.rs == rhs) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
deps = mk_join(deps, lit);
|
||||
m_eqs.push_back(eq(m_eq_id++, lhs, rhs, deps));
|
||||
m_eqs.push_back(depeq(m_eq_id++, lhs, rhs, deps));
|
||||
propagate_eq(deps, l, r, true);
|
||||
TRACE("seq", tout << "propagate eq\n" << m_eqs.size() << "\nlhs: " << lhs << "\nrhs: " << rhs << "\n";);
|
||||
return true;
|
||||
|
@ -1653,11 +1660,11 @@ std::ostream& theory_seq::display_equations(std::ostream& out) const {
|
|||
return out;
|
||||
}
|
||||
|
||||
std::ostream& theory_seq::display_equation(std::ostream& out, eq const& e) const {
|
||||
std::ostream& theory_seq::display_equation(std::ostream& out, depeq const& e) const {
|
||||
bool first = true;
|
||||
for (expr* a : e.ls()) { if (first) first = false; else out << "\n"; out << mk_bounded_pp(a, m, 2); }
|
||||
for (expr* a : e.ls) { if (first) first = false; else out << "\n"; out << mk_bounded_pp(a, m, 2); }
|
||||
out << " = ";
|
||||
for (expr* a : e.rs()) { if (first) first = false; else out << "\n"; out << mk_bounded_pp(a, m, 2); }
|
||||
for (expr* a : e.rs) { if (first) first = false; else out << "\n"; out << mk_bounded_pp(a, m, 2); }
|
||||
out << " <- \n";
|
||||
return display_deps(out, e.dep());
|
||||
}
|
||||
|
@ -2017,11 +2024,9 @@ app* theory_seq::mk_value(app* e) {
|
|||
void theory_seq::validate_model(model& mdl) {
|
||||
return;
|
||||
for (auto const& eq : m_eqs) {
|
||||
expr_ref_vector ls = eq.ls();
|
||||
expr_ref_vector rs = eq.rs();
|
||||
sort* srt = ls[0]->get_sort();
|
||||
expr_ref l(m_util.str.mk_concat(ls, srt), m);
|
||||
expr_ref r(m_util.str.mk_concat(rs, srt), m);
|
||||
sort* srt = eq.ls[0]->get_sort();
|
||||
expr_ref l(m_util.str.mk_concat(eq.ls, srt), m);
|
||||
expr_ref r(m_util.str.mk_concat(eq.rs, srt), m);
|
||||
if (!mdl.are_equal(l, r)) {
|
||||
IF_VERBOSE(0, verbose_stream() << "equality failed: " << l << " = " << r << "\nbut\n" << mdl(l) << " != " << mdl(r) << "\n");
|
||||
}
|
||||
|
|
|
@ -21,6 +21,7 @@ Revision History:
|
|||
#include "ast/seq_decl_plugin.h"
|
||||
#include "ast/rewriter/th_rewriter.h"
|
||||
#include "ast/rewriter/seq_skolem.h"
|
||||
#include "ast/rewriter/seq_eq_solver.h"
|
||||
#include "ast/ast_trail.h"
|
||||
#include "util/scoped_vector.h"
|
||||
#include "util/scoped_ptr_vector.h"
|
||||
|
@ -146,26 +147,21 @@ namespace smt {
|
|||
};
|
||||
|
||||
// Asserted or derived equality with dependencies
|
||||
class eq {
|
||||
class depeq : public seq::eq {
|
||||
unsigned m_id;
|
||||
expr_ref_vector m_lhs;
|
||||
expr_ref_vector m_rhs;
|
||||
dependency* m_dep;
|
||||
public:
|
||||
|
||||
eq(unsigned id, expr_ref_vector& l, expr_ref_vector& r, dependency* d):
|
||||
m_id(id), m_lhs(l), m_rhs(r), m_dep(d) {}
|
||||
expr_ref_vector const& ls() const { return m_lhs; }
|
||||
expr_ref_vector const& rs() const { return m_rhs; }
|
||||
public:
|
||||
depeq(unsigned id, expr_ref_vector& l, expr_ref_vector& r, dependency* d):
|
||||
eq(l, r), m_id(id), m_dep(d) {}
|
||||
dependency* dep() const { return m_dep; }
|
||||
unsigned id() const { return m_id; }
|
||||
};
|
||||
|
||||
eq mk_eqdep(expr* l, expr* r, dependency* dep) {
|
||||
depeq mk_eqdep(expr* l, expr* r, dependency* dep) {
|
||||
expr_ref_vector ls(m), rs(m);
|
||||
m_util.str.get_concat_units(l, ls);
|
||||
m_util.str.get_concat_units(r, rs);
|
||||
return eq(m_eq_id++, ls, rs, dep);
|
||||
return depeq(m_eq_id++, ls, rs, dep);
|
||||
}
|
||||
|
||||
// equalities that are decomposed by conacatenations
|
||||
|
@ -324,7 +320,7 @@ namespace smt {
|
|||
|
||||
dependency_manager m_dm;
|
||||
solution_map m_rep; // unification representative.
|
||||
scoped_vector<eq> m_eqs; // set of current equations.
|
||||
scoped_vector<depeq> m_eqs; // set of current equations.
|
||||
scoped_vector<ne> m_nqs; // set of current disequalities.
|
||||
scoped_vector<nc> m_ncs; // set of non-contains constraints.
|
||||
scoped_vector<expr*> m_lts; // set of asserted str.<, str.<= literals
|
||||
|
@ -357,6 +353,7 @@ namespace smt {
|
|||
arith_util m_autil;
|
||||
seq::skolem m_sk;
|
||||
seq_axioms m_ax;
|
||||
seq::eq_solver m_eq;
|
||||
seq_regex m_regex;
|
||||
arith_value m_arith_value;
|
||||
th_trail_stack m_trail_stack;
|
||||
|
@ -432,28 +429,29 @@ namespace smt {
|
|||
bool fixed_length(bool is_zero = false);
|
||||
bool fixed_length(expr* e, bool is_zero);
|
||||
bool branch_unit_variable(dependency* dep, expr* X, expr_ref_vector const& units);
|
||||
bool branch_variable_eq(eq const& e);
|
||||
bool branch_binary_variable(eq const& e);
|
||||
bool branch_variable_eq(depeq const& e);
|
||||
bool branch_binary_variable(depeq const& e);
|
||||
bool can_align_from_lhs(expr_ref_vector const& ls, expr_ref_vector const& rs);
|
||||
bool can_align_from_rhs(expr_ref_vector const& ls, expr_ref_vector const& rs);
|
||||
bool branch_ternary_variable_rhs(eq const& e);
|
||||
bool branch_ternary_variable_lhs(eq const& e);
|
||||
bool branch_ternary_variable_rhs(depeq const& e);
|
||||
bool branch_ternary_variable_lhs(depeq const& e);
|
||||
literal mk_alignment(expr* e1, expr* e2);
|
||||
bool branch_quat_variable(eq const& e);
|
||||
bool len_based_split(eq const& e);
|
||||
bool branch_quat_variable(depeq const& e);
|
||||
bool len_based_split(depeq const& e);
|
||||
bool is_unit_eq(expr_ref_vector const& ls, expr_ref_vector const& rs);
|
||||
bool propagate_length_coherence(expr* e);
|
||||
bool split_lengths(dependency* dep,
|
||||
expr_ref_vector const& ls, expr_ref_vector const& rs,
|
||||
vector<rational> const& ll, vector<rational> const& rl);
|
||||
bool set_empty(expr* x);
|
||||
bool is_complex(eq const& e);
|
||||
bool is_complex(depeq const& e);
|
||||
lbool regex_are_equal(expr* r1, expr* r2);
|
||||
void add_unhandled_expr(expr* e);
|
||||
|
||||
bool check_extensionality();
|
||||
bool check_contains();
|
||||
bool check_lts();
|
||||
dependency* m_eq_deps { nullptr };
|
||||
bool solve_eqs(unsigned start);
|
||||
bool solve_eq(unsigned idx);
|
||||
bool simplify_eq(expr_ref_vector& l, expr_ref_vector& r, dependency* dep);
|
||||
|
@ -560,6 +558,7 @@ namespace smt {
|
|||
void deque_axiom(expr* e);
|
||||
void add_axiom(literal l1, literal l2 = null_literal, literal l3 = null_literal, literal l4 = null_literal, literal l5 = null_literal);
|
||||
void add_axiom(literal_vector& lits);
|
||||
void add_consequence(bool uses_eq, expr_ref_vector const& clause);
|
||||
|
||||
bool has_length(expr *e) const { return m_has_length.contains(e); }
|
||||
void add_length(expr* e, expr* l);
|
||||
|
@ -613,7 +612,7 @@ namespace smt {
|
|||
|
||||
// diagnostics
|
||||
std::ostream& display_equations(std::ostream& out) const;
|
||||
std::ostream& display_equation(std::ostream& out, eq const& e) const;
|
||||
std::ostream& display_equation(std::ostream& out, depeq const& e) const;
|
||||
std::ostream& display_disequations(std::ostream& out) const;
|
||||
std::ostream& display_disequation(std::ostream& out, ne const& e) const;
|
||||
std::ostream& display_deps(std::ostream& out, dependency* deps) const;
|
||||
|
|
|
@ -25,7 +25,7 @@ Notes:
|
|||
#include "tactic/smtlogics/qfnra_tactic.h"
|
||||
#include "sat/tactic/sat_tactic.h"
|
||||
#include "sat/sat_solver/inc_sat_solver.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
#include "ackermannization/ackermannize_bv_tactic.h"
|
||||
|
||||
#include "tactic/fpa/qffp_tactic.h"
|
||||
|
|
|
@ -19,7 +19,6 @@ Notes:
|
|||
#include "tactic/portfolio/default_tactic.h"
|
||||
#include "tactic/core/simplify_tactic.h"
|
||||
#include "tactic/smtlogics/qfbv_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/smtlogics/qflia_tactic.h"
|
||||
#include "tactic/smtlogics/qflra_tactic.h"
|
||||
#include "tactic/smtlogics/qfnia_tactic.h"
|
||||
|
@ -32,6 +31,7 @@ Notes:
|
|||
#include "tactic/smtlogics/qfaufbv_tactic.h"
|
||||
#include "tactic/smtlogics/qfauflia_tactic.h"
|
||||
#include "tactic/fd_solver/fd_solver.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
|
||||
tactic * mk_default_tactic(ast_manager & m, params_ref const & p) {
|
||||
tactic * st = using_params(and_then(mk_simplify_tactic(m),
|
||||
|
|
|
@ -13,7 +13,7 @@ z3_add_component(smtlogic_tactics
|
|||
qfufbv_tactic.cpp
|
||||
qfuf_tactic.cpp
|
||||
quant_tactics.cpp
|
||||
smt_tactic_select.cpp
|
||||
smt_tactic.cpp
|
||||
COMPONENT_DEPENDENCIES
|
||||
ackermannization
|
||||
aig_tactic
|
||||
|
@ -28,6 +28,7 @@ z3_add_component(smtlogic_tactics
|
|||
PYG_FILES
|
||||
qfufbv_tactic_params.pyg
|
||||
TACTIC_HEADERS
|
||||
smt_tactic.h
|
||||
nra_tactic.h
|
||||
qfaufbv_tactic.h
|
||||
qfauflia_tactic.h
|
||||
|
|
|
@ -21,7 +21,7 @@ Notes:
|
|||
#include "tactic/core/propagate_values_tactic.h"
|
||||
#include "tactic/core/nnf_tactic.h"
|
||||
#include "tactic/arith/probe_arith.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
#include "qe/qe_tactic.h"
|
||||
#include "qe/nlqsat.h"
|
||||
#include "qe/qe_lite.h"
|
||||
|
|
|
@ -25,9 +25,8 @@ Notes:
|
|||
#include "tactic/bv/bv_size_reduction_tactic.h"
|
||||
#include "tactic/core/ctx_simplify_tactic.h"
|
||||
#include "tactic/smtlogics/qfbv_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic_select.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
#include "ackermannization/ackermannize_bv_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
|
||||
static tactic * mk_qfaufbv_preamble(ast_manager & m, params_ref const & p) {
|
||||
|
||||
|
@ -60,7 +59,7 @@ tactic * mk_qfaufbv_tactic(ast_manager & m, params_ref const & p) {
|
|||
|
||||
tactic * st = using_params(
|
||||
and_then(preamble_st,
|
||||
cond(mk_is_qfbv_probe(), mk_qfbv_tactic(m), mk_smt_tactic_select(m, p))), main_p);
|
||||
cond(mk_is_qfbv_probe(), mk_qfbv_tactic(m), mk_smt_tactic(m, p))), main_p);
|
||||
|
||||
st->updt_params(p);
|
||||
return st;
|
||||
|
|
|
@ -22,7 +22,8 @@ Notes:
|
|||
#include "tactic/arith/propagate_ineqs_tactic.h"
|
||||
#include "tactic/core/solve_eqs_tactic.h"
|
||||
#include "tactic/core/elim_uncnstr_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
|
||||
|
||||
tactic * mk_qfauflia_tactic(ast_manager & m, params_ref const & p) {
|
||||
params_ref main_p;
|
||||
|
|
|
@ -26,11 +26,10 @@ Notes:
|
|||
#include "tactic/bv/max_bv_sharing_tactic.h"
|
||||
#include "tactic/bv/bv_size_reduction_tactic.h"
|
||||
#include "tactic/aig/aig_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "sat/tactic/sat_tactic.h"
|
||||
#include "sat/sat_solver/inc_sat_solver.h"
|
||||
#include "ackermannization/ackermannize_bv_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic_select.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
|
||||
#define MEMLIMIT 300
|
||||
|
||||
|
@ -123,8 +122,8 @@ static tactic * mk_qfbv_tactic(ast_manager& m, params_ref const & p, tactic* sat
|
|||
|
||||
tactic * mk_qfbv_tactic(ast_manager & m, params_ref const & p) {
|
||||
tactic * new_sat = cond(mk_produce_proofs_probe(),
|
||||
and_then(mk_simplify_tactic(m), mk_smt_tactic_select(m, p)),
|
||||
and_then(mk_simplify_tactic(m), mk_smt_tactic(m, p)),
|
||||
mk_psat_tactic(m, p));
|
||||
return mk_qfbv_tactic(m, p, new_sat, mk_smt_tactic_select(m, p));
|
||||
return mk_qfbv_tactic(m, p, new_sat, mk_smt_tactic(m, p));
|
||||
|
||||
}
|
||||
|
|
|
@ -24,7 +24,7 @@ Notes:
|
|||
#include "tactic/core/elim_uncnstr_tactic.h"
|
||||
#include "tactic/arith/normalize_bounds_tactic.h"
|
||||
#include "tactic/arith/fix_dl_var_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
#include "tactic/arith/lia2pb_tactic.h"
|
||||
#include "tactic/arith/pb2bv_tactic.h"
|
||||
#include "tactic/arith/diff_neq_tactic.h"
|
||||
|
|
|
@ -23,7 +23,6 @@ Notes:
|
|||
#include "tactic/arith/normalize_bounds_tactic.h"
|
||||
#include "tactic/core/solve_eqs_tactic.h"
|
||||
#include "tactic/core/elim_uncnstr_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/arith/add_bounds_tactic.h"
|
||||
#include "tactic/arith/pb2bv_tactic.h"
|
||||
#include "tactic/arith/lia2pb_tactic.h"
|
||||
|
@ -31,6 +30,7 @@ Notes:
|
|||
#include "tactic/bv/bit_blaster_tactic.h"
|
||||
#include "tactic/bv/max_bv_sharing_tactic.h"
|
||||
#include "tactic/aig/aig_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
#include "sat/tactic/sat_tactic.h"
|
||||
#include "tactic/arith/bound_manager.h"
|
||||
#include "tactic/arith/probe_arith.h"
|
||||
|
|
|
@ -21,10 +21,10 @@ Notes:
|
|||
#include "tactic/core/propagate_values_tactic.h"
|
||||
#include "tactic/core/solve_eqs_tactic.h"
|
||||
#include "tactic/core/elim_uncnstr_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/arith/recover_01_tactic.h"
|
||||
#include "tactic/core/ctx_simplify_tactic.h"
|
||||
#include "tactic/arith/probe_arith.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
|
||||
tactic * mk_qflra_tactic(ast_manager & m, params_ref const & p) {
|
||||
params_ref pivot_p;
|
||||
|
|
|
@ -21,7 +21,6 @@ Notes:
|
|||
#include "tactic/core/propagate_values_tactic.h"
|
||||
#include "tactic/core/solve_eqs_tactic.h"
|
||||
#include "tactic/core/elim_uncnstr_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/bv/bit_blaster_tactic.h"
|
||||
#include "tactic/bv/max_bv_sharing_tactic.h"
|
||||
#include "sat/tactic/sat_tactic.h"
|
||||
|
@ -30,6 +29,7 @@ Notes:
|
|||
#include "tactic/arith/card2bv_tactic.h"
|
||||
#include "tactic/core/ctx_simplify_tactic.h"
|
||||
#include "tactic/core/cofactor_term_ite_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
#include "nlsat/tactic/qfnra_nlsat_tactic.h"
|
||||
|
||||
static tactic * mk_qfnia_bv_solver(ast_manager & m, params_ref const & p_ref) {
|
||||
|
|
|
@ -20,8 +20,8 @@ Notes:
|
|||
#include "tactic/core/simplify_tactic.h"
|
||||
#include "tactic/core/propagate_values_tactic.h"
|
||||
#include "tactic/arith/nla2bv_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "nlsat/tactic/qfnra_nlsat_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
|
||||
static tactic * mk_qfnra_sat_solver(ast_manager& m, params_ref const& p, unsigned bv_size) {
|
||||
params_ref nra2sat_p = p;
|
||||
|
|
|
@ -22,8 +22,7 @@ Notes:
|
|||
#include "tactic/core/symmetry_reduce_tactic.h"
|
||||
#include "tactic/core/solve_eqs_tactic.h"
|
||||
#include "tactic/core/propagate_values_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic_select.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
|
||||
tactic * mk_qfuf_tactic(ast_manager & m, params_ref const & p) {
|
||||
params_ref s2_p;
|
||||
|
@ -35,7 +34,7 @@ tactic * mk_qfuf_tactic(ast_manager & m, params_ref const & p) {
|
|||
mk_solve_eqs_tactic(m, p),
|
||||
using_params(mk_simplify_tactic(m, p), s2_p),
|
||||
if_no_proofs(if_no_unsat_cores(mk_symmetry_reduce_tactic(m, p))),
|
||||
mk_smt_tactic_select(m, p));
|
||||
mk_smt_tactic(m, p));
|
||||
}
|
||||
|
||||
|
||||
|
|
|
@ -22,7 +22,6 @@ Notes:
|
|||
#include "tactic/core/propagate_values_tactic.h"
|
||||
#include "tactic/core/solve_eqs_tactic.h"
|
||||
#include "tactic/core/elim_uncnstr_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/bv/max_bv_sharing_tactic.h"
|
||||
#include "tactic/bv/bv_size_reduction_tactic.h"
|
||||
#include "tactic/core/reduce_args_tactic.h"
|
||||
|
@ -37,7 +36,7 @@ Notes:
|
|||
#include "sat/sat_solver/inc_sat_solver.h"
|
||||
#include "tactic/smtlogics/qfaufbv_tactic.h"
|
||||
#include "tactic/smtlogics/qfbv_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic_select.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
#include "solver/tactic2solver.h"
|
||||
#include "tactic/bv/bv_bound_chk_tactic.h"
|
||||
#include "ackermannization/ackermannize_bv_tactic.h"
|
||||
|
@ -186,7 +185,7 @@ tactic * mk_qfufbv_tactic(ast_manager & m, params_ref const & p) {
|
|||
and_then(preamble_st,
|
||||
cond(mk_is_qfbv_probe(),
|
||||
mk_qfbv_tactic(m),
|
||||
mk_smt_tactic_select(m, p))),
|
||||
mk_smt_tactic(m, p))),
|
||||
main_p);
|
||||
|
||||
st->updt_params(p);
|
||||
|
@ -198,5 +197,5 @@ 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_select(m, p)));
|
||||
cond(mk_is_qfufbv_probe(), actual_tactic, mk_smt_tactic(m, p)));
|
||||
}
|
||||
|
|
|
@ -24,9 +24,9 @@ Revision History:
|
|||
#include "qe/qe_lite.h"
|
||||
#include "qe/qsat.h"
|
||||
#include "tactic/core/ctx_simplify_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/core/elim_term_ite_tactic.h"
|
||||
#include "tactic/arith/probe_arith.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
|
||||
static tactic * mk_quant_preprocessor(ast_manager & m, bool disable_gaussian = false) {
|
||||
params_ref pull_ite_p;
|
||||
|
|
30
src/tactic/smtlogics/smt_tactic.cpp
Normal file
30
src/tactic/smtlogics/smt_tactic.cpp
Normal file
|
@ -0,0 +1,30 @@
|
|||
/*++
|
||||
Copyright (c) 2020 Microsoft Corporation
|
||||
|
||||
Module Name:
|
||||
|
||||
smt_tactic.cpp
|
||||
|
||||
Abstract:
|
||||
|
||||
Tactic that selects SMT backend.
|
||||
|
||||
Author:
|
||||
|
||||
Nikolaj Bjorner (nbjorner) 2020-09-14
|
||||
|
||||
|
||||
--*/
|
||||
#include "smt/tactic/smt_tactic_core.h"
|
||||
#include "sat/tactic/sat_tactic.h"
|
||||
#include "sat/sat_params.hpp"
|
||||
|
||||
tactic * mk_smt_tactic(ast_manager & m, params_ref const & p) {
|
||||
sat_params sp(p);
|
||||
return sp.euf() ? mk_sat_tactic(m, p) : mk_smt_tactic_core(m, p);
|
||||
}
|
||||
|
||||
tactic * mk_smt_tactic_using(ast_manager& m, bool auto_config, params_ref const& p) {
|
||||
sat_params sp(p);
|
||||
return sp.euf() ? mk_sat_tactic(m, p) : mk_smt_tactic_core_using(m, auto_config, p);
|
||||
}
|
32
src/tactic/smtlogics/smt_tactic.h
Normal file
32
src/tactic/smtlogics/smt_tactic.h
Normal file
|
@ -0,0 +1,32 @@
|
|||
/*++
|
||||
Copyright (c) 2020 Microsoft Corporation
|
||||
|
||||
Module Name:
|
||||
|
||||
smt_tactic.h
|
||||
|
||||
Abstract:
|
||||
|
||||
Tactic that selects SMT backend.
|
||||
|
||||
Author:
|
||||
|
||||
Nikolaj Bjorner (nbjorner) 2020-09-14
|
||||
|
||||
|
||||
--*/
|
||||
#pragma once
|
||||
|
||||
#include "util/params.h"
|
||||
class ast_manager;
|
||||
class tactic;
|
||||
|
||||
tactic * mk_smt_tactic(ast_manager & m, params_ref const & p = params_ref());
|
||||
|
||||
tactic * mk_smt_tactic_using(ast_manager& m, bool auto_config = true, params_ref const& p = params_ref());
|
||||
|
||||
/*
|
||||
ADD_TACTIC("smt", "apply a SAT based SMT solver.", "mk_smt_tactic(m, p)")
|
||||
*/
|
||||
|
||||
|
|
@ -1,26 +0,0 @@
|
|||
/*++
|
||||
Copyright (c) 2020 Microsoft Corporation
|
||||
|
||||
Module Name:
|
||||
|
||||
smt_tactic_select.cpp
|
||||
|
||||
Abstract:
|
||||
|
||||
Tactic that selects SMT backend.
|
||||
|
||||
Author:
|
||||
|
||||
Nikolaj Bjorner (nbjorner) 2020-09-14
|
||||
|
||||
|
||||
--*/
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "sat/tactic/sat_tactic.h"
|
||||
#include "sat/sat_params.hpp"
|
||||
|
||||
tactic * mk_smt_tactic_select(ast_manager & m, params_ref const & p) {
|
||||
sat_params sp(p);
|
||||
return sp.euf() ? mk_sat_tactic(m, p) : mk_smt_tactic(m, p);
|
||||
}
|
||||
|
|
@ -1,25 +0,0 @@
|
|||
/*++
|
||||
Copyright (c) 2020 Microsoft Corporation
|
||||
|
||||
Module Name:
|
||||
|
||||
smt_tactic_select.h
|
||||
|
||||
Abstract:
|
||||
|
||||
Tactic that selects SMT backend.
|
||||
|
||||
Author:
|
||||
|
||||
Nikolaj Bjorner (nbjorner) 2020-09-14
|
||||
|
||||
|
||||
--*/
|
||||
#pragma once
|
||||
|
||||
#include "util/params.h"
|
||||
class ast_manager;
|
||||
class tactic;
|
||||
|
||||
tactic * mk_smt_tactic_select(ast_manager & m, params_ref const & p);
|
||||
|
|
@ -23,7 +23,7 @@ Notes:
|
|||
#include "tactic/core/distribute_forall_tactic.h"
|
||||
#include "tactic/core/der_tactic.h"
|
||||
#include "tactic/core/reduce_args_tactic.h"
|
||||
#include "smt/tactic/smt_tactic.h"
|
||||
#include "tactic/smtlogics/smt_tactic.h"
|
||||
#include "tactic/core/nnf_tactic.h"
|
||||
#include "tactic/ufbv/macro_finder_tactic.h"
|
||||
#include "tactic/ufbv/ufbv_rewriter_tactic.h"
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue