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/strings/string_view.h"
32#include "absl/types/span.h"
45#include "ortools/sat/sat_parameters.pb.h"
93 return BooleanVariable(num_vars);
129 std::vector<LiteralWithCoeff>* cst);
146 owned_propagators_.push_back(std::move(propagator));
163 const std::vector<std::pair<Literal, double>>& prefs) {
165 for (
const std::pair<Literal, double>& p : prefs) {
212 const std::vector<Literal>& assumptions);
335 template <
typename Output>
343 if (num_processed_fixed_variables_ < trail_->
Index()) {
354 out->AddClause(clause->AsSpan());
375 const std::vector<Decision>&
Decisions()
const {
return decisions_; }
406 drat_proof_handler_ = drat_proof_handler;
438 shared_binary_clauses_callback) {
439 shared_binary_clauses_callback_ = shared_binary_clauses_callback;
447 current - deterministic_time_at_last_advanced_time_limit_);
448 deterministic_time_at_last_advanced_time_limit_ = current;
455 if (!decisions_.empty())
return decisions_[0].trail_index;
457 return trail_->
Index();
467 bool PropagateAndStopAfterOneConflictResolution();
482 bool ClauseIsValidUnderDebugAssignment(
483 const std::vector<Literal>& clause)
const;
484 bool PBConstraintIsValidUnderDebugAssignment(
485 const std::vector<LiteralWithCoeff>& cst,
const Coefficient rhs)
const;
493 Status SolveInternal(
int assumption_level);
510 Status ReapplyDecisionsUpTo(
int level,
int* first_propagation_index);
513 bool IsMemoryLimitReached()
const;
516 bool SetModelUnsat();
519 int DecisionLevel(BooleanVariable
var)
const {
526 SatClause* ReasonClauseOrNull(BooleanVariable
var)
const;
527 UpperBoundedLinearConstraint* ReasonPbConstraintOrNull(
528 BooleanVariable
var)
const;
539 bool ResolvePBConflict(BooleanVariable
var,
540 MutableUpperBoundedLinearConstraint* conflict,
549 bool ClauseIsUsedAsReason(SatClause* clause)
const {
550 const BooleanVariable
var = clause->PropagatedLiteral().Variable();
553 ReasonClauseOrNull(
var) == clause;
558 bool AddProblemClauseInternal(absl::Span<const Literal> literals);
563 bool AddLinearConstraintInternal(
const std::vector<LiteralWithCoeff>& cst,
567 void CanonicalizeLinear(std::vector<LiteralWithCoeff>* cst,
576 int AddLearnedClauseAndEnqueueUnitPropagation(
577 const std::vector<Literal>& literals,
bool is_redundant);
581 void EnqueueNewDecision(Literal
literal);
587 bool PropagationIsDone()
const;
590 void InitializePropagators();
594 void Untrail(
int target_trail_index);
597 void ProcessNewlyFixedVariablesForDratProof();
601 int ComputeMaxTrailIndex(absl::Span<const Literal> clause)
const;
615 void ComputeFirstUIPConflict(
616 int max_trail_index, std::vector<Literal>* conflict,
617 std::vector<Literal>* reason_used_to_infer_the_conflict,
618 std::vector<SatClause*>* subsumed_clauses);
623 void ComputeUnionOfReasons(
const std::vector<Literal>&
input,
624 std::vector<Literal>* literals);
630 void ComputePBConflict(
int max_trail_index,
Coefficient initial_slack,
631 MutableUpperBoundedLinearConstraint* conflict,
632 int* backjump_level);
642 void MinimizeConflict(
643 std::vector<Literal>* conflict,
644 std::vector<Literal>* reason_used_to_infer_the_conflict);
645 void MinimizeConflictExperimental(std::vector<Literal>* conflict);
646 void MinimizeConflictSimple(std::vector<Literal>* conflict);
647 void MinimizeConflictRecursively(std::vector<Literal>* conflict);
650 bool CanBeInferedFromConflictVariables(BooleanVariable variable);
657 bool IsConflictValid(
const std::vector<Literal>& literals);
661 int ComputeBacktrackLevel(
const std::vector<Literal>& literals);
676 template <
typename LiteralList>
677 int ComputeLbd(
const LiteralList& literals);
681 void CleanClauseDatabaseIfNeeded();
685 void BumpReasonActivities(
const std::vector<Literal>& literals);
686 void BumpClauseActivity(SatClause* clause);
687 void RescaleClauseActivities(
double scaling_factor);
688 void UpdateClauseActivityIncrement();
690 std::string DebugString(
const SatClause& clause)
const;
692 std::string RunningStatisticsString()
const;
696 void KeepAllClauseUsedToInfer(BooleanVariable variable);
702 void TryToMinimizeClause(SatClause* clause);
706 std::unique_ptr<Model> owned_model_;
708 BooleanVariable num_variables_ = BooleanVariable(0);
712 BinaryImplicationGraph* binary_implication_graph_;
713 LiteralWatchers* clauses_propagator_;
714 PbConstraints* pb_constraints_;
717 std::vector<SatPropagator*> propagators_;
718 std::vector<SatPropagator*> non_empty_propagators_;
721 std::vector<SatPropagator*> external_propagators_;
722 SatPropagator* last_propagator_ =
nullptr;
725 std::vector<std::unique_ptr<SatPropagator>> owned_propagators_;
728 bool track_binary_clauses_;
729 BinaryClauseManager binary_clauses_;
733 TimeLimit* time_limit_;
734 SatParameters* parameters_;
735 RestartPolicy* restart_;
736 SatDecisionPolicy* decision_policy_;
737 SolverLogger* logger_;
740 VariablesAssignment debug_assignment_;
746 int current_decision_level_ = 0;
747 std::vector<Decision> decisions_;
751 int last_decision_or_backtrack_trail_index_ = 0;
754 int assumption_level_ = 0;
755 std::vector<Literal> assumptions_;
760 int num_processed_fixed_variables_ = 0;
761 double deterministic_time_of_last_fixed_variables_cleanup_ = 0.0;
764 int drat_num_processed_fixed_variables_ = 0;
773 int64_t num_minimizations = 0;
774 int64_t num_literals_removed = 0;
777 int64_t num_learned_pb_literals = 0;
780 int64_t num_literals_learned = 0;
781 int64_t num_literals_forgotten = 0;
782 int64_t num_subsumed_clauses = 0;
785 int64_t minimization_num_clauses = 0;
786 int64_t minimization_num_decisions = 0;
787 int64_t minimization_num_true = 0;
788 int64_t minimization_num_subsumed = 0;
789 int64_t minimization_num_removed_literals = 0;
798 bool model_is_unsat_ =
false;
801 double clause_activity_increment_;
805 int num_learned_clause_before_cleanup_ = 0;
808 SparseBitset<BooleanVariable> is_marked_;
809 SparseBitset<BooleanVariable> is_independent_;
810 SparseBitset<BooleanVariable> tmp_mark_;
811 std::vector<int> min_trail_index_per_level_;
814 std::vector<BooleanVariable> dfs_stack_;
815 std::vector<BooleanVariable> variable_to_process_;
818 std::vector<Literal> literals_scratchpad_;
821 DEFINE_STRONG_INDEX_TYPE(SatDecisionLevel);
822 SparseBitset<SatDecisionLevel> is_level_marked_;
825 std::vector<Literal> learned_conflict_;
826 std::vector<Literal> reason_used_to_infer_the_conflict_;
827 std::vector<Literal> extra_reason_literals_;
828 std::vector<SatClause*> subsumed_clauses_;
835 bool block_clause_deletion_ =
false;
839 VariableWithSameReasonIdentifier same_reason_identifier_;
842 bool is_relevant_for_core_computation_;
845 MutableUpperBoundedLinearConstraint pb_conflict_;
850 double deterministic_time_at_last_advanced_time_limit_ = 0;
853 bool problem_is_pure_sat_;
855 DratProofHandler* drat_proof_handler_;
857 mutable StatsGroup stats_;
859 std::function<void(Literal, Literal)> shared_binary_clauses_callback_ =
874void MinimizeCore(SatSolver* solver, std::vector<Literal>* core);
884 std::vector<LiteralWithCoeff>* cst) {
894 const std::vector<Literal>& literals) {
896 std::vector<LiteralWithCoeff> cst;
897 cst.reserve(literals.size());
898 for (
int i = 0; i < literals.size(); ++i) {
899 cst.emplace_back(literals[i], 1);
908 const std::vector<Literal>& literals) {
910 std::vector<LiteralWithCoeff> cst;
911 cst.reserve(literals.size());
912 for (
const Literal l : literals) {
922 const std::vector<Literal>& literals) {
924 std::vector<LiteralWithCoeff> cst;
925 cst.reserve(literals.size());
926 for (
const Literal l : literals) {
936 absl::Span<const Literal> literals) {
938 std::vector<LiteralWithCoeff> cst;
939 cst.reserve(literals.size());
940 for (
const Literal l : literals) {
966 const std::vector<Literal>& literals,
Literal r) {
968 std::vector<Literal> clause;
969 for (
const Literal l : literals) {
982 absl::Span<const Literal> enforcement_literals,
983 absl::Span<const Literal> clause) {
985 std::vector<Literal> tmp;
986 for (
const Literal l : enforcement_literals) {
987 tmp.push_back(l.Negated());
989 for (
const Literal l : clause) {
1000 const std::vector<Literal>& literals,
Literal r) {
1002 std::vector<Literal> clause;
1003 for (
const Literal l : literals) {
1005 clause.push_back(l.Negated());
1009 clause.push_back(r);
1053 std::vector<Literal> clause_to_exclude_solution;
1054 clause_to_exclude_solution.reserve(current_level);
1055 for (
int i = 0; i < current_level; ++i) {
1056 clause_to_exclude_solution.push_back(
1057 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 SetShareBinaryClauseCallback(const std::function< void(Literal, Literal)> &shared_binary_clauses_callback)
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
SharedClausesManager * clauses
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)