-
Notifications
You must be signed in to change notification settings - Fork 9
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
WIP: Feature/xpress solver for mathopt #137
base: main
Are you sure you want to change the base?
Changes from 11 commits
c648ee7
073e8e3
6a6d3c0
95357b3
19805c0
d15a1b6
d825a91
d0502ea
dc317c0
62e6b26
38a13ba
1fb56d8
c9419e7
0ae4346
63a2a87
823a95d
b3d57f8
127e691
f3ffa94
6796e77
46e755a
1637a76
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,263 @@ | ||
// Copyright 2010-2024 Google LLC | ||
// Licensed under the Apache License, Version 2.0 (the "License"); | ||
// you may not use this file except in compliance with the License. | ||
// You may obtain a copy of the License at | ||
// | ||
// http://www.apache.org/licenses/LICENSE-2.0 | ||
// | ||
// Unless required by applicable law or agreed to in writing, software | ||
// distributed under the License is distributed on an "AS IS" BASIS, | ||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
// See the License for the specific language governing permissions and | ||
// limitations under the License. | ||
|
||
#include "ortools/math_opt/solvers/xpress/g_xpress.h" | ||
|
||
#include <memory> | ||
#include <optional> | ||
#include <string> | ||
#include <utility> | ||
#include <vector> | ||
|
||
#include "absl/log/check.h" | ||
#include "absl/log/die_if_null.h" | ||
#include "absl/memory/memory.h" | ||
#include "absl/status/status.h" | ||
#include "absl/status/statusor.h" | ||
#include "absl/strings/str_format.h" | ||
#include "absl/types/span.h" | ||
#include "ortools/base/logging.h" | ||
#include "ortools/base/source_location.h" | ||
#include "ortools/base/status_builder.h" | ||
#include "ortools/base/status_macros.h" | ||
#include "ortools/xpress/environment.h" | ||
|
||
namespace operations_research::math_opt { | ||
constexpr int kXpressOk = 0; | ||
|
||
absl::Status Xpress::ToStatus(const int xprs_err, | ||
const absl::StatusCode code) const { | ||
if (xprs_err == kXpressOk) { | ||
return absl::OkStatus(); | ||
} | ||
char errmsg[512]; | ||
XPRSgetlasterror(xpress_model_, errmsg); | ||
return util::StatusBuilder(code) | ||
<< "Xpress error code: " << xprs_err << ", message: " << errmsg; | ||
} | ||
|
||
Xpress::Xpress(XPRSprob& model) : xpress_model_(ABSL_DIE_IF_NULL(model)) {} | ||
|
||
absl::StatusOr<std::unique_ptr<Xpress>> Xpress::New( | ||
const std::string& model_name) { | ||
bool correctlyLoaded = initXpressEnv(); | ||
CHECK(correctlyLoaded); | ||
XPRSprob* model; | ||
CHECK_EQ(kXpressOk, XPRScreateprob(model)); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This looks wrong to me.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We should try to only DCHECK if there is a performance reason to do so. In the common case, it is better to either just CHECK, or propagate a status error here. At least that is the convention we attempt to follow throughout the mathopt codebase (we are not 100% consistent). Do we expect XPRSprob* to be nullptr if XPRScreateprob returns OK? It seems like things are already pretty busted in this case and we can just CHECK. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. you are right, I guess this is redundant and with no actual debug value. I'll remove it. |
||
DCHECK(model != nullptr); // should not be NULL if status=0 | ||
CHECK_EQ(kXpressOk, XPRSaddcbmessage(*model, printXpressMessage, NULL, 0)); | ||
return absl::WrapUnique(new Xpress(*model)); | ||
} | ||
|
||
void XPRS_CC Xpress::printXpressMessage(XPRSprob prob, void* data, | ||
const char* sMsg, int nLen, | ||
int nMsgLvl) { | ||
if (sMsg) { | ||
std::cout << sMsg << std::endl; | ||
} | ||
} | ||
|
||
Xpress::~Xpress() { | ||
CHECK_EQ(kXpressOk, XPRSdestroyprob(xpress_model_)); | ||
CHECK_EQ(kXpressOk, XPRSfree()); | ||
} | ||
|
||
absl::Status Xpress::AddVars(const absl::Span<const double> obj, | ||
const absl::Span<const double> lb, | ||
const absl::Span<const double> ub, | ||
const absl::Span<const char> vtype) { | ||
return AddVars({}, {}, {}, obj, lb, ub, vtype); | ||
} | ||
|
||
absl::Status Xpress::AddVars(const absl::Span<const int> vbegin, | ||
const absl::Span<const int> vind, | ||
const absl::Span<const double> vval, | ||
const absl::Span<const double> obj, | ||
const absl::Span<const double> lb, | ||
const absl::Span<const double> ub, | ||
const absl::Span<const char> vtype) { | ||
const int num_vars = static_cast<int>(lb.size()); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. we are not quite ready yet, but we are working up to int64 support (models with more than 2**31 variables). Not sure if xpress supports this, but if you do, maybe use int64_t instead of int here. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. XPRESS does support this but through separate API calls of course (in this instance, XPRSaddcols64 instead of XPRSaddcols). If it's OK for you, I'll add a TODO here to look into it, and add support for int64 later (it will need some reflexion on how to support both 32 and 64 archs, maybe we'll wait for you to do it for gurobi and then copy) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Xpress supports only 64bit counts of non-zeros. Number of variables, rows, etc. are all limited to 32bit signed integers. Since you are not adding any non-zeros here, the code is already safe for the kind of 64bit stuff that Xpress supports. |
||
if (vind.size() != vval.size() || ub.size() != num_vars || | ||
vtype.size() != num_vars || (!obj.empty() && obj.size() != num_vars) || | ||
(!vbegin.empty() && vbegin.size() != num_vars)) { | ||
return absl::InvalidArgumentError( | ||
"Xpress::AddVars arguments are of inconsistent sizes"); | ||
} | ||
double* c_obj = nullptr; | ||
if (!obj.empty()) { | ||
c_obj = const_cast<double*>(obj.data()); | ||
} | ||
// TODO: look into int64 support for number of vars (use XPRSaddcols64) | ||
return ToStatus(XPRSaddcols(xpress_model_, num_vars, 0, c_obj, nullptr, | ||
nullptr, nullptr, lb.data(), ub.data())); | ||
} | ||
|
||
absl::Status Xpress::AddConstrs(const absl::Span<const char> sense, | ||
const absl::Span<const double> rhs, | ||
const absl::Span<const double> rng) { | ||
const int num_cons = static_cast<int>(sense.size()); | ||
if (rhs.size() != num_cons) { | ||
return absl::InvalidArgumentError( | ||
"RHS must have one element per constraint."); | ||
} | ||
return ToStatus(XPRSaddrows(xpress_model_, num_cons, 0, sense.data(), | ||
rhs.data(), rng.data(), NULL, NULL, NULL)); | ||
} | ||
|
||
absl::Status Xpress::SetObjective(bool maximize, double offset, | ||
const absl::Span<const int> colind, | ||
const absl::Span<const double> values) { | ||
RETURN_IF_ERROR(ToStatus(XPRSchgobjsense( | ||
xpress_model_, maximize ? XPRS_OBJ_MAXIMIZE : XPRS_OBJ_MINIMIZE))) | ||
<< "Failed to change objective sense in XPRESS"; | ||
|
||
static int indexes[1] = {-1}; | ||
double xprs_values[1] = {-offset}; | ||
RETURN_IF_ERROR(ToStatus(XPRSchgobj(xpress_model_, 1, indexes, xprs_values))) | ||
<< "Failed to set objective offset in XPRESS"; | ||
|
||
const int n_cols = static_cast<int>(colind.size()); | ||
auto c_colind = const_cast<int*>(colind.data()); | ||
auto c_values = const_cast<double*>(values.data()); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why |
||
return ToStatus(XPRSchgobj(xpress_model_, n_cols, c_colind, c_values)); | ||
} | ||
|
||
absl::Status Xpress::ChgCoeffs(absl::Span<const int> rowind, | ||
absl::Span<const int> colind, | ||
absl::Span<const double> values) { | ||
const int n_coefs = static_cast<int>(rowind.size()); | ||
auto c_rowind = const_cast<int*>(rowind.data()); | ||
auto c_colind = const_cast<int*>(colind.data()); | ||
auto c_values = const_cast<double*>(values.data()); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Again, no need to cast away |
||
return ToStatus( | ||
XPRSchgmcoef(xpress_model_, n_coefs, c_rowind, c_colind, c_values)); | ||
} | ||
|
||
absl::StatusOr<int> Xpress::LpOptimizeAndGetStatus(std::string flags) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Is this something that is required by ORtools? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't think that solving MIPs as LPs is important. But we need to be able to set the LP algorithm to be used, and LPSTATUS to be set in order to have details about the LP solve status. Is that possible with XPRSoptimize() ? |
||
RETURN_IF_ERROR(ToStatus(XPRSlpoptimize(xpress_model_, flags.c_str()))) | ||
<< "XPRESS LP solve failed"; | ||
int xpress_status; | ||
RETURN_IF_ERROR( | ||
ToStatus(XPRSgetintattrib(xpress_model_, XPRS_LPSTATUS, &xpress_status))) | ||
<< "Could not get XPRESS status"; | ||
return xpress_status; | ||
} | ||
absl::Status Xpress::PostSolve() { | ||
return ToStatus(XPRSpostsolve(xpress_model_)); | ||
} | ||
|
||
absl::StatusOr<int> Xpress::MipOptimizeAndGetStatus() { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Same comment as for the LP optimizing function above. |
||
RETURN_IF_ERROR(ToStatus(XPRSmipoptimize(xpress_model_, nullptr))) | ||
<< "XPRESS MIP solve failed"; | ||
int xpress_status; | ||
RETURN_IF_ERROR( | ||
ToStatus(XPRSgetintattrib(xpress_model_, XPRS_MIPSTATUS, &xpress_status))) | ||
<< "Could not get XPRESS status"; | ||
return xpress_status; | ||
} | ||
|
||
void Xpress::Terminate() { XPRSinterrupt(xpress_model_, XPRS_STOP_USER); }; | ||
|
||
absl::StatusOr<int> Xpress::GetIntAttr(int attribute) const { | ||
int result; | ||
RETURN_IF_ERROR(ToStatus(XPRSgetintattrib(xpress_model_, attribute, &result))) | ||
<< "Error getting Xpress int attribute: " << attribute; | ||
return result; | ||
} | ||
|
||
absl::Status Xpress::SetIntAttr(int attribute, int value) { | ||
return ToStatus(XPRSsetintcontrol(xpress_model_, attribute, value)); | ||
} | ||
|
||
absl::StatusOr<double> Xpress::GetDoubleAttr(int attribute) const { | ||
double result; | ||
RETURN_IF_ERROR(ToStatus(XPRSgetdblattrib(xpress_model_, attribute, &result))) | ||
<< "Error getting Xpress double attribute: " << attribute; | ||
return result; | ||
} | ||
|
||
absl::StatusOr<std::vector<double>> Xpress::GetPrimalValues() const { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. In which context is this function supposed to be called? There are only very special situations (inside callbacks) in which you would want to use |
||
int nVars = GetNumberOfVariables(); | ||
XPRSgetintattrib(xpress_model_, XPRS_COLS, &nVars); | ||
std::vector<double> values(nVars); | ||
RETURN_IF_ERROR(ToStatus( | ||
XPRSgetlpsol(xpress_model_, values.data(), nullptr, nullptr, nullptr))) | ||
<< "Error getting Xpress LP solution"; | ||
return values; | ||
} | ||
|
||
int Xpress::GetNumberOfConstraints() const { | ||
int n; | ||
XPRSgetintattrib(xpress_model_, XPRS_ROWS, &n); | ||
return n; | ||
} | ||
|
||
int Xpress::GetNumberOfVariables() const { | ||
int n; | ||
XPRSgetintattrib(xpress_model_, XPRS_COLS, &n); | ||
return n; | ||
} | ||
|
||
absl::StatusOr<std::vector<double>> Xpress::GetConstraintDuals() const { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. XPRSgetduals() is the preferred way to query duals. |
||
int nCons = GetNumberOfConstraints(); | ||
double values[nCons]; | ||
RETURN_IF_ERROR( | ||
ToStatus(XPRSgetlpsol(xpress_model_, nullptr, nullptr, values, nullptr))) | ||
<< "Failed to retrieve LP solution from XPRESS"; | ||
std::vector<double> result(values, values + nCons); | ||
return result; | ||
} | ||
absl::StatusOr<std::vector<double>> Xpress::GetReducedCostValues() const { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. XPRSgetredcosts() is the preferred way to query reduced costs. |
||
int nVars = GetNumberOfVariables(); | ||
double values[nVars]; | ||
RETURN_IF_ERROR( | ||
ToStatus(XPRSgetlpsol(xpress_model_, nullptr, nullptr, nullptr, values))) | ||
<< "Failed to retrieve LP solution from XPRESS"; | ||
std::vector<double> result(values, values + nVars); | ||
return result; | ||
} | ||
|
||
absl::Status Xpress::GetBasis(std::vector<int>& rowBasis, | ||
std::vector<int>& colBasis) const { | ||
rowBasis.resize(GetNumberOfConstraints()); | ||
colBasis.resize(GetNumberOfVariables()); | ||
return ToStatus( | ||
XPRSgetbasis(xpress_model_, rowBasis.data(), colBasis.data())); | ||
} | ||
|
||
absl::Status Xpress::SetStartingBasis(std::vector<int>& rowBasis, | ||
std::vector<int>& colBasis) const { | ||
return ToStatus( | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Maybe assert that |
||
XPRSloadbasis(xpress_model_, rowBasis.data(), colBasis.data())); | ||
} | ||
|
||
absl::StatusOr<std::vector<double>> Xpress::GetVarLb() const { | ||
int nVars = GetNumberOfVariables(); | ||
std::vector<double> bounds; | ||
bounds.reserve(nVars); | ||
RETURN_IF_ERROR( | ||
ToStatus(XPRSgetlb(xpress_model_, bounds.data(), 0, nVars - 1))) | ||
<< "Failed to retrieve variable LB from XPRESS"; | ||
return bounds; | ||
} | ||
absl::StatusOr<std::vector<double>> Xpress::GetVarUb() const { | ||
int nVars = GetNumberOfVariables(); | ||
std::vector<double> bounds; | ||
bounds.reserve(nVars); | ||
RETURN_IF_ERROR( | ||
ToStatus(XPRSgetub(xpress_model_, bounds.data(), 0, nVars - 1))) | ||
<< "Failed to retrieve variable UB from XPRESS"; | ||
return bounds; | ||
} | ||
|
||
} // namespace operations_research::math_opt |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In theory
XPRSgetlasterror()
can fail. To make this rock solid I suggest to check the return value of that function and seterrmsg
to some generic string in case the function returns non-zero.