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:
parent
a0ece6dd2c
commit
083926c658
|
@ -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:
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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;
|
||||||
|
|
Loading…
Reference in a new issue