14#ifndef OR_TOOLS_SAT_PB_CONSTRAINT_H_
15#define OR_TOOLS_SAT_PB_CONSTRAINT_H_
25#include "absl/container/flat_hash_map.h"
26#include "absl/log/check.h"
27#include "absl/strings/string_view.h"
28#include "absl/types/span.h"
33#include "ortools/sat/sat_parameters.pb.h"
48 std::numeric_limits<Coefficient::ValueType>::max());
91 std::vector<LiteralWithCoeff>* cst, Coefficient* bound_shift,
92 Coefficient* max_value);
106 std::vector<LiteralWithCoeff>* cst, Coefficient* bound_shift,
107 Coefficient* max_value);
117 Coefficient bound_shift, Coefficient max_value);
125 Coefficient bound_shift,
126 Coefficient max_value);
134 std::vector<LiteralWithCoeff>* cst, Coefficient* rhs);
164 bool use_upper_bound, Coefficient upper_bound,
165 std::vector<LiteralWithCoeff>* cst);
169 Coefficient
Rhs(
int i)
const {
return rhs_[
i]; }
171 return constraints_[
i];
175 bool AddConstraint(absl::Span<const LiteralWithCoeff> cst,
176 Coefficient max_value, Coefficient rhs);
178 std::vector<Coefficient> rhs_;
179 std::vector<std::vector<LiteralWithCoeff>> constraints_;
197 return AbsCoefficient(terms_[var]);
204 return Literal(var, terms_[var] > 0);
221 const Coefficient bound = max_sum_ - rhs_;
226 terms_[var] = (terms_[var] > 0) ? bound : -bound;
233 int trail_index)
const;
245 const Trail& trail,
int trail_index);
270 Coefficient initial_slack, Coefficient target);
280 Coefficient
Rhs()
const {
return rhs_; }
281 Coefficient
MaxSum()
const {
return max_sum_; }
287 const BooleanVariable var = literal.
Variable();
288 const Coefficient term_encoding = literal.
IsPositive() ? coeff : -coeff;
295 rhs_ -= std::min(coeff, AbsCoefficient(terms_[var]));
296 max_sum_ += AbsCoefficient(term_encoding + terms_[var]) -
297 AbsCoefficient(terms_[var]);
302 CHECK_GE(max_sum_, 0) <<
"Overflow";
303 terms_[var] += term_encoding;
310 const BooleanVariable var = literal.
Variable();
311 if (literal ==
GetLiteral(var))
return Coefficient(0);
312 return std::min(coeff, AbsCoefficient(terms_[var]));
318 return non_zeros_.PositionsSetAtLeastOnce();
325 Coefficient AbsCoefficient(Coefficient a)
const {
return a > 0 ? a : -a; }
328 Coefficient ComputeMaxSum()
const;
340 Coefficient max_sum_;
348class UpperBoundedLinearConstraint;
389 const std::vector<LiteralWithCoeff>& cst);
393 Coefficient
Rhs()
const {
return rhs_; }
401 bool InitializeRhs(Coefficient rhs,
int trail_index, Coefficient* threshold,
417 bool Propagate(
int trail_index, Coefficient* threshold,
Trail* trail,
424 void Untrail(Coefficient* threshold,
int trail_index);
442 BooleanVariable propagated_variable,
443 std::vector<Literal>* reason);
449 Coefficient* conflict_slack);
464 const Trail& trail,
int trail_index,
484 uint64_t
hash()
const {
return hash_; }
491 Coefficient GetSlackFromThreshold(Coefficient threshold) {
492 return (index_ < 0) ? threshold : coeffs_[index_] + threshold;
494 void Update(Coefficient slack, Coefficient* threshold) {
495 *threshold = (index_ < 0) ? slack : slack - coeffs_[index_];
496 already_propagated_end_ = starts_[index_ + 1];
501 bool is_marked_for_deletion_;
503 int first_reason_trail_index_;
508 int already_propagated_end_;
517 std::vector<Coefficient> coeffs_;
518 std::vector<int> starts_;
519 std::vector<Literal> literals_;
531 conflicting_constraint_index_(-1),
532 num_learned_constraint_before_cleanup_(0),
533 constraint_activity_increment_(1.0),
534 parameters_(model->GetOrCreate<SatParameters>()),
535 stats_(
"PbConstraints"),
536 num_constraint_lookups_(0),
537 num_inspected_constraint_literals_(0),
538 num_threshold_updates_(0) {
547 LOG(INFO) << stats_.StatString();
548 LOG(INFO) <<
"num_constraint_lookups_: " << num_constraint_lookups_;
549 LOG(INFO) <<
"num_threshold_updates_: " << num_threshold_updates_;
554 void Untrail(
const Trail& trail,
int trail_index)
final;
555 absl::Span<const Literal>
Reason(
const Trail& trail,
int trail_index,
556 int64_t conflict_id)
const final;
563 if (!constraints_.empty()) {
564 to_update_.resize(num_variables << 1);
565 enqueue_helper_.reasons.resize(num_variables);
579 bool AddConstraint(
const std::vector<LiteralWithCoeff>& cst, Coefficient rhs,
585 Coefficient rhs,
Trail* trail);
589 bool IsEmpty() const final {
return constraints_.empty(); }
598 if (conflicting_constraint_index_ == -1)
return nullptr;
599 return constraints_[conflicting_constraint_index_.value()].get();
614 constraints_[index]->MarkForDeletion();
615 DeleteConstraintMarkedForDeletion();
621 return num_inspected_constraint_literals_;
626 bool PropagateNext(
Trail* trail);
630 void ComputeNewLearnedConstraintLimit();
631 void DeleteSomeLearnedConstraintIfNeeded();
636 void DeleteConstraintMarkedForDeletion();
646 struct ConstraintIndexWithCoeff {
647 ConstraintIndexWithCoeff() =
default;
648 ConstraintIndexWithCoeff(
bool n, ConstraintIndex
i, Coefficient c)
649 : need_untrail_inspection(n), index(
i), coefficient(c) {}
650 bool need_untrail_inspection;
651 ConstraintIndex index;
652 Coefficient coefficient;
656 std::vector<std::unique_ptr<UpperBoundedLinearConstraint>> constraints_;
667 SparseBitset<ConstraintIndex> to_untrail_;
671 absl::flat_hash_map<int64_t, std::vector<UpperBoundedLinearConstraint*>>
672 possible_duplicates_;
675 PbConstraintsEnqueueHelper enqueue_helper_;
679 ConstraintIndex conflicting_constraint_index_;
682 int target_number_of_learned_constraint_;
683 int num_learned_constraint_before_cleanup_;
684 double constraint_activity_increment_;
687 SatParameters* parameters_;
690 mutable StatsGroup stats_;
691 int64_t num_constraint_lookups_;
692 int64_t num_inspected_constraint_literals_;
693 int64_t num_threshold_updates_;
715 first_variable_.resize(num_variables);
716 seen_.ClearAndResize(BooleanVariable(num_variables));
726 if (seen_[var])
return first_variable_[var];
727 const BooleanVariable reference_var =
728 trail_.ReferenceVarWithSameReason(var);
729 if (reference_var == var)
return var;
730 if (seen_[reference_var])
return first_variable_[reference_var];
731 seen_.Set(reference_var);
732 first_variable_[reference_var] = var;
int NumConstraints() const
Getters. All the constraints are guaranteed to be in canonical form.
CanonicalBooleanLinearProblem & operator=(const CanonicalBooleanLinearProblem &)=delete
Coefficient Rhs(int i) const
const std::vector< LiteralWithCoeff > & Constraint(int i) const
CanonicalBooleanLinearProblem()=default
CanonicalBooleanLinearProblem(const CanonicalBooleanLinearProblem &)=delete
This type is neither copyable nor movable.
bool AddLinearConstraint(bool use_lower_bound, Coefficient lower_bound, bool use_upper_bound, Coefficient upper_bound, std::vector< LiteralWithCoeff > *cst)
std::string DebugString() const
LiteralIndex Index() const
BooleanVariable Variable() const
Coefficient MaxSum() const
std::string DebugString()
Returns a string representation of the constraint.
Coefficient ReduceCoefficientsAndComputeSlackForTrailPrefix(const Trail &trail, int trail_index)
Coefficient CancelationAmount(Literal literal, Coefficient coeff) const
Returns the "cancelation" amount of AddTerm(literal, coeff).
void ReduceCoefficients()
Coefficient GetCoefficient(BooleanVariable var) const
Returns the coefficient (>= 0) of the given variable.
Coefficient ComputeSlackForTrailPrefix(const Trail &trail, int trail_index) const
void AddToRhs(Coefficient value)
Adds a non-negative value to this constraint Rhs().
void CopyIntoVector(std::vector< LiteralWithCoeff > *output)
Copies this constraint into a vector<LiteralWithCoeff> representation.
const std::vector< BooleanVariable > & PossibleNonZeros() const
void ReduceSlackTo(const Trail &trail, int trail_index, Coefficient initial_slack, Coefficient target)
void ReduceGivenCoefficient(BooleanVariable var)
void AddTerm(Literal literal, Coefficient coeff)
void ClearAndResize(int num_variables)
Literal GetLiteral(BooleanVariable var) const
void UpdateActivityIncrement()
bool AddConstraint(const std::vector< LiteralWithCoeff > &cst, Coefficient rhs, Trail *trail)
bool Propagate(Trail *trail) final
PbConstraints & operator=(const PbConstraints &)=delete
int64_t num_constraint_lookups() const
Some statistics.
void Resize(int num_variables)
Changes the number of variables.
bool IsEmpty() const final
PbConstraints(const PbConstraints &)=delete
This type is neither copyable nor movable.
void ClearConflictingConstraint()
PbConstraints(Model *model)
~PbConstraints() override
int NumberOfConstraints() const
Returns the number of constraints managed by this class.
UpperBoundedLinearConstraint * ReasonPbConstraint(int trail_index) const
absl::Span< const Literal > Reason(const Trail &trail, int trail_index, int64_t conflict_id) const final
int64_t num_inspected_constraint_literals() const
void BumpActivity(UpperBoundedLinearConstraint *constraint)
bool AddLearnedConstraint(const std::vector< LiteralWithCoeff > &cst, Coefficient rhs, Trail *trail)
int64_t num_threshold_updates() const
void RescaleActivities(double scaling_factor)
void DeleteConstraint(int index)
Only used for testing.
UpperBoundedLinearConstraint * ConflictingConstraint()
void Untrail(const Trail &trail, int trail_index) final
SatPropagator(const std::string &name)
void Enqueue(Literal true_literal, int propagator_id)
void RegisterPropagator(SatPropagator *propagator)
void MarkForDeletion()
API to mark a constraint for deletion before actually deleting it.
void ResolvePBConflict(const Trail &trail, BooleanVariable var, MutableUpperBoundedLinearConstraint *conflict, Coefficient *conflict_slack)
UpperBoundedLinearConstraint(const std::vector< LiteralWithCoeff > &cst)
Takes a pseudo-Boolean formula in canonical form.
Coefficient ComputeCancelation(const Trail &trail, int trail_index, const MutableUpperBoundedLinearConstraint &conflict)
bool Propagate(int trail_index, Coefficient *threshold, Trail *trail, PbConstraintsEnqueueHelper *helper)
void set_activity(double activity)
void AddToConflict(MutableUpperBoundedLinearConstraint *conflict)
void FillReason(const Trail &trail, int source_trail_index, BooleanVariable propagated_variable, std::vector< Literal > *reason)
void Untrail(Coefficient *threshold, int trail_index)
bool is_marked_for_deletion() const
int already_propagated_end() const
bool HasIdenticalTerms(absl::Span< const LiteralWithCoeff > cst)
Returns true if the given terms are the same as the one in this constraint.
bool InitializeRhs(Coefficient rhs, int trail_index, Coefficient *threshold, Trail *trail, PbConstraintsEnqueueHelper *helper)
bool is_used_as_a_reason() const
void set_is_learned(bool is_learned)
BooleanVariable FirstVariableWithSameReason(BooleanVariable var)
VariableWithSameReasonIdentifier(const VariableWithSameReasonIdentifier &)=delete
This type is neither copyable nor movable.
VariableWithSameReasonIdentifier & operator=(const VariableWithSameReasonIdentifier &)=delete
void Clear()
Clears the cache. Call this before each conflict analysis.
void Resize(int num_variables)
VariableWithSameReasonIdentifier(const Trail &trail)
void SimplifyCanonicalBooleanLinearConstraint(std::vector< LiteralWithCoeff > *cst, Coefficient *rhs)
bool ApplyLiteralMapping(const util_intops::StrongVector< LiteralIndex, LiteralIndex > &mapping, std::vector< LiteralWithCoeff > *cst, Coefficient *bound_shift, Coefficient *max_value)
Coefficient ComputeNegatedCanonicalRhs(Coefficient lower_bound, Coefficient bound_shift, Coefficient max_value)
std::ostream & operator<<(std::ostream &os, const BoolVar &var)
bool BooleanLinearExpressionIsCanonical(absl::Span< const LiteralWithCoeff > cst)
Returns true iff the Boolean linear expression is in canonical form.
Coefficient ComputeCanonicalRhs(Coefficient upper_bound, Coefficient bound_shift, Coefficient max_value)
H AbslHashValue(H h, const IntVar &i)
– ABSL HASHING SUPPORT --------------------------------------------------—
const Coefficient kCoefficientMax(std::numeric_limits< Coefficient::ValueType >::max())
bool ComputeBooleanLinearExpressionCanonicalForm(std::vector< LiteralWithCoeff > *cst, Coefficient *bound_shift, Coefficient *max_value)
In SWIG mode, we don't want anything besides these top-level includes.
#define IF_STATS_ENABLED(instructions)
#define DEFINE_STRONG_INT64_TYPE(integer_type_name)
#define DEFINE_STRONG_INDEX_TYPE(index_type_name)
Represents a term in a pseudo-Boolean formula.
LiteralWithCoeff()=default
LiteralWithCoeff(Literal l, Coefficient c)
bool operator==(const LiteralWithCoeff &other) const
LiteralWithCoeff(Literal l, int64_t c)
UpperBoundedLinearConstraint * pb_constraint
void Enqueue(Literal l, int source_trail_index, UpperBoundedLinearConstraint *ct, Trail *trail)
int propagator_id
The propagator id of PbConstraints.
std::vector< Literal > conflict
A temporary vector to store the last conflict.
std::vector< ReasonInfo > reasons