27#include "absl/container/flat_hash_map.h"
28#include "absl/container/flat_hash_set.h"
29#include "absl/memory/memory.h"
30#include "absl/status/status.h"
31#include "absl/status/statusor.h"
32#include "absl/strings/str_cat.h"
33#include "absl/strings/str_format.h"
34#include "absl/strings/string_view.h"
35#include "absl/types/span.h"
38#include "ortools/gscip/gscip.pb.h"
44#include "scip/cons_linear.h"
45#include "scip/cons_quadratic.h"
47#include "scip/scip_general.h"
48#include "scip/scip_param.h"
49#include "scip/scip_prob.h"
50#include "scip/scip_solvingstats.h"
51#include "scip/scipdefplugins.h"
52#include "scip/type_cons.h"
53#include "scip/type_scip.h"
54#include "scip/type_var.h"
58#define RETURN_ERROR_UNLESS(x) \
60 return util::StatusBuilder(absl::InvalidArgumentError(absl::StrFormat( \
61 "Condition violated at %s:%d: %s", __FILE__, __LINE__, #x)))
65constexpr absl::string_view kLinearConstraintHandlerName =
"linear";
67SCIP_VARTYPE ConvertVarType(
const GScipVarType var_type) {
70 return SCIP_VARTYPE_CONTINUOUS;
72 return SCIP_VARTYPE_IMPLINT;
74 return SCIP_VARTYPE_INTEGER;
78GScipVarType ConvertVarType(
const SCIP_VARTYPE var_type) {
80 case SCIP_VARTYPE_CONTINUOUS:
82 case SCIP_VARTYPE_IMPLINT:
84 case SCIP_VARTYPE_INTEGER:
85 case SCIP_VARTYPE_BINARY:
90GScipOutput::Status ConvertStatus(
const SCIP_STATUS scip_status) {
91 switch (scip_status) {
92 case SCIP_STATUS_UNKNOWN:
93 return GScipOutput::UNKNOWN;
94 case SCIP_STATUS_USERINTERRUPT:
95 return GScipOutput::USER_INTERRUPT;
96 case SCIP_STATUS_BESTSOLLIMIT:
97 return GScipOutput::BEST_SOL_LIMIT;
98 case SCIP_STATUS_MEMLIMIT:
99 return GScipOutput::MEM_LIMIT;
100 case SCIP_STATUS_NODELIMIT:
101 return GScipOutput::NODE_LIMIT;
102 case SCIP_STATUS_RESTARTLIMIT:
103 return GScipOutput::RESTART_LIMIT;
104 case SCIP_STATUS_SOLLIMIT:
105 return GScipOutput::SOL_LIMIT;
106 case SCIP_STATUS_STALLNODELIMIT:
107 return GScipOutput::STALL_NODE_LIMIT;
108 case SCIP_STATUS_TIMELIMIT:
109 return GScipOutput::TIME_LIMIT;
110 case SCIP_STATUS_TOTALNODELIMIT:
111 return GScipOutput::TOTAL_NODE_LIMIT;
112 case SCIP_STATUS_OPTIMAL:
113 return GScipOutput::OPTIMAL;
114 case SCIP_STATUS_GAPLIMIT:
115 return GScipOutput::GAP_LIMIT;
116 case SCIP_STATUS_INFEASIBLE:
117 return GScipOutput::INFEASIBLE;
118 case SCIP_STATUS_UNBOUNDED:
119 return GScipOutput::UNBOUNDED;
120 case SCIP_STATUS_INFORUNBD:
121 return GScipOutput::INF_OR_UNBD;
122 case SCIP_STATUS_TERMINATE:
123 return GScipOutput::TERMINATE;
125 LOG(
FATAL) <<
"Unrecognized scip status: " << scip_status;
129SCIP_PARAMEMPHASIS ConvertEmphasis(
131 switch (gscip_emphasis) {
132 case GScipParameters::DEFAULT_EMPHASIS:
133 return SCIP_PARAMEMPHASIS_DEFAULT;
134 case GScipParameters::CP_SOLVER:
135 return SCIP_PARAMEMPHASIS_CPSOLVER;
136 case GScipParameters::EASY_CIP:
137 return SCIP_PARAMEMPHASIS_EASYCIP;
138 case GScipParameters::FEASIBILITY:
139 return SCIP_PARAMEMPHASIS_FEASIBILITY;
140 case GScipParameters::HARD_LP:
141 return SCIP_PARAMEMPHASIS_HARDLP;
142 case GScipParameters::OPTIMALITY:
143 return SCIP_PARAMEMPHASIS_OPTIMALITY;
145 return SCIP_PARAMEMPHASIS_COUNTER;
146 case GScipParameters::PHASE_FEAS:
147 return SCIP_PARAMEMPHASIS_PHASEFEAS;
148 case GScipParameters::PHASE_IMPROVE:
149 return SCIP_PARAMEMPHASIS_PHASEIMPROVE;
150 case GScipParameters::PHASE_PROOF:
151 return SCIP_PARAMEMPHASIS_PHASEPROOF;
153 LOG(
FATAL) <<
"Unrecognized gscip_emphasis: "
158SCIP_PARAMSETTING ConvertMetaParamValue(
159 const GScipParameters::MetaParamValue gscip_meta_param_value) {
160 switch (gscip_meta_param_value) {
161 case GScipParameters::DEFAULT_META_PARAM_VALUE:
162 return SCIP_PARAMSETTING_DEFAULT;
163 case GScipParameters::AGGRESSIVE:
164 return SCIP_PARAMSETTING_AGGRESSIVE;
165 case GScipParameters::FAST:
166 return SCIP_PARAMSETTING_FAST;
167 case GScipParameters::OFF:
168 return SCIP_PARAMSETTING_OFF;
170 LOG(
FATAL) <<
"Unrecognized gscip_meta_param_value: "
175absl::Status CheckSolutionsInOrder(
const GScipResult& result,
176 const bool is_maximize) {
177 auto objective_as_good_as = [is_maximize](
double left,
double right) {
179 return left >= right;
181 return left <= right;
183 for (
int i = 1; i < result.objective_values.size(); ++i) {
184 const double previous = result.objective_values[i - 1];
185 const double current = result.objective_values[i];
186 if (!objective_as_good_as(previous, current)) {
188 <<
"Expected SCIP solutions to be in best objective order "
190 << (is_maximize ?
"maximization" :
"minimization")
191 <<
" problem, the " << i - 1 <<
" objective is " << previous
192 <<
" and the " << i <<
" objective is " << current;
195 return absl::OkStatus();
207 return constraint_options;
210absl::Status GScip::SetParams(
const GScipParameters& params,
211 const std::string& legacy_params) {
212 if (params.has_silence_output()) {
213 SCIPsetMessagehdlrQuiet(scip_, params.silence_output());
215 if (!params.search_logs_filename().empty()) {
216 SCIPsetMessagehdlrLogfile(scip_, params.search_logs_filename().c_str());
219 const SCIP_Bool set_param_quiet =
220 static_cast<SCIP_Bool
>(!params.silence_output());
223 scip_, ConvertEmphasis(params.emphasis()), set_param_quiet));
224 if (params.has_heuristics()) {
226 scip_, ConvertMetaParamValue(params.heuristics()), set_param_quiet));
228 if (params.has_presolve()) {
230 scip_, ConvertMetaParamValue(params.presolve()), set_param_quiet));
232 if (params.has_separating()) {
234 scip_, ConvertMetaParamValue(params.separating()), set_param_quiet));
236 for (
const auto& bool_param : params.bool_params()) {
238 (SCIPsetBoolParam(scip_, bool_param.first.c_str(), bool_param.second)));
240 for (
const auto& int_param : params.int_params()) {
242 (SCIPsetIntParam(scip_, int_param.first.c_str(), int_param.second)));
244 for (
const auto& long_param : params.long_params()) {
246 long_param.second)));
248 for (
const auto& char_param : params.char_params()) {
249 if (char_param.second.size() != 1) {
250 return absl::InvalidArgumentError(
251 absl::StrCat(
"Character parameters must be single character strings, "
253 char_param.first,
" was: ", char_param.second));
256 char_param.second[0])));
258 for (
const auto& string_param : params.string_params()) {
260 string_param.second.c_str())));
262 for (
const auto& real_param : params.real_params()) {
264 (SCIPsetRealParam(scip_, real_param.first.c_str(), real_param.second)));
266 if (!legacy_params.empty()) {
270 return absl::OkStatus();
274 const std::string& problem_name) {
275 SCIP*
scip =
nullptr;
280 return absl::WrapUnique(
new GScip(
scip));
283GScip::GScip(SCIP* scip) : scip_(scip) {}
287absl::Status GScip::FreeTransform() {
292 return absl::StrFormat(
"SCIP %d.%d.%d [LP solver: %s]", SCIPmajorVersion(),
293 SCIPminorVersion(), SCIPtechVersion(),
294 SCIPlpiGetSolverName());
298 if (scip_ ==
nullptr) {
301 return SCIPinterruptSolve(scip_) == SCIP_OKAY;
304absl::Status GScip::CleanUp() {
305 if (scip_ !=
nullptr) {
306 for (SCIP_VAR* variable : variables_) {
307 if (variable !=
nullptr) {
311 for (SCIP_CONS* constraint : constraints_) {
312 if (constraint !=
nullptr) {
318 return absl::OkStatus();
322 const absl::Status clean_up_status = CleanUp();
323 LOG_IF(DFATAL, !clean_up_status.ok()) << clean_up_status;
327 double lb,
double ub,
double obj_coef,
GScipVarType var_type,
329 SCIP_VAR*
var =
nullptr;
336 ConvertVarType(var_type)));
341 variables_.insert(
var);
348absl::Status GScip::MaybeKeepConstraintAlive(
351 constraints_.insert(constraint);
355 return absl::OkStatus();
361 SCIP_CONS* constraint =
nullptr;
363 <<
"Error adding constraint: " <<
name <<
".";
365 _ <<
"invalid lower bound");
367 _ <<
"invalid upper bound");
370 const_cast<SCIP_VAR**
>(range.
variables.data()),
391 SCIP_CONS* constraint =
nullptr;
394 <<
"Error adding quadratic constraint: " <<
name <<
" in linear term.";
397 <<
"Error adding quadratic constraint: " <<
name <<
" in quadratic term.";
399 <<
"Error adding quadratic constraint: " <<
name <<
" in quadratic term.";
401 _ <<
"invalid lower bound");
403 _ <<
"invalid upper bound");
405 scip_, &constraint,
name.c_str(), num_lin_vars,
431 <<
"Error adding indicator constraint: " <<
name <<
".";
436 SCIP_CONS* constraint =
nullptr;
439 <<
"Error adding indicator constraint: " <<
name <<
".";
442 _ <<
"invalid upper bound");
444 scip_, &constraint,
name.c_str(), indicator,
446 const_cast<SCIP_Var**
>(indicator_constraint.
variables.data()),
447 const_cast<double*
>(indicator_constraint.
coefficients.data()),
467 <<
"Error adding and constraint: " <<
name <<
".";
468 SCIP_CONS* constraint =
nullptr;
470 SCIPcreateConsAnd(scip_, &constraint,
name.c_str(),
472 const_cast<SCIP_VAR**
>(logical_data.
operators.data()),
492 <<
"Error adding or constraint: " <<
name <<
".";
493 SCIP_CONS* constraint =
nullptr;
495 SCIPcreateConsOr(scip_, &constraint,
name.c_str(), logical_data.
resultant,
497 const_cast<SCIP_Var**
>(logical_data.
operators.data()),
515absl::Status ValidateSOSData(
const GScipSOSData& sos_data,
516 const std::string&
name) {
518 <<
"Error adding SOS constraint: " <<
name <<
".";
519 if (!sos_data.
weights.empty()) {
521 <<
" Error adding SOS constraint: " <<
name <<
".";
523 absl::flat_hash_set<double> distinct_weights;
524 for (
const double w : sos_data.
weights) {
526 <<
"Error adding SOS constraint: " <<
name
527 <<
", weights must be distinct, but found value " << w <<
" twice.";
528 distinct_weights.insert(w);
530 return absl::OkStatus();
539 SCIP_CONS* constraint =
nullptr;
540 double* weights =
nullptr;
541 if (!sos_data.
weights.empty()) {
542 weights =
const_cast<double*
>(sos_data.
weights.data());
547 const_cast<SCIP_Var**
>(sos_data.
variables.data()), weights,
566 SCIP_CONS* constraint =
nullptr;
567 double* weights =
nullptr;
568 if (!sos_data.
weights.empty()) {
569 weights =
const_cast<double*
>(sos_data.
weights.data());
573 const_cast<SCIP_Var**
>(sos_data.
variables.data()), weights,
590 scip_, is_maximize ? SCIP_OBJSENSE_MAXIMIZE : SCIP_OBJSENSE_MINIMIZE));
591 return absl::OkStatus();
595 double old_offset = SCIPgetOrigObjoffset(scip_);
596 double delta_offset = offset - old_offset;
598 return absl::OkStatus();
602 return SCIPgetObjsense(scip_) == SCIP_OBJSENSE_MAXIMIZE;
609 return absl::OkStatus();
615 return absl::OkStatus();
621 return absl::OkStatus();
626 return absl::OkStatus();
630 SCIP_Bool infeasible;
632 SCIPchgVarType(scip_,
var, ConvertVarType(var_type), &infeasible));
633 return absl::OkStatus();
637 SCIP_Bool did_delete;
640 <<
"Failed to delete variable named: " <<
Name(
var);
641 variables_.erase(
var);
643 return absl::OkStatus();
647 const absl::flat_hash_set<SCIP_VAR*>& vars) {
648 for (SCIP_CONS* constraint : constraints_) {
650 return absl::InvalidArgumentError(absl::StrCat(
651 "Model contains nonlinear constraint: ",
Name(constraint)));
654 return absl::OkStatus();
660 for (SCIP_CONS* constraint : constraints_) {
661 const absl::Span<SCIP_VAR* const> nonzeros =
663 const std::vector<SCIP_VAR*> nonzeros_copy(nonzeros.begin(),
665 for (SCIP_VAR*
var : nonzeros_copy) {
666 if (vars.contains(
var)) {
671 for (SCIP_VAR*
const var : vars) {
674 return absl::OkStatus();
678 return ScipInfUnclamp(SCIPvarGetLbOriginal(
var));
682 return ScipInfUnclamp(SCIPvarGetUbOriginal(
var));
688 return ConvertVarType(SCIPvarGetType(
var));
694 return absl::string_view(SCIPconshdlrGetName(SCIPconsGetHdlr(constraint)));
698 return ConstraintType(constraint) == kLinearConstraintHandlerName;
702 SCIP_CONS* constraint) {
703 int num_vars = SCIPgetNVarsLinear(scip_, constraint);
704 return absl::MakeConstSpan(SCIPgetValsLinear(scip_, constraint), num_vars);
708 SCIP_CONS* constraint) {
709 int num_vars = SCIPgetNVarsLinear(scip_, constraint);
710 return absl::MakeConstSpan(SCIPgetVarsLinear(scip_, constraint), num_vars);
714 return ScipInfUnclamp(SCIPgetLhsLinear(scip_, constraint));
718 return ScipInfUnclamp(SCIPgetRhsLinear(scip_, constraint));
722 return SCIPconsGetName(constraint);
728 return absl::OkStatus();
734 return absl::OkStatus();
739 constraints_.erase(constraint);
741 return absl::OkStatus();
750 return absl::OkStatus();
755 const double value) {
757 return absl::OkStatus();
763 const int scip_num_vars = SCIPgetNOrigVars(scip_);
764 const bool is_solution_partial = partial_solution.size() < scip_num_vars;
765 if (is_solution_partial) {
770 <<
"Error suggesting hint.";
773 for (
const auto& var_value_pair : partial_solution) {
775 var_value_pair.second));
777 if (!is_solution_partial) {
778 SCIP_Bool is_feasible;
780 scip_, solution,
false,
true,
783 if (!
static_cast<bool>(is_feasible)) {
790 if (
static_cast<bool>(is_stored)) {
798 const GScipParameters& params,
const std::string& legacy_params,
809 const absl::Status param_status = SetParams(params, legacy_params);
810 if (!param_status.ok()) {
811 result.
gscip_output.set_status(GScipOutput::INVALID_SOLVER_PARAMETERS);
814 std::string(param_status.message()));
817 if (params.print_scip_model()) {
820 if (!params.scip_model_filename().empty()) {
822 scip_, params.scip_model_filename().c_str(),
"cip", FALSE));
824 if (params.has_objective_limit()) {
826 ScipInfClamp(params.objective_limit()),
827 _ <<
"invalid objective_limit");
839 if (message_handler !=
nullptr) {
843 SCIPsetMessagehdlr(scip_, new_handler.get());
858 const SCIP_STAGE stage = SCIPgetStage(scip_);
859 if (stage != SCIP_STAGE_PRESOLVING && stage != SCIP_STAGE_SOLVING &&
860 stage != SCIP_STAGE_SOLVED) {
863 absl::StrCat(
"Unpexpected SCIP final stage= ", stage,
864 " was expected to be either SCIP_STAGE_PRESOLVING, "
865 "SCIP_STAGE_SOLVING, or SCIP_STAGE_SOLVED"));
868 if (params.print_detailed_solving_stats()) {
871 if (!params.detailed_solving_stats_filename().empty()) {
872 FILE*
file = fopen(params.detailed_solving_stats_filename().c_str(),
"w");
873 if (
file ==
nullptr) {
874 return absl::InvalidArgumentError(absl::StrCat(
875 "Could not open file: ", params.detailed_solving_stats_filename(),
876 " to write SCIP solve stats."));
879 int close_result = fclose(
file);
880 if (close_result != 0) {
881 return absl::InvalidArgumentError(absl::StrCat(
882 "Error: ", close_result,
883 " closing file: ", params.detailed_solving_stats_filename(),
884 " when writing solve stats."));
890 GScipSolvingStats* stats = result.
gscip_output.mutable_stats();
891 const int num_scip_solutions = SCIPgetNSols(scip_);
892 const int num_returned_solutions =
894 SCIP_SOL** all_solutions = SCIPgetSols(scip_);
895 stats->set_best_objective(ScipInfUnclamp(SCIPgetPrimalbound(scip_)));
896 for (
int i = 0; i < num_returned_solutions; ++i) {
897 SCIP_SOL* scip_sol = all_solutions[i];
898 const double obj_value = ScipInfUnclamp(SCIPgetSolOrigObj(scip_, scip_sol));
900 for (SCIP_VAR* v : variables_) {
901 solution[v] = SCIPgetSolVal(scip_, scip_sol, v);
908 if (stage != SCIP_STAGE_PRESOLVING && SCIPhasPrimalRay(scip_)) {
909 for (SCIP_VAR* v : variables_) {
910 result.
primal_ray[v] = SCIPgetPrimalRayVal(scip_, v);
914 stats->set_best_bound(ScipInfUnclamp(SCIPgetDualbound(scip_)));
915 stats->set_node_count(SCIPgetNTotalNodes(scip_));
916 stats->set_first_lp_relaxation_bound(SCIPgetFirstLPDualboundRoot(scip_));
917 stats->set_root_node_bound(SCIPgetDualboundRoot(scip_));
918 if (stage != SCIP_STAGE_PRESOLVING) {
919 stats->set_total_lp_iterations(SCIPgetNLPIterations(scip_));
920 stats->set_primal_simplex_iterations(SCIPgetNPrimalLPIterations(scip_));
921 stats->set_dual_simplex_iterations(SCIPgetNDualLPIterations(scip_));
922 stats->set_deterministic_time(SCIPgetDeterministicTime(scip_));
924 result.
gscip_output.set_status(ConvertStatus(SCIPgetStatus(scip_)));
932 if (message_handler !=
nullptr) {
941 if (params.has_objective_limit()) {
949 SCIPsetMessagehdlrQuiet(scip_,
false);
950 SCIPsetMessagehdlrLogfile(scip_,
nullptr);
956 const std::string& parameter_name) {
957 SCIP_Bool default_value;
959 SCIPgetBoolParam(scip_, parameter_name.c_str(), &default_value));
960 return static_cast<bool>(default_value);
964 const std::string& parameter_name) {
967 SCIPgetIntParam(scip_, parameter_name.c_str(), &default_value));
968 return default_value;
972 const std::string& parameter_name) {
975 SCIPgetLongintParam(scip_, parameter_name.c_str(), &result));
976 return static_cast<int64_t
>(result);
980 const std::string& parameter_name) {
983 SCIPgetRealParam(scip_, parameter_name.c_str(), &result));
988 const std::string& parameter_name) {
991 SCIPgetCharParam(scip_, parameter_name.c_str(), &result));
996 const std::string& parameter_name) {
999 SCIPgetStringParam(scip_, parameter_name.c_str(), &result));
1000 return std::string(result);
1003absl::StatusOr<double> GScip::ScipInfClamp(
const double d) {
1004 const double kScipInf =
ScipInf();
1005 if (d == std::numeric_limits<double>::infinity()) {
1008 if (d == -std::numeric_limits<double>::infinity()) {
1012 if (d >= kScipInf || d <= -kScipInf) {
1014 << d <<
" is not in SCIP's finite range: (" << -kScipInf <<
", "
1020double GScip::ScipInfUnclamp(
double d) {
1021 const double kScipInf =
ScipInf();
1022 if (d >= kScipInf)
return std::numeric_limits<double>::infinity();
1023 if (d <= -kScipInf)
return -std::numeric_limits<double>::infinity();
1027#undef RETURN_ERROR_UNLESS
#define LOG_IF(severity, condition)
#define ASSIGN_OR_RETURN(lhs, rexpr)
#define RETURN_IF_ERROR(expr)
double LinearConstraintUb(SCIP_CONS *constraint)
absl::Status SetObjectiveOffset(double offset)
absl::StatusOr< SCIP_VAR * > AddVariable(double lb, double ub, double obj_coef, GScipVarType var_type, const std::string &var_name="", const GScipVariableOptions &options=DefaultGScipVariableOptions())
absl::Status DeleteVariable(SCIP_VAR *var)
absl::StatusOr< SCIP_CONS * > AddOrConstraint(const GScipLogicalConstraintData &logical_data, const std::string &name="", const GScipConstraintOptions &options=DefaultGScipConstraintOptions())
absl::StatusOr< double > DefaultRealParamValue(const std::string ¶meter_name)
absl::Status CanSafeBulkDelete(const absl::flat_hash_set< SCIP_VAR * > &vars)
absl::StatusOr< SCIP_CONS * > AddAndConstraint(const GScipLogicalConstraintData &logical_data, const std::string &name="", const GScipConstraintOptions &options=DefaultGScipConstraintOptions())
absl::StatusOr< SCIP_CONS * > AddLinearConstraint(const GScipLinearRange &range, const std::string &name="", const GScipConstraintOptions &options=DefaultGScipConstraintOptions())
bool IsConstraintLinear(SCIP_CONS *constraint)
absl::string_view ConstraintType(SCIP_CONS *constraint)
absl::StatusOr< int64_t > DefaultLongParamValue(const std::string ¶meter_name)
absl::StatusOr< int > DefaultIntParamValue(const std::string ¶meter_name)
absl::Status DeleteConstraint(SCIP_CONS *constraint)
absl::Status SetLinearConstraintUb(SCIP_CONS *constraint, double ub)
absl::Status SafeBulkDelete(const absl::flat_hash_set< SCIP_VAR * > &vars)
static absl::StatusOr< std::unique_ptr< GScip > > Create(const std::string &problem_name)
double ObjCoef(SCIP_VAR *var)
absl::Status SetMaximize(bool is_maximize)
absl::StatusOr< std::string > DefaultStringParamValue(const std::string ¶meter_name)
absl::StatusOr< bool > DefaultBoolParamValue(const std::string ¶meter_name)
absl::StatusOr< SCIP_CONS * > AddIndicatorConstraint(const GScipIndicatorConstraint &indicator_constraint, const std::string &name="", const GScipConstraintOptions &options=DefaultGScipConstraintOptions())
absl::Status SetLb(SCIP_VAR *var, double lb)
absl::Span< SCIP_VAR *const > LinearConstraintVariables(SCIP_CONS *constraint)
double LinearConstraintLb(SCIP_CONS *constraint)
absl::Status SetLinearConstraintCoef(SCIP_CONS *constraint, SCIP_VAR *var, double value)
absl::Status SetLinearConstraintLb(SCIP_CONS *constraint, double lb)
absl::StatusOr< GScipHintResult > SuggestHint(const GScipSolution &partial_solution)
absl::StatusOr< GScipResult > Solve(const GScipParameters ¶ms=GScipParameters(), const std::string &legacy_params="", GScipMessageHandler message_handler=nullptr)
GScipVarType VarType(SCIP_VAR *var)
absl::Status SetVarType(SCIP_VAR *var, GScipVarType var_type)
absl::Status SetBranchingPriority(SCIP_VAR *var, int priority)
absl::StatusOr< char > DefaultCharParamValue(const std::string ¶meter_name)
absl::Span< const double > LinearConstraintCoefficients(SCIP_CONS *constraint)
absl::Status SetUb(SCIP_VAR *var, double ub)
absl::Status SetObjCoef(SCIP_VAR *var, double obj_coef)
absl::StatusOr< SCIP_CONS * > AddSOS2Constraint(const GScipSOSData &sos_data, const std::string &name="", const GScipConstraintOptions &options=DefaultGScipConstraintOptions())
static std::string ScipVersion()
absl::StatusOr< SCIP_CONS * > AddQuadraticConstraint(const GScipQuadraticRange &range, const std::string &name="", const GScipConstraintOptions &options=DefaultGScipConstraintOptions())
absl::string_view Name(SCIP_VAR *var)
absl::Status AddLinearConstraintCoef(SCIP_CONS *constraint, SCIP_VAR *var, double value)
absl::StatusOr< SCIP_CONS * > AddSOS1Constraint(const GScipSOSData &sos_data, const std::string &name="", const GScipConstraintOptions &options=DefaultGScipConstraintOptions())
bool ObjectiveIsMaximize()
#define RETURN_ERROR_UNLESS(x)
absl::StatusOr< MessageHandlerPtr > MakeSCIPMessageHandler(const GScipMessageHandler gscip_message_handler)
std::unique_ptr< SCIP_MESSAGEHDLR, ReleaseSCIPMessageHandler > MessageHandlerPtr
MessageHandlerPtr CaptureMessageHandlerPtr(SCIP_MESSAGEHDLR *const handler)
Collection of objects used to extend the Constraint Solver library.
const GScipConstraintOptions & DefaultGScipConstraintOptions()
std::function< void(GScipMessageType type, absl::string_view message)> GScipMessageHandler
int GScipMaxNumThreads(const GScipParameters ¶meters)
absl::Status LegacyScipSetSolverSpecificParameters(const std::string ¶meters, SCIP *scip)
std::string ProtoEnumToString(ProtoEnumType enum_value)
const GScipVariableOptions & DefaultGScipVariableOptions()
absl::flat_hash_map< SCIP_VAR *, double > GScipSolution
StatusBuilder InternalErrorBuilder()
StatusBuilder InvalidArgumentErrorBuilder()
#define SCIP_TO_STATUS(x)
#define RETURN_IF_SCIP_ERROR(x)
std::vector< double > coefficients
std::vector< SCIP_Var * > variables
SCIP_VAR * indicator_variable
std::vector< SCIP_VAR * > variables
std::vector< double > coefficients
std::vector< SCIP_VAR * > operators
std::vector< SCIP_Var * > quadratic_variables1
std::vector< SCIP_Var * > quadratic_variables2
std::vector< SCIP_Var * > linear_variables
std::vector< double > linear_coefficients
std::vector< double > quadratic_coefficients
absl::flat_hash_map< SCIP_VAR *, double > primal_ray
std::vector< double > objective_values
std::vector< GScipSolution > solutions
std::vector< SCIP_VAR * > variables
std::vector< double > weights
#define OR_ASSIGN_OR_RETURN3(lhs, rexpr, error_expression)