mirror of
				https://github.com/Z3Prover/z3
				synced 2025-10-26 17:29:21 +00:00 
			
		
		
		
	add demodulator simplifier
refactor demodulator-rewriter a bit to separate reusable features.
This commit is contained in:
		
							parent
							
								
									9acbfa3923
								
							
						
					
					
						commit
						1974c224ab
					
				
					 8 changed files with 624 additions and 155 deletions
				
			
		|  | @ -39,7 +39,7 @@ def init_project_def(): | ||||||
|     add_lib('macros', ['rewriter'], 'ast/macros') |     add_lib('macros', ['rewriter'], 'ast/macros') | ||||||
|     add_lib('model',  ['macros']) |     add_lib('model',  ['macros']) | ||||||
|     add_lib('converters', ['model'], 'ast/converters') |     add_lib('converters', ['model'], 'ast/converters') | ||||||
|     add_lib('simplifiers', ['euf', 'normal_forms', 'bit_blaster', 'converters'], 'ast/simplifiers') |     add_lib('simplifiers', ['euf', 'normal_forms', 'bit_blaster', 'converters', 'substitution'], 'ast/simplifiers') | ||||||
|     add_lib('tactic', ['simplifiers']) |     add_lib('tactic', ['simplifiers']) | ||||||
|     add_lib('solver', ['params', 'model', 'tactic', 'proofs']) |     add_lib('solver', ['params', 'model', 'tactic', 'proofs']) | ||||||
|     add_lib('cmd_context', ['solver', 'rewriter', 'params']) |     add_lib('cmd_context', ['solver', 'rewriter', 'params']) | ||||||
|  |  | ||||||
|  | @ -52,9 +52,9 @@ add_subdirectory(ast/macros) | ||||||
| add_subdirectory(model) | add_subdirectory(model) | ||||||
| add_subdirectory(ast/euf) | add_subdirectory(ast/euf) | ||||||
| add_subdirectory(ast/converters) | add_subdirectory(ast/converters) | ||||||
|  | add_subdirectory(ast/substitution) | ||||||
| add_subdirectory(ast/simplifiers) | add_subdirectory(ast/simplifiers) | ||||||
| add_subdirectory(tactic) | add_subdirectory(tactic) | ||||||
| add_subdirectory(ast/substitution) |  | ||||||
| add_subdirectory(smt/params) | add_subdirectory(smt/params) | ||||||
| add_subdirectory(parsers/util) | add_subdirectory(parsers/util) | ||||||
| add_subdirectory(math/grobner) | add_subdirectory(math/grobner) | ||||||
|  |  | ||||||
|  | @ -52,6 +52,7 @@ public: | ||||||
| 
 | 
 | ||||||
|     expr_ref mk_app(func_decl* f, unsigned num_args, expr* const* args); |     expr_ref mk_app(func_decl* f, unsigned num_args, expr* const* args); | ||||||
|     expr_ref mk_app(func_decl* f, ptr_vector<expr> const& args) { return mk_app(f, args.size(), args.data()); } |     expr_ref mk_app(func_decl* f, ptr_vector<expr> const& args) { return mk_app(f, args.size(), args.data()); } | ||||||
|  |     expr_ref mk_app(func_decl* f, expr_ref_vector const& args) { return mk_app(f, args.size(), args.data()); } | ||||||
|     expr_ref mk_eq(expr* a, expr* b); |     expr_ref mk_eq(expr* a, expr* b); | ||||||
| 
 | 
 | ||||||
