mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-26 09:24:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			222 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			222 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /************************************************************************************[SimpSolver.h]
 | |
| Copyright (c) 2006,      Niklas Een, Niklas Sorensson
 | |
| Copyright (c) 2007-2010, Niklas Sorensson
 | |
| 
 | |
| Permission is hereby granted, free of charge, to any person obtaining a copy of this software and
 | |
| associated documentation files (the "Software"), to deal in the Software without restriction,
 | |
| including without limitation the rights to use, copy, modify, merge, publish, distribute,
 | |
| sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is
 | |
| furnished to do so, subject to the following conditions:
 | |
| 
 | |
| The above copyright notice and this permission notice shall be included in all copies or
 | |
| substantial portions of the Software.
 | |
| 
 | |
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT
 | |
| NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 | |
| NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
 | |
| DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT
 | |
| OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 | |
| **************************************************************************************************/
 | |
| 
 | |
| #ifndef Minisat_SimpSolver_h
 | |
| #define Minisat_SimpSolver_h
 | |
| 
 | |
| #include "Queue.h"
 | |
| #include "Solver.h"
 | |
| 
 | |
| 
 | |
| namespace Minisat {
 | |
| 
 | |
| //=================================================================================================
 | |
| 
 | |
| 
 | |
| class SimpSolver : public Solver {
 | |
|  public:
 | |
|     // Constructor/Destructor:
 | |
|     //
 | |
|     SimpSolver();
 | |
|     ~SimpSolver();
 | |
| 
 | |
|     // Problem specification:
 | |
|     //
 | |
|     Var     newVar    (lbool upol = l_Undef, bool dvar = true);
 | |
|     void    releaseVar(Lit l);
 | |
|     bool    addClause (const vec<Lit>& ps);
 | |
|     bool    addEmptyClause();                // Add the empty clause to the solver.
 | |
|     bool    addClause (Lit p);               // Add a unit clause to the solver.
 | |
|     bool    addClause (Lit p, Lit q);        // Add a binary clause to the solver.
 | |
|     bool    addClause (Lit p, Lit q, Lit r); // Add a ternary clause to the solver.
 | |
|     bool    addClause (Lit p, Lit q, Lit r, Lit s); // Add a quaternary clause to the solver. 
 | |
|     bool    addClause_(      vec<Lit>& ps);
 | |
|     bool    substitute(Var v, Lit x);  // Replace all occurences of v with x (may cause a contradiction).
 | |
| 
 | |
|     // Variable mode:
 | |
|     // 
 | |
|     void    setFrozen (Var v, bool b); // If a variable is frozen it will not be eliminated.
 | |
|     bool    isEliminated(Var v) const;
 | |
| 
 | |
|     // Alternative freeze interface (may replace 'setFrozen()'):
 | |
|     void    freezeVar (Var v);         // Freeze one variable so it will not be eliminated.
 | |
|     void    thaw      ();              // Thaw all frozen variables.
 | |
| 
 | |
| 
 | |
|     // Solving:
 | |
|     //
 | |
|     bool    solve       (const vec<Lit>& assumps, bool do_simp = true, bool turn_off_simp = false);
 | |
|     lbool   solveLimited(const vec<Lit>& assumps, bool do_simp = true, bool turn_off_simp = false);
 | |
|     bool    solve       (                     bool do_simp = true, bool turn_off_simp = false);
 | |
|     bool    solve       (Lit p       ,        bool do_simp = true, bool turn_off_simp = false);       
 | |
|     bool    solve       (Lit p, Lit q,        bool do_simp = true, bool turn_off_simp = false);
 | |
|     bool    solve       (Lit p, Lit q, Lit r, bool do_simp = true, bool turn_off_simp = false);
 | |
|     bool    eliminate   (bool turn_off_elim = false);  // Perform variable elimination based simplification. 
 | |
| 
 | |
|     // Memory managment:
 | |
|     //
 | |
|     virtual void garbageCollect();
 | |
| 
 | |
| 
 | |
|     // Generate a (possibly simplified) DIMACS file:
 | |
|     //
 | |
| #if 0
 | |
|     void    toDimacs  (const char* file, const vec<Lit>& assumps);
 | |
|     void    toDimacs  (const char* file);
 | |
|     void    toDimacs  (const char* file, Lit p);
 | |
|     void    toDimacs  (const char* file, Lit p, Lit q);
 | |
|     void    toDimacs  (const char* file, Lit p, Lit q, Lit r);
 | |
| #endif
 | |
| 
 | |
|     // Mode of operation:
 | |
|     //
 | |
|     int     grow;              // Allow a variable elimination step to grow by a number of clauses (default to zero).
 | |
|     int     clause_lim;        // Variables are not eliminated if it produces a resolvent with a length above this limit.
 | |
|                                // -1 means no limit.
 | |
|     int     subsumption_lim;   // Do not check if subsumption against a clause larger than this. -1 means no limit.
 | |
|     double  simp_garbage_frac; // A different limit for when to issue a GC during simplification (Also see 'garbage_frac').
 | |
| 
 | |
|     bool    use_asymm;         // Shrink clauses by asymmetric branching.
 | |
|     bool    use_rcheck;        // Check if a clause is already implied. Prett costly, and subsumes subsumptions :)
 | |
|     bool    use_elim;          // Perform variable elimination.
 | |
|     bool    extend_model;      // Flag to indicate whether the user needs to look at the full model.
 | |
| 
 | |
|     // Statistics:
 | |
|     //
 | |
|     int     merges;
 | |
|     int     asymm_lits;
 | |
|     int     eliminated_vars;
 | |
| 
 | |
|  protected:
 | |
| 
 | |
|     // Helper structures:
 | |
|     //
 | |
|     struct ElimLt {
 | |
|         const LMap<int>& n_occ;
 | |
|         explicit ElimLt(const LMap<int>& no) : n_occ(no) {}
 | |
| 
 | |
|         // TODO: are 64-bit operations here noticably bad on 32-bit platforms? Could use a saturating
 | |
|         // 32-bit implementation instead then, but this will have to do for now.
 | |
|         uint64_t cost  (Var x)        const { return (uint64_t)n_occ[mkLit(x)] * (uint64_t)n_occ[~mkLit(x)]; }
 | |
|         bool operator()(Var x, Var y) const { return cost(x) < cost(y); }
 | |
|         
 | |
|         // TODO: investigate this order alternative more.
 | |
|         // bool operator()(Var x, Var y) const { 
 | |
|         //     int c_x = cost(x);
 | |
|         //     int c_y = cost(y);
 | |
|         //     return c_x < c_y || c_x == c_y && x < y; }
 | |
|     };
 | |
| 
 | |
|     struct ClauseDeleted {
 | |
|         const ClauseAllocator& ca;
 | |
|         explicit ClauseDeleted(const ClauseAllocator& _ca) : ca(_ca) {}
 | |
|         bool operator()(const CRef& cr) const { return ca[cr].mark() == 1; } };
 | |
| 
 | |
|     // Solver state:
 | |
|     //
 | |
|     int                 elimorder;
 | |
|     bool                use_simplification;
 | |
|     Var                 max_simp_var;        // Max variable at the point simplification was turned off.
 | |
|     vec<uint32_t>       elimclauses;
 | |
|     VMap<char>          touched;
 | |
|     OccLists<Var, vec<CRef>, ClauseDeleted>
 | |
|                         occurs;
 | |
|     LMap<int>           n_occ;
 | |
|     Heap<Var,ElimLt>    elim_heap;
 | |
|     Queue<CRef>         subsumption_queue;
 | |
|     VMap<char>          frozen;
 | |
|     vec<Var>            frozen_vars;
 | |
|     VMap<char>          eliminated;
 | |
|     int                 bwdsub_assigns;
 | |
|     int                 n_touched;
 | |
| 
 | |
|     // Temporaries:
 | |
|     //
 | |
|     CRef                bwdsub_tmpunit;
 | |
| 
 | |
|     // Main internal methods:
 | |
|     //
 | |
|     lbool         solve_                   (bool do_simp = true, bool turn_off_simp = false);
 | |
|     bool          asymm                    (Var v, CRef cr);
 | |
|     bool          asymmVar                 (Var v);
 | |
|     void          updateElimHeap           (Var v);
 | |
|     void          gatherTouchedClauses     ();
 | |
|     bool          merge                    (const Clause& _ps, const Clause& _qs, Var v, vec<Lit>& out_clause);
 | |
|     bool          merge                    (const Clause& _ps, const Clause& _qs, Var v, int& size);
 | |
|     bool          backwardSubsumptionCheck (bool verbose = false);
 | |
|     bool          eliminateVar             (Var v);
 | |
|     void          extendModel              ();
 | |
| 
 | |
|     void          removeClause             (CRef cr);
 | |
|     bool          strengthenClause         (CRef cr, Lit l);
 | |
|     bool          implied                  (const vec<Lit>& c);
 | |
|     void          relocAll                 (ClauseAllocator& to);
 | |
| };
 | |
| 
 | |
| 
 | |
| //=================================================================================================
 | |
| // Implementation of inline methods:
 | |
| 
 | |
| 
 | |
| inline bool SimpSolver::isEliminated (Var v) const { return eliminated[v]; }
 | |
| inline void SimpSolver::updateElimHeap(Var v) {
 | |
|     assert(use_simplification);
 | |
|     // if (!frozen[v] && !isEliminated(v) && value(v) == l_Undef)
 | |
|     if (elim_heap.inHeap(v) || (!frozen[v] && !isEliminated(v) && value(v) == l_Undef))
 | |
|         elim_heap.update(v); }
 | |
| 
 | |
| 
 | |
| inline bool SimpSolver::addClause    (const vec<Lit>& ps)    { ps.copyTo(add_tmp); return addClause_(add_tmp); }
 | |
| inline bool SimpSolver::addEmptyClause()                     { add_tmp.clear(); return addClause_(add_tmp); }
 | |
| inline bool SimpSolver::addClause    (Lit p)                 { add_tmp.clear(); add_tmp.push(p); return addClause_(add_tmp); }
 | |
| inline bool SimpSolver::addClause    (Lit p, Lit q)          { add_tmp.clear(); add_tmp.push(p); add_tmp.push(q); return addClause_(add_tmp); }
 | |
| inline bool SimpSolver::addClause    (Lit p, Lit q, Lit r)   { add_tmp.clear(); add_tmp.push(p); add_tmp.push(q); add_tmp.push(r); return addClause_(add_tmp); }
 | |
| inline bool SimpSolver::addClause    (Lit p, Lit q, Lit r, Lit s){ add_tmp.clear(); add_tmp.push(p); add_tmp.push(q); add_tmp.push(r); add_tmp.push(s); return addClause_(add_tmp); }
 | |
| inline void SimpSolver::setFrozen    (Var v, bool b) { frozen[v] = (char)b; if (use_simplification && !b) { updateElimHeap(v); } }
 | |
| 
 | |
| inline void SimpSolver::freezeVar(Var v){
 | |
|     if (!frozen[v]){
 | |
|         frozen[v] = 1;
 | |
|         frozen_vars.push(v); 
 | |
|     } }
 | |
| 
 | |
| inline void SimpSolver::thaw(){
 | |
|     for (int i = 0; i < frozen_vars.size(); i++){
 | |
|         Var v = frozen_vars[i];
 | |
|         frozen[v] = 0;
 | |
|         if (use_simplification)
 | |
|             updateElimHeap(v);
 | |
|     }
 | |
|     frozen_vars.clear(); }
 | |
| 
 | |
| inline bool SimpSolver::solve        (                     bool do_simp, bool turn_off_simp)  { budgetOff(); assumptions.clear(); return solve_(do_simp, turn_off_simp) == l_True; }
 | |
| inline bool SimpSolver::solve        (Lit p       ,        bool do_simp, bool turn_off_simp)  { budgetOff(); assumptions.clear(); assumptions.push(p); return solve_(do_simp, turn_off_simp) == l_True; }
 | |
| inline bool SimpSolver::solve        (Lit p, Lit q,        bool do_simp, bool turn_off_simp)  { budgetOff(); assumptions.clear(); assumptions.push(p); assumptions.push(q); return solve_(do_simp, turn_off_simp) == l_True; }
 | |
| inline bool SimpSolver::solve        (Lit p, Lit q, Lit r, bool do_simp, bool turn_off_simp)  { budgetOff(); assumptions.clear(); assumptions.push(p); assumptions.push(q); assumptions.push(r); return solve_(do_simp, turn_off_simp) == l_True; }
 | |
| inline bool SimpSolver::solve        (const vec<Lit>& assumps, bool do_simp, bool turn_off_simp){ 
 | |
|     budgetOff(); assumps.copyTo(assumptions); return solve_(do_simp, turn_off_simp) == l_True; }
 | |
| 
 | |
| inline lbool SimpSolver::solveLimited (const vec<Lit>& assumps, bool do_simp, bool turn_off_simp){ 
 | |
|     assumps.copyTo(assumptions); return solve_(do_simp, turn_off_simp); }
 | |
| 
 | |
| //=================================================================================================
 | |
| }
 | |
| 
 | |
| #endif
 |