mirror of
https://github.com/Z3Prover/z3
synced 2025-04-06 17:44:08 +00:00
delay (lazy) process equalities.
This commit is contained in:
parent
ddbcd08d46
commit
156139622c
|
@ -304,7 +304,7 @@ namespace arith {
|
|||
return lp::EQ;
|
||||
}
|
||||
|
||||
void solver::mk_eq_axiom(bool is_eq, euf::th_eq const& e) {
|
||||
void solver::new_eq_eh(euf::th_eq const& e) {
|
||||
theory_var v1 = e.v1();
|
||||
theory_var v2 = e.v2();
|
||||
if (is_bool(v1))
|
||||
|
@ -316,23 +316,35 @@ namespace arith {
|
|||
if (e1->get_id() > e2->get_id())
|
||||
std::swap(e1, e2);
|
||||
|
||||
if (is_eq && m.are_equal(e1, e2))
|
||||
if (m.are_equal(e1, e2))
|
||||
return;
|
||||
if (!is_eq && m.are_distinct(e1, e2))
|
||||
return;
|
||||
|
||||
if (is_eq) {
|
||||
++m_stats.m_assert_eq;
|
||||
m_new_eq = true;
|
||||
euf::enode* n1 = var2enode(v1);
|
||||
euf::enode* n2 = var2enode(v2);
|
||||
lpvar w1 = register_theory_var_in_lar_solver(v1);
|
||||
lpvar w2 = register_theory_var_in_lar_solver(v2);
|
||||
auto cs = lp().add_equality(w1, w2);
|
||||
add_eq_constraint(cs.first, n1, n2);
|
||||
add_eq_constraint(cs.second, n1, n2);
|
||||
++m_stats.m_assert_eq;
|
||||
m_new_eq = true;
|
||||
euf::enode* n1 = var2enode(v1);
|
||||
euf::enode* n2 = var2enode(v2);
|
||||
lpvar w1 = register_theory_var_in_lar_solver(v1);
|
||||
lpvar w2 = register_theory_var_in_lar_solver(v2);
|
||||
auto cs = lp().add_equality(w1, w2);
|
||||
add_eq_constraint(cs.first, n1, n2);
|
||||
add_eq_constraint(cs.second, n1, n2);
|
||||
}
|
||||
|
||||
void solver::new_diseq_eh(euf::th_eq const& e) {
|
||||
m_delayed_eqs.push_back(std::make_pair(e, false));
|
||||
ctx.push(push_back_vector<svector<std::pair<euf::th_eq, bool>>>(m_delayed_eqs));
|
||||
}
|
||||
|
||||
void solver::mk_diseq_axiom(euf::th_eq const& e) {
|
||||
if (is_bool(e.v1()))
|
||||
return;
|
||||
}
|
||||
force_push();
|
||||
expr* e1 = var2expr(e.v1());
|
||||
expr* e2 = var2expr(e.v2());
|
||||
if (e1->get_id() > e2->get_id())
|
||||
std::swap(e1, e2);
|
||||
if (m.are_distinct(e1, e2))
|
||||
return;
|
||||
literal le, ge;
|
||||
if (a.is_numeral(e1))
|
||||
std::swap(e1, e2);
|
||||
|
@ -347,9 +359,7 @@ namespace arith {
|
|||
expr_ref zero(a.mk_numeral(rational(0), a.is_int(e1)), m);
|
||||
rewrite(diff);
|
||||
if (a.is_numeral(diff)) {
|
||||
if (is_eq && a.is_zero(diff))
|
||||
return;
|
||||
if (!is_eq && !a.is_zero(diff))
|
||||
if (!a.is_zero(diff))
|
||||
return;
|
||||
if (a.is_zero(diff))
|
||||
add_unit(eq);
|
||||
|
|
|
@ -969,6 +969,9 @@ namespace arith {
|
|||
|
||||
TRACE("arith", ctx.display(tout););
|
||||
|
||||
if (!check_delayed_eqs())
|
||||
return sat::check_result::CR_CONTINUE;
|
||||
|
||||
switch (check_lia()) {
|
||||
case l_true:
|
||||
break;
|
||||
|
@ -1064,6 +1067,19 @@ namespace arith {
|
|||
}
|
||||
}
|
||||
|
||||
bool solver::check_delayed_eqs() {
|
||||
for (auto p : m_delayed_eqs) {
|
||||
auto const& e = p.first;
|
||||
if (p.second)
|
||||
new_eq_eh(e);
|
||||
else if (is_eq(e.v1(), e.v2())) {
|
||||
mk_diseq_axiom(e);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
lbool solver::check_lia() {
|
||||
TRACE("arith", );
|
||||
if (!m.inc())
|
||||
|
|
|
@ -164,6 +164,7 @@ namespace arith {
|
|||
svector<literal> m_inequalities; // asserted rows corresponding to inequality literals.
|
||||
svector<euf::enode_pair> m_equalities; // asserted rows corresponding to equalities.
|
||||
svector<theory_var> m_definitions; // asserted rows corresponding to definitions
|
||||
svector<std::pair<euf::th_eq, bool>> m_delayed_eqs;
|
||||
|
||||
literal_vector m_asserted;
|
||||
expr* m_not_handled{ nullptr };
|
||||
|
@ -305,6 +306,7 @@ namespace arith {
|
|||
literal is_bound_implied(lp::lconstraint_kind k, rational const& value, api_bound const& b) const;
|
||||
void assert_bound(bool is_true, api_bound& b);
|
||||
void mk_eq_axiom(bool is_eq, euf::th_eq const& eq);
|
||||
void mk_diseq_axiom(euf::th_eq const& eq);
|
||||
void assert_idiv_mod_axioms(theory_var u, theory_var v, theory_var w, rational const& r);
|
||||
api_bound* mk_var_bound(sat::literal lit, theory_var v, lp_api::bound_kind bk, rational const& bound);
|
||||
lp::lconstraint_kind bound2constraint_kind(bool is_int, lp_api::bound_kind bk, bool is_true);
|
||||
|
@ -348,6 +350,7 @@ namespace arith {
|
|||
bool use_nra_model();
|
||||
|
||||
lbool make_feasible();
|
||||
bool check_delayed_eqs();
|
||||
lbool check_lia();
|
||||
lbool check_nla();
|
||||
bool is_infeasible() const;
|
||||
|
@ -423,8 +426,8 @@ namespace arith {
|
|||
void collect_statistics(statistics& st) const override;
|
||||
euf::th_solver* clone(euf::solver& ctx) override;
|
||||
bool use_diseqs() const override { return true; }
|
||||
void new_eq_eh(euf::th_eq const& eq) override { mk_eq_axiom(true, eq); }
|
||||
void new_diseq_eh(euf::th_eq const& de) override { mk_eq_axiom(false, de); }
|
||||
void new_eq_eh(euf::th_eq const& eq) override;
|
||||
void new_diseq_eh(euf::th_eq const& de) override;
|
||||
bool unit_propagate() override;
|
||||
void init_model() override;
|
||||
void finalize_model(model& mdl) override { DEBUG_CODE(dbg_finalize_model(mdl);); }
|
||||
|
|
Loading…
Reference in a new issue