|     bool reduce_quantifier(quantifier * old_q,  |     bool reduce_quantifier(quantifier * old_q,  | ||||||
|  |  | ||||||
|  | @ -3,6 +3,7 @@ z3_add_component(simplifiers | ||||||
|     bit_blaster.cpp |     bit_blaster.cpp | ||||||
|     bv_slice.cpp |     bv_slice.cpp | ||||||
|     card2bv.cpp |     card2bv.cpp | ||||||
|  |     demodulator_simplifier.cpp | ||||||
|     dependent_expr_state.cpp |     dependent_expr_state.cpp | ||||||
|     elim_unconstrained.cpp |     elim_unconstrained.cpp | ||||||
|     eliminate_predicates.cpp |     eliminate_predicates.cpp | ||||||
|  | @ -18,4 +19,5 @@ z3_add_component(simplifiers | ||||||
|     rewriter |     rewriter | ||||||
|     bit_blaster |     bit_blaster | ||||||
|     normal_forms |     normal_forms | ||||||
|  |     substitution | ||||||
| ) | ) | ||||||
|  |  | ||||||
							
								
								
									
										199
									
								
								src/ast/simplifiers/demodulator_simplifier.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										199
									
								
								src/ast/simplifiers/demodulator_simplifier.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,199 @@ | ||||||
|  | /*++
 | ||||||
|  | Copyright (c) 2022 Microsoft Corporation | ||||||
|  | 
 | ||||||
|  | Module Name: | ||||||
|  | 
 | ||||||
|  |     demodulator_simplifier.cpp | ||||||
|  | 
 | ||||||
|  | Author: | ||||||
|  | 
 | ||||||
|  |     Nikolaj Bjorner (nbjorner) 2022-12-4 | ||||||
|  | 
 | ||||||
|  | --*/ | ||||||
|  | 
 | ||||||
|  | #include "ast/simplifiers/demodulator_simplifier.h" | ||||||
|  | 
 | ||||||
|  | demodulator_index::~demodulator_index() { | ||||||
|  |     reset(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_index::reset() { | ||||||
|  |     for (auto& [k, v] : m_fwd_index) | ||||||
|  |         dealloc(v); | ||||||
|  |     for (auto& [k, v] : m_bwd_index) | ||||||
|  |         dealloc(v); | ||||||
|  |     m_fwd_index.reset(); | ||||||
|  |     m_bwd_index.reset(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_index::add(func_decl* f, unsigned i, obj_map<func_decl, uint_set*>& map) { | ||||||
|  |     uint_set* s; | ||||||
|  |     if (!map.find(f, s)) { | ||||||
|  |         s = alloc(uint_set); | ||||||
|  |         map.insert(f, s); | ||||||
|  |     } | ||||||
|  |     s->insert(i); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_index::del(func_decl* f, unsigned i, obj_map<func_decl, uint_set*>& map) { | ||||||
|  |     uint_set* s; | ||||||
|  |     if (map.find(f, s)) | ||||||
|  |         s->remove(i); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_index::insert_bwd(expr* e, unsigned i) { | ||||||
|  |     struct proc { | ||||||
|  |         unsigned i; | ||||||
|  |         demodulator_index& idx; | ||||||
|  |         proc(unsigned i, demodulator_index& idx) :i(i), idx(idx) {} | ||||||
|  |         void operator()(app* a) { | ||||||
|  |             if (a->get_num_args() > 0 && is_uninterp(a)) | ||||||
|  |                 idx.add(a->get_decl(), i, idx.m_bwd_index); | ||||||
|  |         } | ||||||
|  |         void operator()(expr* e) {} | ||||||
|  |     }; | ||||||
|  |     proc p(i, *this); | ||||||
|  |     for_each_expr(p, e); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_index::remove_bwd(expr* e, unsigned i) { | ||||||
|  |     struct proc { | ||||||
|  |         unsigned i; | ||||||
|  |         demodulator_index& idx; | ||||||
|  |         proc(unsigned i, demodulator_index& idx) :i(i), idx(idx) {} | ||||||
|  |         void operator()(app* a) { | ||||||
|  |             if (a->get_num_args() > 0 && is_uninterp(a)) | ||||||
|  |                 idx.del(a->get_decl(), i, idx.m_bwd_index); | ||||||
|  |         } | ||||||
|  |         void operator()(expr* e) {} | ||||||
|  |     }; | ||||||
|  |     proc p(i, *this); | ||||||
|  |     for_each_expr(p, e); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | demodulator_simplifier::demodulator_simplifier(ast_manager& m, params_ref const& p, dependent_expr_state& st): | ||||||
|  |     dependent_expr_simplifier(m, st), | ||||||
|  |     m_util(m), | ||||||
|  |     m_match_subst(m), | ||||||
|  |     m_rewriter(m), | ||||||
|  |     m_pinned(m) | ||||||
|  | { | ||||||
|  |     std::function<bool(func_decl* f, expr_ref_vector const& args, expr_ref& r)> rw = [&](func_decl* f, expr_ref_vector const& args, expr_ref& r) { | ||||||
|  |         return rewrite1(f, args, r); | ||||||
|  |     }; | ||||||
|  |     m_rewriter.set_rewrite1(rw); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_simplifier::rewrite(unsigned i) { | ||||||
|  |     if (m_index.empty()) | ||||||
|  |         return; | ||||||
|  | 
 | ||||||
|  |     m_dependencies.reset(); | ||||||
|  |     expr* f = fml(i); | ||||||
|  |     expr_ref r = m_rewriter.rewrite(f); | ||||||
|  |     if (r == f) | ||||||
|  |         return; | ||||||
|  |     expr_dependency_ref d(dep(i), m); | ||||||
|  |     for (unsigned j : m_dependencies) | ||||||
|  |         d = m.mk_join(d, dep(j)); | ||||||
|  |     m_fmls.update(i, dependent_expr(m, r, d)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool demodulator_simplifier::rewrite1(func_decl* f, expr_ref_vector const& args, expr_ref& np) { | ||||||
|  |     uint_set* set; | ||||||
|  |     if (!m_index.find_fwd(f, set)) | ||||||
|  |         return false; | ||||||
|  | 
 | ||||||
|  |     TRACE("demodulator", tout << "trying to rewrite: " << f->get_name() << " args:\n" << m_new_args << "\n";); | ||||||
|  | 
 | ||||||
|  |     for (unsigned i : *set) { | ||||||
|  | 
 | ||||||
|  |         auto const& [lhs, rhs] = m_rewrites[i]; | ||||||
|  | 
 | ||||||
|  |         if (lhs->get_num_args() != args.size()) | ||||||
|  |             continue; | ||||||
|  | 
 | ||||||
|  |         SASSERT(lhs->get_decl() == f); | ||||||
|  | 
 | ||||||
|  |         TRACE("demodulator", tout << "Matching with demodulator: " << mk_pp(d, m) << std::endl; ); | ||||||
|  | 
 | ||||||
|  |         if (m_match_subst(lhs, rhs, args.data(), np)) { | ||||||
|  |             TRACE("demodulator_bug", tout << "succeeded...\n" << mk_pp(rhs, m) << "\n===>\n" << np << "\n";); | ||||||
|  |             m_dependencies.insert(i); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_simplifier::reschedule_processed(func_decl* f) { | ||||||
|  |     uint_set* set = nullptr; | ||||||
|  |     if (!m_index.find_bwd(f, set)) | ||||||
|  |         return; | ||||||
|  |     uint_set tmp; | ||||||
|  |     for (auto i : *set) | ||||||
|  |         if (m_processed.contains(i)) | ||||||
|  |             tmp.insert(i); | ||||||
|  |     for (auto i : tmp) { | ||||||
|  |         m_processed.remove(i); | ||||||
|  |         m_index.remove_fwd(f, i); | ||||||
|  |         m_index.remove_bwd(fml(i), i); | ||||||
|  |         m_todo.push_back(i); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_simplifier::reschedule_demodulators(func_decl* f, expr* lhs) { | ||||||
|  |     uint_set* set; | ||||||
|  |     if (!m_index.find_bwd(f, set)) | ||||||
|  |         return; | ||||||
|  |     uint_set all_occurrences(*set); | ||||||
|  |     for (unsigned i : all_occurrences) { | ||||||
|  |         app_expr_pair p; | ||||||
|  |         if (!m_rewrites.find(i, p)) | ||||||
|  |             continue;         | ||||||
|  |         if (!m_match_subst.can_rewrite(fml(i), lhs)) | ||||||
|  |             continue; | ||||||
|  |         func_decl* f = p.first->get_decl(); | ||||||
|  |         m_index.remove_fwd(f, i); | ||||||
|  |         m_index.remove_bwd(fml(i), i); | ||||||
|  |         m_todo.push_back(i); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_simplifier::reset() { | ||||||
|  |     m_pinned.reset(); | ||||||
|  |     m_index.reset(); | ||||||
|  |     m_processed.reset(); | ||||||
|  |     m_todo.reset(); | ||||||
|  |     unsigned max_vid = 1; | ||||||
|  |     for (unsigned i : indices()) | ||||||
|  |         max_vid = std::max(max_vid, m_util.max_var_id(fml(i))); | ||||||
|  |     m_match_subst.reserve(max_vid); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_simplifier::reduce() { | ||||||
|  |     reset(); | ||||||
|  |     for (unsigned i : indices()) | ||||||
|  |         m_todo.push_back(i); | ||||||
|  | 
 | ||||||
|  |     app_ref large(m); | ||||||
|  |     expr_ref small(m); | ||||||
|  |     while (!m_todo.empty()) { | ||||||
|  |         unsigned i = m_todo.back(); | ||||||
|  |         m_todo.pop_back(); | ||||||
|  |         rewrite(i); | ||||||
|  |         if (m_util.is_demodulator(fml(i), large, small)) { | ||||||
|  |             func_decl* f = large->get_decl(); | ||||||
|  |             reschedule_processed(f); | ||||||
|  |             reschedule_demodulators(f, large); | ||||||
|  |             m_index.insert_fwd(f, i); | ||||||
|  |             m_rewrites.insert(i, app_expr_pair(large, small)); | ||||||
|  |             m_pinned.push_back(large); | ||||||
|  |             m_pinned.push_back(small); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |             m_processed.insert(i); | ||||||
|  |         m_index.insert_bwd(fml(i), i); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										60
									
								
								src/ast/simplifiers/demodulator_simplifier.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								src/ast/simplifiers/demodulator_simplifier.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,60 @@ | ||||||
|  | /*++
 | ||||||
|  | Copyright (c) 2022 Microsoft Corporation | ||||||
|  | 
 | ||||||
|  | Module Name: | ||||||
|  | 
 | ||||||
|  |     demodulator_simplifier.h | ||||||
|  | 
 | ||||||
|  | Author: | ||||||
|  | 
 | ||||||
|  |     Nikolaj Bjorner (nbjorner) 2022-12-4 | ||||||
|  | 
 | ||||||
|  | --*/ | ||||||
|  | 
 | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include "ast/substitution/demodulator_rewriter.h" | ||||||
|  | #include "ast/simplifiers/dependent_expr_state.h" | ||||||
|  | #include "util/uint_set.h" | ||||||
|  | 
 | ||||||
|  | class demodulator_index { | ||||||
|  |     obj_map<func_decl, uint_set*> m_fwd_index, m_bwd_index;  | ||||||
|  |     void add(func_decl* f, unsigned i, obj_map<func_decl, uint_set*>& map); | ||||||
|  |     void del(func_decl* f, unsigned i, obj_map<func_decl, uint_set*>& map); | ||||||
|  |  public: | ||||||
|  |     ~demodulator_index(); | ||||||
|  |     void reset(); | ||||||
|  |     void insert_fwd(func_decl* f, unsigned i) { add(f, i, m_fwd_index); } | ||||||
|  |     void remove_fwd(func_decl* f, unsigned i) { del(f, i, m_fwd_index); } | ||||||
|  |     void insert_bwd(expr* e, unsigned i); | ||||||
|  |     void remove_bwd(expr* e, unsigned i); | ||||||
|  |     bool find_fwd(func_decl* f, uint_set*& s) { return m_bwd_index.find(f, s); } | ||||||
|  |     bool find_bwd(func_decl* f, uint_set*& s) { return m_fwd_index.find(f, s); } | ||||||
|  |     bool empty() const { return m_fwd_index.empty(); } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | class demodulator_simplifier : public dependent_expr_simplifier { | ||||||
|  |     typedef std::pair<app*, expr*> app_expr_pair; | ||||||
|  |     demodulator_index         m_index; | ||||||
|  |     demodulator_util          m_util; | ||||||
|  |     demodulator_match_subst   m_match_subst; | ||||||
|  |     demodulator_rewriter_util m_rewriter; | ||||||
|  |     u_map<app_expr_pair>      m_rewrites; | ||||||
|  |     uint_set                  m_processed, m_dependencies; | ||||||
|  |     unsigned_vector           m_todo; | ||||||
|  |     expr_ref_vector           m_pinned; | ||||||
|  | 
 | ||||||
|  |     void rewrite(unsigned i); | ||||||
|  |     bool rewrite1(func_decl* f, expr_ref_vector const& args, expr_ref& np); | ||||||
|  |     expr* fml(unsigned i) { return m_fmls[i].fml(); } | ||||||
|  |     expr_dependency* dep(unsigned i) { return m_fmls[i].dep(); } | ||||||
|  |     void reschedule_processed(func_decl* f); | ||||||
|  |     void reschedule_demodulators(func_decl* f, expr* lhs); | ||||||
|  |     void reset(); | ||||||
|  | 
 | ||||||
|  |  public: | ||||||
|  |      demodulator_simplifier(ast_manager& m, params_ref const& p, dependent_expr_state& st); | ||||||
|  |      void reduce() override; | ||||||
|  | 
 | ||||||
|  |      | ||||||
|  | }; | ||||||
|  | @ -27,32 +27,73 @@ Revision History: | ||||||
| #include "ast/rewriter/var_subst.h" | #include "ast/rewriter/var_subst.h" | ||||||
| #include "ast/substitution/demodulator_rewriter.h" | #include "ast/substitution/demodulator_rewriter.h" | ||||||
| 
 | 
 | ||||||
| demodulator_rewriter::demodulator_rewriter(ast_manager & m): | 
 | ||||||
|     m(m), | class var_set_proc { | ||||||
|     m_match_subst(m), |     uint_set & m_set; | ||||||
|     m_bsimp(m), | public: | ||||||
|     m_todo(m), |     var_set_proc(uint_set &s):m_set(s) {} | ||||||
|     m_in_processed(m), |     void operator()(var * n) { m_set.insert(n->get_idx()); } | ||||||
|     m_new_args(m),  |     void operator()(quantifier * n) {} | ||||||
|     m_rewrite_todo(m), |     void operator()(app * n) {} | ||||||
|     m_rewrite_cache(m), | }; | ||||||
|     m_new_exprs(m) { | 
 | ||||||
|     params_ref p; | int demodulator_util::is_subset(expr * e1, expr * e2) const { | ||||||
|     p.set_bool("elim_and", true); |     uint_set ev1, ev2; | ||||||
|     m_bsimp.updt_params(p); | 
 | ||||||
|  |     if (m.is_value(e1)) | ||||||
|  |         return 1; // values are always a subset!
 | ||||||
|  | 
 | ||||||
|  |     var_set_proc proc1(ev1); | ||||||
|  |     for_each_expr(proc1, e1); | ||||||
|  |     var_set_proc proc2(ev2); | ||||||
|  |     for_each_expr(proc2, e2); | ||||||
|  | 
 | ||||||
|  |     return (ev1==ev2          ) ? +2 : // We return +2 if the sets are equal.
 | ||||||
|  |            (ev1.subset_of(ev2)) ? +1 : | ||||||
|  |            (ev2.subset_of(ev1)) ? -1 : | ||||||
|  |                                    0 ; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| demodulator_rewriter::~demodulator_rewriter() { | int demodulator_util::is_smaller(expr * e1, expr * e2) const { | ||||||
|     reset_dealloc_values(m_fwd_idx); |     unsigned sz1 = 0, sz2 = 0; | ||||||
|     reset_dealloc_values(m_back_idx); | 
 | ||||||
|     for (auto & kv : m_demodulator2lhs_rhs) { |     // values are always smaller!
 | ||||||
|         m.dec_ref(kv.m_key); |     if (m.is_value(e1)) | ||||||
|         m.dec_ref(kv.m_value.first); |         return +1; | ||||||
|         m.dec_ref(kv.m_value.second); |     else if (m.is_value(e2)) | ||||||
|  |         return -1; | ||||||
|  | 
 | ||||||
|  |     // interpreted stuff is always better than uninterpreted.
 | ||||||
|  |     if (!is_uninterp(e1) && is_uninterp(e2)) | ||||||
|  |         return +1; | ||||||
|  |     else if (is_uninterp(e1) && !is_uninterp(e2)) | ||||||
|  |         return -1; | ||||||
|  | 
 | ||||||
|  |     // two uninterpreted functions are ordered first by the number of
 | ||||||
|  |     // arguments, then by their id.
 | ||||||
|  |     if (is_uninterp(e1) && is_uninterp(e2)) { | ||||||
|  |         if (to_app(e1)->get_num_args() < to_app(e2)->get_num_args()) | ||||||
|  |             return +1; | ||||||
|  |         else if (to_app(e1)->get_num_args() > to_app(e2)->get_num_args()) | ||||||
|  |             return -1; | ||||||
|  |         else { | ||||||
|  |             unsigned a = to_app(e1)->get_decl()->get_id(); | ||||||
|  |             unsigned b = to_app(e2)->get_decl()->get_id(); | ||||||
|  |             if (a < b) | ||||||
|  |                 return +1; | ||||||
|  |             else if (a > b) | ||||||
|  |                 return -1; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |     sz1 = get_depth(e1); | ||||||
|  |     sz2 = get_depth(e2); | ||||||
|  | 
 | ||||||
|  |     return (sz1 == sz2) ?  0 : | ||||||
|  |            (sz1  < sz2) ? +1 : | ||||||
|  |                           -1 ; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool demodulator_rewriter::is_demodulator(expr * e, app_ref & large, expr_ref & small) const { | bool demodulator_util::is_demodulator(expr * e, app_ref & large, expr_ref & small) const { | ||||||
|     if (!is_forall(e)) { |     if (!is_forall(e)) { | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  | @ -109,71 +150,6 @@ bool demodulator_rewriter::is_demodulator(expr * e, app_ref & large, expr_ref & | ||||||
|     return false; |     return false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| class var_set_proc { |  | ||||||
|     uint_set & m_set; |  | ||||||
| public: |  | ||||||
|     var_set_proc(uint_set &s):m_set(s) {} |  | ||||||
|     void operator()(var * n) { m_set.insert(n->get_idx()); } |  | ||||||
|     void operator()(quantifier * n) {} |  | ||||||
|     void operator()(app * n) {} |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| int demodulator_rewriter::is_subset(expr * e1, expr * e2) const { |  | ||||||
|     uint_set ev1, ev2; |  | ||||||
| 
 |  | ||||||
|     if (m.is_value(e1)) |  | ||||||
|         return 1; // values are always a subset!
 |  | ||||||
| 
 |  | ||||||
|     var_set_proc proc1(ev1); |  | ||||||
|     for_each_expr(proc1, e1); |  | ||||||
|     var_set_proc proc2(ev2); |  | ||||||
|     for_each_expr(proc2, e2); |  | ||||||
| 
 |  | ||||||
|     return (ev1==ev2          ) ? +2 : // We return +2 if the sets are equal.
 |  | ||||||
|            (ev1.subset_of(ev2)) ? +1 : |  | ||||||
|            (ev2.subset_of(ev1)) ? -1 : |  | ||||||
|                                    0 ; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| int demodulator_rewriter::is_smaller(expr * e1, expr * e2) const { |  | ||||||
|     unsigned sz1 = 0, sz2 = 0; |  | ||||||
| 
 |  | ||||||
|     // values are always smaller!
 |  | ||||||
|     if (m.is_value(e1)) |  | ||||||
|         return +1; |  | ||||||
|     else if (m.is_value(e2)) |  | ||||||
|         return -1; |  | ||||||
| 
 |  | ||||||
|     // interpreted stuff is always better than uninterpreted.
 |  | ||||||
|     if (!is_uninterp(e1) && is_uninterp(e2)) |  | ||||||
|         return +1; |  | ||||||
|     else if (is_uninterp(e1) && !is_uninterp(e2)) |  | ||||||
|         return -1; |  | ||||||
| 
 |  | ||||||
|     // two uninterpreted functions are ordered first by the number of
 |  | ||||||
|     // arguments, then by their id.
 |  | ||||||
|     if (is_uninterp(e1) && is_uninterp(e2)) { |  | ||||||
|         if (to_app(e1)->get_num_args() < to_app(e2)->get_num_args()) |  | ||||||
|             return +1; |  | ||||||
|         else if (to_app(e1)->get_num_args() > to_app(e2)->get_num_args()) |  | ||||||
|             return -1; |  | ||||||
|         else { |  | ||||||
|             unsigned a = to_app(e1)->get_decl()->get_id(); |  | ||||||
|             unsigned b = to_app(e2)->get_decl()->get_id(); |  | ||||||
|             if (a < b) |  | ||||||
|                 return +1; |  | ||||||
|             else if (a > b) |  | ||||||
|                 return -1; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     sz1 = get_depth(e1); |  | ||||||
|     sz2 = get_depth(e2); |  | ||||||
| 
 |  | ||||||
|     return (sz1 == sz2) ?  0 : |  | ||||||
|            (sz1  < sz2) ? +1 : |  | ||||||
|                           -1 ; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| class max_var_id_proc { | class max_var_id_proc { | ||||||
|     unsigned    m_max_var_id; |     unsigned    m_max_var_id; | ||||||
| public: | public: | ||||||
|  | @ -187,13 +163,202 @@ public: | ||||||
|     unsigned get_max() { return m_max_var_id; } |     unsigned get_max() { return m_max_var_id; } | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| unsigned demodulator_rewriter::max_var_id(expr_ref_vector const& es) { | unsigned demodulator_util::max_var_id(expr* e) { | ||||||
|  |     max_var_id_proc proc; | ||||||
|  |     for_each_expr(proc, e); | ||||||
|  |     return proc.get_max(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | unsigned demodulator_util::max_var_id(expr_ref_vector const& es) { | ||||||
|     max_var_id_proc proc; |     max_var_id_proc proc; | ||||||
|     for (expr* e : es)  |     for (expr* e : es)  | ||||||
|         for_each_expr(proc, e); |         for_each_expr(proc, e); | ||||||
|     return proc.get_max(); |     return proc.get_max(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | // ------------------
 | ||||||
|  | 
 | ||||||
|  | demodulator_rewriter_util::demodulator_rewriter_util(ast_manager& m): | ||||||
|  |     m(m), | ||||||
|  |     m_th_rewriter(m), | ||||||
|  |     m_rewrite_todo(m), | ||||||
|  |     m_rewrite_cache(m), | ||||||
|  |     m_new_exprs(m), | ||||||
|  |     m_new_args(m) | ||||||
|  | {} | ||||||
|  | 
 | ||||||
|  | expr_ref demodulator_rewriter_util::rewrite(expr * n) { | ||||||
|  | 
 | ||||||
|  |     TRACE("demodulator", tout << "rewrite: " << mk_pp(n, m) << std::endl; ); | ||||||
|  |     app * a; | ||||||
|  | 
 | ||||||
|  |     SASSERT(m_rewrite_todo.empty()); | ||||||
|  |     m_new_exprs.reset(); | ||||||
|  |     m_rewrite_cache.reset(); | ||||||
|  | 
 | ||||||
|  |     m_rewrite_todo.push_back(n); | ||||||
|  |     while (!m_rewrite_todo.empty()) { | ||||||
|  |         TRACE("demodulator_stack", tout << "STACK: " << std::endl; | ||||||
|  |               for (unsigned i = 0; i < m_rewrite_todo.size(); i++) | ||||||
|  |                   tout << std::dec << i << ": " << std::hex << (size_t)m_rewrite_todo[i] << | ||||||
|  |                   " = " << mk_pp(m_rewrite_todo[i], m) << std::endl; | ||||||
|  |               ); | ||||||
|  | 
 | ||||||
|  |         expr * e = m_rewrite_todo.back(); | ||||||
|  |         expr_ref actual(e, m); | ||||||
|  | 
 | ||||||
|  |         if (m_rewrite_cache.contains(e)) { | ||||||
|  |             const expr_bool_pair &ebp = m_rewrite_cache.get(e); | ||||||
|  |             if (ebp.second) { | ||||||
|  |                 m_rewrite_todo.pop_back(); | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 actual = ebp.first; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         switch (actual->get_kind()) { | ||||||
|  |         case AST_VAR: | ||||||
|  |             rewrite_cache(e, actual, true); | ||||||
|  |             m_rewrite_todo.pop_back(); | ||||||
|  |             break; | ||||||
|  |         case AST_APP: | ||||||
|  |             a = to_app(actual); | ||||||
|  |             if (rewrite_visit_children(a)) { | ||||||
|  |                 func_decl * f = a->get_decl(); | ||||||
|  |                 m_new_args.reset(); | ||||||
|  |                 bool all_untouched = true; | ||||||
|  |                 for (expr* o_child : *a) { | ||||||
|  |                     expr * n_child; | ||||||
|  |                     SASSERT(m_rewrite_cache.contains(o_child) && m_rewrite_cache.get(o_child).second); | ||||||
|  |                     expr_bool_pair const & ebp = m_rewrite_cache.get(o_child); | ||||||
|  |                     n_child = ebp.first; | ||||||
|  |                     if (n_child != o_child) | ||||||
|  |                         all_untouched = false; | ||||||
|  |                     m_new_args.push_back(n_child); | ||||||
|  |                 } | ||||||
|  |                 expr_ref np(m); | ||||||
|  |                 if (m_rewrite1(f, m_new_args, np)) { | ||||||
|  |                     rewrite_cache(e, np, false); | ||||||
|  |                     // No pop.
 | ||||||
|  |                 }  | ||||||
|  |                 else { | ||||||
|  |                     if (all_untouched) { | ||||||
|  |                         rewrite_cache(e, actual, true); | ||||||
|  |                     } | ||||||
|  |                     else { | ||||||
|  |                         expr_ref na(m); | ||||||
|  |                         na = m_th_rewriter.mk_app(f, m_new_args); | ||||||
|  |                         TRACE("demodulator_bug", tout << "e:\n" << mk_pp(e, m) << "\nnew_args: \n"; | ||||||
|  |                               tout << m_new_args << "\n"; | ||||||
|  |                               tout << "=====>\n"; | ||||||
|  |                               tout << "na:\n " << na << "\n";); | ||||||
|  |                         rewrite_cache(e, na, true); | ||||||
|  |                     } | ||||||
|  |                     m_rewrite_todo.pop_back(); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |         case AST_QUANTIFIER: { | ||||||
|  |             expr * body = to_quantifier(actual)->get_expr(); | ||||||
|  |             if (m_rewrite_cache.contains(body)) { | ||||||
|  |                 const expr_bool_pair ebp = m_rewrite_cache.get(body); | ||||||
|  |                 SASSERT(ebp.second); | ||||||
|  |                 expr * new_body = ebp.first; | ||||||
|  |                 quantifier_ref q(m); | ||||||
|  |                 q = m.update_quantifier(to_quantifier(actual), new_body); | ||||||
|  |                 m_new_exprs.push_back(q); | ||||||
|  |                 expr_ref new_q = elim_unused_vars(m, q, params_ref()); | ||||||
|  |                 m_new_exprs.push_back(new_q); | ||||||
|  |                 rewrite_cache(e, new_q, true); | ||||||
|  |                 m_rewrite_todo.pop_back(); | ||||||
|  |             } else { | ||||||
|  |                 m_rewrite_todo.push_back(body); | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         default: | ||||||
|  |             UNREACHABLE(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     SASSERT(m_rewrite_cache.contains(n)); | ||||||
|  |     const expr_bool_pair & ebp = m_rewrite_cache.get(n); | ||||||
|  |     SASSERT(ebp.second); | ||||||
|  |     expr * r = ebp.first; | ||||||
|  | 
 | ||||||
|  |     TRACE("demodulator", tout << "rewrite result: " << mk_pp(r, m) << std::endl; ); | ||||||
|  | 
 | ||||||
|  |     return expr_ref(r, m); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool demodulator_rewriter_util::rewrite_visit_children(app * a) { | ||||||
|  |     bool res = true; | ||||||
|  |     for (expr* e : *a) { | ||||||
|  |         if (m_rewrite_cache.contains(e) && m_rewrite_cache.get(e).second) | ||||||
|  |             continue; | ||||||
|  |         bool recursive = false; | ||||||
|  |         expr * v = e; | ||||||
|  |         if (m_rewrite_cache.contains(e)) { | ||||||
|  |             auto const & [t, marked] = m_rewrite_cache.get(e); | ||||||
|  |             if (marked)  | ||||||
|  |                 v = t; | ||||||
|  |         } | ||||||
|  |         for (expr* t : m_rewrite_todo) { | ||||||
|  |             if (t == v) { | ||||||
|  |                 recursive = true; | ||||||
|  |                 TRACE("demodulator", tout << "Detected demodulator cycle: " << | ||||||
|  |                       mk_pp(a, m) << " --> " << mk_pp(v, m) << std::endl;); | ||||||
|  |                 rewrite_cache(e, v, true); | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         if (!recursive) { | ||||||
|  |             m_rewrite_todo.push_back(e); | ||||||
|  |             res = false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return res; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void demodulator_rewriter_util::rewrite_cache(expr * e, expr * new_e, bool done) { | ||||||
|  |     m_rewrite_cache.insert(e, expr_bool_pair(new_e, done)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // ------------------
 | ||||||
|  | 
 | ||||||
|  | demodulator_rewriter::demodulator_rewriter(ast_manager & m): | ||||||
|  |     m(m), | ||||||
|  |     m_match_subst(m), | ||||||
|  |     m_util(m), | ||||||
|  |     m_bsimp(m), | ||||||
|  |     m_todo(m), | ||||||
|  |     m_in_processed(m), | ||||||
|  |     m_new_args(m),  | ||||||
|  |     m_rewrite_todo(m), | ||||||
|  |     m_rewrite_cache(m), | ||||||
|  |     m_new_exprs(m) { | ||||||
|  |     params_ref p; | ||||||
|  |     p.set_bool("elim_and", true); | ||||||
|  |     m_bsimp.updt_params(p); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | demodulator_rewriter::~demodulator_rewriter() { | ||||||
|  |     reset_dealloc_values(m_fwd_idx); | ||||||
|  |     reset_dealloc_values(m_back_idx); | ||||||
|  |     for (auto & kv : m_demodulator2lhs_rhs) { | ||||||
|  |         m.dec_ref(kv.m_key); | ||||||
|  |         m.dec_ref(kv.m_value.first); | ||||||
|  |         m.dec_ref(kv.m_value.second); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| void demodulator_rewriter::insert_fwd_idx(app * large, expr * small, quantifier * demodulator) { | void demodulator_rewriter::insert_fwd_idx(app * large, expr * small, quantifier * demodulator) { | ||||||
|     SASSERT(demodulator); |     SASSERT(demodulator); | ||||||
|     SASSERT(large && small); |     SASSERT(large && small); | ||||||
|  | @ -265,17 +430,18 @@ bool demodulator_rewriter::rewrite1(func_decl * f, expr_ref_vector const & args, | ||||||
| 
 | 
 | ||||||
|     for (quantifier* d : *set) { |     for (quantifier* d : *set) { | ||||||
| 
 | 
 | ||||||
|         auto const& [large, rhs] = m_demodulator2lhs_rhs[d]; |         auto const& [lhs, rhs] = m_demodulator2lhs_rhs[d]; | ||||||
|          |          | ||||||
|         if (large->get_num_args() != args.size()) |         if (lhs->get_num_args() != args.size()) | ||||||
|             continue; |             continue; | ||||||
|          |          | ||||||
|         TRACE("demodulator_bug", tout << "Matching with demodulator: " << mk_pp(d, m) << std::endl; ); |         TRACE("demodulator_bug", tout << "Matching with demodulator: " << mk_pp(d, m) << std::endl; ); | ||||||
|          |          | ||||||
|         SASSERT(large->get_decl() == f); |         SASSERT(lhs->get_decl() == f); | ||||||
|          |          | ||||||
|         if (m_match_subst(large, rhs, args.data(), np)) { |         if (m_match_subst(lhs, rhs, args.data(), np)) { | ||||||
|             TRACE("demodulator_bug", tout << "succeeded...\n" << mk_pp(rhs, m) << "\n===>\n" << mk_pp(np, m) << "\n";); |             TRACE("demodulator_bug", tout << "succeeded...\n" << mk_pp(rhs, m) << "\n===>\n" << mk_pp(np, m) << "\n";); | ||||||
|  |             m_new_exprs.push_back(np); | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -289,15 +455,14 @@ bool demodulator_rewriter::rewrite_visit_children(app * a) { | ||||||
|         if (m_rewrite_cache.contains(e) && m_rewrite_cache.get(e).second) |         if (m_rewrite_cache.contains(e) && m_rewrite_cache.get(e).second) | ||||||
|             continue; |             continue; | ||||||
|         bool recursive = false; |         bool recursive = false; | ||||||
|         unsigned sz = m_rewrite_todo.size(); |  | ||||||
|         expr * v = e; |         expr * v = e; | ||||||
|         if (m_rewrite_cache.contains(e)) { |         if (m_rewrite_cache.contains(e)) { | ||||||
|             expr_bool_pair const & ebp = m_rewrite_cache.get(e); |             auto const & [t, marked] = m_rewrite_cache.get(e); | ||||||
|             if (ebp.second)  |             if (marked)  | ||||||
|                 v = ebp.first; |                 v = t; | ||||||
|         } |         } | ||||||
|         for (unsigned i = sz; i-- > 0;) { |         for (expr* t : m_rewrite_todo) { | ||||||
|             if (m_rewrite_todo[i] == v) { |             if (t == v) { | ||||||
|                 recursive = true; |                 recursive = true; | ||||||
|                 TRACE("demodulator", tout << "Detected demodulator cycle: " << |                 TRACE("demodulator", tout << "Detected demodulator cycle: " << | ||||||
|                       mk_pp(a, m) << " --> " << mk_pp(v, m) << std::endl;); |                       mk_pp(a, m) << " --> " << mk_pp(v, m) << std::endl;); | ||||||
|  | @ -504,7 +669,7 @@ void demodulator_rewriter::reschedule_processed(func_decl * f) { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool demodulator_rewriter::can_rewrite(expr * n, expr * lhs) { | bool demodulator_match_subst::can_rewrite(expr * n, expr * lhs) { | ||||||
|     // this is a quick check, we just traverse d and check if there is an expression in d that is an instance of lhs of n'.
 |     // this is a quick check, we just traverse d and check if there is an expression in d that is an instance of lhs of n'.
 | ||||||
|     // we cannot use the trick used for m_processed, since the main loop would not terminate.
 |     // we cannot use the trick used for m_processed, since the main loop would not terminate.
 | ||||||
| 
 | 
 | ||||||
|  | @ -530,7 +695,7 @@ bool demodulator_rewriter::can_rewrite(expr * n, expr * lhs) { | ||||||
| 
 | 
 | ||||||
|         case AST_APP: |         case AST_APP: | ||||||
|             if (for_each_expr_args(stack, visited, to_app(curr)->get_num_args(), to_app(curr)->get_args())) { |             if (for_each_expr_args(stack, visited, to_app(curr)->get_num_args(), to_app(curr)->get_args())) { | ||||||
|                 if (m_match_subst(lhs, curr)) |                 if ((*this)(lhs, curr)) | ||||||
|                     return true; |                     return true; | ||||||
|                 visited.mark(curr, true); |                 visited.mark(curr, true); | ||||||
|                 stack.pop_back(); |                 stack.pop_back(); | ||||||
|  | @ -582,7 +747,7 @@ void demodulator_rewriter::reschedule_demodulators(func_decl * f, expr * lhs) { | ||||||
|         func_decl_ref df(l->get_decl(), m); |         func_decl_ref df(l->get_decl(), m); | ||||||
|              |              | ||||||
|         // Now we know there is an occurrence of f in d
 |         // Now we know there is an occurrence of f in d
 | ||||||
|         if (!can_rewrite(d, lhs))  |         if (!m_match_subst.can_rewrite(d, lhs))  | ||||||
|             continue; |             continue; | ||||||
| 
 | 
 | ||||||
|         TRACE("demodulator", tout << "Rescheduling: " << std::endl << mk_pp(d, m) << std::endl); |         TRACE("demodulator", tout << "Rescheduling: " << std::endl << mk_pp(d, m) << std::endl); | ||||||
|  | @ -602,7 +767,7 @@ void demodulator_rewriter::operator()(expr_ref_vector const& exprs, | ||||||
|     for (expr* e : exprs)  |     for (expr* e : exprs)  | ||||||
|         m_todo.push_back(e); |         m_todo.push_back(e); | ||||||
| 
 | 
 | ||||||
|     m_match_subst.reserve(max_var_id(exprs)); |     m_match_subst.reserve(m_util.max_var_id(exprs)); | ||||||
| 
 | 
 | ||||||
|     while (!m_todo.empty()) { |     while (!m_todo.empty()) { | ||||||
|         // let n be the next formula in m_todo.
 |         // let n be the next formula in m_todo.
 | ||||||
|  | @ -618,7 +783,7 @@ void demodulator_rewriter::operator()(expr_ref_vector const& exprs, | ||||||
| 
 | 
 | ||||||
|         app_ref large(m); |         app_ref large(m); | ||||||
|         expr_ref small(m); |         expr_ref small(m); | ||||||
|         if (!is_demodulator(np, large, small)) { |         if (!m_util.is_demodulator(np, large, small)) { | ||||||
|             // insert n' into m_processed
 |             // insert n' into m_processed
 | ||||||
|             m_processed.insert(np); |             m_processed.insert(np); | ||||||
|             m_in_processed.push_back(np); |             m_in_processed.push_back(np); | ||||||
|  | @ -661,7 +826,7 @@ void demodulator_rewriter::operator()(expr_ref_vector const& exprs, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| demodulator_rewriter::match_subst::match_subst(ast_manager & m): | demodulator_match_subst::demodulator_match_subst(ast_manager & m): | ||||||
|     m(m), |     m(m), | ||||||
|     m_subst(m) { |     m_subst(m) { | ||||||
| } | } | ||||||
|  | @ -693,7 +858,7 @@ struct match_args_aux_proc { | ||||||
|     void operator()(app * n) {} |     void operator()(app * n) {} | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| bool demodulator_rewriter::match_subst::match_args(app * lhs, expr * const * args) { | bool demodulator_match_subst::match_args(app * lhs, expr * const * args) { | ||||||
|     m_cache.reset(); |     m_cache.reset(); | ||||||
|     m_todo.reset(); |     m_todo.reset(); | ||||||
| 
 | 
 | ||||||
|  | @ -809,7 +974,7 @@ bool demodulator_rewriter::match_subst::match_args(app * lhs, expr * const * arg | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| bool demodulator_rewriter::match_subst::operator()(app * lhs, expr * rhs, expr * const * args, expr_ref & new_rhs) { | bool demodulator_match_subst::operator()(app * lhs, expr * rhs, expr * const * args, expr_ref & new_rhs) { | ||||||
|      |      | ||||||
|     if (match_args(lhs, args)) { |     if (match_args(lhs, args)) { | ||||||
|         if (m_all_args_eq) { |         if (m_all_args_eq) { | ||||||
|  | @ -824,7 +989,7 @@ bool demodulator_rewriter::match_subst::operator()(app * lhs, expr * rhs, expr * | ||||||
|     return false; |     return false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool demodulator_rewriter::match_subst::operator()(expr * t, expr * i) { | bool demodulator_match_subst::operator()(expr * t, expr * i) { | ||||||
|     m_cache.reset(); |     m_cache.reset(); | ||||||
|     m_todo.reset(); |     m_todo.reset(); | ||||||
|     if (is_var(t)) |     if (is_var(t)) | ||||||
|  |  | ||||||
|  | @ -24,6 +24,7 @@ Revision History: | ||||||
| #include "ast/ast.h" | #include "ast/ast.h" | ||||||
| #include "ast/substitution/substitution.h" | #include "ast/substitution/substitution.h" | ||||||
| #include "ast/rewriter/bool_rewriter.h" | #include "ast/rewriter/bool_rewriter.h" | ||||||
|  | #include "ast/rewriter/th_rewriter.h" | ||||||
| #include "util/obj_hashtable.h" | #include "util/obj_hashtable.h" | ||||||
| #include "util/obj_pair_hashtable.h" | #include "util/obj_pair_hashtable.h" | ||||||
| #include "util/array_map.h" | #include "util/array_map.h" | ||||||
|  | @ -92,6 +93,92 @@ The code in spc_rewriter.* does something like that. We cannot reuse this code d | ||||||
| for the superposion engine in Z3, but we can adapt it for our needs in the preprocessor. | for the superposion engine in Z3, but we can adapt it for our needs in the preprocessor. | ||||||
|     |     | ||||||
| */ | */ | ||||||
|  | class demodulator_util { | ||||||
|  |     ast_manager& m; | ||||||
|  |     int is_subset(expr*, expr*) const; | ||||||
|  |     int is_smaller(expr*, expr*) const; | ||||||
|  |  public: | ||||||
|  |     demodulator_util(ast_manager& m):m(m) {} | ||||||
|  |     bool is_demodulator(expr* e, app_ref& large, expr_ref & small) const; | ||||||
|  |     unsigned max_var_id(expr* e); | ||||||
|  |     unsigned max_var_id(expr_ref_vector const& e); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |    \brief Custom matcher & substitution application | ||||||
|  | */ | ||||||
|  | class demodulator_match_subst { | ||||||
|  |     typedef std::pair<expr *, expr *>      expr_pair; | ||||||
|  |     typedef obj_pair_hashtable<expr, expr> cache; | ||||||
|  |      | ||||||
|  |     void reset(); | ||||||
|  |      | ||||||
|  |     ast_manager &         m; | ||||||
|  |     substitution          m_subst; | ||||||
|  |     cache                 m_cache; | ||||||
|  |     svector<expr_pair>    m_todo; | ||||||
|  |     bool                  m_all_args_eq; | ||||||
|  |      | ||||||
|  |     bool match_args(app * t, expr * const * args); | ||||||
|  |      | ||||||
|  | public: | ||||||
|  |     demodulator_match_subst(ast_manager & m); | ||||||
|  | 
 | ||||||
|  |     void reserve(unsigned max_vid) { m_subst.reserve(2, max_vid+1); } | ||||||
|  |     /**
 | ||||||
|  |        \brief Let f be the top symbol of lhs. If (f args) is an | ||||||
|  |        instance of lhs, that is, there is a substitution s | ||||||
|  |        s.t. s[lhs] = (f args), then return true and store s[rhs] | ||||||
|  |        into new_rhs.  Where s[t] represents the application of the | ||||||
|  |        substitution s into t. | ||||||
|  |         | ||||||
|  |        Assumptions, the variables in lhs and (f args) are assumed to be distinct. | ||||||
|  |        So, (f x y) matches (f y x).  | ||||||
|  |        Moreover, the result should be in terms of the variables in (f args). | ||||||
|  |     */ | ||||||
|  |     bool operator()(app * lhs, expr * rhs, expr * const * args, expr_ref & new_rhs); | ||||||
|  |      | ||||||
|  |     /**
 | ||||||
|  |        \brief Return true if \c i is an instance of \c t. | ||||||
|  |     */ | ||||||
|  |     bool operator()(expr * t, expr * i); | ||||||
|  | 
 | ||||||
|  |     bool can_rewrite(expr* n, expr* lhs); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | class demodulator_rewriter_util { | ||||||
|  |     ast_manager& m; | ||||||
|  |     std::function<bool(func_decl*, expr_ref_vector const&, expr_ref&)> m_rewrite1; | ||||||
|  | 
 | ||||||
|  |     typedef std::pair<expr *, bool> expr_bool_pair; | ||||||
|  | 
 | ||||||
|  |     class plugin { | ||||||
|  |         ast_manager& m; | ||||||
|  |     public: | ||||||
|  |         plugin(ast_manager& m): m(m) { }         | ||||||
|  |         void ins_eh(expr* k, expr_bool_pair v) { m.inc_ref(k); m.inc_ref(v.first); } | ||||||
|  |         void del_eh(expr* k, expr_bool_pair v) { m.dec_ref(k); m.dec_ref(v.first); } | ||||||
|  |         static unsigned to_int(expr const * k) { return k->get_id(); } | ||||||
|  |     }; | ||||||
|  |     typedef array_map<expr*, expr_bool_pair, plugin> expr_map; | ||||||
|  | 
 | ||||||
|  |     typedef expr_map rewrite_cache_map; | ||||||
|  | 
 | ||||||
|  |     th_rewriter         m_th_rewriter; | ||||||
|  |     expr_ref_buffer     m_rewrite_todo; | ||||||
|  |     rewrite_cache_map   m_rewrite_cache; | ||||||
|  |     expr_ref_buffer     m_new_exprs; | ||||||
|  |     expr_ref_vector     m_new_args; | ||||||
|  | 
 | ||||||
|  |     bool rewrite_visit_children(app * a); | ||||||
|  |     void rewrite_cache(expr * e, expr * new_e, bool done); | ||||||
|  | 
 | ||||||
|  | public: | ||||||
|  |     demodulator_rewriter_util(ast_manager& m); | ||||||
|  |     void set_rewrite1(std::function<bool(func_decl*, expr_ref_vector const&, expr_ref&)>& fn) { m_rewrite1 = fn; } | ||||||
|  |     expr_ref rewrite(expr * n);     | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| class demodulator_rewriter final { | class demodulator_rewriter final { | ||||||
|     class rewrite_proc; |     class rewrite_proc; | ||||||
|     class add_back_idx_proc; |     class add_back_idx_proc; | ||||||
|  | @ -119,47 +206,10 @@ class demodulator_rewriter final { | ||||||
|     typedef obj_map<quantifier, app_expr_pair> demodulator2lhs_rhs; |     typedef obj_map<quantifier, app_expr_pair> demodulator2lhs_rhs; | ||||||
|     typedef expr_map rewrite_cache_map; |     typedef expr_map rewrite_cache_map; | ||||||
| 
 | 
 | ||||||
|     /**
 |  | ||||||
|        \brief Custom matcher & substitution application |  | ||||||
|     */ |  | ||||||
|     class match_subst { |  | ||||||
|         typedef std::pair<expr *, expr *>      expr_pair; |  | ||||||
|         typedef obj_pair_hashtable<expr, expr> cache; |  | ||||||
| 
 |  | ||||||
|         void reset(); |  | ||||||
| 
 |  | ||||||
|         ast_manager &         m; |  | ||||||
|         substitution          m_subst; |  | ||||||
|         cache                 m_cache; |  | ||||||
|         svector<expr_pair>    m_todo; |  | ||||||
|         bool                  m_all_args_eq; |  | ||||||
|       |  | ||||||
|         bool match_args(app * t, expr * const * args); |  | ||||||
|     |  | ||||||
|     public: |  | ||||||
|         match_subst(ast_manager & m); |  | ||||||
|         void reserve(unsigned max_vid) { m_subst.reserve(2, max_vid+1); } |  | ||||||
|         /**
 |  | ||||||
|            \brief Let f be the top symbol of lhs. If (f args) is an |  | ||||||
|            instance of lhs, that is, there is a substitution s |  | ||||||
|            s.t. s[lhs] = (f args), then return true and store s[rhs] |  | ||||||
|            into new_rhs.  Where s[t] represents the application of the |  | ||||||
|            substitution s into t. |  | ||||||
| 
 |  | ||||||
|            Assumptions, the variables in lhs and (f args) are assumed to be distinct. |  | ||||||
|            So, (f x y) matches (f y x).  |  | ||||||
|            Moreover, the result should be in terms of the variables in (f args). |  | ||||||
|         */ |  | ||||||
|         bool operator()(app * lhs, expr * rhs, expr * const * args, expr_ref & new_rhs); |  | ||||||
|          |  | ||||||
|         /**
 |  | ||||||
|            \brief Return true if \c i is an instance of \c t. |  | ||||||
|         */ |  | ||||||
|         bool operator()(expr * t, expr * i); |  | ||||||
|     }; |  | ||||||
| 
 | 
 | ||||||
|     ast_manager &       m; |     ast_manager &       m; | ||||||
|     match_subst         m_match_subst; |     demodulator_match_subst m_match_subst; | ||||||
|  |     demodulator_util    m_util; | ||||||
|     bool_rewriter       m_bsimp; |     bool_rewriter       m_bsimp; | ||||||
|     fwd_idx_map         m_fwd_idx; |     fwd_idx_map         m_fwd_idx; | ||||||
|     back_idx_map        m_back_idx; |     back_idx_map        m_back_idx; | ||||||
|  | @ -179,7 +229,6 @@ class demodulator_rewriter final { | ||||||
|     void remove_bwd_idx(expr* q); |     void remove_bwd_idx(expr* q); | ||||||
|     bool check_fwd_idx_consistency(); |     bool check_fwd_idx_consistency(); | ||||||
|     void show_fwd_idx(std::ostream & out); |     void show_fwd_idx(std::ostream & out); | ||||||
|     bool is_demodulator(expr * e, app_ref & large, expr_ref & small) const; |  | ||||||
|     bool can_rewrite(expr * n, expr * lhs); |     bool can_rewrite(expr * n, expr * lhs); | ||||||
|      |      | ||||||
|     expr * rewrite(expr * n); |     expr * rewrite(expr * n); | ||||||
|  | @ -188,13 +237,6 @@ class demodulator_rewriter final { | ||||||
|     void rewrite_cache(expr * e, expr * new_e, bool done); |     void rewrite_cache(expr * e, expr * new_e, bool done); | ||||||
|     void reschedule_processed(func_decl * f); |     void reschedule_processed(func_decl * f); | ||||||
|     void reschedule_demodulators(func_decl * f, expr * np); |     void reschedule_demodulators(func_decl * f, expr * np); | ||||||
|     unsigned max_var_id(expr_ref_vector const& es); |  | ||||||
| 
 |  | ||||||
|     // is_smaller returns -1 for e1<e2, 0 for e1==e2 and +1 for e1>e2.
 |  | ||||||
|     int is_smaller(expr * e1, expr * e2) const; |  | ||||||
| 
 |  | ||||||
|     // is_subset returns -1 for e1 subset e2, +1 for e2 subset e1, 0 else.
 |  | ||||||
|     int is_subset(expr * e1, expr * e2) const; |  | ||||||
| 
 | 
 | ||||||
| public: | public: | ||||||
|     demodulator_rewriter(ast_manager & m); |     demodulator_rewriter(ast_manager & m); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue