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

more review of mbp_arrays

This commit is contained in:
Nikolaj Bjorner 2025-02-18 19:48:54 -08:00
parent a4a84ed083
commit 42f6e1300a

View file

@ -44,14 +44,15 @@ struct mbp_array_tg::impl {
obj_pair_hashtable<expr, expr> m_seenp; obj_pair_hashtable<expr, expr> m_seenp;
// apply rules that split on model // apply rules that split on model
bool m_use_mdl; bool m_use_mdl = false;
// m_has_store.is_marked(t) if t has a subterm store(v) where v is a // m_has_store.is_marked(t) if t has a subterm store(v) where v is a
// variable to be eliminated // variable to be eliminated
ast_mark m_has_stores; ast_mark m_has_stores;
// variables required for applying rules // variables required for applying rules
vector<expr_ref_vector> indices; vector<expr_ref_vector> indices;
expr_ref_vector terms, rdTerms; expr_ref_vector terms;
app_ref_vector rdTerms;
bool has_var(expr *t) { return contains_vars(t, m_vars_set, m); } bool has_var(expr *t) { return contains_vars(t, m_vars_set, m); }
@ -157,7 +158,7 @@ struct mbp_array_tg::impl {
impl(ast_manager &man, mbp::term_graph &tg, model &mdl, impl(ast_manager &man, mbp::term_graph &tg, model &mdl,
obj_hashtable<app> &vars_set, expr_sparse_mark &seen) obj_hashtable<app> &vars_set, expr_sparse_mark &seen)
: m(man), m_array_util(m), m_tg(tg), m_mdl(mdl), m_vars_set(vars_set), : m(man), m_array_util(m), m_tg(tg), m_mdl(mdl), m_vars_set(vars_set),
m_new_vars(m), m_seen(seen), m_use_mdl(false), terms(m), rdTerms(m) {} m_new_vars(m), m_seen(seen), terms(m), rdTerms(m) {}
// create a peq where write terms are preferred on the left hand side // create a peq where write terms are preferred on the left hand side
peq mk_wr_peq(expr *e1, expr *e2) { peq mk_wr_peq(expr *e1, expr *e2) {
@ -186,9 +187,9 @@ struct mbp_array_tg::impl {
// !(select(y, j) = elem) // !(select(y, j) = elem)
void elimwreq(peq p, bool is_neg) { void elimwreq(peq p, bool is_neg) {
expr* a = nullptr, *j = nullptr, *elem = nullptr; expr* a = nullptr, *j = nullptr, *elem = nullptr;
VERIFY(is_arr_write(p.lhs(), a, j, elem)); VERIFY(is_arr_write(p.lhs(), a, j, elem)); // TDOO: make this work with multi-arity arrays
TRACE("mbp_tg", TRACE("mbp_tg",
tout << "applying elimwreq on " << expr_ref(p.mk_peq(), m) << " is neg: " << is_neg;); tout << "applying elimwreq on " << expr_ref(p.mk_peq(), m) << " is neg: " << is_neg << "\n");
vector<expr_ref_vector> indices; vector<expr_ref_vector> indices;
bool in = false; bool in = false;
p.get_diff_indices(indices); p.get_diff_indices(indices);
@ -201,7 +202,8 @@ struct mbp_array_tg::impl {
// save for later // save for later
eq_index = i; eq_index = i;
break; break;
} else }
else
deq.push_back(i); deq.push_back(i);
} }
} }
@ -216,7 +218,8 @@ struct mbp_array_tg::impl {
m_tg.add_eq(p_new.mk_peq(), p.mk_peq()); m_tg.add_eq(p_new.mk_peq(), p.mk_peq());
return; return;
} }
for (expr *d : deq) { m_tg.add_deq(j, d); } for (expr *d : deq)
m_tg.add_deq(j, d);
expr_ref_vector setOne(m); expr_ref_vector setOne(m);
setOne.push_back(j); setOne.push_back(j);
indices.push_back(setOne); indices.push_back(setOne);
@ -226,9 +229,11 @@ struct mbp_array_tg::impl {
m_tg.add_lit(p_new.mk_peq()); m_tg.add_lit(p_new.mk_peq());
m_tg.add_eq(rd, elem); m_tg.add_eq(rd, elem);
m_tg.add_eq(p.mk_peq(), p_new.mk_peq()); m_tg.add_eq(p.mk_peq(), p_new.mk_peq());
} else { }
else {
expr_ref rd_eq(m.mk_eq(rd, elem), m); expr_ref rd_eq(m.mk_eq(rd, elem), m);
if (m_mdl.is_false(rd_eq)) { m_tg.add_deq(rd, elem); } if (m_mdl.is_false(rd_eq))
m_tg.add_deq(rd, elem);
else { else {
expr_ref npeq(mk_not(p_new.mk_peq()), m); expr_ref npeq(mk_not(p_new.mk_peq()), m);
m_tg.add_lit(npeq); m_tg.add_lit(npeq);
@ -293,23 +298,22 @@ struct mbp_array_tg::impl {
// iterate through all terms in m_tg and apply all array MBP rules once // iterate through all terms in m_tg and apply all array MBP rules once
// returns true if any rules were applied // returns true if any rules were applied
bool apply() { bool apply() {
TRACE("mbp_tg", tout << "Iterating over terms of tg";); TRACE("mbp_tg", tout << "Iterating over terms of tg\n");
indices.reset(); indices.reset();
rdTerms.reset(); rdTerms.reset();
m_new_vars.reset(); m_new_vars.reset();
expr_ref e(m), rdEq(m), rdDeq(m); expr_ref e(m), rdEq(m), rdDeq(m);
expr *nt, *term; expr *nt = nullptr;
bool progress = false, is_neg = false; bool progress = false, is_neg = false;
// Not resetting terms because get_terms calls resize on terms // Not resetting terms because get_terms calls resize on terms
m_tg.get_terms(terms, false); m_tg.get_terms(terms, false);
for (unsigned i = 0; i < terms.size(); i++) { for (expr* term : terms) {
term = terms.get(i);
if (m_seen.is_marked(term)) if (m_seen.is_marked(term))
continue; continue;
if (m_tg.is_cgr(term)) if (m_tg.is_cgr(term))
continue; continue;
TRACE("mbp_tg", tout << "processing " << expr_ref(term, m);); TRACE("mbp_tg", tout << "processing " << expr_ref(term, m) << "\n");
expr* a, *b; expr* a, *b;
if (is_implicit_peq(term, a, b) || is_neg_peq(term, a, b)) { if (is_implicit_peq(term, a, b) || is_neg_peq(term, a, b)) {
// rewrite array eq as peq // rewrite array eq as peq
@ -337,7 +341,6 @@ struct mbp_array_tg::impl {
if (!m_array_util.is_store(p.lhs()) && has_var(p.lhs()) && !is_neg) { if (!m_array_util.is_store(p.lhs()) && has_var(p.lhs()) && !is_neg) {
// TODO: don't apply this rule if vars in p.lhs() also // TODO: don't apply this rule if vars in p.lhs() also
// appear in p.rhs() // appear in p.rhs()
mark_seen(p.lhs()); mark_seen(p.lhs());
mark_seen(nt); mark_seen(nt);
mark_seen(term); mark_seen(term);
@ -357,6 +360,7 @@ struct mbp_array_tg::impl {
continue; continue;
} }
} }
TRACE("mbp_tg", tout << "not applying any rules on " << expr_ref(nt, m) << " " << is_rd_wr(nt) << " " << m_use_mdl << "\n";);
if (m_use_mdl && is_rd_wr(nt)) { if (m_use_mdl && is_rd_wr(nt)) {
mark_seen(term); mark_seen(term);
progress = true; progress = true;
@ -368,23 +372,22 @@ struct mbp_array_tg::impl {
// iterate over term graph again to collect read terms // iterate over term graph again to collect read terms
// irrespective of whether they have been marked or not // irrespective of whether they have been marked or not
rdTerms.reset(); rdTerms.reset();
for (unsigned i = 0; i < terms.size(); i++) { for (auto term : terms) {
term = terms.get(i);
if (m_array_util.is_select(term) && if (m_array_util.is_select(term) &&
has_var(to_app(term)->get_arg(0))) { has_var(to_app(term)->get_arg(0))) {
rdTerms.push_back(term); rdTerms.push_back(to_app(term));
if (is_seen(term)) continue; if (is_seen(term)) continue;
add_rdVar(term); add_rdVar(term);
mark_seen(term); mark_seen(term);
} }
} }
if (!m_use_mdl) if (!m_use_mdl)
return progress; return progress;
for (unsigned i = 0; i < rdTerms.size(); i++) { for (unsigned i = 0; i < rdTerms.size(); i++) {
app* e1 = to_app(rdTerms.get(i)); app* e1 = rdTerms.get(i);
expr* a1 = e1->get_arg(0); expr* a1 = e1->get_arg(0);
for (unsigned j = i + 1; j < rdTerms.size(); j++) { for (unsigned j = i + 1; j < rdTerms.size(); j++) {
app* e2 = to_app(rdTerms.get(j)); app* e2 = rdTerms.get(j);
expr* a2 = e2->get_arg(0); expr* a2 = e2->get_arg(0);
if (!is_seen(e1, e2) && a1 == e2) { if (!is_seen(e1, e2) && a1 == e2) {
mark_seen(e1, e2); mark_seen(e1, e2);