//-----------------------------------------------------------------------------
// name: OSI Interface for CPLEX
// author: Tobias Pfender
// Konrad-Zuse-Zentrum Berlin (Germany)
// email: pfender@zib.de
// date: 09/25/2000
// license: this file may be freely distributed under the terms of EPL
// comments: please scan this file for '???' and read the comments
//-----------------------------------------------------------------------------
// Copyright (C) 2000, Tobias Pfender, International Business Machines
// Corporation and others. All Rights Reserved.
#ifndef OsiCpxSolverInterface_H
#define OsiCpxSolverInterface_H
#include "OsiSolverInterface.hpp"
#include "CoinWarmStartBasis.hpp"
#include "OsiColCut.hpp"
#include "OsiRowCut.hpp"
#include "OsiCpxConfig.h"
typedef struct cpxlp *CPXLPptr;
typedef struct cpxenv *CPXENVptr;
/** CPLEX Solver Interface
Instantiation of OsiCpxSolverInterface for CPLEX
*/
class OSICPXLIB_EXPORT OsiCpxSolverInterface : virtual public OsiSolverInterface {
friend void OsiCpxSolverInterfaceUnitTest(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;
// Set mipstart option (pass column solution to CPLEX before MIP start)
void setMipStart(bool value) { domipstart = value; }
// Get mipstart option value
bool getMipStart() const { return domipstart; }
//@}
//---------------------------------------------------------------------------
///@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;
//@}
//---------------------------------------------------------------------------
/**@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 CoinBigIndex 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 CoinBigIndex *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 CoinBigIndex *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);
using OsiSolverInterface::readMps;
/** 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 CPLEX specific public interfaces */
//@{
/** Get pointer to CPLEX 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
};
CPXLPptr getLpPtr(int keepCached = KEEPCACHED_NONE);
//@{
/// Method to access CPLEX environment pointer
CPXENVptr getEnvironmentPtr();
//@}
/// return a vector of variable types (continous, binary, integer)
const char *getCtype() const;
/**@name Constructors and destructor */
//@{
/// Default Constructor
OsiCpxSolverInterface();
/// Clone
virtual OsiSolverInterface *clone(bool copyData = true) const;
/// Copy constructor
OsiCpxSolverInterface(const OsiCpxSolverInterface &);
/// Assignment operator
OsiCpxSolverInterface &operator=(const OsiCpxSolverInterface &rhs);
/// Destructor
virtual ~OsiCpxSolverInterface();
/// Resets as if default constructor
virtual void reset();
//@}
/***************************************************************************/
/**@name OsiSimplexInterface methods
Cplex adds a slack with coeff +1 in "<=" and "=" constraints,
with coeff -1 in ">=",
slack being non negative. We switch in order to get a "Clp tableau"
where all the slacks have coefficient +1 in the original tableau.
If a slack for ">=" is non basic, invB is not changed;
column of the slack in the optimal tableau is flipped.
If a slack for ">=" is basic, corresp. row of invB is flipped;
whole row of the optimal tableau is flipped;
then whole column for the slack in opt tableau is flipped.
Ranged rows are not supported. It might work, but no garantee is given.
Code implemented only for Cplex9.0 and higher, lower version number of
Cplex will abort the code.
*/
//@{
/** Returns 1 if can just do getBInv etc
2 if has all OsiSimplex methods
and 0 if it has none */
virtual int canDoSimplexInterface() const;
using OsiSolverInterface::enableSimplexInterface;
/** Useless function, defined only for compatibility with
OsiSimplexInterface
*/
virtual void enableSimplexInterface(int doingPrimal) {};
/** Useless function, defined only for compatibility with
OsiSimplexInterface
*/
virtual void disableSimplexInterface() {};
/** Useless function, defined only for compatibility with
OsiSimplexInterface
*/
virtual void enableFactorization() const {};
/** Useless function, defined only for compatibility with
OsiSimplexInterface
*/
virtual void disableFactorization() const {};
///Returns true if a basis is available
virtual bool basisIsAvailable() const;
/** Returns a basis status of the structural/artificial variables
At present as warm start i.e 0: free, 1: basic, 2: upper, 3: lower
*/
virtual void getBasisStatus(int *cstat, int *rstat) const;
///Get a row of the tableau (slack part in slack if not NULL)
virtual void getBInvARow(int row, double *z, double *slack = NULL) const;
///Get a row of the basis inverse
virtual void getBInvRow(int row, double *z) const;
///Get a column of the tableau
virtual void getBInvACol(int col, double *vec) const;
///Get a column of the basis inverse
virtual void getBInvCol(int col, double *vec) const;
/** Get indices of the pivot variable in each row
(order of indices corresponds to the
order of elements in a vector retured by getBInvACol() and
getBInvCol()).
*/
virtual void getBasics(int *index) const;
/// switches CPLEX to prob type LP
void switchToLP();
/// switches CPLEX to prob type MIP
void switchToMIP();
//@}
/***************************************************************************/
protected:
/// Get LP Pointer for const methods
CPXLPptr getMutableLpPtr() const;
/// Get Environment Pointer for const methods
CPXENVptr getMutableEnvironmentPtr() const;
/**@name Protected methods */
//@{
/// Apply a row cut. Return true if cut was applied.
virtual void applyRowCut(const OsiRowCut &rc);
/** Apply a column cut (bound adjustment).
Return true if cut was applied.
*/
virtual void applyColCut(const OsiColCut &cc);
//@}
private:
/**@name Private static class functions */
//@{
/// resizes coltype_ vector to be able to store at least minsize elements
void resizeColType(int minsize);
/// frees colsize_ vector
void freeColType();
//@}
/**@name Private methods */
//@{
/// The real work of a copy constructor (used by copy and assignment)
void gutsOfCopy(const OsiCpxSolverInterface &source);
/// The real work of the constructor
void gutsOfConstructor();
/// The real work of the destructor
void gutsOfDestructor();
/// free cached column rim vectors
void freeCachedColRim();
/// free cached row rim vectors
void freeCachedRowRim();
/// free cached result vectors
void freeCachedResults();
/// free cached matrices
void freeCachedMatrix();
/// free all cached data (except specified entries, see getLpPtr())
void freeCachedData(int keepCached = KEEPCACHED_NONE);
/// free all allocated memory
void freeAllMemory();
//@}
/**@name Private member data */
//@{
/// CPLEX environment used in this class instance
mutable CPXENVptr env_;
/// CPLEX model represented by this class instance
mutable CPXLPptr lp_;
/// Hotstart information
int *hotStartCStat_;
int hotStartCStatSize_;
int *hotStartRStat_;
int hotStartRStatSize_;
int hotStartMaxIteration_;
/**@name Cached information derived from the CPLEX model */
//@{
/// Pointer to objective vector
mutable double *obj_;
/// Pointer to dense vector of variable lower bounds
mutable double *collower_;
/// Pointer to dense vector of variable lower bounds
mutable double *colupper_;
/// Pointer to dense vector of row sense indicators
mutable char *rowsense_;
/// Pointer to dense vector of row right-hand side values
mutable double *rhs_;
/// Pointer to dense vector of slack upper bounds for range constraints (undefined for non-range rows)
mutable double *rowrange_;
/// Pointer to dense vector of row lower bounds
mutable double *rowlower_;
/// Pointer to dense vector of row upper bounds
mutable double *rowupper_;
/// Pointer to primal solution vector
mutable double *colsol_;
/// Pointer to dual solution vector
mutable double *rowsol_;
/// Pointer to reduced cost vector
mutable double *redcost_;
/// Pointer to row activity (slack) vector
mutable double *rowact_;
/// Pointer to row-wise copy of problem matrix coefficients.
mutable CoinPackedMatrix *matrixByRow_;
/// Pointer to row-wise copy of problem matrix coefficients.
mutable CoinPackedMatrix *matrixByCol_;
//@}
/**@name Additional information needed for storing MIP problems */
//@{
/// Pointer to dense vector of variable types (continous, binary, integer)
char *coltype_;
/// Size of allocated memory for coltype_
int coltypesize_;
/// Stores whether CPLEX' prob type is currently set to MIP
mutable bool probtypemip_;
/// Whether to pass a column solution to CPLEX before starting MIP solve (copymipstart)
bool domipstart;
/// Whether to disable use of advanced basis (if given)
bool disableadvbasis;
//@}
};
#endif
/* vi: softtabstop=2 shiftwidth=2 expandtab tabstop=2
*/