/*++ Copyright (c) 2012 Microsoft Corporation Module Name: expr_safe_replace.cpp Abstract: Version of expr_replace/expr_substitution that is safe for quantifiers. Author: Nikolaj Bjorner (nbjorner) 2012-11-30 Revision History: --*/ #include "ast/rewriter/expr_safe_replace.h" #include "ast/rewriter/var_subst.h" #include "ast/ast_pp.h" void expr_safe_replace::insert(expr* src, expr* dst) { SASSERT(src->get_sort() == dst->get_sort()); m_src.push_back(src); m_dst.push_back(dst); } void expr_safe_replace::operator()(expr_ref_vector& es) { if (empty()) return; expr_ref val(m); for (unsigned i = 0; i < es.size(); ++i) { (*this)(es.get(i), val); es[i] = val; } } void expr_safe_replace::operator()(expr* e, expr_ref& res) { if (empty()) { res = e; return; } for (unsigned i = 0, e = m_src.size(); i < e; ++i) { m_cache.emplace(m_src.get(i), m_dst.get(i)); } m_todo.push_back(e); expr* a, *b; while (!m_todo.empty()) { a = m_todo.back(); auto &cached = m_cache[a]; if (cached) { m_todo.pop_back(); } else if (is_var(a)) { cached = a; m_todo.pop_back(); } else if (is_app(a)) { app* c = to_app(a); m_args.reset(); bool arg_differs = false, has_all_args = true; for (expr* arg : *c) { expr* d = m_cache[arg]; if (d) { if (has_all_args) { m_args.push_back(d); arg_differs |= arg != d; SASSERT(arg->get_sort() == d->get_sort()); } } else { m_todo.push_back(arg); has_all_args = false; } } if (has_all_args) { if (arg_differs) { b = m.mk_app(c->get_decl(), m_args); m_refs.push_back(b); SASSERT(a->get_sort() == b->get_sort()); } else { b = a; } cached = b; m_todo.pop_back(); } } else { SASSERT(is_quantifier(a)); quantifier* q = to_quantifier(a); expr_safe_replace replace(m); var_shifter shift(m); expr_ref new_body(m), src(m), dst(m), tmp(m); expr_ref_vector pats(m), nopats(m); unsigned num_decls = q->get_num_decls(); for (unsigned i = 0; i < m_src.size(); ++i) { shift(m_src.get(i), num_decls, src); shift(m_dst.get(i), num_decls, dst); replace.insert(src, dst); } unsigned np = q->get_num_patterns(); for (unsigned i = 0; i < np; ++i) { replace(q->get_pattern(i), tmp); pats.push_back(tmp); } np = q->get_num_no_patterns(); for (unsigned i = 0; i < np; ++i) { replace(q->get_no_pattern(i), tmp); nopats.push_back(tmp); } replace(q->get_expr(), new_body); b = m.update_quantifier(q, pats.size(), pats.c_ptr(), nopats.size(), nopats.c_ptr(), new_body); m_refs.push_back(b); cached = b; m_todo.pop_back(); } } res = m_cache.at(e); m_cache.clear(); m_todo.reset(); m_args.reset(); m_refs.reset(); } void expr_safe_replace::reset() { m_src.reset(); m_dst.reset(); m_refs.finalize(); m_cache.clear(); } void expr_safe_replace::apply_substitution(expr* s, expr* def, expr_ref& t) { reset(); insert(s, def); (*this)(t, t); reset(); }