src/work/athos/lp/lp_solver_glpk.h
changeset 1263 a490938ad0aa
equal deleted inserted replaced
-1:000000000000 2:483dea4d403f
       
     1 // -*- c++ -*-
       
     2 #ifndef LEMON_LP_SOLVER_GLPK_H
       
     3 #define LEMON_LP_SOLVER_GLPK_H
       
     4 
       
     5 ///\ingroup misc
       
     6 ///\file
       
     7 
       
     8 extern "C" {
       
     9 #include "glpk.h"
       
    10 }
       
    11 #include <lp_solver_base.h>
       
    12 
       
    13 namespace lemon {
       
    14   
       
    15   
       
    16   template <typename Value>
       
    17   const Value LpSolverBase<Value>::INF=std::numeric_limits<Value>::infinity();
       
    18 
       
    19 
       
    20   /// \brief Wrapper for GLPK solver
       
    21   /// 
       
    22   /// This class implements a lemon wrapper for GLPK.
       
    23   class LpGlpk : public LpSolverBase<double> {
       
    24 
       
    25   public:
       
    26 
       
    27     typedef LpSolverBase<double> Parent;
       
    28     
       
    29     /// \e
       
    30     LPX* lp;
       
    31 
       
    32     /// \e
       
    33     LpGlpk() : Parent(), 
       
    34 			lp(lpx_create_prob()) {
       
    35       //int_row_map.push_back(Row());
       
    36       //int_col_map.push_back(Col());
       
    37       lpx_set_int_parm(lp, LPX_K_DUAL, 1);
       
    38     }
       
    39     /// \e
       
    40     ~LpGlpk() {
       
    41       lpx_delete_prob(lp);
       
    42     }
       
    43 
       
    44     //MATRIX INDEPEDENT MANIPULATING FUNCTIONS
       
    45 
       
    46     /// \e
       
    47     void setMinimize() { 
       
    48       lpx_set_obj_dir(lp, LPX_MIN);
       
    49     }
       
    50     /// \e
       
    51     void setMaximize() { 
       
    52       lpx_set_obj_dir(lp, LPX_MAX);
       
    53     }
       
    54 
       
    55     /// \e
       
    56     /// Retrieve number of rows
       
    57     int rowNum() const { return lpx_get_num_rows(lp); }
       
    58     /// \e
       
    59     /// Retrieve number of coloumns
       
    60     int colNum() const { return lpx_get_num_cols(lp); }
       
    61 
       
    62   protected:
       
    63     /// \e
       
    64     int LpGlpk::_addCol() { 
       
    65       int i=lpx_add_cols(lp, 1);
       
    66       _setColLowerBound(i, -INF);
       
    67       _setColUpperBound(i, INF);
       
    68       return i;
       
    69     }
       
    70     /// \e
       
    71     int LpGlpk::_addRow() { 
       
    72       int i=lpx_add_rows(lp, 1);
       
    73       return i;
       
    74     }
       
    75     /// \e
       
    76     virtual void _setRowCoeffs(int i, 
       
    77 			       const std::vector<std::pair<int, double> >& coeffs) {
       
    78       int mem_length=1+colNum();
       
    79       int* indices = new int[mem_length];
       
    80       double* doubles = new double[mem_length];
       
    81       int length=0;
       
    82       for (std::vector<std::pair<int, double> >::
       
    83 	     const_iterator it=coeffs.begin(); it!=coeffs.end(); ++it) {
       
    84 	++length;
       
    85 	indices[length]=it->first;
       
    86 	doubles[length]=it->second;
       
    87       }
       
    88       lpx_set_mat_row(lp, i, length, indices, doubles);
       
    89       delete [] indices;
       
    90       delete [] doubles;
       
    91     }
       
    92     /// \e
       
    93     virtual void _getRowCoeffs(int i, 
       
    94 			       std::vector<std::pair<int, double> >& coeffs) {
       
    95       int mem_length=1+colNum();
       
    96       int* indices = new int[mem_length];
       
    97       double* doubles = new double[mem_length];
       
    98       int length=lpx_get_mat_row(lp, i, indices, doubles);
       
    99       for (int i=1; i<=length; ++i) {
       
   100 	coeffs.push_back(std::make_pair(indices[i], doubles[i]));
       
   101       }
       
   102       delete [] indices;
       
   103       delete [] doubles;
       
   104     }
       
   105     /// \e
       
   106     virtual void _setColCoeffs(int i, 
       
   107 			       const std::vector<std::pair<int, double> >& coeffs) {
       
   108       int mem_length=1+rowNum();
       
   109       int* indices = new int[mem_length];
       
   110       double* doubles = new double[mem_length];
       
   111       int length=0;
       
   112       for (std::vector<std::pair<int, double> >::
       
   113 	     const_iterator it=coeffs.begin(); it!=coeffs.end(); ++it) {
       
   114 	++length;
       
   115 	indices[length]=it->first;
       
   116 	doubles[length]=it->second;
       
   117       }
       
   118       lpx_set_mat_col(lp, i, length, indices, doubles);
       
   119       delete [] indices;
       
   120       delete [] doubles;
       
   121     }
       
   122     /// \e
       
   123     virtual void _getColCoeffs(int i, 
       
   124 			       std::vector<std::pair<int, double> >& coeffs) {
       
   125       int mem_length=1+rowNum();
       
   126       int* indices = new int[mem_length];
       
   127       double* doubles = new double[mem_length];
       
   128       int length=lpx_get_mat_col(lp, i, indices, doubles);
       
   129       for (int i=1; i<=length; ++i) {
       
   130 	coeffs.push_back(std::make_pair(indices[i], doubles[i]));
       
   131       }
       
   132       delete [] indices;
       
   133       delete [] doubles;
       
   134     }
       
   135     /// \e
       
   136     virtual void _eraseCol(int i) {
       
   137       int cols[2];
       
   138       cols[1]=i;
       
   139       lpx_del_cols(lp, 1, cols);
       
   140     }
       
   141     virtual void _eraseRow(int i) {
       
   142       int rows[2];
       
   143       rows[1]=i;
       
   144       lpx_del_rows(lp, 1, rows);
       
   145     }
       
   146 
       
   147     void _setCoeff(int row, int col, double value) {
       
   148       ///FIXME Of course this is not efficient at all, but GLPK knows not more.
       
   149       /// First approach: get one row, apply changes and set it again
       
   150 
       
   151       int mem_length=1+colNum();
       
   152       int* indices = new int[mem_length];
       
   153       double* doubles = new double[mem_length];
       
   154 
       
   155 
       
   156       int length=lpx_get_mat_row(lp, i, indices, doubles);
       
   157 
       
   158       //The following code does not suppose that the elements of the array indices are sorted
       
   159       int i=1;
       
   160       bool found=false;
       
   161       while (i <= length && !found){
       
   162 	if (indices[i]=col){
       
   163 	  found = true;
       
   164 	  doubles[i]=value;
       
   165 	}
       
   166 	++i;
       
   167       }
       
   168       if (!found){
       
   169 	++length;
       
   170 	indices[length]=col;
       
   171 	doubles[length]=value;
       
   172       }
       
   173 
       
   174 // This is a piece of code that assumes that the array 'indices' is sorted. 
       
   175 // Not ready, I suppose. 
       
   176 //       //We need another arrays to put the data back, anyway
       
   177 //       int* new_indices = new int[length+1];
       
   178 //       double* new_doubles = new double[length+1];
       
   179 //       int offset;
       
   180 
       
   181 //       int i=1;
       
   182 //       while (i <= length && indices[i]<col){
       
   183 // 	new_indiced[i]=indices[i];
       
   184 // 	new_doubles[i]=doubles[i];
       
   185 // 	++i;
       
   186 //       }
       
   187 //       if (i>length || indices[i]>col){
       
   188 // 	//Ha atugrottuk, vagy a vegen van
       
   189 // 	new_indices[i]=col;
       
   190 // 	new_doubles[i]=value;
       
   191 // 	offset = 1;
       
   192 //       }
       
   193 //       else{
       
   194 // 	//I.e.: indices[i]=col
       
   195 // 	new_doubles[i]=value;
       
   196 // 	offset = 0;
       
   197 // 	++i;
       
   198 //       }
       
   199 //       while (i <= length){
       
   200 // 	new_indices[i+offset]=indices[i];
       
   201 // 	new_values[i+offset]=values[i];
       
   202 //       }
       
   203 
       
   204       lpx_set_mat_row(lp, row, length, indices, doubles);
       
   205       delete [] indices;
       
   206       delete [] doubles;
       
   207 
       
   208 //       lpx_set_mat_row(lp, row, length+offset, new_indices, new_doubles);
       
   209 //       delete [] new_indices;
       
   210 //       delete [] new_doubles;
       
   211       
       
   212 
       
   213     }
       
   214     double _getCoeff(int col, int row) {
       
   215       /// FIXME not yet implemented
       
   216       return 0.0;
       
   217     }
       
   218     virtual void _setColLowerBound(int i, double lo) {
       
   219       if (lo==INF) {
       
   220 	//FIXME error
       
   221       }
       
   222       int b=lpx_get_col_type(lp, i);
       
   223       double up=lpx_get_col_ub(lp, i);	
       
   224       if (lo==-INF) {
       
   225 	switch (b) {
       
   226 	case LPX_FR:
       
   227 	case LPX_LO:
       
   228 	  lpx_set_col_bnds(lp, i, LPX_FR, lo, up);
       
   229 	  break;
       
   230 	case LPX_UP:
       
   231 	  break;
       
   232 	case LPX_DB:
       
   233 	case LPX_FX:
       
   234 	  lpx_set_col_bnds(lp, i, LPX_UP, lo, up);
       
   235 	  break;
       
   236 	default: ;
       
   237 	  //FIXME error
       
   238 	}
       
   239       } else {
       
   240 	switch (b) {
       
   241 	case LPX_FR:
       
   242 	case LPX_LO:
       
   243 	  lpx_set_col_bnds(lp, i, LPX_LO, lo, up);
       
   244 	  break;
       
   245 	case LPX_UP:	  
       
   246 	case LPX_DB:
       
   247 	case LPX_FX:
       
   248 	  if (lo==up) 
       
   249 	    lpx_set_col_bnds(lp, i, LPX_FX, lo, up);
       
   250 	  else 
       
   251 	    lpx_set_col_bnds(lp, i, LPX_DB, lo, up);
       
   252 	  break;
       
   253 	default: ;
       
   254 	  //FIXME error
       
   255 	}
       
   256       }
       
   257     }
       
   258     virtual double _getColLowerBound(int i) {
       
   259       int b=lpx_get_col_type(lp, i);
       
   260       switch (b) {
       
   261       case LPX_FR:
       
   262 	return -INF;
       
   263       case LPX_LO:
       
   264 	return lpx_get_col_lb(lp, i);
       
   265       case LPX_UP:
       
   266 	return -INF;
       
   267       case LPX_DB:
       
   268       case LPX_FX:
       
   269 	return lpx_get_col_lb(lp, i);
       
   270       default: ;
       
   271 	//FIXME error
       
   272 	return 0.0;
       
   273       }
       
   274     }
       
   275     virtual void _setColUpperBound(int i, double up) {
       
   276       if (up==-INF) {
       
   277 	//FIXME error
       
   278       }
       
   279       int b=lpx_get_col_type(lp, i);
       
   280       double lo=lpx_get_col_lb(lp, i);
       
   281       if (up==INF) {
       
   282 	switch (b) {
       
   283 	case LPX_FR:
       
   284 	case LPX_LO:
       
   285 	  break;
       
   286 	case LPX_UP:
       
   287 	  lpx_set_col_bnds(lp, i, LPX_FR, lo, up);
       
   288 	  break;
       
   289 	case LPX_DB:
       
   290 	case LPX_FX:
       
   291 	  lpx_set_col_bnds(lp, i, LPX_LO, lo, up);
       
   292 	  break;
       
   293 	default: ;
       
   294 	  //FIXME error
       
   295 	}
       
   296       } else {
       
   297 	switch (b) {
       
   298 	case LPX_FR:
       
   299 	  lpx_set_col_bnds(lp, i, LPX_UP, lo, up);
       
   300 	case LPX_LO:
       
   301 	  if (lo==up) 
       
   302 	    lpx_set_col_bnds(lp, i, LPX_FX, lo, up);
       
   303 	  else
       
   304 	    lpx_set_col_bnds(lp, i, LPX_DB, lo, up);
       
   305 	  break;
       
   306 	case LPX_UP:
       
   307 	  lpx_set_col_bnds(lp, i, LPX_UP, lo, up);
       
   308 	  break;
       
   309 	case LPX_DB:
       
   310 	case LPX_FX:
       
   311 	  if (lo==up) 
       
   312 	    lpx_set_col_bnds(lp, i, LPX_FX, lo, up);
       
   313 	  else 
       
   314 	    lpx_set_col_bnds(lp, i, LPX_DB, lo, up);
       
   315 	  break;
       
   316 	default: ;
       
   317 	  //FIXME error
       
   318 	}
       
   319       }
       
   320     }
       
   321     virtual double _getColUpperBound(int i) {
       
   322       int b=lpx_get_col_type(lp, i);
       
   323       switch (b) {
       
   324       case LPX_FR:
       
   325       case LPX_LO:
       
   326 	return INF;
       
   327       case LPX_UP:
       
   328       case LPX_DB:
       
   329       case LPX_FX:
       
   330 	return lpx_get_col_ub(lp, i);
       
   331       default: ;
       
   332 	//FIXME error
       
   333 	return 0.0;
       
   334       }
       
   335     }
       
   336     virtual void _setRowLowerBound(int i, double lo) {
       
   337       if (lo==INF) {
       
   338 	//FIXME error
       
   339       }
       
   340       int b=lpx_get_row_type(lp, i);
       
   341       double up=lpx_get_row_ub(lp, i);	
       
   342       if (lo==-INF) {
       
   343 	switch (b) {
       
   344 	case LPX_FR:
       
   345 	case LPX_LO:
       
   346 	  lpx_set_row_bnds(lp, i, LPX_FR, lo, up);
       
   347 	  break;
       
   348 	case LPX_UP:
       
   349 	  break;
       
   350 	case LPX_DB:
       
   351 	case LPX_FX:
       
   352 	  lpx_set_row_bnds(lp, i, LPX_UP, lo, up);
       
   353 	  break;
       
   354 	default: ;
       
   355 	  //FIXME error
       
   356 	}
       
   357       } else {
       
   358 	switch (b) {
       
   359 	case LPX_FR:
       
   360 	case LPX_LO:
       
   361 	  lpx_set_row_bnds(lp, i, LPX_LO, lo, up);
       
   362 	  break;
       
   363 	case LPX_UP:	  
       
   364 	case LPX_DB:
       
   365 	case LPX_FX:
       
   366 	  if (lo==up) 
       
   367 	    lpx_set_row_bnds(lp, i, LPX_FX, lo, up);
       
   368 	  else 
       
   369 	    lpx_set_row_bnds(lp, i, LPX_DB, lo, up);
       
   370 	  break;
       
   371 	default: ;
       
   372 	  //FIXME error
       
   373 	}
       
   374       }
       
   375     }
       
   376     virtual double _getRowLowerBound(int i) {
       
   377       int b=lpx_get_row_type(lp, i);
       
   378       switch (b) {
       
   379       case LPX_FR:
       
   380 	return -INF;
       
   381       case LPX_LO:
       
   382 	return lpx_get_row_lb(lp, i);
       
   383       case LPX_UP:
       
   384 	return -INF;
       
   385       case LPX_DB:
       
   386       case LPX_FX:
       
   387 	return lpx_get_row_lb(lp, i);
       
   388       default: ;
       
   389 	//FIXME error
       
   390 	return 0.0;
       
   391       }
       
   392     }
       
   393     virtual void _setRowUpperBound(int i, double up) {
       
   394       if (up==-INF) {
       
   395 	//FIXME error
       
   396       }
       
   397       int b=lpx_get_row_type(lp, i);
       
   398       double lo=lpx_get_row_lb(lp, i);
       
   399       if (up==INF) {
       
   400 	switch (b) {
       
   401 	case LPX_FR:
       
   402 	case LPX_LO:
       
   403 	  break;
       
   404 	case LPX_UP:
       
   405 	  lpx_set_row_bnds(lp, i, LPX_FR, lo, up);
       
   406 	  break;
       
   407 	case LPX_DB:
       
   408 	case LPX_FX:
       
   409 	  lpx_set_row_bnds(lp, i, LPX_LO, lo, up);
       
   410 	  break;
       
   411 	default: ;
       
   412 	  //FIXME error
       
   413 	}
       
   414       } else {
       
   415 	switch (b) {
       
   416 	case LPX_FR:
       
   417 	  lpx_set_row_bnds(lp, i, LPX_UP, lo, up);
       
   418 	case LPX_LO:
       
   419 	  if (lo==up) 
       
   420 	    lpx_set_row_bnds(lp, i, LPX_FX, lo, up);
       
   421 	  else
       
   422 	    lpx_set_row_bnds(lp, i, LPX_DB, lo, up);
       
   423 	  break;
       
   424 	case LPX_UP:
       
   425 	  lpx_set_row_bnds(lp, i, LPX_UP, lo, up);
       
   426 	  break;
       
   427 	case LPX_DB:
       
   428 	case LPX_FX:
       
   429 	  if (lo==up) 
       
   430 	    lpx_set_row_bnds(lp, i, LPX_FX, lo, up);
       
   431 	  else 
       
   432 	    lpx_set_row_bnds(lp, i, LPX_DB, lo, up);
       
   433 	  break;
       
   434 	default: ;
       
   435 	  //FIXME error
       
   436 	}
       
   437       }
       
   438     }
       
   439     virtual double _getRowUpperBound(int i) {
       
   440       int b=lpx_get_row_type(lp, i);
       
   441       switch (b) {
       
   442       case LPX_FR:
       
   443       case LPX_LO:
       
   444 	return INF;
       
   445       case LPX_UP:
       
   446       case LPX_DB:
       
   447       case LPX_FX:
       
   448 	return lpx_get_row_ub(lp, i);
       
   449       default: ;
       
   450 	//FIXME error
       
   451 	return 0.0;
       
   452       }
       
   453     }
       
   454     /// \e
       
   455     virtual double _getObjCoeff(int i) { 
       
   456       return lpx_get_obj_coef(lp, i);
       
   457     }
       
   458     /// \e
       
   459     virtual void _setObjCoeff(int i, double obj_coef) { 
       
   460       lpx_set_obj_coef(lp, i, obj_coef);
       
   461     }
       
   462 
       
   463   protected:
       
   464     virtual double _getPrimal(int i) {
       
   465       return lpx_get_col_prim(lp, i);
       
   466     }
       
   467 
       
   468     //MIP
       
   469   protected:
       
   470     /// \e
       
   471     void _setColCont(int i) { lpx_set_col_kind(lp, i, LPX_CV); }
       
   472     /// \e
       
   473     void _setColInt(int i) { lpx_set_col_kind(lp, i, LPX_IV); }
       
   474     /// \e
       
   475     double _getMIPPrimal(int i) { return lpx_mip_col_val(lp, i); }
       
   476 
       
   477 
       
   478 //   public:
       
   479 //     /// \e
       
   480 //     void solveSimplex() { lpx_simplex(lp); }
       
   481 //     /// \e
       
   482 //     void solvePrimalSimplex() { lpx_simplex(lp); }
       
   483 //     /// \e
       
   484 //     void solveDualSimplex() { lpx_simplex(lp); }
       
   485 //     /// \e
       
   486 //     double getObjVal() { return lpx_get_obj_val(lp); }
       
   487 //     /// \e
       
   488 //     int warmUp() { return lpx_warm_up(lp); }
       
   489 //     /// \e
       
   490 //     void printWarmUpStatus(int i) {
       
   491 //       switch (i) {
       
   492 //       case LPX_E_OK: cout << "LPX_E_OK" << endl; break;
       
   493 //       case LPX_E_EMPTY: cout << "LPX_E_EMPTY" << endl; break;	
       
   494 //       case LPX_E_BADB: cout << "LPX_E_BADB" << endl; break;
       
   495 //       case LPX_E_SING: cout << "LPX_E_SING" << endl; break;
       
   496 //       }
       
   497 //     }
       
   498 //     /// \e
       
   499 //     int getPrimalStatus() { return lpx_get_prim_stat(lp); }
       
   500 //     /// \e
       
   501 //     void printPrimalStatus(int i) {
       
   502 //       switch (i) {
       
   503 //       case LPX_P_UNDEF: cout << "LPX_P_UNDEF" << endl; break;
       
   504 //       case LPX_P_FEAS: cout << "LPX_P_FEAS" << endl; break;	
       
   505 //       case LPX_P_INFEAS: cout << "LPX_P_INFEAS" << endl; break;
       
   506 //       case LPX_P_NOFEAS: cout << "LPX_P_NOFEAS" << endl; break;
       
   507 //       }
       
   508 //     }
       
   509 //     /// \e
       
   510 //     int getDualStatus() { return lpx_get_dual_stat(lp); }
       
   511 //     /// \e
       
   512 //     void printDualStatus(int i) {
       
   513 //       switch (i) {
       
   514 //       case LPX_D_UNDEF: cout << "LPX_D_UNDEF" << endl; break;
       
   515 //       case LPX_D_FEAS: cout << "LPX_D_FEAS" << endl; break;	
       
   516 //       case LPX_D_INFEAS: cout << "LPX_D_INFEAS" << endl; break;
       
   517 //       case LPX_D_NOFEAS: cout << "LPX_D_NOFEAS" << endl; break;
       
   518 //       }
       
   519 //     }
       
   520 //     /// Returns the status of the slack variable assigned to row \c row.
       
   521 //     int getRowStat(const Row& row) { 
       
   522 //       return lpx_get_row_stat(lp, row_iter_map[row]); 
       
   523 //     }
       
   524 //     /// \e
       
   525 //     void printRowStatus(int i) {
       
   526 //       switch (i) {
       
   527 //       case LPX_BS: cout << "LPX_BS" << endl; break;
       
   528 //       case LPX_NL: cout << "LPX_NL" << endl; break;	
       
   529 //       case LPX_NU: cout << "LPX_NU" << endl; break;
       
   530 //       case LPX_NF: cout << "LPX_NF" << endl; break;
       
   531 //       case LPX_NS: cout << "LPX_NS" << endl; break;
       
   532 //       }
       
   533 //     }
       
   534 //     /// Returns the status of the variable assigned to column \c col.
       
   535 //     int getColStat(const Col& col) { 
       
   536 //       return lpx_get_col_stat(lp, col_iter_map[col]); 
       
   537 //     }
       
   538 //     /// \e
       
   539 //     void printColStatus(int i) {
       
   540 //       switch (i) {
       
   541 //       case LPX_BS: cout << "LPX_BS" << endl; break;
       
   542 //       case LPX_NL: cout << "LPX_NL" << endl; break;	
       
   543 //       case LPX_NU: cout << "LPX_NU" << endl; break;
       
   544 //       case LPX_NF: cout << "LPX_NF" << endl; break;
       
   545 //       case LPX_NS: cout << "LPX_NS" << endl; break;
       
   546 //       }
       
   547 //     }
       
   548 
       
   549 //     // MIP
       
   550 //     /// \e
       
   551 //     void solveBandB() { lpx_integer(lp); }
       
   552 //     /// \e
       
   553 //     void setLP() { lpx_set_class(lp, LPX_LP); }
       
   554 //     /// \e
       
   555 //     void setMIP() { lpx_set_class(lp, LPX_MIP); }
       
   556 
       
   557 
       
   558 
       
   559   };
       
   560   
       
   561   /// @}
       
   562 
       
   563 } //namespace lemon
       
   564 
       
   565 #endif //LEMON_LP_SOLVER_GLPK_H