29#include "absl/container/flat_hash_map.h"
30#include "absl/memory/memory.h"
31#include "absl/status/status.h"
32#include "absl/status/statusor.h"
33#include "absl/strings/str_cat.h"
34#include "absl/strings/str_join.h"
35#include "absl/strings/str_split.h"
36#include "absl/strings/string_view.h"
37#include "absl/time/clock.h"
38#include "absl/time/time.h"
39#include "absl/types/span.h"
47#include "ortools/math_opt/callback.pb.h"
52#include "ortools/math_opt/model.pb.h"
53#include "ortools/math_opt/model_parameters.pb.h"
54#include "ortools/math_opt/model_update.pb.h"
55#include "ortools/math_opt/parameters.pb.h"
56#include "ortools/math_opt/result.pb.h"
57#include "ortools/math_opt/solution.pb.h"
58#include "ortools/math_opt/sparse_containers.pb.h"
62#include "absl/status/status.h"
71absl::string_view SafeName(
const VariablesProto& variables,
int index) {
72 if (variables.names().empty()) {
75 return variables.names(
index);
78absl::string_view SafeName(
const LinearConstraintsProto& linear_constraints,
80 if (linear_constraints.names().empty()) {
83 return linear_constraints.names(
index);
86absl::StatusOr<TerminationProto> BuildTermination(
88 const SolveInterrupter*
const interrupter) {
108 interrupter->IsInterrupted()
110 : LIMIT_UNDETERMINED);
118 interrupter->IsInterrupted()
120 : LIMIT_UNDETERMINED);
125 return absl::InternalError(
126 absl::StrCat(
"Unexpected GLOP termination reason: ",
129 LOG(
FATAL) <<
"Unimplemented GLOP termination reason: "
135GlopSolver::GlopSolver() : linear_program_(), lp_solver_() {}
137void GlopSolver::AddVariables(
const VariablesProto& variables) {
139 const glop::ColIndex col_index = linear_program_.CreateNewVariable();
140 linear_program_.SetVariableBounds(col_index, variables.lower_bounds(i),
141 variables.upper_bounds(i));
142 linear_program_.SetVariableName(col_index, SafeName(variables, i));
150template <
typename IndexType>
153 IndexType num_indices,
154 absl::flat_hash_map<int64_t, IndexType>& id_index_map) {
156 num_indices.value(), IndexType(0));
157 IndexType new_index(0);
159 if (indices_to_delete[
index]) {
161 new_indices[
index] = IndexType(-1);
163 new_indices[
index] = new_index;
167 for (
auto it = id_index_map.begin(); it != id_index_map.end();) {
168 IndexType
index = it->second;
169 if (indices_to_delete[
index]) {
171 id_index_map.erase(it++);
173 it->second = new_indices[
index];
179void GlopSolver::DeleteVariables(absl::Span<const int64_t> ids_to_delete) {
180 const glop::ColIndex num_cols = linear_program_.num_variables();
183 for (
const int64_t deleted_variable_id : ids_to_delete) {
184 columns_to_delete[variables_.at(deleted_variable_id)] =
true;
186 linear_program_.DeleteColumns(columns_to_delete);
187 UpdateIdIndexMap<glop::ColIndex>(columns_to_delete, num_cols, variables_);
190void GlopSolver::DeleteLinearConstraints(
191 absl::Span<const int64_t> ids_to_delete) {
192 const glop::RowIndex num_rows = linear_program_.num_constraints();
194 for (
const int64_t deleted_constraint_id : ids_to_delete) {
195 rows_to_delete[linear_constraints_.at(deleted_constraint_id)] =
true;
197 linear_program_.DeleteRows(rows_to_delete);
198 UpdateIdIndexMap<glop::RowIndex>(rows_to_delete, num_rows,
199 linear_constraints_);
202void GlopSolver::AddLinearConstraints(
203 const LinearConstraintsProto& linear_constraints) {
205 const glop::RowIndex row_index = linear_program_.CreateNewConstraint();
206 linear_program_.SetConstraintBounds(row_index,
207 linear_constraints.lower_bounds(i),
208 linear_constraints.upper_bounds(i));
209 linear_program_.SetConstraintName(row_index,
210 SafeName(linear_constraints, i));
216void GlopSolver::SetOrUpdateObjectiveCoefficients(
217 const SparseDoubleVectorProto& linear_objective_coefficients) {
218 for (
int i = 0; i < linear_objective_coefficients.ids_size(); ++i) {
219 const glop::ColIndex col_index =
220 variables_.at(linear_objective_coefficients.ids(i));
221 linear_program_.SetObjectiveCoefficient(
222 col_index, linear_objective_coefficients.values(i));
226void GlopSolver::SetOrUpdateConstraintMatrix(
227 const SparseDoubleMatrixProto& linear_constraint_matrix) {
229 const glop::ColIndex col_index =
230 variables_.at(linear_constraint_matrix.column_ids(j));
231 const glop::RowIndex row_index =
232 linear_constraints_.at(linear_constraint_matrix.row_ids(j));
233 const double coefficient = linear_constraint_matrix.coefficients(j);
234 linear_program_.SetCoefficient(row_index, col_index,
coefficient);
238void GlopSolver::UpdateVariableBounds(
239 const VariableUpdatesProto& variable_updates) {
240 for (
const auto [
id, lb] :
MakeView(variable_updates.lower_bounds())) {
241 const auto col_index = variables_.at(
id);
242 linear_program_.SetVariableBounds(
243 col_index, lb, linear_program_.variable_upper_bounds()[col_index]);
245 for (
const auto [
id, ub] :
MakeView(variable_updates.upper_bounds())) {
246 const auto col_index = variables_.at(
id);
247 linear_program_.SetVariableBounds(
248 col_index, linear_program_.variable_lower_bounds()[col_index], ub);
252void GlopSolver::UpdateLinearConstraintBounds(
253 const LinearConstraintUpdatesProto& linear_constraint_updates) {
254 for (
const auto [
id, lb] :
255 MakeView(linear_constraint_updates.lower_bounds())) {
256 const auto row_index = linear_constraints_.at(
id);
257 linear_program_.SetConstraintBounds(
258 row_index, lb, linear_program_.constraint_upper_bounds()[row_index]);
260 for (
const auto [
id, ub] :
261 MakeView(linear_constraint_updates.upper_bounds())) {
262 const auto row_index = linear_constraints_.at(
id);
263 linear_program_.SetConstraintBounds(
264 row_index, linear_program_.constraint_lower_bounds()[row_index], ub);
268std::pair<glop::GlopParameters, std::vector<std::string>>
269GlopSolver::MergeSolveParameters(
const SolveParametersProto& solver_parameters,
270 const bool setting_initial_basis,
271 const bool has_message_callback) {
273 std::vector<std::string> warnings;
279 if (has_message_callback) {
296 if (!result.
has_random_seed() && solver_parameters.has_random_seed()) {
297 const int random_seed =
std::max(0, solver_parameters.random_seed());
301 solver_parameters.iteration_limit()) {
305 solver_parameters.lp_algorithm() != LP_ALGORITHM_UNSPECIFIED) {
306 switch (solver_parameters.lp_algorithm()) {
307 case LP_ALGORITHM_PRIMAL_SIMPLEX:
310 case LP_ALGORITHM_DUAL_SIMPLEX:
313 case LP_ALGORITHM_BARRIER:
315 "GLOP does not support 'LP_ALGORITHM_BARRIER' value for "
316 "'lp_algorithm' parameter.");
321 <<
" unknown, error setting GLOP parameters";
325 solver_parameters.scaling() != EMPHASIS_UNSPECIFIED) {
326 switch (solver_parameters.scaling()) {
331 case EMPHASIS_MEDIUM:
336 case EMPHASIS_VERY_HIGH:
343 <<
" unknown, error setting GLOP parameters";
346 if (setting_initial_basis) {
349 solver_parameters.presolve() != EMPHASIS_UNSPECIFIED) {
350 switch (solver_parameters.presolve()) {
355 case EMPHASIS_MEDIUM:
357 case EMPHASIS_VERY_HIGH:
363 <<
" unknown, error setting GLOP parameters";
366 if (solver_parameters.cuts() != EMPHASIS_UNSPECIFIED) {
367 warnings.push_back(absl::StrCat(
368 "GLOP does not support 'cuts' parameters, but cuts was set to: ",
371 if (solver_parameters.heuristics() != EMPHASIS_UNSPECIFIED) {
373 absl::StrCat(
"GLOP does not support 'heuristics' parameter, but "
374 "heuristics was set to: ",
377 if (solver_parameters.has_cutoff_limit()) {
378 warnings.push_back(
"GLOP does not support 'cutoff_limit' parameter");
380 if (solver_parameters.has_objective_limit()) {
381 warnings.push_back(
"GLOP does not support 'objective_limit' parameter");
383 if (solver_parameters.has_best_bound_limit()) {
384 warnings.push_back(
"GLOP does not support 'best_bound_limit' parameter");
386 if (solver_parameters.has_solution_limit()) {
387 warnings.push_back(
"GLOP does not support 'solution_limit' parameter");
389 return std::make_pair(std::move(result), std::move(warnings));
392bool GlopSolver::CanUpdate(
const ModelUpdateProto& model_update) {
393 return model_update.objective_updates()
394 .quadratic_coefficients()
399template <
typename IndexType>
401 const std::vector<int64_t>& ids_in_order,
402 const absl::flat_hash_map<int64_t, IndexType>& id_map,
404 const SparseVectorFilterProto& filter) {
406 SparseDoubleVectorProto result;
407 for (
const int64_t variable_id : ids_in_order) {
408 const double value = values[id_map.at(variable_id)];
410 result.add_ids(variable_id);
411 result.add_values(
value);
418template <
typename ValueType>
420 switch (glop_basis_status) {
421 case ValueType::BASIC:
422 return BasisStatusProto::BASIS_STATUS_BASIC;
423 case ValueType::FIXED_VALUE:
424 return BasisStatusProto::BASIS_STATUS_FIXED_VALUE;
425 case ValueType::AT_LOWER_BOUND:
426 return BasisStatusProto::BASIS_STATUS_AT_LOWER_BOUND;
427 case ValueType::AT_UPPER_BOUND:
428 return BasisStatusProto::BASIS_STATUS_AT_UPPER_BOUND;
429 case ValueType::FREE:
430 return BasisStatusProto::BASIS_STATUS_FREE;
432 return BasisStatusProto::BASIS_STATUS_UNSPECIFIED;
435template <
typename IndexType,
typename ValueType>
437 const std::vector<int64_t>& ids_in_order,
438 const absl::flat_hash_map<int64_t, IndexType>& id_map,
440 SparseBasisStatusVector result;
441 for (
const int64_t variable_id : ids_in_order) {
442 const ValueType
value = values[id_map.at(variable_id)];
443 result.add_ids(variable_id);
450template <
typename ValueType>
452 switch (basis_status) {
453 case BASIS_STATUS_BASIC:
454 return ValueType::BASIC;
455 case BASIS_STATUS_FIXED_VALUE:
456 return ValueType::FIXED_VALUE;
457 case BASIS_STATUS_AT_LOWER_BOUND:
458 return ValueType::AT_LOWER_BOUND;
459 case BASIS_STATUS_AT_UPPER_BOUND:
460 return ValueType::AT_UPPER_BOUND;
461 case BASIS_STATUS_FREE:
462 return ValueType::FREE;
464 LOG(
FATAL) <<
"Unexpected invalid initial_basis.";
465 return ValueType::FREE;
471 const absl::flat_hash_map<int64_t, T>& id_map) {
472 std::vector<int64_t> sorted;
473 sorted.reserve(id_map.size());
474 for (
const auto& entry : id_map) {
475 sorted.emplace_back(entry.first);
477 std::sort(sorted.begin(), sorted.end());
482 const ModelSolveParametersProto& model_parameters,
483 SolveResultProto& solve_result) {
488 const bool phase_I_solution_available =
489 (
status == glop::ProblemStatus::INIT) &&
490 (lp_solver_.GetNumberOfSimplexIterations() > 0);
492 status != glop::ProblemStatus::PRIMAL_FEASIBLE &&
493 status != glop::ProblemStatus::DUAL_FEASIBLE &&
494 status != glop::ProblemStatus::PRIMAL_UNBOUNDED &&
495 status != glop::ProblemStatus::DUAL_UNBOUNDED &&
496 !phase_I_solution_available) {
500 auto sorted_constraints =
GetSortedIs(linear_constraints_);
501 SolutionProto*
const solution = solve_result.add_solutions();
502 BasisProto*
const basis = solution->mutable_basis();
503 PrimalSolutionProto*
const primal_solution =
504 solution->mutable_primal_solution();
505 DualSolutionProto*
const dual_solution = solution->mutable_dual_solution();
511 primal_solution->set_feasibility_status(SOLUTION_STATUS_FEASIBLE);
512 basis->set_basic_dual_feasibility(SOLUTION_STATUS_FEASIBLE);
513 dual_solution->set_feasibility_status(SOLUTION_STATUS_FEASIBLE);
514 }
else if (
status == glop::ProblemStatus::PRIMAL_FEASIBLE) {
519 primal_solution->set_feasibility_status(SOLUTION_STATUS_FEASIBLE);
520 dual_solution->set_feasibility_status(SOLUTION_STATUS_UNDETERMINED);
521 basis->set_basic_dual_feasibility(SOLUTION_STATUS_INFEASIBLE);
522 }
else if (
status == glop::ProblemStatus::DUAL_FEASIBLE) {
529 primal_solution->set_feasibility_status(SOLUTION_STATUS_INFEASIBLE);
530 dual_solution->set_feasibility_status(SOLUTION_STATUS_FEASIBLE);
531 basis->set_basic_dual_feasibility(SOLUTION_STATUS_FEASIBLE);
534 if (lp_solver_.GetParameters().use_dual_simplex()) {
540 primal_solution->set_feasibility_status(SOLUTION_STATUS_UNDETERMINED);
541 dual_solution->set_feasibility_status(SOLUTION_STATUS_INFEASIBLE);
542 basis->set_basic_dual_feasibility(SOLUTION_STATUS_INFEASIBLE);
545 primal_solution->set_feasibility_status(SOLUTION_STATUS_INFEASIBLE);
546 dual_solution->set_feasibility_status(SOLUTION_STATUS_UNDETERMINED);
547 basis->set_basic_dual_feasibility(SOLUTION_STATUS_UNDETERMINED);
552 primal_solution->set_objective_value(lp_solver_.GetObjectiveValue());
553 if (basis->basic_dual_feasibility() == SOLUTION_STATUS_FEASIBLE) {
556 dual_solution->set_objective_value(primal_solution->objective_value());
560 *basis->mutable_constraint_status() = *basis->mutable_variable_status() =
562 lp_solver_.variable_statuses());
563 *basis->mutable_constraint_status() =
565 lp_solver_.constraint_statuses());
568 sorted_variables, variables_, lp_solver_.variable_values(),
569 model_parameters.variable_values_filter());
572 sorted_constraints, linear_constraints_, lp_solver_.dual_values(),
573 model_parameters.dual_values_filter());
575 sorted_variables, variables_, lp_solver_.reduced_costs(),
576 model_parameters.reduced_costs_filter());
578 if (!lp_solver_.primal_ray().empty()) {
579 PrimalRayProto*
const primal_ray = solve_result.add_primal_rays();
582 sorted_variables, variables_, lp_solver_.primal_ray(),
583 model_parameters.variable_values_filter());
585 if (!lp_solver_.constraints_dual_ray().empty() &&
586 !lp_solver_.variable_bounds_dual_ray().empty()) {
587 DualRayProto*
const dual_ray = solve_result.add_dual_rays();
588 *dual_ray->mutable_dual_values() =
590 lp_solver_.constraints_dual_ray(),
591 model_parameters.dual_values_filter());
593 sorted_variables, variables_, lp_solver_.variable_bounds_dual_ray(),
594 model_parameters.reduced_costs_filter());
599 const absl::Duration solve_time,
600 SolveStatsProto& solve_stats) {
601 const bool is_maximize = linear_program_.IsMaximizationProblem();
602 constexpr double kInf = std::numeric_limits<double>::infinity();
605 solve_stats.mutable_problem_status()->set_primal_status(
606 FEASIBILITY_STATUS_UNDETERMINED);
607 solve_stats.set_best_primal_bound(is_maximize ? -
kInf :
kInf);
608 solve_stats.mutable_problem_status()->set_dual_status(
609 FEASIBILITY_STATUS_UNDETERMINED);
610 solve_stats.set_best_dual_bound(is_maximize ?
kInf : -
kInf);
615 solve_stats.mutable_problem_status()->set_primal_status(
616 FEASIBILITY_STATUS_FEASIBLE);
617 solve_stats.mutable_problem_status()->set_dual_status(
618 FEASIBILITY_STATUS_FEASIBLE);
619 solve_stats.set_best_primal_bound(lp_solver_.GetObjectiveValue());
620 solve_stats.set_best_dual_bound(lp_solver_.GetObjectiveValue());
622 case glop::ProblemStatus::PRIMAL_INFEASIBLE:
623 solve_stats.mutable_problem_status()->set_primal_status(
624 FEASIBILITY_STATUS_INFEASIBLE);
626 case glop::ProblemStatus::DUAL_UNBOUNDED:
627 solve_stats.mutable_problem_status()->set_primal_status(
628 FEASIBILITY_STATUS_INFEASIBLE);
629 solve_stats.mutable_problem_status()->set_dual_status(
630 FEASIBILITY_STATUS_FEASIBLE);
631 solve_stats.set_best_dual_bound(is_maximize ? -
kInf :
kInf);
633 case glop::ProblemStatus::PRIMAL_UNBOUNDED:
634 solve_stats.mutable_problem_status()->set_primal_status(
635 FEASIBILITY_STATUS_FEASIBLE);
636 solve_stats.mutable_problem_status()->set_dual_status(
637 FEASIBILITY_STATUS_INFEASIBLE);
638 solve_stats.set_best_primal_bound(is_maximize ?
kInf : -
kInf);
640 case glop::ProblemStatus::DUAL_INFEASIBLE:
641 solve_stats.mutable_problem_status()->set_dual_status(
642 FEASIBILITY_STATUS_INFEASIBLE);
644 case glop::ProblemStatus::INFEASIBLE_OR_UNBOUNDED:
645 solve_stats.mutable_problem_status()->set_primal_or_dual_infeasible(
true);
647 case glop::ProblemStatus::PRIMAL_FEASIBLE:
648 solve_stats.mutable_problem_status()->set_primal_status(
649 FEASIBILITY_STATUS_FEASIBLE);
650 solve_stats.set_best_primal_bound(lp_solver_.GetObjectiveValue());
652 case glop::ProblemStatus::DUAL_FEASIBLE:
653 solve_stats.mutable_problem_status()->set_dual_status(
654 FEASIBILITY_STATUS_FEASIBLE);
655 solve_stats.set_best_dual_bound(lp_solver_.GetObjectiveValue());
657 case glop::ProblemStatus::INIT:
658 case glop::ProblemStatus::IMPRECISE:
662 case glop::ProblemStatus::ABNORMAL:
663 case glop::ProblemStatus::INVALID_PROBLEM:
664 return absl::InternalError(
665 absl::StrCat(
"Unexpected GLOP termination reason: ",
670 solve_stats.set_simplex_iterations(lp_solver_.GetNumberOfSimplexIterations());
672 solve_time, solve_stats.mutable_solve_time()));
674 return absl::OkStatus();
677absl::Status GlopSolver::FillSolveResult(
679 const ModelSolveParametersProto& model_parameters,
680 const SolveInterrupter*
const interrupter,
const absl::Duration solve_time,
681 SolveResultProto& solve_result) {
683 BuildTermination(
status, interrupter));
684 FillSolution(
status, model_parameters, solve_result);
686 FillSolveStats(
status, solve_time, *solve_result.mutable_solve_stats()));
687 return absl::OkStatus();
690void GlopSolver::SetGlopBasis(
const BasisProto& basis) {
692 for (
const auto [
id,
value] :
MakeView(basis.variable_status())) {
693 variable_statuses[variables_.at(
id)] =
694 ToGlopBasisStatus<glop::VariableStatus>(
695 static_cast<BasisStatusProto
>(
value));
698 linear_program_.num_constraints());
699 for (
const auto [
id,
value] :
MakeView(basis.constraint_status())) {
700 constraint_statuses[linear_constraints_.at(
id)] =
701 ToGlopBasisStatus<glop::ConstraintStatus>(
702 static_cast<BasisStatusProto
>(
value));
704 lp_solver_.SetInitialBasis(variable_statuses, constraint_statuses);
709 const ModelSolveParametersProto& model_parameters,
711 const CallbackRegistrationProto& callback_registration,
const Callback cb,
716 const absl::Time
start = absl::Now();
717 SolveResultProto result;
719 auto [glop_parameters, warnings] = MergeSolveParameters(
721 model_parameters.has_initial_basis(),
722 message_cb !=
nullptr);
723 if (!warnings.empty()) {
724 if (
parameters.strictness().bad_parameter()) {
725 return absl::InvalidArgumentError(absl::StrJoin(warnings,
"; "));
727 for (std::string& warning : warnings) {
728 result.add_warnings(std::move(warning));
732 lp_solver_.SetParameters(glop_parameters);
735 if (model_parameters.has_initial_basis()) {
736 SetGlopBasis(model_parameters.initial_basis());
739 std::atomic<bool> interrupt_solve =
false;
741 TimeLimit::FromParameters(lp_solver_.GetParameters());
742 time_limit->RegisterExternalBooleanAsLimit(&interrupt_solve);
746 interrupt_solve =
true;
749 if (message_cb !=
nullptr) {
757 CHECK_EQ(lp_solver_.GetSolverLogger().NumInfoLoggingCallbacks(), 0);
758 lp_solver_.GetSolverLogger().AddInfoLoggingCallback(
759 [&](
const std::string&
message) {
760 message_cb(absl::StrSplit(
message,
'\n'));
764 if (message_cb !=
nullptr) {
766 CHECK_EQ(lp_solver_.GetSolverLogger().NumInfoLoggingCallbacks(), 1);
767 lp_solver_.GetSolverLogger().ClearInfoLoggingCallbacks();
772 lp_solver_.SolveWithTimeLimit(linear_program_,
time_limit.get());
773 const absl::Duration solve_time = absl::Now() -
start;
776 solve_time, result));
781absl::StatusOr<std::unique_ptr<SolverInterface>> GlopSolver::New(
783 if (!
model.objective().quadratic_coefficients().row_ids().empty()) {
784 return absl::InvalidArgumentError(
785 "Glop does not support quadratic objectives");
787 auto solver = absl::WrapUnique(
new GlopSolver);
791 solver->linear_program_.SetDcheckBounds(
false);
793 solver->linear_program_.SetName(
model.name());
794 solver->linear_program_.SetMaximizationProblem(
model.objective().maximize());
795 solver->linear_program_.SetObjectiveOffset(
model.objective().offset());
797 solver->AddVariables(
model.variables());
798 solver->SetOrUpdateObjectiveCoefficients(
799 model.objective().linear_coefficients());
801 solver->AddLinearConstraints(
model.linear_constraints());
802 solver->SetOrUpdateConstraintMatrix(
model.linear_constraint_matrix());
803 solver->linear_program_.CleanUp();
807absl::Status GlopSolver::Update(
const ModelUpdateProto& model_update) {
808 if (model_update.objective_updates().has_direction_update()) {
809 linear_program_.SetMaximizationProblem(
810 model_update.objective_updates().direction_update());
812 if (model_update.objective_updates().has_offset_update()) {
813 linear_program_.SetObjectiveOffset(
814 model_update.objective_updates().offset_update());
817 DeleteVariables(model_update.deleted_variable_ids());
818 AddVariables(model_update.new_variables());
820 SetOrUpdateObjectiveCoefficients(
821 model_update.objective_updates().linear_coefficients());
822 UpdateVariableBounds(model_update.variable_updates());
824 DeleteLinearConstraints(model_update.deleted_linear_constraint_ids());
825 AddLinearConstraints(model_update.new_linear_constraints());
826 UpdateLinearConstraintBounds(model_update.linear_constraint_updates());
828 SetOrUpdateConstraintMatrix(model_update.linear_constraint_matrix_updates());
830 linear_program_.CleanUp();
832 return absl::OkStatus();
#define CHECK_EQ(val1, val2)
#define CHECK_NE(val1, val2)
bool has_max_number_of_iterations() const
void set_num_omp_threads(int32_t value)
bool has_random_seed() const
void set_scaling_method(::operations_research::glop::GlopParameters_ScalingAlgorithm value)
bool has_use_dual_simplex() const
void set_max_number_of_iterations(int64_t value)
bool has_use_scaling() const
bool has_max_time_in_seconds() const
void set_random_seed(int32_t value)
void set_use_preprocessing(bool value)
void set_use_scaling(bool value)
void set_log_search_progress(bool value)
bool has_use_preprocessing() const
bool has_log_search_progress() const
void set_log_to_stdout(bool value)
void set_max_time_in_seconds(double value)
bool has_num_omp_threads() const
void set_use_dual_simplex(bool value)
bool has_scaling_method() const
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)
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)
#define ASSIGN_OR_RETURN(lhs, rexpr)
#define RETURN_IF_ERROR(expr)