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

optimize dio changes

Signed-off-by: Lev Nachmanson <levnach@hotmail.com>
This commit is contained in:
Lev Nachmanson 2024-08-29 08:46:05 -10:00 committed by Lev Nachmanson
parent 18c75e6333
commit 66c6bad01c
2 changed files with 52 additions and 16 deletions

View file

@ -266,15 +266,23 @@ namespace lp {
} }
void substitute_var_on_f(unsigned k, int k_sign, const term_o& k_subst, const lar_term& l_term) { void substitute_var_on_f(unsigned k, int k_sign, const term_o& k_subst, const lar_term& l_term, unsigned index_to_avoid) {
TRACE("dioph_eq", print_term_o(k_subst, tout<< "x" << k<< " -> ") << std::endl; tout << "l_term:"; print_lar_term_L(l_term, tout) << std::endl;); TRACE("dioph_eq", print_term_o(k_subst, tout<< "x" << k<< " -> ") << std::endl; tout << "l_term:"; print_lar_term_L(l_term, tout) << std::endl;);
for (unsigned e_index: m_f) { for (unsigned e_index: m_f) {
if (e_index == index_to_avoid) continue;
term_o& e = m_eprime[e_index].m_e; term_o& e = m_eprime[e_index].m_e;
if (!e.contains(k)) continue; if (!e.contains(k)) continue;
const mpq& k_coeff = e.get_coeff(k); const mpq& k_coeff = e.get_coeff(k);
TRACE("dioph_eq", print_eprime_entry(e_index, tout << "before:") << std::endl; TRACE("dioph_eq", print_eprime_entry(e_index, tout << "before:") << std::endl;
tout << "k_coeff:" << k_coeff << std::endl;); tout << "k_coeff:" << k_coeff << std::endl;);
m_eprime[e_index].m_l += mpq(-k_sign)*k_coeff*l_term; if (!l_term.is_empty()) {
if (k_sign == 1)
m_eprime[e_index].m_l -= k_coeff*l_term;
else {
m_eprime[e_index].m_l += k_coeff*l_term;
}
}
e.substitute(k_subst, k); e.substitute(k_subst, k);
TRACE("dioph_eq", print_eprime_entry(e_index, tout << "after :") << std::endl;); TRACE("dioph_eq", print_eprime_entry(e_index, tout << "after :") << std::endl;);
} }
@ -310,47 +318,54 @@ namespace lp {
TRACE("dioph_eq", tout << "subst_term:"; print_term_o(t, tout) << std::endl;); TRACE("dioph_eq", tout << "subst_term:"; print_term_o(t, tout) << std::endl;);
return t; return t;
} }
void eliminate_var(eprime_pair& e_pair, unsigned k, int k_sign) { void eliminate_var_in_f(eprime_pair& e_pair, unsigned k, int k_sign) {
term_o t = get_term_to_subst(e_pair.m_e, k, k_sign); term_o t = get_term_to_subst(e_pair.m_e, k, k_sign);
substitute_var_on_f(k, k_sign, t, e_pair.m_l) ; substitute_var_on_f(k, k_sign, t, e_pair.m_l, -1) ; // -1 is for the index to ignore
} }
// k is the variable to substitute // k is the variable to substitute
void fresh_var_step(eprime_pair & e_pair, mpq& ahk, unsigned k) { void fresh_var_step(unsigned e_index, unsigned k) {
eprime_pair & e_pair = m_eprime[e_index];
// step 7 from the paper // step 7 from the paper
auto & eh = e_pair.m_e; auto & eh = e_pair.m_e;
// xt is the fresh variable // xt is the fresh variable
unsigned xt = m_last_fresh_x_var--; unsigned xt = m_last_fresh_x_var--;
TRACE("dioph_eq", tout << "introduce fresh xt:" << "~x"<< fresh_index(xt) << std::endl; TRACE("dioph_eq", tout << "introduce fresh xt:" << "~x"<< fresh_index(xt) << std::endl;
tout << "eh:"; print_term_o(eh,tout) << std::endl;); tout << "eh:"; print_term_o(eh,tout) << std::endl;);
/* Let eh = sum (a_i*x_i) + c = 0. /* Let eh = sum (a_i*x_i) + c
For each i != k, let a_i = a_qi*ahk + a_ri, and let c = c_q * ahk + c_r For each i != k, let a_i = a_qi*ahk + a_ri, and let c = c_q * ahk + c_r
eh = ahk * (x_k + sum {a_qi*x_i) + c_q | i != k }) + sum {a_ri*x_i | i != k} + c_r = 0 eh = ahk * (x_k + sum {a_qi*x_i) + c_q | i != k }) + sum {a_ri*x_i | i != k} + c_r = 0
x_t = x_k + sum {a_qi*x_i) + c_q | i != k }, it will be xt_term xt = x_k + sum {a_qi*x_i) + c_q | i != k }, it will be xt_term
Then x_k -> - sum {a_qi*x_i) + c_q | i != k } + x_t, it will be subs_k Then x_k -> - sum {a_qi*x_i) + c_q | i != k } + xt, it will be subs_k
eh = ahk*x_t + ... eh = ahk*xt + ...
*/ */
term_o xt_term; term_o xt_term;
term_o k_subs; term_o k_subs;
// copy it aside
const mpq ahk = eh.get_coeff(k);
mpq r, q = machine_div_rem(eh.c(), ahk, r); mpq r, q = machine_div_rem(eh.c(), ahk, r);
xt_term.add_var(k); xt_term.add_var(k);
xt_term.c() = q; xt_term.c() = q;
xt_term.add_monomial(mpq(-1), xt); xt_term.add_monomial(mpq(-1), xt);
k_subs.add_var(xt); k_subs.add_var(xt);
k_subs.c() = - q; k_subs.c() = - q;
term_o et; //et is the elimination k from eh, which is an optimization
et.add_monomial(ahk, xt);
et.c() = r;
for (const auto & p: eh) { for (const auto & p: eh) {
if (p.j() == k) continue; if (p.j() == k) continue;
q = machine_div_rem(p.coeff(), ahk, r); q = machine_div_rem(p.coeff(), ahk, r);
xt_term.add_monomial(q, p.j()); xt_term.add_monomial(q, p.j());
k_subs.add_monomial(-q, p.j()); k_subs.add_monomial(-q, p.j());
et.add_monomial(r, p.j());
} }
m_eprime[e_index].m_e = et;
eprime_pair xt_entry = {xt_term, lar_term()}; // 0 for m_l field eprime_pair xt_entry = {xt_term, lar_term()}; // 0 for m_l field
m_eprime.push_back(xt_entry); m_eprime.push_back(xt_entry);
TRACE("dioph_eq", tout << "xt_term:"; print_term_o(xt_term, tout) << std::endl; TRACE("dioph_eq", tout << "xt_term:"; print_term_o(xt_term, tout) << std::endl;
tout << "k_subs:"; print_term_o(k_subs, tout) << std::endl; tout << "k_subs:"; print_term_o(k_subs, tout) << std::endl;
print_eprime_entry(m_eprime.size()-1, tout);); print_eprime_entry(m_eprime.size()-1, tout););
substitute_var_on_f(k, 1, k_subs, xt_entry.m_l); substitute_var_on_f(k, 1, k_subs, xt_entry.m_l, e_index);
// the term to eliminate the fresh variable // the term to eliminate the fresh variable
term_o xt_subs = xt_term.clone(); term_o xt_subs = xt_term.clone();
@ -377,10 +392,9 @@ namespace lp {
TRACE("dioph_eq", tout << "push to S:\n"; print_eprime_entry(*eh_it, tout);); TRACE("dioph_eq", tout << "push to S:\n"; print_eprime_entry(*eh_it, tout););
m_s.push_back(*eh_it); m_s.push_back(*eh_it);
m_f.erase(eh_it); m_f.erase(eh_it);
eliminate_var(eprime_entry, k , k_sign); eliminate_var_in_f(eprime_entry, k , k_sign);
} else { } else {
ahk *= k_sign; // restore the original coefficient fresh_var_step(*eh_it, k);
fresh_var_step(eprime_entry, ahk, k);
} }
} }

