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

debug dio

Signed-off-by: Lev Nachmanson <levnach@hotmail.com>
This commit is contained in:
Lev Nachmanson 2025-01-03 12:34:51 -10:00 committed by Lev Nachmanson
parent a0ece6dd2c
commit 083926c658
3 changed files with 146 additions and 60 deletions

View file

@ -56,7 +56,7 @@ namespace lp {
ret.add_monomial(p.coeff(), p.j()); ret.add_monomial(p.coeff(), p.j());
} }
ret.c() = c(); ret.c() = c();
ret.j() = j(); ret.set_j(j());
return ret; return ret;
} }
term_o(const lar_term& t) : lar_term(t), m_c(0) { term_o(const lar_term& t) : lar_term(t), m_c(0) {
@ -233,7 +233,12 @@ namespace lp {
// {coeff*lar.get_term(k)}) // {coeff*lar.get_term(k)})
std_vector<unsigned> m_k2s; std_vector<unsigned> m_k2s;
std_vector<unsigned> m_fresh_definitions; struct fresh_definition {
unsigned m_ei;
unsigned m_origin;
fresh_definition(unsigned i, unsigned o) : m_ei(i), m_origin(o) {}
};
std_vector<fresh_definition> m_fresh_definitions;
unsigned m_conflict_index = -1; // m_entries[m_conflict_index] gives the conflict unsigned m_conflict_index = -1; // m_entries[m_conflict_index] gives the conflict
unsigned m_max_number_of_iterations = 100; unsigned m_max_number_of_iterations = 100;
@ -281,9 +286,14 @@ namespace lp {
undo_add_term(imp& s, const lar_term &t): m_s(s), m_t(t) { undo_add_term(imp& s, const lar_term &t): m_s(s), m_t(t) {
} }
void undo () { void undo () {
TRACE("dioph_eq", m_s.lra.print_term(m_t, tout););
for (const auto & p: m_t) { for (const auto & p: m_t) {
auto it = m_s.m_columns_to_terms.find(p.var()); auto it = m_s.m_columns_to_terms.find(p.var());
it->second.erase(m_t.j()); it->second.erase(m_t.j());
if (it->second.size() == 0) {
m_s.m_columns_to_terms.erase(it);
}
} }
} }
}; };
@ -303,7 +313,7 @@ namespace lp {
}; };
std::unordered_set<unsigned> m_changed_columns; std::unordered_set<unsigned> m_changed_columns;
// m_column_to_terms[j] is the set of k such lra.get_term(k) depends on j // m_column_to_terms[j] is the set of all k such lra.get_term(k) depends on j
std::unordered_map<unsigned, std::unordered_set<unsigned>> m_columns_to_terms; std::unordered_map<unsigned, std::unordered_set<unsigned>> m_columns_to_terms;
friend undo_fixed_column; friend undo_fixed_column;
void add_changed_column(unsigned j) { void add_changed_column(unsigned j) {
@ -326,7 +336,6 @@ namespace lp {
TRACE("dioph_eq", tout << "not all vars are integrall\n";); TRACE("dioph_eq", tout << "not all vars are integrall\n";);
return; return;
} }
TRACE("dioph_eq", tout << "inserted" << std::endl;);
m_added_terms.push_back(t); m_added_terms.push_back(t);
auto undo = undo_add_term(*this, *t); auto undo = undo_add_term(*this, *t);
lra.trail().push(undo); lra.trail().push(undo);
@ -364,6 +373,7 @@ namespace lp {
} }
void register_columns_to_term(const lar_term& t) { void register_columns_to_term(const lar_term& t) {
TRACE("dioph_eq", tout << "register term:"; lra.print_term(t, tout););
for (const auto &p: t) { for (const auto &p: t) {
auto it = m_columns_to_terms.find(p.var()); auto it = m_columns_to_terms.find(p.var());
if (it != m_columns_to_terms.end()) { if (it != m_columns_to_terms.end()) {
@ -373,12 +383,12 @@ namespace lp {
std::unordered_set<unsigned> s; std::unordered_set<unsigned> s;
s.insert(t.j()); s.insert(t.j());
m_columns_to_terms[p.var()] = s; m_columns_to_terms[p.var()] = s;
TRACE("dioph_eq", tout << "insert " << p.var(););
} }
} }
} }
// the term has form sum(a_i*x_i) - t.j() = 0, // the term has form sum(a_i*x_i) - t.j() = 0,
void fill_entry(const lar_term& t) { void fill_entry(const lar_term& t) {
register_columns_to_term(t);
TRACE("dioph_eq", print_lar_term_L(t, tout) << std::endl;); TRACE("dioph_eq", print_lar_term_L(t, tout) << std::endl;);
entry te = { mpq(0), entry_status::F}; entry te = { mpq(0), entry_status::F};
unsigned entry_index = (unsigned) m_entries.size(); unsigned entry_index = (unsigned) m_entries.size();
@ -462,45 +472,54 @@ namespace lp {
std::unordered_set<unsigned> entries_to_recalculate; std::unordered_set<unsigned> entries_to_recalculate;
std::unordered_set<unsigned> changed_terms; // a term is signified by the term column, like j in lra.get_term(j) std::unordered_set<unsigned> changed_terms; // a term is signified by the term column, like j in lra.get_term(j)
std_vector<unsigned> fresh_entries_to_remove; std_vector<unsigned> fresh_entries_to_remove;
for (unsigned j = 0; j < m_fresh_definitions.size(); j++) {
unsigned k = m_fresh_definitions[j]; for (unsigned j : m_changed_columns) {
if (k == UINT_MAX) continue; const auto it = m_columns_to_terms.find(j);
for (const auto & p: m_e_matrix.m_rows[k]) { if (it != m_columns_to_terms.end())
if (contains(m_changed_columns, p.var())) { for (unsigned k : it->second) {
fresh_entries_to_remove.push_back(k); changed_terms.insert(k);
continue;
} }
if (!m_var_register.external_is_used(j))
} continue;
}
TRACE("dioph_eq", tout << "found " << fresh_entries_to_remove.size() << " fresh entries to remove\n";
tout << "m_changed_columns:\n";
for (unsigned j : m_changed_columns) {
tout << j << " ";
}
tout << std::endl;
);
if (fresh_entries_to_remove.size()) {
NOT_IMPLEMENTED_YET();
}
for (unsigned j : m_changed_columns) {
for (unsigned k : m_columns_to_terms[j]) {
changed_terms.insert(k);
}
for (const auto& p : m_e_matrix.column(this->lar_solver_to_local(j))) { for (const auto& p : m_e_matrix.column(this->lar_solver_to_local(j))) {
TRACE("dioph_eq", tout << "insert into entries_to_recalculate " << p.var() << " for changed_column j=" << j<< std::endl;
tout << "p.coeff:" << p.coeff() << std::endl;
);
entries_to_recalculate.insert(p.var()); entries_to_recalculate.insert(p.var());
} }
} }
for (unsigned j : changed_terms) { for (unsigned j : changed_terms) {
for (const auto & cs: m_l_matrix.column(j)) { for (const auto & cs: m_l_matrix.column(j)) {
TRACE("dioph_eq", tout << "insert into entries_to_recalculate " << cs.var() << " for changed_term j=" << j<< std::endl;);
entries_to_recalculate.insert(cs.var()); entries_to_recalculate.insert(cs.var());
} }
} }
for(unsigned k : entries_to_recalculate) {
for (const lar_term* t : m_added_terms) {
register_columns_to_term(*t);
}
SASSERT(is_in_sync());
TRACE("dioph_eq", tout << "entries_to_recalculate:"; for (unsigned j : entries_to_recalculate) {tout << " " << j;});
for (unsigned j = 0; j < m_fresh_definitions.size(); j++) {
const fresh_definition& fd = m_fresh_definitions[j];
if (fd.m_ei == UINT_MAX) continue;
if (contains(entries_to_recalculate, fd.m_origin)) {
fresh_entries_to_remove.push_back(j);
continue;
}
}
TRACE("dioph_eq", tout << "found " << fresh_entries_to_remove.size() << " fresh entries to remove\n";
tout << "m_changed_columns:\n";
std::vector<unsigned> v(m_changed_columns.begin(), m_changed_columns.end());
std::sort(v.begin(), v.end());
for (unsigned j : v) {
tout << j << " ";
}
tout << std::endl;
);
if (fresh_entries_to_remove.size()) {
NOT_IMPLEMENTED_YET();
}
for(unsigned k : entries_to_recalculate) {
recalculate_entry(k); recalculate_entry(k);
} }
move_recalculated_to_F(entries_to_recalculate); move_recalculated_to_F(entries_to_recalculate);
@ -528,6 +547,16 @@ namespace lp {
} }
} }
bool entries_are_ok() {
for (unsigned ei = 0; ei < m_entries.size(); ei++) {
if (entry_invariant(ei) == false) {
TRACE("dioph_eq", tout << "bad entry:"; print_entry(ei, tout););
return false;
}
}
return true;
}
void init() { void init() {
m_report_branch = false; m_report_branch = false;
m_conflict_index = -1; m_conflict_index = -1;
@ -541,18 +570,8 @@ namespace lp {
fill_entry(*t); fill_entry(*t);
} }
m_added_terms.clear(); m_added_terms.clear();
/* SASSERT(entries_are_ok());
for (unsigned j = 0; j < lra.column_count(); j++) { }
if (!lra.column_is_int(j) || !lra.column_has_term(j))
continue;
const lar_term& t = lra.get_term(j);
if (!all_vars_are_int(t)) {
TRACE("dioph_eq", tout << "not all vars are integrall\n";);
continue;
}
fill_entry(t);
}*/
}
template <typename K> template <typename K>
mpq gcd_of_coeffs(const K& k) { mpq gcd_of_coeffs(const K& k) {
@ -573,10 +592,6 @@ namespace lp {
return lra.print_expl(out, ex); return lra.print_expl(out, ex);
} }
std::string var_str(unsigned j) {
return std::string(is_fresh_var(j) ? "~" : "") + std::to_string(j);
}
bool has_fresh_var(unsigned row_index) const { bool has_fresh_var(unsigned row_index) const {
for (const auto& p : m_e_matrix.m_rows[row_index]) { for (const auto& p : m_e_matrix.m_rows[row_index]) {
if (is_fresh_var(p.var())) if (is_fresh_var(p.var()))
@ -671,6 +686,7 @@ namespace lp {
if (m_indexed_work_vector[k].is_zero()) if (m_indexed_work_vector[k].is_zero())
return; return;
const entry& e = entry_for_subs(k); const entry& e = entry_for_subs(k);
SASSERT(e.m_entry_status == entry_status::S);
TRACE("dioph_eq", tout << "k:" << k << ", in "; TRACE("dioph_eq", tout << "k:" << k << ", in ";
print_term_o(create_term_from_ind_c(), tout) << std::endl; print_term_o(create_term_from_ind_c(), tout) << std::endl;
tout << "subs with e:"; tout << "subs with e:";
@ -854,7 +870,10 @@ namespace lp {
tout << "term_to_tighten + open_ml:"; tout << "term_to_tighten + open_ml:";
print_term_o(term_to_tighten + open_ml(m_tmp_l), tout) print_term_o(term_to_tighten + open_ml(m_tmp_l), tout)
<< std::endl; << std::endl;
); tout << "ls:"; print_term_o(fix_vars(term_to_tighten + open_ml(m_tmp_l)),tout) << std::endl;
tout << "rs:"; print_term_o(term_to_lar_solver(remove_fresh_vars(create_term_from_ind_c())), tout ) << std::endl;
);
SASSERT( SASSERT(
fix_vars(term_to_tighten + open_ml(m_tmp_l)) == fix_vars(term_to_tighten + open_ml(m_tmp_l)) ==
term_to_lar_solver(remove_fresh_vars(create_term_from_ind_c()))); term_to_lar_solver(remove_fresh_vars(create_term_from_ind_c())));
@ -1283,6 +1302,69 @@ namespace lp {
return br; return br;
} }
bool columns_to_terms_is_correct() const {
std::unordered_map<unsigned, std::unordered_set<unsigned>> c2t;
for (unsigned k = 0; k < lra.terms().size(); k ++ ) {
const lar_term* t = lra.terms()[k];
if (!all_vars_are_int(*t)) continue;
for (const auto& p: *t) {
unsigned j = p.var();
auto it = c2t.find(j);
if (it == c2t.end()) {
std::unordered_set<unsigned> s;
s.insert(t->j());
c2t[j] = s;
} else {
it->second.insert(t->j());
}
}
}
for (const auto & p : c2t) {
unsigned j = p.first;
const auto it = m_columns_to_terms.find(j);
if (it == m_columns_to_terms.end()) {
TRACE("dioph_eq", tout << "column j" << j << " is not registered" << std::endl;
tout << "the column belongs to the the following terms:";
for (unsigned tj : p.second) {
tout << " " << tj;
}
tout << std::endl;
);
return false;
}
if (it->second != p.second) {
return false;
}
}
// reverse inclusion
for (const auto & p : m_columns_to_terms) {
unsigned j = p.first;
const auto it = c2t.find(j);
if (it == c2t.end()) {
TRACE("dioph_eq", tout << "should not be registered j " << j << std::endl;
lra.print_terms(tout););
return false;
}
if (it->second != p.second) {
return false;
}
}
return true;
}
bool is_in_sync() const {
unsigned n_local_columns = m_e_matrix.column_count();
for (unsigned j = 0; j < n_local_columns; j++) {
unsigned external_j = m_var_register.local_to_external(j);
if (external_j == UINT_MAX) continue;
if (external_j >= lra.column_count())
return false;
}
return columns_to_terms_is_correct();
}
public: public:
lia_move check() { lia_move check() {
lra.stats().m_dio_calls++; lra.stats().m_dio_calls++;
@ -1443,9 +1525,7 @@ namespace lp {
} }
while (!q.empty()) { while (!q.empty()) {
unsigned xt = pop_front(q); unsigned xt = pop_front(q);
term_o fresh_t = get_term_from_entry(m_fresh_definitions[xt]); term_o fresh_t = get_term_from_entry(m_fresh_definitions[xt].m_ei);
if (fresh_t.get_coeff(xt).is_minus_one() == false)
std::cout << "coeff:" << fresh_t.get_coeff(xt) << std::endl;
SASSERT(fresh_t.get_coeff(xt).is_minus_one()); SASSERT(fresh_t.get_coeff(xt).is_minus_one());
fresh_t.erase_var(xt); fresh_t.erase_var(xt);
if (!t.contains(xt)) if (!t.contains(xt))
@ -1551,8 +1631,10 @@ namespace lp {
m_k2s.resize(k + 1, -1); m_k2s.resize(k + 1, -1);
m_k2s[k] = fresh_row; m_k2s[k] = fresh_row;
m_fresh_definitions.resize(xt + 1, -1); fresh_definition fd(-1, -1);
m_fresh_definitions[xt] = fresh_row;
m_fresh_definitions.resize(xt + 1, fd);
m_fresh_definitions[xt] = fresh_definition(fresh_row, h);
TRACE("dioph_eq", tout << "changed entry:"; TRACE("dioph_eq", tout << "changed entry:";
print_entry(h, tout) << std::endl; print_entry(h, tout) << std::endl;
tout << "added entry for fresh var:\n"; tout << "added entry for fresh var:\n";
@ -1576,8 +1658,10 @@ namespace lp {
if (need_print_dep) { if (need_print_dep) {
out << "\tm_l:{"; out << "\tm_l:{";
print_lar_term_L(l_term_from_row(ei), out) << "}, "; print_lar_term_L(l_term_from_row(ei), out) << "}, ";
print_ml(l_term_from_row(ei), out << "\n\texpl of fixed in m_l:\n") << "\n"; print_ml(l_term_from_row(ei), out) << std::endl;
out << "expl of fixed in m_l:{\n";
print_dep(out, explain_fixed_in_meta_term(l_term_from_row(ei))); print_dep(out, explain_fixed_in_meta_term(l_term_from_row(ei)));
out << "}\n";
} }
switch (e.m_entry_status) { switch (e.m_entry_status) {
case entry_status::F: case entry_status::F:

View file

@ -1714,7 +1714,7 @@ namespace lp {
unsigned j = A_r().column_count(); unsigned j = A_r().column_count();
SASSERT(ext_index == null_lpvar || external_to_local(ext_index) == j); SASSERT(ext_index == null_lpvar || external_to_local(ext_index) == j);
column ul(term); column ul(term);
term->j() = j; // point from the term to the column term->set_j(j); // point from the term to the column
m_columns.push_back(ul); m_columns.push_back(ul);
m_trail.push(undo_add_column(*this)); m_trail.push(undo_add_column(*this));
add_basic_var_to_core_fields(); add_basic_var_to_core_fields();

View file

@ -32,7 +32,9 @@ protected:
public: public:
// the column index related to the term // the column index related to the term
lpvar j() const { return m_j; } lpvar j() const { return m_j; }
lpvar& j() { return m_j; } void set_j(unsigned j) {
m_j = j;
}
void add_monomial(const mpq& c, unsigned j) { void add_monomial(const mpq& c, unsigned j) {
if (c.is_zero()) if (c.is_zero())
return; return;