Skip to content

Commit 00adfb5

Browse files
committed
[CP-SAT] more bugfixes
1 parent 12cb30c commit 00adfb5

17 files changed

+149
-89
lines changed

ortools/sat/all_different.cc

+4-1
Original file line numberDiff line numberDiff line change
@@ -275,7 +275,7 @@ bool AllDifferentConstraint::Propagate() {
275275
successor_.AppendToLastVector(value);
276276

277277
// Seed with previous matching.
278-
if (prev_matching_[x] == value) {
278+
if (prev_matching_[x] == value && value_to_variable_[value] == -1) {
279279
variable_to_value_[x] = prev_matching_[x];
280280
value_to_variable_[prev_matching_[x]] = x;
281281
}
@@ -373,6 +373,7 @@ bool AllDifferentConstraint::Propagate() {
373373
if (assignment.LiteralIsFalse(x_lit)) continue;
374374

375375
const int value_node = value + num_variables_;
376+
DCHECK_LT(value_node, component_number_.size());
376377
if (variable_to_value_[x] != value &&
377378
component_number_[x] != component_number_[value_node]) {
378379
// We can deduce that x != value. To explain, force x == value,
@@ -383,6 +384,8 @@ bool AllDifferentConstraint::Propagate() {
383384
variable_visited_.assign(num_variables_, false);
384385
// Undo x -> old_value and old_variable -> value.
385386
const int old_variable = value_to_variable_[value];
387+
DCHECK_GE(old_variable, 0);
388+
DCHECK_LT(old_variable, num_variables_);
386389
variable_to_value_[old_variable] = -1;
387390
const int old_value = variable_to_value_[x];
388391
value_to_variable_[old_value] = -1;

ortools/sat/cp_constraints.cc

+7-5
Original file line numberDiff line numberDiff line change
@@ -135,20 +135,22 @@ bool GreaterThanAtLeastOneOfPropagator::Propagate() {
135135

136136
int first_non_false = 0;
137137
const int size = exprs_.size();
138+
Literal* const selectors = selectors_.data();
139+
AffineExpression* const exprs = exprs_.data();
138140
for (int i = 0; i < size; ++i) {
139-
if (assignment.LiteralIsTrue(selectors_[i])) return true;
141+
if (assignment.LiteralIsTrue(selectors[i])) return true;
140142

141143
// The permutation is needed to have proper lazy reason.
142-
if (assignment.LiteralIsFalse(selectors_[i])) {
144+
if (assignment.LiteralIsFalse(selectors[i])) {
143145
if (i != first_non_false) {
144-
std::swap(selectors_[i], selectors_[first_non_false]);
145-
std::swap(exprs_[i], exprs_[first_non_false]);
146+
std::swap(selectors[i], selectors[first_non_false]);
147+
std::swap(exprs[i], exprs[first_non_false]);
146148
}
147149
++first_non_false;
148150
continue;
149151
}
150152

151-
const IntegerValue min = integer_trail_->LowerBound(exprs_[i]);
153+
const IntegerValue min = integer_trail_->LowerBound(exprs[i]);
152154
if (min < target_min) {
153155
target_min = min;
154156

ortools/sat/cp_model_loader.cc

+4-4
Original file line numberDiff line numberDiff line change
@@ -1286,14 +1286,14 @@ void LoadLinearConstraint(const ConstraintProto& ct, Model* m) {
12861286
rhs_min = std::max(rhs_min, min_sum.value());
12871287
rhs_max = std::min(rhs_max, max_sum.value());
12881288

1289-
auto* detector = m->GetOrCreate<GreaterThanAtLeastOneOfDetector>();
1289+
auto* repository = m->GetOrCreate<BinaryRelationRepository>();
12901290
const Literal lit = mapping->Literal(ct.enforcement_literal(0));
12911291
const Domain domain = ReadDomainFromProto(ct.linear());
12921292
if (vars.size() == 1) {
1293-
detector->Add(lit, {vars[0], coeffs[0]}, {}, rhs_min, rhs_max);
1293+
repository->Add(lit, {vars[0], coeffs[0]}, {}, rhs_min, rhs_max);
12941294
} else if (vars.size() == 2) {
1295-
detector->Add(lit, {vars[0], coeffs[0]}, {vars[1], coeffs[1]}, rhs_min,
1296-
rhs_max);
1295+
repository->Add(lit, {vars[0], coeffs[0]}, {vars[1], coeffs[1]}, rhs_min,
1296+
rhs_max);
12971297
}
12981298
}
12991299

ortools/sat/cp_model_solver_helpers.cc

+1
Original file line numberDiff line numberDiff line change
@@ -1253,6 +1253,7 @@ void LoadCpModel(const CpModelProto& model_proto, Model* model) {
12531253
// Note that we do that before we finish loading the problem (objective and
12541254
// LP relaxation), because propagation will be faster at this point and it
12551255
// should be enough for the purpose of this auto-detection.
1256+
model->GetOrCreate<BinaryRelationRepository>()->Build();
12561257
if (parameters.auto_detect_greater_than_at_least_one_of()) {
12571258
model->GetOrCreate<GreaterThanAtLeastOneOfDetector>()
12581259
->AddGreaterThanAtLeastOneOfConstraints(model);

ortools/sat/cuts.cc

+1-2
Original file line numberDiff line numberDiff line change
@@ -1666,8 +1666,7 @@ BoolRLTCutHelper::~BoolRLTCutHelper() {
16661666
shared_stats_->AddStats(stats);
16671667
}
16681668

1669-
void BoolRLTCutHelper::Initialize(
1670-
const absl::flat_hash_map<IntegerVariable, glop::ColIndex>& lp_vars) {
1669+
void BoolRLTCutHelper::Initialize(absl::Span<const IntegerVariable> lp_vars) {
16711670
product_detector_->InitializeBooleanRLTCuts(lp_vars, *lp_values_);
16721671
enabled_ = !product_detector_->BoolRLTCandidates().empty();
16731672
}

ortools/sat/cuts.h

+1-2
Original file line numberDiff line numberDiff line change
@@ -582,8 +582,7 @@ class BoolRLTCutHelper {
582582

583583
// Precompute data according to the current lp relaxation.
584584
// This also restrict any Boolean to be currently appearing in the LP.
585-
void Initialize(
586-
const absl::flat_hash_map<IntegerVariable, glop::ColIndex>& lp_vars);
585+
void Initialize(absl::Span<const IntegerVariable> lp_vars);
587586

588587
// Tries RLT separation of the input constraint. Returns true on success.
589588
bool TrySimpleSeparation(const CutData& input_ct);

ortools/sat/implied_bounds.cc

+13-4
Original file line numberDiff line numberDiff line change
@@ -797,7 +797,7 @@ void ProductDetector::UpdateRLTMaps(
797797

798798
// TODO(user): limit work if too many ternary.
799799
void ProductDetector::InitializeBooleanRLTCuts(
800-
const absl::flat_hash_map<IntegerVariable, glop::ColIndex>& lp_vars,
800+
absl::Span<const IntegerVariable> lp_vars,
801801
const util_intops::StrongVector<IntegerVariable, double>& lp_values) {
802802
// TODO(user): Maybe we shouldn't reconstruct this every time, but it is hard
803803
// in case of multiple lps to make sure we don't use variables not in the lp
@@ -808,14 +808,19 @@ void ProductDetector::InitializeBooleanRLTCuts(
808808
// We will list all interesting multiplicative candidate for each variable.
809809
bool_rlt_candidates_.clear();
810810
const int size = ternary_clauses_with_view_.size();
811+
if (size == 0) return;
812+
813+
is_in_lp_vars_.resize(integer_trail_->NumIntegerVariables().value());
814+
for (const IntegerVariable var : lp_vars) is_in_lp_vars_.Set(var);
815+
811816
for (int i = 0; i < size; i += 3) {
812817
const IntegerVariable var1 = ternary_clauses_with_view_[i];
813818
const IntegerVariable var2 = ternary_clauses_with_view_[i + 1];
814819
const IntegerVariable var3 = ternary_clauses_with_view_[i + 2];
815820

816-
if (!lp_vars.contains(PositiveVariable(var1))) continue;
817-
if (!lp_vars.contains(PositiveVariable(var2))) continue;
818-
if (!lp_vars.contains(PositiveVariable(var3))) continue;
821+
if (!is_in_lp_vars_[PositiveVariable(var1)]) continue;
822+
if (!is_in_lp_vars_[PositiveVariable(var2)]) continue;
823+
if (!is_in_lp_vars_[PositiveVariable(var3)]) continue;
819824

820825
// If we have l1 + l2 + l3 >= 1, then for all (i, j) pair we have
821826
// !li * !lj <= lk. We are looking for violation like this.
@@ -830,6 +835,10 @@ void ProductDetector::InitializeBooleanRLTCuts(
830835
UpdateRLTMaps(lp_values, NegationOf(var2), 1.0 - lp2, NegationOf(var3),
831836
1.0 - lp3, var1, lp1);
832837
}
838+
839+
// Clear.
840+
// TODO(user): Just switch to memclear() when dense.
841+
for (const IntegerVariable var : lp_vars) is_in_lp_vars_.ClearBucket(var);
833842
}
834843

835844
} // namespace sat

ortools/sat/implied_bounds.h

+3-1
Original file line numberDiff line numberDiff line change
@@ -298,7 +298,7 @@ class ProductDetector {
298298
// Experimental. Find violated inequality of the form l1 * l2 <= l3.
299299
// And set-up data structure to query this efficiently.
300300
void InitializeBooleanRLTCuts(
301-
const absl::flat_hash_map<IntegerVariable, glop::ColIndex>& lp_vars,
301+
absl::Span<const IntegerVariable> lp_vars,
302302
const util_intops::StrongVector<IntegerVariable, double>& lp_values);
303303

304304
// BoolRLTCandidates()[var] contains the list of factor for which we have
@@ -385,6 +385,8 @@ class ProductDetector {
385385
// as NegatedVariable(). This is a flat vector of size multiple of 3.
386386
std::vector<IntegerVariable> ternary_clauses_with_view_;
387387

388+
Bitset64<IntegerVariable> is_in_lp_vars_;
389+
388390
// Stats.
389391
int64_t num_products_ = 0;
390392
int64_t num_int_products_ = 0;

ortools/sat/implied_bounds_test.cc

+1-2
Original file line numberDiff line numberDiff line change
@@ -665,8 +665,7 @@ TEST(ProductDetectorTest, RLT) {
665665
lp_values[x] = 0.7;
666666
lp_values[y] = 0.9;
667667
lp_values[z] = 0.2;
668-
const absl::flat_hash_map<IntegerVariable, glop::ColIndex> lp_vars = {
669-
{x, glop::ColIndex(0)}, {y, glop::ColIndex(1)}, {z, glop::ColIndex(2)}};
668+
std::vector<IntegerVariable> lp_vars = {x, y, z};
670669
detector->InitializeBooleanRLTCuts(lp_vars, lp_values);
671670

672671
// (1 - X) * Y <= Z, 0.3 * 0.9 == 0.27 <= 0.2, interesting!

ortools/sat/linear_constraint_manager.h

+7
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,13 @@ struct ModelReducedCosts
5656
ModelReducedCosts() = default;
5757
};
5858

59+
// Stores the mapping integer_variable -> glop::ColIndex.
60+
// This is shared across all LP, which is fine since there are disjoint.
61+
struct ModelLpVariableMapping
62+
: public util_intops::StrongVector<IntegerVariable, glop::ColIndex> {
63+
ModelLpVariableMapping() = default;
64+
};
65+
5966
// Knowing the symmetry of the IP problem should allow us to
6067
// solve the LP faster via "folding" techniques.
6168
//

ortools/sat/linear_programming_constraint.cc

+11-8
Original file line numberDiff line numberDiff line change
@@ -296,6 +296,7 @@ LinearProgrammingConstraint::LinearProgrammingConstraint(
296296
implied_bounds_processor_({}, integer_trail_,
297297
model->GetOrCreate<ImpliedBounds>()),
298298
dispatcher_(model->GetOrCreate<LinearProgrammingDispatcher>()),
299+
mirror_lp_variable_(*model->GetOrCreate<ModelLpVariableMapping>()),
299300
expanded_lp_solution_(*model->GetOrCreate<ModelLpValues>()),
300301
expanded_reduced_costs_(*model->GetOrCreate<ModelReducedCosts>()) {
301302
// Tweak the default parameters to make the solve incremental.
@@ -314,6 +315,8 @@ LinearProgrammingConstraint::LinearProgrammingConstraint(
314315

315316
// Register our local rev int repository.
316317
integer_trail_->RegisterReversibleClass(&rc_rev_int_repository_);
318+
mirror_lp_variable_.resize(integer_trail_->NumIntegerVariables(),
319+
glop::kInvalidCol);
317320

318321
// Register SharedStatistics with the cut helpers.
319322
auto* stats = model->GetOrCreate<SharedStatistics>();
@@ -336,6 +339,7 @@ LinearProgrammingConstraint::LinearProgrammingConstraint(
336339

337340
integer_variables_.push_back(positive_variable);
338341
extended_integer_variables_.push_back(positive_variable);
342+
DCHECK_EQ(mirror_lp_variable_[positive_variable], glop::kInvalidCol);
339343
mirror_lp_variable_[positive_variable] = col;
340344
++col;
341345
}
@@ -347,6 +351,7 @@ LinearProgrammingConstraint::LinearProgrammingConstraint(
347351
const int orbit_index = symmetrizer_->OrbitIndex(var);
348352
for (const IntegerVariable var : symmetrizer_->Orbit(orbit_index)) {
349353
extended_integer_variables_.push_back(var);
354+
DCHECK_EQ(mirror_lp_variable_[var], glop::kInvalidCol);
350355
mirror_lp_variable_[var] = col;
351356
++col;
352357
}
@@ -458,7 +463,7 @@ void LinearProgrammingConstraint::RegisterWith(Model* model) {
458463
glop::ColIndex LinearProgrammingConstraint::GetMirrorVariable(
459464
IntegerVariable positive_variable) {
460465
DCHECK(VariableIsPositive(positive_variable));
461-
return mirror_lp_variable_.at(positive_variable);
466+
return mirror_lp_variable_[positive_variable];
462467
}
463468

464469
void LinearProgrammingConstraint::SetObjectiveCoefficient(IntegerVariable ivar,
@@ -898,7 +903,7 @@ glop::Fractional LinearProgrammingConstraint::GetVariableValueAtCpScale(
898903

899904
double LinearProgrammingConstraint::GetSolutionValue(
900905
IntegerVariable variable) const {
901-
return lp_solution_[mirror_lp_variable_.at(variable).value()];
906+
return lp_solution_[mirror_lp_variable_[variable].value()];
902907
}
903908

904909
void LinearProgrammingConstraint::UpdateBoundsOfLpVariables() {
@@ -1539,8 +1544,7 @@ bool LinearProgrammingConstraint::PostprocessAndAddCut(
15391544

15401545
// Simple copy for non-slack variables.
15411546
if (var < first_slack) {
1542-
const glop::ColIndex col =
1543-
mirror_lp_variable_.at(PositiveVariable(var));
1547+
const glop::ColIndex col = mirror_lp_variable_[PositiveVariable(var)];
15441548
if (VariableIsPositive(var)) {
15451549
tmp_scattered_vector_.Add(col, coeff);
15461550
} else {
@@ -2166,7 +2170,7 @@ bool LinearProgrammingConstraint::Propagate() {
21662170
implied_bounds_processor_.RecomputeCacheAndSeparateSomeImpliedBoundCuts(
21672171
expanded_lp_solution_);
21682172
if (parameters_.add_rlt_cuts()) {
2169-
rlt_cut_helper_.Initialize(mirror_lp_variable_);
2173+
rlt_cut_helper_.Initialize(extended_integer_variables_);
21702174
}
21712175

21722176
// The "generic" cuts are currently part of this class as they are using
@@ -2610,12 +2614,11 @@ bool LinearProgrammingConstraint::PropagateExactLpReason() {
26102614
// For the corner case of an objective of size 1, we do not want or need
26112615
// to take it into account.
26122616
bool take_objective_into_account = true;
2613-
if (mirror_lp_variable_.contains(objective_cp_)) {
2617+
if (objective_cp_is_part_of_lp_) {
26142618
// The objective is part of the lp.
26152619
// This should only happen for objective with a single term.
26162620
CHECK_EQ(integer_objective_.size(), 1);
2617-
CHECK_EQ(integer_objective_[0].first,
2618-
mirror_lp_variable_.at(objective_cp_));
2621+
CHECK_EQ(integer_objective_[0].first, mirror_lp_variable_[objective_cp_]);
26192622
CHECK_EQ(integer_objective_[0].second, IntegerValue(1));
26202623

26212624
take_objective_into_account = false;

ortools/sat/linear_programming_constraint.h

+13-6
Original file line numberDiff line numberDiff line change
@@ -154,7 +154,16 @@ class LinearProgrammingConstraint : public PropagatorInterface,
154154

155155
// The main objective variable should be equal to the linear sum of
156156
// the arguments passed to SetObjectiveCoefficient().
157-
void SetMainObjectiveVariable(IntegerVariable ivar) { objective_cp_ = ivar; }
157+
void SetMainObjectiveVariable(IntegerVariable ivar) {
158+
objective_cp_ = ivar;
159+
objective_cp_is_part_of_lp_ = false;
160+
for (const IntegerVariable var : integer_variables_) {
161+
if (var == objective_cp_) {
162+
objective_cp_is_part_of_lp_ = true;
163+
break;
164+
}
165+
}
166+
}
158167
IntegerVariable ObjectiveVariable() const { return objective_cp_; }
159168

160169
// Register a new cut generator with this constraint.
@@ -500,14 +509,10 @@ class LinearProgrammingConstraint : public PropagatorInterface,
500509

501510
// Structures used for mirroring IntegerVariables inside the underlying LP
502511
// solver: an integer variable var is mirrored by mirror_lp_variable_[var].
503-
// Note that these indices are dense in [0, mirror_lp_variable_.size()] so
512+
// Note that these indices are dense in [0, integer_variables.size()] so
504513
// they can be used as vector indices.
505-
//
506-
// TODO(user): This should be util_intops::StrongVector<glop::ColIndex,
507-
// IntegerVariable>.
508514
std::vector<IntegerVariable> integer_variables_;
509515
std::vector<IntegerVariable> extended_integer_variables_;
510-
absl::flat_hash_map<IntegerVariable, glop::ColIndex> mirror_lp_variable_;
511516

512517
// This is only used if we use symmetry folding.
513518
// Refer to relevant orbit in the LinearConstraintSymmetrizer.
@@ -516,6 +521,7 @@ class LinearProgrammingConstraint : public PropagatorInterface,
516521
// We need to remember what to optimize if an objective is given, because
517522
// then we will switch the objective between feasibility and optimization.
518523
bool objective_is_defined_ = false;
524+
bool objective_cp_is_part_of_lp_ = false;
519525
IntegerVariable objective_cp_;
520526

521527
// Singletons from Model.
@@ -587,6 +593,7 @@ class LinearProgrammingConstraint : public PropagatorInterface,
587593
bool lp_at_level_zero_is_final_ = false;
588594

589595
// Same as lp_solution_ but this vector is indexed by IntegerVariable.
596+
ModelLpVariableMapping& mirror_lp_variable_;
590597
ModelLpValues& expanded_lp_solution_;
591598
ModelReducedCosts& expanded_reduced_costs_;
592599

0 commit comments

Comments
 (0)