3
0
Fork 0
mirror of https://github.com/Z3Prover/z3 synced 2025-04-06 09:34:08 +00:00

collect the explanation correctly

This commit is contained in:
Lev Nachmanson 2024-11-18 19:17:53 -08:00 committed by Lev Nachmanson
parent bc0fdfe158
commit 174185582a
2 changed files with 88 additions and 40 deletions

View file

@ -23,10 +23,13 @@
-- lia: point to int_solver.
-- m_entries: it keeps all "entry" objects.
-- m_e_matrix: i-th row of this matrix keeps the term corresponding to
m_entries[i]. The actual term is the sum of the matrix row and the constant
m_c of the entry. The column j of the matrix corresponds to j column of
lar_solver if j < lra.column_count(). Otherwise, j is a fresh column. It has
to change in the interactive version. Implementation remarks:
m_entries[i]. The actual term corresponding to m_entry[i] is the sum of the
matrix i-th row and the constant m_entry[].m_c.
The mapping between the columns of lar_solver and m_e_matrix is controlled by m_var_register.
local_to_lar_solver(lar_solver_to_local(j)) == j. If local_to_lar_solver(j) == -1
then j is a fresh variable, that is such that got introduced when normalizing a term like 3x-6y + 5z +11 = 0,
when no variable has a coefficient +-1.
-- get_term_from_entry(unsigned i) return a term corresponding i-th entry.
If t = get_term_from_entry(i) then we have equality t = 0. Initially
get_term_from_entry(i) is set to initt(j) = lra.get_term(j) - j, for some
@ -246,6 +249,7 @@ namespace lp {
bool m_left;
bool m_fully_explored = false;
void flip() {
SASSERT(m_fully_explored == false);
m_left = !m_left;
m_fully_explored = true;
}
@ -415,6 +419,8 @@ namespace lp {
// A conflict is reported when the gcd of the monomial coefficients does not divide the free coefficent.
// If there is no conflict the entry is divided, normalized, by gcd.
// The function returns true if and only if there is no conflict. In the case of a conflict a branch
// can be returned as well.
bool normalize_e_by_gcd(unsigned ei) {
entry& e = m_entries[ei];
TRACE("dioph_eq", print_entry(ei, tout) << std::endl;);
@ -437,7 +443,7 @@ namespace lp {
return true;
}
// c_g is not integral
if (lra.settings().stats().m_dio_conflicts %
if (lra.stats().m_dio_calls %
lra.settings().dio_cut_from_proof_period() ==
0 &&
!has_fresh_var(ei))
@ -563,7 +569,7 @@ namespace lp {
subs_front_in_indexed_vector(q);
}
lia_move tighten_with_S() {
lia_move tighten_terms_with_S() {
for (unsigned j = 0; j < lra.column_count(); j++) {
if (!lra.column_has_term(j) || lra.column_is_free(j) ||
is_fixed(j) || !lia.column_is_int(j))
@ -792,26 +798,35 @@ namespace lp {
return dep;
}
lia_move iterate() {
lia_move process_f() {
while (m_f.size()) {
if (!normalize_by_gcd()) {
lra.settings().stats().m_dio_conflicts++;
if (m_report_branch) {
lra.stats().m_dio_cut_from_proofs++;
m_report_branch = false;
return lia_move::branch;
} else {
lra.stats().m_dio_normalize_conflicts++;
}
return lia_move::conflict;
}
rewrite_eqs();
if (m_conflict_index != UINT_MAX) {
lra.settings().stats().m_dio_conflicts++;
lra.stats().m_dio_rewrite_conflicts++;
return lia_move::conflict;
}
}
return lia_move::undef;
}
lia_move process_f_and_tighten_terms() {
lia_move ret = process_f();
if (ret != lia_move::undef)
return ret;
TRACE("dioph_eq", print_S(tout););
lia_move ret = tighten_with_S();
ret = tighten_terms_with_S();
if (ret == lia_move::conflict) {
lra.settings().stats().m_dio_conflicts++;
lra.stats().m_dio_tighten_conflicts++;
return lia_move::conflict;
}
return lia_move::undef;
@ -835,16 +850,13 @@ namespace lp {
}
// returns true if the left and the right branches were explored
bool undo_last_branch() {
if (m_branch_stack.back().m_fully_explored) {
TRACE("dio_br", tout << "pop fully explored, m_branch_stack.size():" << m_branch_stack.size() << std::endl;);
void undo_explored_branches() {
TRACE("dio_br", tout << "m_branch_stack.size():" << m_branch_stack.size() << std::endl;);
while (m_branch_stack.size() && m_branch_stack.back().m_fully_explored) {
m_branch_stack.pop_back();
return true;
lra_pop();
}
// exploring the other side of the branch
TRACE("dio_br", tout << "flipped branch" << std::endl;);
m_branch_stack.back().flip();
return false;
TRACE("dio_br", tout << "after pop:m_branch_stack.size():" << m_branch_stack.size() << std::endl;);
}
lia_move check_fixing(unsigned j) const {
@ -903,6 +915,7 @@ namespace lp {
if (br.m_j == UINT_MAX)
return false;
m_branch_stack.push_back(br);
lra.stats().m_dio_branching_depth = std::max(lra.stats().m_dio_branching_depth, (unsigned)m_branch_stack.size());
return true;
}
@ -912,10 +925,10 @@ namespace lp {
} else {
lra.add_var_bound(b->m_j, lconstraint_kind::GE, b->m_rs + mpq(1));
}
if (lra.column_is_fixed(b->m_j)) {
if (fix_var(lar_solver_to_local(b->m_j)) == lia_move::conflict)
return lia_move::conflict;
}
// if (lra.column_is_fixed(b->m_j)) {
// if (fix_var(lar_solver_to_local(b->m_j)) == lia_move::conflict)
// return lia_move::conflict;
// }
return lia_move::undef;
}
@ -927,20 +940,32 @@ namespace lp {
}
void lra_pop() {
m_lra_level--;
SASSERT(m_lra_level != UINT_MAX);
lra.pop();
lra.find_feasible_solution();
SASSERT(lra.get_status() == lp_status::CANCELLED || lra.is_feasible());
}
lia_move process_undef() {
void transfer_explanations_from_closed_branches() {
m_infeas_explanation.clear();
for (auto ci : m_explanation_of_branches) {
if (this->lra.constraints().valid_index(ci))
m_infeas_explanation.push_back(ci);
}
}
lia_move branching_on_undef() {
m_explanation_of_branches.clear();
branch* b;
bool need_create_branch = true;
m_number_of_iterations = 0;
while (++m_number_of_iterations < m_max_number_of_iterations) {
lra.stats().m_dio_branch_iterations++;
if (need_create_branch) {
if (!push_branch()) {
undo_branching();
lra.stats().m_dio_branching_sats++;
return lia_move::sat;
}
need_create_branch = false;
@ -950,13 +975,16 @@ namespace lp {
if (add_var_bound_for_branch(b) == lia_move::conflict) {
collect_evidence();
if (m_branch_stack.size() == 0)
undo_explored_branches();
if (m_branch_stack.size() == 0) {
lra.stats().m_dio_branching_infeasibles++;
transfer_explanations_from_closed_branches();
return lia_move::conflict;
}
TRACE("dio_br", tout << lp_status_to_string(lra.get_status()) << std::endl;
tout << "explanation:\n"; lra.print_expl(tout, m_infeas_explanation););
if (undo_last_branch())
need_create_branch = true;
lra_pop();
need_create_branch = false;
m_branch_stack.back().flip();
continue;
}
auto st = lra.find_feasible_solution();
@ -966,6 +994,7 @@ namespace lp {
TRACE("dio_br", tout << "n_of_ii:" << n_of_ii << "\n";);
if (n_of_ii == 0) {
undo_branching();
lra.stats().m_dio_branching_sats++;
return lia_move::sat;
}
// got to create a new branch
@ -974,13 +1003,19 @@ namespace lp {
} else {
if (st == lp_status::CANCELLED) return lia_move::undef;
collect_evidence();
if (m_branch_stack.size() == 0)
return lia_move::conflict;
undo_explored_branches();
if (m_branch_stack.size() == 0) {
lra.stats().m_dio_branching_infeasibles++;
transfer_explanations_from_closed_branches();
enable_trace("dioph_eq");
return lia_move::conflict;
}
TRACE("dio_br", tout << lp_status_to_string(lra.get_status()) << std::endl;
tout << "explanation:\n"; lra.print_expl(tout, m_infeas_explanation););
if (undo_last_branch())
need_create_branch = true;
need_create_branch = false;
lra_pop();
m_branch_stack.back().flip();
}
}
undo_branching();
@ -1049,20 +1084,20 @@ namespace lp {
br.m_left = (lra.settings().random_next() % 2 == 0);
br.m_rs = floor(lra.get_column_value(bj).x);
TRACE("dio_br", tout << " br.m_j:" << br.m_j << ","
TRACE("dio_br", tout << "score:" << score << "; br.m_j:" << br.m_j << ","
<< (br.m_left ? "left" : "right") << ", br.m_rs:" << br.m_rs << std::endl;);
return br;
}
public:
lia_move check() {
lra.settings().stats().m_dio_calls++;
lra.stats().m_dio_calls++;
init();
lia_move ret = iterate();
lia_move ret = process_f_and_tighten_terms();
if (ret == lia_move::branch || ret == lia_move::conflict)
return ret;
SASSERT(ret == lia_move::undef);
ret = process_undef();
ret = branching_on_undef();
if (ret == lia_move::sat || ret == lia_move::conflict)
return ret;
SASSERT(ret == lia_move::undef);
@ -1382,8 +1417,6 @@ namespace lp {
public:
void explain(explanation& ex) {
if (m_conflict_index == UINT_MAX) {
SASSERT(!(lra.get_status() == lp_status::FEASIBLE ||
lra.get_status() == lp_status::OPTIMAL));
for (auto ci : m_infeas_explanation) {
ex.push_back(ci.ci());
}

View file

@ -128,8 +128,15 @@ struct statistics {
unsigned m_grobner_conflicts = 0;
unsigned m_offset_eqs = 0;
unsigned m_fixed_eqs = 0;
unsigned m_dio_conflicts = 0;
unsigned m_dio_calls = 0;
unsigned m_dio_normalize_conflicts = 0;
unsigned m_dio_tighten_conflicts = 0;
unsigned m_dio_branch_iterations= 0;
unsigned m_dio_branching_depth = 0;
unsigned m_dio_cut_from_proofs = 0;
unsigned m_dio_branching_infeasibles = 0;
unsigned m_dio_rewrite_conflicts = 0;
unsigned m_dio_branching_sats = 0;
::statistics m_st = {};
void reset() {
@ -163,7 +170,15 @@ struct statistics {
st.update("arith-nra-calls", m_nra_calls);
st.update("arith-bounds-improvements", m_nla_bounds_improvements);
st.update("arith-dio-calls", m_dio_calls);
st.update("arith-dio-conflicts", m_dio_conflicts);
st.update("arith-dio-normalize-conflicts", m_dio_normalize_conflicts);
st.update("arith-dio-tighten-conflicts", m_dio_tighten_conflicts);
st.update("arith-dio-branch-iterations", m_dio_branch_iterations);
st.update("arith-dio-branch-depths", m_dio_branching_depth);
st.update("arith-dio-cut-from-proofs", m_dio_cut_from_proofs);
st.update("arith-dio-branching-infeasibles", m_dio_branching_infeasibles);
st.update("arith-dio-rewrite-conflicts", m_dio_rewrite_conflicts);
st.update("arith-dio-branching-sats", m_dio_branching_sats);
st.update("arith-dio-branching-depth", m_dio_branching_depth);
st.copy(m_st);
}
};