// Osi interface for Mosek ver. 5.0
// Lower versions are not supported
//-----------------------------------------------------------------------------
// name: OSI Interface for MOSEK
// author: Bo Jensen
// email: support@MOSEK.com
//-----------------------------------------------------------------------------
// This code is licensed under the terms of the Eclipse Public License (EPL).
#ifndef OsiMskSolverInterface_H
#define OsiMskSolverInterface_H
#include "OsiSolverInterface.hpp"
#include "OsiMskConfig.h"
typedef void* MSKtask_t;
typedef void* MSKenv_t;
/* MOSEK Solver Interface
Instantiation of OsiMskSolverInterface for MOSEK
*/
class OSIMSKLIB_EXPORT OsiMskSolverInterface : virtual public OsiSolverInterface {
friend void OsiMskSolverInterfaceUnitTest(const std::string & mpsDir, const std::string & netlibDir);
public:
//---------------------------------------------------------------------------
/**@name Solve methods */
//@{
/// Solve initial LP relaxation
virtual void initialSolve();
/// Resolve an LP relaxation after problem modification
virtual void resolve();
/// Invoke solver's built-in enumeration algorithm
virtual void branchAndBound();
//@}
//---------------------------------------------------------------------------
/**@name Parameter set/get methods
The set methods return true if the parameter was set to the given value,
false otherwise. There can be various reasons for failure: the given
parameter is not applicable for the solver (e.g., refactorization
frequency for the volume algorithm), the parameter is not yet implemented
for the solver or simply the value of the parameter is out of the range
the solver accepts. If a parameter setting call returns false check the
details of your solver.
The get methods return true if the given parameter is applicable for the
solver and is implemented. In this case the value of the parameter is
returned in the second argument. Otherwise they return false.
*/
//@{
// Set an integer parameter
bool setIntParam(OsiIntParam key, int value);
// Set an double parameter
bool setDblParam(OsiDblParam key, double value);
// Set a string parameter
bool setStrParam(OsiStrParam key, const std::string & value);
// Get an integer parameter
bool getIntParam(OsiIntParam key, int& value) const;
// Get an double parameter
bool getDblParam(OsiDblParam key, double& value) const;
// Get a string parameter
bool getStrParam(OsiStrParam key, std::string& value) const;
//@}
//---------------------------------------------------------------------------
///@name Methods returning info on how the solution process terminated
//@{
/// Are there a numerical difficulties?
virtual bool isAbandoned() const;
/// Is optimality proven?
virtual bool isProvenOptimal() const;
/// Is primal infeasiblity proven?
virtual bool isProvenPrimalInfeasible() const;
/// Is dual infeasiblity proven?
virtual bool isProvenDualInfeasible() const;
/// Is the given primal objective limit reached?
virtual bool isPrimalObjectiveLimitReached() const;
/// Is the given dual objective limit reached?
virtual bool isDualObjectiveLimitReached() const;
/// Iteration limit reached?
virtual bool isIterationLimitReached() const;
/// Has there been a license problem?
virtual bool isLicenseError() const;
/// Get rescode return of last Mosek optimizer call
int getRescode() const { return Mskerr; }
//@}
//---------------------------------------------------------------------------
/**@name WarmStart related methods */
/*! \brief Get an empty warm start object
This routine returns an empty CoinWarmStartBasis object. Its purpose is
to provide a way to give a client a warm start basis object of the
appropriate type, which can resized and modified as desired.
*/
CoinWarmStart* getEmptyWarmStart () const;
//@{
/// Get warmstarting information
virtual CoinWarmStart* getWarmStart() const;
/** Set warmstarting information. Return true/false depending on whether
the warmstart information was accepted or not. */
virtual bool setWarmStart(const CoinWarmStart* warmstart);
//@}
//---------------------------------------------------------------------------
/**@name Hotstart related methods (primarily used in strong branching).
The user can create a hotstart (a snapshot) of the optimization process
then reoptimize over and over again always starting from there.
NOTE: between hotstarted optimizations only
bound changes are allowed. */
//@{
/// Create a hotstart point of the optimization process
virtual void markHotStart();
/// Optimize starting from the hotstart
virtual void solveFromHotStart();
/// Delete the snapshot
virtual void unmarkHotStart();
//@}
//---------------------------------------------------------------------------
/**@name Problem information methods
These methods call the solver's query routines to return
information about the problem referred to by the current object.
Querying a problem that has no data associated with it result in
zeros for the number of rows and columns, and NULL pointers from
the methods that return vectors.
Const pointers returned from any data-query method are valid as
long as the data is unchanged and the solver is not called.
*/
//@{
/**@name Methods related to querying the input data */
//@{
/// Get number of columns
virtual int getNumCols() const;
/// Get number of rows
virtual int getNumRows() const;
/// Get number of nonzero elements
virtual int getNumElements() const;
/// Get pointer to array[getNumCols()] of column lower bounds
virtual const double * getColLower() const;
/// Get pointer to array[getNumCols()] of column upper bounds
virtual const double * getColUpper() const;
/** Get pointer to array[getNumRows()] of row constraint senses.
setColLower()
and
setColUpper()
*/
virtual void setColBounds( int elementIndex,
double lower, double upper );
/** Set the bounds on a number of columns simultaneouslysetCollower()
and
setColupper()
over and over again.
@param [indexfirst,indexLast]
contains the indices of
the constraints whose either bound changes
@param boundList the new lower/upper bound pairs for the variables
*/
virtual void setColSetBounds(const int* indexFirst,
const int* indexLast,
const double* boundList);
/** Set a single row lower boundsetRowLower()
and
setRowUpper()
*/
virtual void setRowBounds( int elementIndex,
double lower, double upper );
/** Set the type of a single rowsetRowLower()
and
setRowUpper()
over and over again.
@param [indexfirst,indexLast]
contains the indices of
the constraints whose either bound changes
@param boundList the new lower/upper bound pairs for the constraints
*/
virtual void setRowSetBounds(const int* indexFirst,
const int* indexLast,
const double* boundList);
/** Set the type of a number of rows simultaneouslysetRowType()
and
over and over again.
@param [indexfirst,indexLast]
contains the indices of
the constraints whose type changes
@param senseList the new senses
@param rhsList the new right hand sides
@param rangeList the new ranges
*/
virtual void setRowSetTypes(const int* indexFirst,
const int* indexLast,
const char* senseList,
const double* rhsList,
const double* rangeList);
//@}
//-------------------------------------------------------------------------
/**@name Integrality related changing methods */
//@{
/** Set the index-th variable to be a continuous variable */
virtual void setContinuous(int index);
/** Set the index-th variable to be an integer variable */
virtual void setInteger(int index);
/** Set the variables listed in indices (which is of length len) to be
continuous variables */
virtual void setContinuous(const int* indices, int len);
/** Set the variables listed in indices (which is of length len) to be
integer variables */
virtual void setInteger(const int* indices, int len);
//@}
//-------------------------------------------------------------------------
/// Set objective function sense (1 for min (default), -1 for max,)
virtual void setObjSense(double s);
/** Set the primal solution column values
colsol[numcols()] is an array of values of the problem column
variables. These values are copied to memory owned by the
solver object or the solver. They will be returned as the
result of colsol() until changed by another call to
setColsol() or by a call to any solver routine. Whether the
solver makes use of the solution in any way is
solver-dependent.
*/
virtual void setColSolution(const double * colsol);
/** Set dual solution vector
rowprice[numrows()] is an array of values of the problem row
dual variables. These values are copied to memory owned by the
solver object or the solver. They will be returned as the
result of rowprice() until changed by another call to
setRowprice() or by a call to any solver routine. Whether the
solver makes use of the solution in any way is
solver-dependent.
*/
virtual void setRowPrice(const double * rowprice);
//-------------------------------------------------------------------------
/**@name Methods to expand a problem.effectiveness >= effectivenessLb
are applied.
effectiveness < effectivenessLb
colub
: all columns have upper bound infinity
collb
: all columns have lower bound 0
rowub
: all rows have upper bound infinity
rowlb
: all rows have lower bound -infinity
obj
: all variables have 0 objective coefficient
delete
and delete[]
functions.
*/
virtual void assignProblem(CoinPackedMatrix*& matrix,
double*& collb, double*& colub, double*& obj,
double*& rowlb, double*& rowub);
/** Load in an problem by copying the arguments (the constraints on the
rows are given by sense/rhs/range triplets). If a pointer is 0 then the
following values are the default:
colub
: all columns have upper bound infinity
collb
: all columns have lower bound 0
obj
: all variables have 0 objective coefficient
rowsen
: all rows are >=
rowrhs
: all right hand sides are 0
rowrng
: 0 for the ranged rows
delete
and delete[]
functions.
*/
virtual void assignProblem(CoinPackedMatrix*& matrix,
double*& collb, double*& colub, double*& obj,
char*& rowsen, double*& rowrhs,
double*& rowrng);
/** Just like the other loadProblem() methods except that the matrix is
given in a standard column major ordered format (without gaps). */
virtual void loadProblem(const int numcols, const int numrows,
const int* start, const int* index,
const double* value,
const double* collb, const double* colub,
const double* obj,
const double* rowlb, const double* rowub);
/** Just like the other loadProblem() methods except that the matrix is
given in a standard column major ordered format (without gaps). */
virtual void loadProblem(const int numcols, const int numrows,
const int* start, const int* index,
const double* value,
const double* collb, const double* colub,
const double* obj,
const char* rowsen, const double* rowrhs,
const double* rowrng);
/** Read an mps file from the given filename */
virtual int readMps(const char *filename,
const char *extension = "mps");
/** Write the problem into an mps file of the given filename.
If objSense is non zero then -1.0 forces the code to write a
maximization objective and +1.0 to write a minimization one.
If 0.0 then solver can do what it wants */
virtual void writeMps(const char *filename,
const char *extension = "mps",
double objSense=0.0) const;
//@}
/**@name Message handling */
//@{
/** Pass in a message handler
It is the client's responsibility to destroy a message handler installed
by this routine; it will not be destroyed when the solver interface is
destroyed.
*/
void passInMessageHandler(CoinMessageHandler * handler);
//@}
//---------------------------------------------------------------------------
/**@name MOSEK specific public interfaces */
//@{
/** Get pointer to MOSEK model and free all specified cached data entries
(combined with logical or-operator '|' ):
*/
enum keepCachedFlag
{
/// discard all cached data (default)
KEEPCACHED_NONE = 0,
/// column information: objective values, lower and upper bounds, variable types
KEEPCACHED_COLUMN = 1,
/// row information: right hand sides, ranges and senses, lower and upper bounds for row
KEEPCACHED_ROW = 2,
/// problem matrix: matrix ordered by column and by row
KEEPCACHED_MATRIX = 4,
/// LP solution: primal and dual solution, reduced costs, row activities
KEEPCACHED_RESULTS = 8,
/// only discard cached LP solution
KEEPCACHED_PROBLEM = KEEPCACHED_COLUMN | KEEPCACHED_ROW | KEEPCACHED_MATRIX,
/// keep all cached data (similar to getMutableLpPtr())
KEEPCACHED_ALL = KEEPCACHED_PROBLEM | KEEPCACHED_RESULTS,
/// free only cached column and LP solution information
FREECACHED_COLUMN = KEEPCACHED_PROBLEM & ~KEEPCACHED_COLUMN,
/// free only cached row and LP solution information
FREECACHED_ROW = KEEPCACHED_PROBLEM & ~KEEPCACHED_ROW,
/// free only cached matrix and LP solution information
FREECACHED_MATRIX = KEEPCACHED_PROBLEM & ~KEEPCACHED_MATRIX,
/// free only cached LP solution information
FREECACHED_RESULTS = KEEPCACHED_ALL & ~KEEPCACHED_RESULTS
};
MSKtask_t getLpPtr( int keepCached = KEEPCACHED_NONE );
//@{
/// Method to access MOSEK environment pointer
MSKenv_t getEnvironmentPtr();
//@}
/// return a vector of variable types (continous, binary, integer)
const char* getCtype() const;
/**@name Static instance counter methods */
/** MOSEK has a context which must be created prior to all other MOSEK calls.
This method: