19#ifndef OR_TOOLS_SAT_SAT_SOLVER_H_
20#define OR_TOOLS_SAT_SAT_SOLVER_H_
30#include "absl/container/flat_hash_map.h"
31#include "absl/types/span.h"
90 return BooleanVariable(num_vars);
127 std::vector<LiteralWithCoeff>* cst);
144 owned_propagators_.push_back(std::move(propagator));
161 const std::vector<std::pair<Literal, double>>& prefs) {
163 for (
const std::pair<Literal, double> p : prefs) {
210 const std::vector<Literal>& assumptions);
320 template <
typename Output>
328 if (num_processed_fixed_variables_ < trail_->
Index()) {
339 out->AddClause(clause->AsSpan());
360 const std::vector<Decision>&
Decisions()
const {
return decisions_; }
391 drat_proof_handler_ = drat_proof_handler;
426 current - deterministic_time_at_last_advanced_time_limit_);
427 deterministic_time_at_last_advanced_time_limit_ = current;
434 if (!decisions_.empty())
return decisions_[0].trail_index;
436 return trail_->
Index();
443 bool PropagateAndStopAfterOneConflictResolution();
455 bool ClauseIsValidUnderDebugAssignement(
456 const std::vector<Literal>& clause)
const;
457 bool PBConstraintIsValidUnderDebugAssignment(
458 const std::vector<LiteralWithCoeff>& cst,
const Coefficient rhs)
const;
466 Status SolveInternal(
int assumption_level);
483 Status ReapplyDecisionsUpTo(
int level,
int* first_propagation_index);
486 bool IsMemoryLimitReached()
const;
489 bool SetModelUnsat();
492 int DecisionLevel(BooleanVariable
var)
const {
499 SatClause* ReasonClauseOrNull(BooleanVariable
var)
const;
500 UpperBoundedLinearConstraint* ReasonPbConstraintOrNull(
501 BooleanVariable
var)
const;
512 bool ResolvePBConflict(BooleanVariable
var,
513 MutableUpperBoundedLinearConstraint* conflict,
522 bool ClauseIsUsedAsReason(SatClause* clause)
const {
523 const BooleanVariable
var = clause->PropagatedLiteral().Variable();
526 ReasonClauseOrNull(
var) == clause;
531 bool AddProblemClauseInternal(absl::Span<const Literal> literals);
536 bool AddLinearConstraintInternal(
const std::vector<LiteralWithCoeff>& cst,
545 int AddLearnedClauseAndEnqueueUnitPropagation(
546 const std::vector<Literal>& literals,
bool is_redundant);
550 void EnqueueNewDecision(Literal
literal);
556 bool PropagationIsDone()
const;
559 void InitializePropagators();
563 void Untrail(
int target_trail_index);
566 void ProcessNewlyFixedVariablesForDratProof();
570 int ComputeMaxTrailIndex(absl::Span<const Literal> clause)
const;
584 void ComputeFirstUIPConflict(
585 int max_trail_index, std::vector<Literal>* conflict,
586 std::vector<Literal>* reason_used_to_infer_the_conflict,
587 std::vector<SatClause*>* subsumed_clauses);
592 void ComputeUnionOfReasons(
const std::vector<Literal>&
input,
593 std::vector<Literal>* literals);
600 void FillUnsatAssumptions(Literal false_assumption,
601 std::vector<Literal>* unsat_assumptions);
607 void ComputePBConflict(
int max_trail_index,
Coefficient initial_slack,
608 MutableUpperBoundedLinearConstraint* conflict,
609 int* backjump_level);
619 void MinimizeConflict(
620 std::vector<Literal>* conflict,
621 std::vector<Literal>* reason_used_to_infer_the_conflict);
622 void MinimizeConflictExperimental(std::vector<Literal>* conflict);
623 void MinimizeConflictSimple(std::vector<Literal>* conflict);
624 void MinimizeConflictRecursively(std::vector<Literal>* conflict);
627 bool CanBeInferedFromConflictVariables(BooleanVariable variable);
634 bool IsConflictValid(
const std::vector<Literal>& literals);
638 int ComputeBacktrackLevel(
const std::vector<Literal>& literals);
653 template <
typename LiteralList>
654 int ComputeLbd(
const LiteralList& literals);
658 void CleanClauseDatabaseIfNeeded();
662 void BumpReasonActivities(
const std::vector<Literal>& literals);
663 void BumpClauseActivity(SatClause* clause);
664 void RescaleClauseActivities(
double scaling_factor);
665 void UpdateClauseActivityIncrement();
667 std::string DebugString(
const SatClause& clause)
const;
669 std::string RunningStatisticsString()
const;
673 void KeepAllClauseUsedToInfer(BooleanVariable variable);
679 void TryToMinimizeClause(SatClause* clause);
683 std::unique_ptr<Model> owned_model_;
685 BooleanVariable num_variables_ = BooleanVariable(0);
689 BinaryImplicationGraph* binary_implication_graph_;
690 LiteralWatchers* clauses_propagator_;
691 PbConstraints* pb_constraints_;
694 std::vector<SatPropagator*> propagators_;
697 std::vector<SatPropagator*> external_propagators_;
698 SatPropagator* last_propagator_ =
nullptr;
701 std::vector<std::unique_ptr<SatPropagator>> owned_propagators_;
704 bool track_binary_clauses_;
705 BinaryClauseManager binary_clauses_;
709 TimeLimit* time_limit_;
710 SatParameters* parameters_;
711 RestartPolicy* restart_;
712 SatDecisionPolicy* decision_policy_;
715 VariablesAssignment debug_assignment_;
721 int current_decision_level_ = 0;
722 std::vector<Decision> decisions_;
726 int last_decision_or_backtrack_trail_index_ = 0;
729 int assumption_level_ = 0;
734 int num_processed_fixed_variables_ = 0;
735 double deterministic_time_of_last_fixed_variables_cleanup_ = 0.0;
738 int drat_num_processed_fixed_variables_ = 0;
747 int64_t num_minimizations = 0;
748 int64_t num_literals_removed = 0;
751 int64_t num_learned_pb_literals = 0;
754 int64_t num_literals_learned = 0;
755 int64_t num_literals_forgotten = 0;
756 int64_t num_subsumed_clauses = 0;
759 int64_t minimization_num_clauses = 0;
760 int64_t minimization_num_decisions = 0;
761 int64_t minimization_num_true = 0;
762 int64_t minimization_num_subsumed = 0;
763 int64_t minimization_num_removed_literals = 0;
772 bool model_is_unsat_ =
false;
775 double clause_activity_increment_;
779 int num_learned_clause_before_cleanup_ = 0;
782 SparseBitset<BooleanVariable> is_marked_;
783 SparseBitset<BooleanVariable> is_independent_;
784 SparseBitset<BooleanVariable> tmp_mark_;
785 std::vector<int> min_trail_index_per_level_;
788 std::vector<BooleanVariable> dfs_stack_;
789 std::vector<BooleanVariable> variable_to_process_;
792 std::vector<Literal> literals_scratchpad_;
795 DEFINE_INT_TYPE(SatDecisionLevel,
int);
796 SparseBitset<SatDecisionLevel> is_level_marked_;
799 std::vector<Literal> learned_conflict_;
800 std::vector<Literal> reason_used_to_infer_the_conflict_;
801 std::vector<Literal> extra_reason_literals_;
802 std::vector<SatClause*> subsumed_clauses_;
809 bool block_clause_deletion_ =
false;
813 VariableWithSameReasonIdentifier same_reason_identifier_;
816 std::vector<LiteralWithCoeff> tmp_pb_constraint_;
819 bool is_relevant_for_core_computation_;
822 MutableUpperBoundedLinearConstraint pb_conflict_;
827 double deterministic_time_at_last_advanced_time_limit_ = 0;
830 bool problem_is_pure_sat_;
832 DratProofHandler* drat_proof_handler_;
834 mutable StatsGroup stats_;
845void MinimizeCore(SatSolver* solver, std::vector<Literal>* core);
855 std::vector<LiteralWithCoeff>* cst) {
865 const std::vector<Literal>& literals) {
867 std::vector<LiteralWithCoeff> cst;
868 cst.reserve(literals.size());
869 for (
int i = 0; i < literals.size(); ++i) {
870 cst.emplace_back(literals[i], 1);
879 const std::vector<Literal>& literals) {
881 std::vector<LiteralWithCoeff> cst;
882 cst.reserve(literals.size());
883 for (
const Literal l : literals) {
893 const std::vector<Literal>& literals) {
895 std::vector<LiteralWithCoeff> cst;
896 cst.reserve(literals.size());
897 for (
const Literal l : literals) {
907 absl::Span<const Literal> literals) {
909 std::vector<LiteralWithCoeff> cst;
910 cst.reserve(literals.size());
911 for (
const Literal l : literals) {
937 const std::vector<Literal>& literals,
Literal r) {
939 std::vector<Literal> clause;
940 for (
const Literal l : literals) {
953 absl::Span<const Literal> enforcement_literals,
954 absl::Span<const Literal> clause) {
956 std::vector<Literal> tmp;
957 for (
const Literal l : enforcement_literals) {
958 tmp.push_back(l.Negated());
960 for (
const Literal l : clause) {
971 const std::vector<Literal>& literals,
Literal r) {
973 std::vector<Literal> clause;
974 for (
const Literal l : literals) {
976 clause.push_back(l.Negated());
1024 std::vector<Literal> clause_to_exclude_solution;
1025 clause_to_exclude_solution.reserve(current_level);
1026 for (
int i = 0; i < current_level; ++i) {
1027 clause_to_exclude_solution.push_back(
1028 sat_solver->
Decisions()[i].literal.Negated());
#define CHECK_LT(val1, val2)
#define CHECK_EQ(val1, val2)
A simple class to enforce both an elapsed time limit and a deterministic time limit in the same threa...
void AdvanceDeterministicTime(double deterministic_duration)
Advances the deterministic time.
void ExtractAllBinaryClauses(Output *out) const
void SetDratProofHandler(DratProofHandler *drat_proof_handler)
BooleanVariable Variable() const
void SetDratProofHandler(DratProofHandler *drat_proof_handler)
bool IsRemovable(SatClause *const clause) const
const std::vector< SatClause * > & AllClausesInCreationOrder() const
void DeleteRemovedClauses()
Class that owns everything related to a particular optimization model.
std::vector< std::pair< Literal, double > > AllPreferences() const
void ResetDecisionHeuristic()
void SetAssignmentPreference(Literal literal, double weight)
bool AddLinearConstraint(bool use_lower_bound, Coefficient lower_bound, bool use_upper_bound, Coefficient upper_bound, std::vector< LiteralWithCoeff > *cst)
bool EnqueueDecisionIfNotConflicting(Literal true_literal)
void SetNumVariables(int num_variables)
std::vector< std::pair< Literal, double > > AllPreferences() const
bool AddTernaryClause(Literal a, Literal b, Literal c)
void AddLastPropagator(SatPropagator *propagator)
const SatParameters & parameters() const
void ResetDecisionHeuristic()
bool AddClauseDuringSearch(absl::Span< const Literal > literals)
Status SolveWithTimeLimit(TimeLimit *time_limit)
void ProcessNewlyFixedVariables()
Status ResetAndSolveWithGivenAssumptions(const std::vector< Literal > &assumptions)
void AddPropagator(SatPropagator *propagator)
BooleanVariable NewBooleanVariable()
const VariablesAssignment & Assignment() const
const std::vector< BinaryClause > & NewlyAddedBinaryClauses()
void NotifyThatModelIsUnsat()
const std::vector< Decision > & Decisions() const
int64_t NumFixedVariables() const
bool AddBinaryClauses(const std::vector< BinaryClause > &clauses)
const Trail & LiteralTrail() const
Status UnsatStatus() const
void SetAssumptionLevel(int assumption_level)
bool ProblemIsPureSat() const
void SaveDebugAssignment()
void AdvanceDeterministicTime(TimeLimit *limit)
void SetDratProofHandler(DratProofHandler *drat_proof_handler)
int64_t num_restarts() const
int AssumptionLevel() const
void MinimizeSomeClauses(int decisions_budget)
int64_t num_branches() const
void SetAssignmentPreference(Literal literal, double weight)
void ResetDecisionHeuristicAndSetAllPreferences(const std::vector< std::pair< Literal, double > > &prefs)
int EnqueueDecisionAndBackjumpOnConflict(Literal true_literal)
void SetParameters(const SatParameters ¶meters)
void TrackBinaryClauses(bool value)
bool AddBinaryClause(Literal a, Literal b)
void ExtractClauses(Output *out)
int EnqueueDecisionAndBacktrackOnConflict(Literal true_literal)
int64_t num_propagations() const
void Backtrack(int target_level)
bool RestoreSolverToAssumptionLevel()
int64_t num_failures() const
std::vector< Literal > GetLastIncompatibleDecisions()
void TakePropagatorOwnership(std::unique_ptr< SatPropagator > propagator)
bool ReapplyAssumptionsIfNeeded()
bool IsModelUnsat() const
bool ResetWithGivenAssumptions(const std::vector< Literal > &assumptions)
int CurrentDecisionLevel() const
double deterministic_time() const
bool AddProblemClause(absl::Span< const Literal > literals)
bool AddUnitClause(Literal true_literal)
void ClearNewlyAddedBinaryClauses()
const VariablesAssignment & Assignment() const
const AssignmentInfo & Info(BooleanVariable var) const
bool VariableIsAssigned(BooleanVariable var) const
bool LiteralIsTrue(Literal literal) const
ModelSharedTimeLimit * time_limit
std::tuple< int64_t, int64_t, const double > Coefficient
std::function< void(Model *)> Equality(IntegerVariable v, int64_t value)
std::ostream & operator<<(std::ostream &os, const BoolVar &var)
std::function< void(Model *)> ReifiedBoolOr(const std::vector< Literal > &literals, Literal r)
std::function< void(Model *)> ClauseConstraint(absl::Span< const Literal > literals)
std::function< void(Model *)> EnforcedClause(absl::Span< const Literal > enforcement_literals, absl::Span< const Literal > clause)
std::function< void(Model *)> ReifiedBoolLe(Literal a, Literal b, Literal r)
std::function< void(Model *)> Implication(const std::vector< Literal > &enforcement_literals, IntegerLiteral i)
void MinimizeCore(SatSolver *solver, std::vector< Literal > *core)
std::string SatStatusString(SatSolver::Status status)
std::function< void(Model *)> AtMostOneConstraint(const std::vector< Literal > &literals)
std::function< int64_t(const Model &)> Value(IntegerVariable v)
std::function< void(Model *)> ReifiedBoolAnd(const std::vector< Literal > &literals, Literal r)
std::function< void(Model *)> CardinalityConstraint(int64_t lower_bound, int64_t upper_bound, const std::vector< Literal > &literals)
std::function< void(Model *)> BooleanLinearConstraint(int64_t lower_bound, int64_t upper_bound, std::vector< LiteralWithCoeff > *cst)
std::function< void(Model *)> ExactlyOneConstraint(const std::vector< Literal > &literals)
std::function< void(Model *)> ExcludeCurrentSolutionAndBacktrack()
const int kUnsatTrailIndex
Collection of objects used to extend the Constraint Solver library.
static int input(yyscan_t yyscanner)
Decision(int i, Literal l)