/****************************************** Copyright (c) 2016, Mate Soos 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 _GATEFINDER_H_ #define _GATEFINDER_H_ #include "solvertypes.h" #include "cset.h" #include #include "watcharray.h" #include namespace CMSat { class Solver; class OccSimplifier; using std::set; class OrGate { public: OrGate(const Lit& _rhs, Lit _lit1, Lit _lit2, const bool _red) : lit1(_lit1) , lit2(_lit2) , rhs(_rhs) , red(_red) { if (lit1 > lit2) std::swap(lit1, lit2); } bool operator==(const OrGate& other) const { return rhs == other.rhs && lit1 == other.lit1 && lit2 == other.lit2 ; } std::array getLits() const { return std::array{{lit1, lit2}}; } //LHS Lit lit1; Lit lit2; //RHS Lit rhs; //Data about gate bool red; }; struct GateCompareForEq { bool operator()(const OrGate& a, const OrGate& b) const { if (a.lit1 != b.lit1) { return (a.lit1 < b.lit1); } if (a.lit2 != b.lit2) { return (a.lit2 < b.lit2); } return (a.rhs < b.rhs); } }; inline std::ostream& operator<<(std::ostream& os, const OrGate& gate) { os << " gate " << " lits: " << gate.lit1 << ", " << gate.lit2 << " rhs: " << gate.rhs << " learnt " << gate.red ; return os; } class GateFinder { public: GateFinder(OccSimplifier *simplifier, Solver *control); bool doAll(); //Stats struct Stats { void clear() { Stats tmp; *this = tmp; } double total_time() const { return findGateTime + orBasedTime + varReplaceTime + andBasedTime + erTime; } Stats& operator+=(const Stats& other); void print(const size_t nVars) const; //Time double findGateTime = 0.0; uint32_t find_gate_timeout = 0; double orBasedTime = 0.0; uint32_t or_based_timeout = 0; double varReplaceTime = 0.0; double andBasedTime = 0.0; uint32_t and_based_timeout = 0; double erTime = 0.0; //OR-gate uint64_t orGateUseful = 0; uint64_t numLongCls = 0; uint64_t numLongClsLits = 0; int64_t litsRem = 0; //Var-replace uint64_t varReplaced = 0; //And-gate uint64_t andGateUseful = 0; uint64_t clauseSizeRem = 0; //ER uint64_t numERVars = 0; //Gates uint64_t learntGatesSize = 0; uint64_t numRed = 0; uint64_t irredGatesSize = 0; uint64_t numIrred = 0; }; const Stats& get_stats() const; private: void print_graphviz_dot(); //Setup void link_in_gate(const OrGate& gate); void add_gate_if_not_already_inside(Lit rhs, Lit lit1, Lit lit2); void find_or_gates_in_sweep_mode(Lit lit); //High-level functions bool remove_clauses_with_all_or_gates(); bool shorten_with_all_or_gates(); //Finding void find_or_gates_and_update_stats(); void find_or_gates(); void findOrGate( const Lit rhs , const Lit lit1 , const Lit lit2 ); bool all_simplifications_with_gates(); vector subs; //to reduce overhead of allocation bool shortenWithOrGate(const OrGate& gate); size_t findEqOrGates(); //And gate treatment bool remove_clauses_using_and_gate( const OrGate& gate , const bool reallyRemove , const bool only_irred ); cl_abst_type calc_sorted_occ_and_set_seen2( const OrGate& gate , uint32_t& maxSize , uint32_t& minSize , const bool only_irred ); void set_seen2_and_abstraction( const Clause& cl , cl_abst_type& abstraction ); bool check_seen_and_gate_against_cl( const Clause& this_cl , const OrGate& gate ); void treatAndGateClause( const ClOffset other_cl_offset , const OrGate& gate , const ClOffset this_cl_offset ); cl_abst_type calc_abst_and_set_seen( const Clause& cl , const OrGate& gate ); ClOffset find_pair_for_and_gate_reduction( const Watched& ws , const size_t minSize , const size_t maxSize , const cl_abst_type abstraction , const OrGate& gate , const bool only_irred ); ClOffset findAndGateOtherCl( const vector& this_sizeSortedOcc , const Lit lit , const cl_abst_type abst2 , const bool gate_is_red , const bool only_irred ); bool findAndGateOtherCl_tri( watch_subarray_const ws_list , const bool gate_is_red , const bool only_irred , Watched& ret ); bool find_pair_for_and_gate_reduction_tri( const Watched& ws , const OrGate& gate , const bool only_irred , Watched& found_pair ); bool remove_clauses_using_and_gate_tri( const OrGate& gate , const bool really_remove , const bool only_irred ); void set_seen2_tri( const OrGate& gate , const bool only_irred ); bool check_seen_and_gate_against_lit( const Lit lit , const OrGate& gate ); ///temporary for and-gate treatment. Cleared at every treatAndGate() call vector > sizeSortedOcc; //Indexes, gate data vector orGates; //List of OR gates //For temporaries vector seen2Set; //Bits that have been set in seen2, and later need to be cleared set clToUnlink; struct TriToUnlink { TriToUnlink(Lit _lit2, Lit _lit3, bool _red) : lit2(_lit2) , lit3(_lit3) , red(_red) {} const Lit lit2; const Lit lit3; const bool red; bool operator<(const TriToUnlink& other) const { if (lit2 != other.lit2) return lit2 < other.lit2; if (lit3 != other.lit3) return lit3 < other.lit3; return red < other.red; } }; set tri_to_unlink; //Graph void print_graphviz_dot2(); ///& seen; vector& seen2; vector& toClear; }; inline const GateFinder::Stats& GateFinder::get_stats() const { return globalStats; } } //end namespace #endif //_GATEFINDER_H_