26#include "absl/container/flat_hash_map.h"
27#include "absl/memory/memory.h"
28#include "absl/status/status.h"
29#include "absl/status/statusor.h"
30#include "absl/strings/str_cat.h"
31#include "absl/strings/str_join.h"
32#include "absl/strings/str_split.h"
33#include "absl/strings/string_view.h"
34#include "absl/time/clock.h"
35#include "absl/time/time.h"
36#include "absl/types/span.h"
46#include "ortools/glop/parameters.pb.h"
49#include "ortools/math_opt/callback.pb.h"
54#include "ortools/math_opt/model.pb.h"
55#include "ortools/math_opt/model_parameters.pb.h"
56#include "ortools/math_opt/model_update.pb.h"
57#include "ortools/math_opt/parameters.pb.h"
58#include "ortools/math_opt/result.pb.h"
59#include "ortools/math_opt/solution.pb.h"
60#include "ortools/math_opt/sparse_containers.pb.h"
70constexpr double kInf = std::numeric_limits<double>::infinity();
72absl::string_view SafeName(
const VariablesProto& variables,
int index) {
73 if (variables.names().empty()) {
76 return variables.names(
index);
79absl::string_view SafeName(
const LinearConstraintsProto& linear_constraints,
81 if (linear_constraints.names().empty()) {
84 return linear_constraints.names(
index);
87absl::StatusOr<TerminationProto> BuildTermination(
89 const SolveInterrupter*
const interrupter) {
109 interrupter->IsInterrupted()
111 : LIMIT_UNDETERMINED);
119 interrupter->IsInterrupted()
121 : LIMIT_UNDETERMINED);
126 return absl::InternalError(
127 absl::StrCat(
"Unexpected GLOP termination reason: ",
130 LOG(
FATAL) <<
"Unimplemented GLOP termination reason: "
136GlopSolver::GlopSolver() : linear_program_(), lp_solver_() {}
138void GlopSolver::AddVariables(
const VariablesProto& variables) {
140 const glop::ColIndex col_index = linear_program_.CreateNewVariable();
141 linear_program_.SetVariableBounds(col_index, variables.lower_bounds(i),
142 variables.upper_bounds(i));
143 linear_program_.SetVariableName(col_index, SafeName(variables, i));
151template <
typename IndexType>
154 IndexType num_indices,
155 absl::flat_hash_map<int64_t, IndexType>& id_index_map) {
158 IndexType new_index(0);
160 if (indices_to_delete[
index]) {
162 new_indices[
index] = IndexType(-1);
164 new_indices[
index] = new_index;
168 for (
auto it = id_index_map.begin(); it != id_index_map.end();) {
169 IndexType
index = it->second;
170 if (indices_to_delete[
index]) {
172 id_index_map.erase(it++);
174 it->second = new_indices[
index];
180void GlopSolver::DeleteVariables(absl::Span<const int64_t> ids_to_delete) {
181 const glop::ColIndex num_cols = linear_program_.num_variables();
184 for (
const int64_t deleted_variable_id : ids_to_delete) {
185 columns_to_delete[variables_.at(deleted_variable_id)] =
true;
187 linear_program_.DeleteColumns(columns_to_delete);
188 UpdateIdIndexMap<glop::ColIndex>(columns_to_delete, num_cols, variables_);
191void GlopSolver::DeleteLinearConstraints(
192 absl::Span<const int64_t> ids_to_delete) {
193 const glop::RowIndex num_rows = linear_program_.num_constraints();
195 for (
const int64_t deleted_constraint_id : ids_to_delete) {
196 rows_to_delete[linear_constraints_.at(deleted_constraint_id)] =
true;
198 linear_program_.DeleteRows(rows_to_delete);
199 UpdateIdIndexMap<glop::RowIndex>(rows_to_delete, num_rows,
200 linear_constraints_);
203void GlopSolver::AddLinearConstraints(
204 const LinearConstraintsProto& linear_constraints) {
206 const glop::RowIndex row_index = linear_program_.CreateNewConstraint();
207 linear_program_.SetConstraintBounds(row_index,
208 linear_constraints.lower_bounds(i),
209 linear_constraints.upper_bounds(i));
210 linear_program_.SetConstraintName(row_index,
211 SafeName(linear_constraints, i));
217void GlopSolver::SetOrUpdateObjectiveCoefficients(
218 const SparseDoubleVectorProto& linear_objective_coefficients) {
219 for (
int i = 0; i < linear_objective_coefficients.ids_size(); ++i) {
220 const glop::ColIndex col_index =
221 variables_.at(linear_objective_coefficients.ids(i));
222 linear_program_.SetObjectiveCoefficient(
223 col_index, linear_objective_coefficients.values(i));
227void GlopSolver::SetOrUpdateConstraintMatrix(
228 const SparseDoubleMatrixProto& linear_constraint_matrix) {
230 const glop::ColIndex col_index =
231 variables_.at(linear_constraint_matrix.column_ids(j));
232 const glop::RowIndex row_index =
233 linear_constraints_.at(linear_constraint_matrix.row_ids(j));
234 const double coefficient = linear_constraint_matrix.coefficients(j);
235 linear_program_.SetCoefficient(row_index, col_index,
coefficient);
239void GlopSolver::UpdateVariableBounds(
240 const VariableUpdatesProto& variable_updates) {
241 for (
const auto [
id, lb] :
MakeView(variable_updates.lower_bounds())) {
242 const auto col_index = variables_.at(
id);
243 linear_program_.SetVariableBounds(
244 col_index, lb, linear_program_.variable_upper_bounds()[col_index]);
246 for (
const auto [
id, ub] :
MakeView(variable_updates.upper_bounds())) {
247 const auto col_index = variables_.at(
id);
248 linear_program_.SetVariableBounds(
249 col_index, linear_program_.variable_lower_bounds()[col_index], ub);
253void GlopSolver::UpdateLinearConstraintBounds(
254 const LinearConstraintUpdatesProto& linear_constraint_updates) {
255 for (
const auto [
id, lb] :
256 MakeView(linear_constraint_updates.lower_bounds())) {
257 const auto row_index = linear_constraints_.at(
id);
258 linear_program_.SetConstraintBounds(
259 row_index, lb, linear_program_.constraint_upper_bounds()[row_index]);
261 for (
const auto [
id, ub] :
262 MakeView(linear_constraint_updates.upper_bounds())) {
263 const auto row_index = linear_constraints_.at(
id);
264 linear_program_.SetConstraintBounds(
265 row_index, linear_program_.constraint_lower_bounds()[row_index], ub);
269absl::StatusOr<glop::GlopParameters> GlopSolver::MergeSolveParameters(
270 const SolveParametersProto& solver_parameters,
271 const bool setting_initial_basis,
const bool has_message_callback) {
272 glop::GlopParameters result = solver_parameters.glop();
273 std::vector<std::string> warnings;
274 if (!result.has_max_time_in_seconds() && solver_parameters.has_time_limit()) {
277 result.set_max_time_in_seconds(absl::ToDoubleSeconds(
time_limit));
279 if (has_message_callback) {
283 result.set_log_search_progress(
true);
289 result.set_log_to_stdout(
false);
290 }
else if (!result.has_log_search_progress()) {
291 result.set_log_search_progress(solver_parameters.enable_output());
293 if (!result.has_num_omp_threads() && solver_parameters.has_threads()) {
294 result.set_num_omp_threads(solver_parameters.threads());
296 if (!result.has_random_seed() && solver_parameters.has_random_seed()) {
297 const int random_seed =
std::max(0, solver_parameters.random_seed());
298 result.set_random_seed(random_seed);
300 if (!result.has_max_number_of_iterations() &&
301 solver_parameters.iteration_limit()) {
302 result.set_max_number_of_iterations(solver_parameters.iteration_limit());
304 if (solver_parameters.has_node_limit()) {
305 warnings.emplace_back(
"GLOP does snot support 'node_limit' parameter");
307 if (!result.has_use_dual_simplex() &&
308 solver_parameters.lp_algorithm() != LP_ALGORITHM_UNSPECIFIED) {
309 switch (solver_parameters.lp_algorithm()) {
310 case LP_ALGORITHM_PRIMAL_SIMPLEX:
311 result.set_use_dual_simplex(
false);
313 case LP_ALGORITHM_DUAL_SIMPLEX:
314 result.set_use_dual_simplex(
true);
316 case LP_ALGORITHM_BARRIER:
317 warnings.emplace_back(
318 "GLOP does not support 'LP_ALGORITHM_BARRIER' value for "
319 "'lp_algorithm' parameter.");
324 <<
" unknown, error setting GLOP parameters";
327 if (!result.has_use_scaling() && !result.has_scaling_method() &&
328 solver_parameters.scaling() != EMPHASIS_UNSPECIFIED) {
329 switch (solver_parameters.scaling()) {
331 result.set_use_scaling(
false);
334 case EMPHASIS_MEDIUM:
335 result.set_use_scaling(
true);
336 result.set_scaling_method(glop::GlopParameters::EQUILIBRATION);
339 case EMPHASIS_VERY_HIGH:
340 result.set_use_scaling(
true);
341 result.set_scaling_method(glop::GlopParameters::LINEAR_PROGRAM);
346 <<
" unknown, error setting GLOP parameters";
349 if (setting_initial_basis) {
350 result.set_use_preprocessing(
false);
351 }
else if (!result.has_use_preprocessing() &&
352 solver_parameters.presolve() != EMPHASIS_UNSPECIFIED) {
353 switch (solver_parameters.presolve()) {
355 result.set_use_preprocessing(
false);
358 case EMPHASIS_MEDIUM:
360 case EMPHASIS_VERY_HIGH:
361 result.set_use_preprocessing(
true);
366 <<
" unknown, error setting GLOP parameters";
369 if (solver_parameters.cuts() != EMPHASIS_UNSPECIFIED) {
370 warnings.push_back(absl::StrCat(
371 "GLOP does not support 'cuts' parameters, but cuts was set to: ",
374 if (solver_parameters.heuristics() != EMPHASIS_UNSPECIFIED) {
376 absl::StrCat(
"GLOP does not support 'heuristics' parameter, but "
377 "heuristics was set to: ",
380 if (solver_parameters.has_cutoff_limit()) {
381 warnings.push_back(
"GLOP does not support 'cutoff_limit' parameter");
383 if (solver_parameters.has_objective_limit()) {
384 warnings.push_back(
"GLOP does not support 'objective_limit' parameter");
386 if (solver_parameters.has_best_bound_limit()) {
387 warnings.push_back(
"GLOP does not support 'best_bound_limit' parameter");
389 if (solver_parameters.has_solution_limit()) {
390 warnings.push_back(
"GLOP does not support 'solution_limit' parameter");
392 if (!warnings.empty()) {
393 return absl::InvalidArgumentError(absl::StrJoin(warnings,
"; "));
398bool GlopSolver::CanUpdate(
const ModelUpdateProto& model_update) {
399 return model_update.objective_updates()
400 .quadratic_coefficients()
405template <
typename IndexType>
407 const std::vector<int64_t>& ids_in_order,
408 const absl::flat_hash_map<int64_t, IndexType>& id_map,
410 const SparseVectorFilterProto& filter) {
412 SparseDoubleVectorProto result;
413 for (
const int64_t variable_id : ids_in_order) {
414 const double value = values[id_map.at(variable_id)];
416 result.add_ids(variable_id);
417 result.add_values(
value);
424template <
typename ValueType>
426 switch (glop_basis_status) {
427 case ValueType::BASIC:
428 return BasisStatusProto::BASIS_STATUS_BASIC;
429 case ValueType::FIXED_VALUE:
430 return BasisStatusProto::BASIS_STATUS_FIXED_VALUE;
431 case ValueType::AT_LOWER_BOUND:
432 return BasisStatusProto::BASIS_STATUS_AT_LOWER_BOUND;
433 case ValueType::AT_UPPER_BOUND:
434 return BasisStatusProto::BASIS_STATUS_AT_UPPER_BOUND;
435 case ValueType::FREE:
436 return BasisStatusProto::BASIS_STATUS_FREE;
438 return BasisStatusProto::BASIS_STATUS_UNSPECIFIED;
441template <
typename IndexType,
typename ValueType>
443 const std::vector<int64_t>& ids_in_order,
444 const absl::flat_hash_map<int64_t, IndexType>& id_map,
446 SparseBasisStatusVector result;
447 for (
const int64_t variable_id : ids_in_order) {
448 const ValueType
value = values[id_map.at(variable_id)];
449 result.add_ids(variable_id);
456template <
typename ValueType>
458 switch (basis_status) {
459 case BASIS_STATUS_BASIC:
460 return ValueType::BASIC;
461 case BASIS_STATUS_FIXED_VALUE:
462 return ValueType::FIXED_VALUE;
463 case BASIS_STATUS_AT_LOWER_BOUND:
464 return ValueType::AT_LOWER_BOUND;
465 case BASIS_STATUS_AT_UPPER_BOUND:
466 return ValueType::AT_UPPER_BOUND;
467 case BASIS_STATUS_FREE:
468 return ValueType::FREE;
470 LOG(
FATAL) <<
"Unexpected invalid initial_basis.";
471 return ValueType::FREE;
477 const absl::flat_hash_map<int64_t, T>& id_map) {
478 std::vector<int64_t> sorted;
479 sorted.reserve(id_map.size());
480 for (
const auto& entry : id_map) {
481 sorted.emplace_back(entry.first);
483 std::sort(sorted.begin(), sorted.end());
492 const absl::flat_hash_map<int64_t, T>& id_map) {
494 constexpr int64_t kEmptyId = -1;
496 for (
const auto& [
id,
index] : id_map) {
499 index_to_id[
index] = id;
509InvertedBounds GlopSolver::ListInvertedBounds()
const {
511 std::vector<glop::ColIndex> inverted_columns;
512 const glop::ColIndex num_cols = linear_program_.num_variables();
513 for (glop::ColIndex
col(0);
col < num_cols; ++
col) {
514 if (linear_program_.variable_lower_bounds()[
col] >
515 linear_program_.variable_upper_bounds()[
col]) {
519 std::vector<glop::RowIndex> inverted_rows;
520 const glop::RowIndex num_rows = linear_program_.num_constraints();
521 for (glop::RowIndex
row(0);
row < num_rows; ++
row) {
522 if (linear_program_.constraint_lower_bounds()[
row] >
523 linear_program_.constraint_upper_bounds()[
row]) {
524 inverted_rows.push_back(
row);
530 InvertedBounds inverted_bounds;
531 if (!inverted_columns.empty()) {
532 const glop::StrictITIVector<glop::ColIndex, int64_t> ids =
535 inverted_bounds.variables.reserve(inverted_columns.size());
536 for (
const glop::ColIndex
col : inverted_columns) {
537 inverted_bounds.variables.push_back(ids[
col]);
540 if (!inverted_rows.empty()) {
541 const glop::StrictITIVector<glop::RowIndex, int64_t> ids =
544 inverted_bounds.linear_constraints.reserve(inverted_rows.size());
545 for (
const glop::RowIndex
row : inverted_rows) {
546 inverted_bounds.linear_constraints.push_back(ids[
row]);
550 return inverted_bounds;
554 const ModelSolveParametersProto& model_parameters,
555 SolveResultProto& solve_result) {
560 const bool phase_I_solution_available =
561 (
status == glop::ProblemStatus::INIT) &&
562 (lp_solver_.GetNumberOfSimplexIterations() > 0);
563 if (
status != glop::ProblemStatus::OPTIMAL &&
564 status != glop::ProblemStatus::PRIMAL_FEASIBLE &&
565 status != glop::ProblemStatus::DUAL_FEASIBLE &&
566 status != glop::ProblemStatus::PRIMAL_UNBOUNDED &&
567 status != glop::ProblemStatus::DUAL_UNBOUNDED &&
568 !phase_I_solution_available) {
572 auto sorted_constraints =
GetSortedIs(linear_constraints_);
573 SolutionProto*
const solution = solve_result.add_solutions();
574 BasisProto*
const basis = solution->mutable_basis();
575 PrimalSolutionProto*
const primal_solution =
576 solution->mutable_primal_solution();
577 DualSolutionProto*
const dual_solution = solution->mutable_dual_solution();
582 if (
status == glop::ProblemStatus::OPTIMAL) {
583 primal_solution->set_feasibility_status(SOLUTION_STATUS_FEASIBLE);
584 basis->set_basic_dual_feasibility(SOLUTION_STATUS_FEASIBLE);
585 dual_solution->set_feasibility_status(SOLUTION_STATUS_FEASIBLE);
586 }
else if (
status == glop::ProblemStatus::PRIMAL_FEASIBLE) {
591 primal_solution->set_feasibility_status(SOLUTION_STATUS_FEASIBLE);
592 dual_solution->set_feasibility_status(SOLUTION_STATUS_UNDETERMINED);
593 basis->set_basic_dual_feasibility(SOLUTION_STATUS_INFEASIBLE);
594 }
else if (
status == glop::ProblemStatus::DUAL_FEASIBLE) {
601 primal_solution->set_feasibility_status(SOLUTION_STATUS_INFEASIBLE);
602 dual_solution->set_feasibility_status(SOLUTION_STATUS_FEASIBLE);
603 basis->set_basic_dual_feasibility(SOLUTION_STATUS_FEASIBLE);
606 if (lp_solver_.GetParameters().use_dual_simplex()) {
612 primal_solution->set_feasibility_status(SOLUTION_STATUS_UNDETERMINED);
613 dual_solution->set_feasibility_status(SOLUTION_STATUS_INFEASIBLE);
614 basis->set_basic_dual_feasibility(SOLUTION_STATUS_INFEASIBLE);
617 primal_solution->set_feasibility_status(SOLUTION_STATUS_INFEASIBLE);
618 dual_solution->set_feasibility_status(SOLUTION_STATUS_UNDETERMINED);
619 basis->set_basic_dual_feasibility(SOLUTION_STATUS_UNDETERMINED);
624 primal_solution->set_objective_value(lp_solver_.GetObjectiveValue());
625 if (basis->basic_dual_feasibility() == SOLUTION_STATUS_FEASIBLE) {
628 dual_solution->set_objective_value(primal_solution->objective_value());
632 *basis->mutable_constraint_status() = *basis->mutable_variable_status() =
634 lp_solver_.variable_statuses());
635 *basis->mutable_constraint_status() =
637 lp_solver_.constraint_statuses());
640 sorted_variables, variables_, lp_solver_.variable_values(),
641 model_parameters.variable_values_filter());
644 sorted_constraints, linear_constraints_, lp_solver_.dual_values(),
645 model_parameters.dual_values_filter());
647 sorted_variables, variables_, lp_solver_.reduced_costs(),
648 model_parameters.reduced_costs_filter());
650 if (!lp_solver_.primal_ray().empty()) {
651 PrimalRayProto*
const primal_ray = solve_result.add_primal_rays();
654 sorted_variables, variables_, lp_solver_.primal_ray(),
655 model_parameters.variable_values_filter());
657 if (!lp_solver_.constraints_dual_ray().empty() &&
658 !lp_solver_.variable_bounds_dual_ray().empty()) {
659 DualRayProto*
const dual_ray = solve_result.add_dual_rays();
660 *dual_ray->mutable_dual_values() =
662 lp_solver_.constraints_dual_ray(),
663 model_parameters.dual_values_filter());
665 sorted_variables, variables_, lp_solver_.variable_bounds_dual_ray(),
666 model_parameters.reduced_costs_filter());
671 const absl::Duration solve_time,
672 SolveStatsProto& solve_stats) {
673 const bool is_maximize = linear_program_.IsMaximizationProblem();
676 solve_stats.mutable_problem_status()->set_primal_status(
677 FEASIBILITY_STATUS_UNDETERMINED);
678 solve_stats.set_best_primal_bound(is_maximize ? -
kInf :
kInf);
679 solve_stats.mutable_problem_status()->set_dual_status(
680 FEASIBILITY_STATUS_UNDETERMINED);
681 solve_stats.set_best_dual_bound(is_maximize ?
kInf : -
kInf);
685 case glop::ProblemStatus::OPTIMAL:
686 solve_stats.mutable_problem_status()->set_primal_status(
687 FEASIBILITY_STATUS_FEASIBLE);
688 solve_stats.mutable_problem_status()->set_dual_status(
689 FEASIBILITY_STATUS_FEASIBLE);
690 solve_stats.set_best_primal_bound(lp_solver_.GetObjectiveValue());
691 solve_stats.set_best_dual_bound(lp_solver_.GetObjectiveValue());
693 case glop::ProblemStatus::PRIMAL_INFEASIBLE:
694 solve_stats.mutable_problem_status()->set_primal_status(
695 FEASIBILITY_STATUS_INFEASIBLE);
697 case glop::ProblemStatus::DUAL_UNBOUNDED:
698 solve_stats.mutable_problem_status()->set_primal_status(
699 FEASIBILITY_STATUS_INFEASIBLE);
700 solve_stats.mutable_problem_status()->set_dual_status(
701 FEASIBILITY_STATUS_FEASIBLE);
702 solve_stats.set_best_dual_bound(is_maximize ? -
kInf :
kInf);
704 case glop::ProblemStatus::PRIMAL_UNBOUNDED:
705 solve_stats.mutable_problem_status()->set_primal_status(
706 FEASIBILITY_STATUS_FEASIBLE);
707 solve_stats.mutable_problem_status()->set_dual_status(
708 FEASIBILITY_STATUS_INFEASIBLE);
709 solve_stats.set_best_primal_bound(is_maximize ?
kInf : -
kInf);
711 case glop::ProblemStatus::DUAL_INFEASIBLE:
712 solve_stats.mutable_problem_status()->set_dual_status(
713 FEASIBILITY_STATUS_INFEASIBLE);
715 case glop::ProblemStatus::INFEASIBLE_OR_UNBOUNDED:
716 solve_stats.mutable_problem_status()->set_primal_or_dual_infeasible(
true);
718 case glop::ProblemStatus::PRIMAL_FEASIBLE:
719 solve_stats.mutable_problem_status()->set_primal_status(
720 FEASIBILITY_STATUS_FEASIBLE);
721 solve_stats.set_best_primal_bound(lp_solver_.GetObjectiveValue());
723 case glop::ProblemStatus::DUAL_FEASIBLE:
724 solve_stats.mutable_problem_status()->set_dual_status(
725 FEASIBILITY_STATUS_FEASIBLE);
726 solve_stats.set_best_dual_bound(lp_solver_.GetObjectiveValue());
728 case glop::ProblemStatus::INIT:
729 case glop::ProblemStatus::IMPRECISE:
733 case glop::ProblemStatus::ABNORMAL:
734 case glop::ProblemStatus::INVALID_PROBLEM:
735 return absl::InternalError(
736 absl::StrCat(
"Unexpected GLOP termination reason: ",
741 solve_stats.set_simplex_iterations(lp_solver_.GetNumberOfSimplexIterations());
743 solve_time, solve_stats.mutable_solve_time()));
745 return absl::OkStatus();
748absl::StatusOr<SolveResultProto> GlopSolver::MakeSolveResult(
750 const ModelSolveParametersProto& model_parameters,
751 const SolveInterrupter*
const interrupter,
752 const absl::Duration solve_time) {
753 SolveResultProto solve_result;
755 BuildTermination(
status, interrupter));
756 FillSolution(
status, model_parameters, solve_result);
758 FillSolveStats(
status, solve_time, *solve_result.mutable_solve_stats()));
762void GlopSolver::SetGlopBasis(
const BasisProto& basis) {
764 for (
const auto [
id,
value] :
MakeView(basis.variable_status())) {
765 variable_statuses[variables_.at(
id)] =
766 ToGlopBasisStatus<glop::VariableStatus>(
767 static_cast<BasisStatusProto
>(
value));
770 linear_program_.num_constraints());
771 for (
const auto [
id,
value] :
MakeView(basis.constraint_status())) {
772 constraint_statuses[linear_constraints_.at(
id)] =
773 ToGlopBasisStatus<glop::ConstraintStatus>(
774 static_cast<BasisStatusProto
>(
value));
776 lp_solver_.SetInitialBasis(variable_statuses, constraint_statuses);
781 const ModelSolveParametersProto& model_parameters,
783 const CallbackRegistrationProto& callback_registration,
const Callback cb,
788 const absl::Time
start = absl::Now();
790 const glop::GlopParameters glop_parameters,
791 MergeSolveParameters(
793 model_parameters.has_initial_basis(),
794 message_cb !=
nullptr));
795 lp_solver_.SetParameters(glop_parameters);
797 if (model_parameters.has_initial_basis()) {
798 SetGlopBasis(model_parameters.initial_basis());
801 std::atomic<bool> interrupt_solve =
false;
803 TimeLimit::FromParameters(lp_solver_.GetParameters());
804 time_limit->RegisterExternalBooleanAsLimit(&interrupt_solve);
808 interrupt_solve =
true;
811 if (message_cb !=
nullptr) {
819 CHECK_EQ(lp_solver_.GetSolverLogger().NumInfoLoggingCallbacks(), 0);
820 lp_solver_.GetSolverLogger().AddInfoLoggingCallback(
821 [&](
const std::string&
message) {
822 message_cb(absl::StrSplit(
message,
'\n'));
826 if (message_cb !=
nullptr) {
828 CHECK_EQ(lp_solver_.GetSolverLogger().NumInfoLoggingCallbacks(), 1);
829 lp_solver_.GetSolverLogger().ClearInfoLoggingCallbacks();
839 lp_solver_.SolveWithTimeLimit(linear_program_,
time_limit.get());
840 const absl::Duration solve_time = absl::Now() -
start;
841 return MakeSolveResult(
status, model_parameters, interrupter, solve_time);
844absl::StatusOr<std::unique_ptr<SolverInterface>> GlopSolver::New(
846 if (!
model.objective().quadratic_coefficients().row_ids().empty()) {
847 return absl::InvalidArgumentError(
848 "Glop does not support quadratic objectives");
850 auto solver = absl::WrapUnique(
new GlopSolver);
854 solver->linear_program_.SetDcheckBounds(
false);
856 solver->linear_program_.SetName(
model.name());
857 solver->linear_program_.SetMaximizationProblem(
model.objective().maximize());
858 solver->linear_program_.SetObjectiveOffset(
model.objective().offset());
860 solver->AddVariables(
model.variables());
861 solver->SetOrUpdateObjectiveCoefficients(
862 model.objective().linear_coefficients());
864 solver->AddLinearConstraints(
model.linear_constraints());
865 solver->SetOrUpdateConstraintMatrix(
model.linear_constraint_matrix());
866 solver->linear_program_.CleanUp();
870absl::Status GlopSolver::Update(
const ModelUpdateProto& model_update) {
871 if (model_update.objective_updates().has_direction_update()) {
872 linear_program_.SetMaximizationProblem(
873 model_update.objective_updates().direction_update());
875 if (model_update.objective_updates().has_offset_update()) {
876 linear_program_.SetObjectiveOffset(
877 model_update.objective_updates().offset_update());
880 DeleteVariables(model_update.deleted_variable_ids());
881 AddVariables(model_update.new_variables());
883 SetOrUpdateObjectiveCoefficients(
884 model_update.objective_updates().linear_coefficients());
885 UpdateVariableBounds(model_update.variable_updates());
887 DeleteLinearConstraints(model_update.deleted_linear_constraint_ids());
888 AddLinearConstraints(model_update.new_linear_constraints());
889 UpdateLinearConstraintBounds(model_update.linear_constraint_updates());
891 SetOrUpdateConstraintMatrix(model_update.linear_constraint_matrix_updates());
893 linear_program_.CleanUp();
895 return absl::OkStatus();
#define CHECK_EQ(val1, val2)
#define CHECK_NE(val1, val2)
#define ASSIGN_OR_RETURN(lhs, rexpr)
#define RETURN_IF_ERROR(expr)
void push_back(const value_type &x)
std::function< void(const std::vector< std::string > &)> MessageCallback
std::function< absl::StatusOr< CallbackResultProto >(const CallbackDataProto &)> Callback
bool AcceptsAndUpdate(const int64_t id, const Value &value)
ModelSharedTimeLimit * time_limit
absl::Cleanup< absl::decay_t< Callback > > MakeCleanup(Callback &&callback)
void InsertOrDie(Collection *const collection, const typename Collection::value_type &value)
std::string GetProblemStatusString(ProblemStatus problem_status)
StrictITIVector< ColIndex, VariableStatus > VariableStatusRow
StrictITIVector< RowIndex, ConstraintStatus > ConstraintStatusColumn
@ INFEASIBLE_OR_UNBOUNDED
StrictITIVector< RowIndex, bool > DenseBooleanColumn
TerminationProto FeasibleTermination(const LimitProto limit, const absl::string_view detail)
absl::Status CheckRegisteredCallbackEvents(const CallbackRegistrationProto ®istration, const absl::flat_hash_set< CallbackEventProto > &supported_events)
int NumMatrixNonzeros(const SparseDoubleMatrixProto &matrix)
void UpdateIdIndexMap(glop::StrictITIVector< IndexType, bool > indices_to_delete, IndexType num_indices, absl::flat_hash_map< int64_t, IndexType > &id_index_map)
int NumVariables(const VariablesProto &variables)
SparseDoubleVectorProto FillSparseDoubleVector(const std::vector< int64_t > &ids_in_order, const absl::flat_hash_map< int64_t, IndexType > &id_map, const glop::StrictITIVector< IndexType, glop::Fractional > &values, const SparseVectorFilterProto &filter)
BasisStatusProto FromGlopBasisStatus(const ValueType glop_basis_status)
absl::StatusOr< SolveResult > Solve(const Model &model, const SolverType solver_type, const SolveArguments &solve_args, const SolverInitArguments &init_args)
ValueType ToGlopBasisStatus(const BasisStatusProto basis_status)
glop::StrictITIVector< T, int64_t > IndexToId(const absl::flat_hash_map< int64_t, T > &id_map)
std::vector< int64_t > GetSortedIs(const absl::flat_hash_map< int64_t, T > &id_map)
SparseVectorView< T > MakeView(absl::Span< const int64_t > ids, const Collection &values)
TerminationProto NoSolutionFoundTermination(const LimitProto limit, const absl::string_view detail)
int NumConstraints(const LinearConstraintsProto &linear_constraints)
TerminationProto TerminateForReason(const TerminationReasonProto reason, const absl::string_view detail)
SparseBasisStatusVector FillSparseBasisStatusVector(const std::vector< int64_t > &ids_in_order, const absl::flat_hash_map< int64_t, IndexType > &id_map, const glop::StrictITIVector< IndexType, ValueType > &values)
Collection of objects used to extend the Constraint Solver library.
std::string ProtoEnumToString(ProtoEnumType enum_value)
inline ::absl::StatusOr< absl::Duration > DecodeGoogleApiProto(const google::protobuf::Duration &proto)
inline ::absl::StatusOr< google::protobuf::Duration > EncodeGoogleApiProto(absl::Duration d)
#define MATH_OPT_REGISTER_SOLVER(solver_type, solver_factory)