View file

@ -45,6 +45,20 @@ public:
m_coeffs.erase(j); m_coeffs.erase(j);
} }
} }
void sub_monomial(const mpq& c, unsigned j) {
if (c.is_zero())
return;
auto *e = m_coeffs.find_core(j);
if (e == nullptr) {
m_coeffs.insert(j, c);
} else {
e->get_data().m_value -= c;
if (e->get_data().m_value.is_zero())
m_coeffs.erase(j);
}
}
void add_var(lpvar j) { void add_var(lpvar j) {
rational c(1); rational c(1);
@ -172,9 +186,9 @@ public:
friend lar_term operator*(const mpq& k, const lar_term& term) { friend lar_term operator*(const mpq& k, const lar_term& term) {
lar_term r; lar_term r;
for (const auto& p : term) { for (const auto& p : term) {
r.add_monomial(p.coeff()*k, p.j()); r.add_monomial(p.coeff()*k, p.j());
} }
return r; return r;
} }
@ -184,6 +198,14 @@ public:
} }
return *this; return *this;
} }
lar_term& operator-=(const lar_term& a) {
for (const auto& p : a) {
sub_monomial(p.coeff(), p.j());
}
return *this;
}
lar_term& operator*=(mpq const& k) { lar_term& operator*=(mpq const& k) {
for (auto & t : m_coeffs) for (auto & t : m_coeffs)