SoPlex Doxygen Documentation
spxbasis.cpp
Go to the documentation of this file.
1 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2 /* */
3 /* This file is part of the class library */
4 /* SoPlex --- the Sequential object-oriented simPlex. */
5 /* */
6 /* Copyright (C) 1996-2012 Konrad-Zuse-Zentrum */
7 /* fuer Informationstechnik Berlin */
8 /* */
9 /* SoPlex is distributed under the terms of the ZIB Academic Licence. */
10 /* */
11 /* You should have received a copy of the ZIB Academic License */
12 /* along with SoPlex; see the file COPYING. If not email to soplex@zib.de. */
13 /* */
14 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
15 
16 //#define DEBUGGING 1
17 
18 #include <assert.h>
19 #include <cstdio>
20 #include <iostream>
21 #include <iomanip>
22 #include <sstream>
23 
24 #include "spxdefines.h"
25 #include "spxbasis.h"
26 #include "didxset.h"
27 #include "dvector.h"
28 #include "spxsolver.h"
29 #include "mpsinput.h"
30 #include "spxout.h"
31 #include "exceptions.h"
32 
33 namespace soplex
34 {
37 {
38  METHOD( "SPxBasis::dualStatus()" );
39  return dualColStatus(static_cast<SPxLP*>(theLP)->number(id));
40 }
41 
44 {
45  METHOD( "SPxBasis::dualStatus()" );
46  return dualRowStatus((static_cast<SPxLP*>(theLP))->number(id));
47 }
48 
51 {
52  METHOD( "SPxBasis::dualRowStatus()" );
53  assert(theLP != 0);
54 
55  if (theLP->rhs(i) < infinity)
56  {
57  if (theLP->lhs(i) > -infinity)
58  {
59  if (theLP->lhs(i) == theLP->rhs(i))
60  return Desc::D_FREE;
61  else
62  return Desc::D_ON_BOTH;
63  }
64  else
65  return Desc::D_ON_LOWER;
66  }
67  else if (theLP->lhs(i) > -infinity)
68  return Desc::D_ON_UPPER;
69  else
70  return Desc::D_UNDEFINED;
71 }
72 
75 {
76  METHOD( "SPxBasis::dualColStatus()" );
77  assert(theLP != 0);
78 
79  if (theLP->SPxLP::upper(i) < infinity)
80  {
81  if (theLP->SPxLP::lower(i) > -infinity)
82  {
83  if (theLP->SPxLP::lower(i) == theLP->SPxLP::upper(i))
84  return Desc::D_FREE;
85  else
86  return Desc::D_ON_BOTH;
87  }
88  else
89  return Desc::D_ON_LOWER;
90  }
91  else if (theLP->SPxLP::lower(i) > -infinity)
92  return Desc::D_ON_UPPER;
93  else
94  return Desc::D_UNDEFINED;
95 }
96 
98 {
99  METHOD( "SPxBasis::loadMatrixVecs()" );
100  assert(theLP != 0);
101  assert(theLP->dim() == matrix.size());
102 
103  MSG_INFO3( spxout << "IBASIS01 loadMatrixVecs() invalidates factorization"
104  << std::endl; )
105 
106  int i;
107  nzCount = 0;
108  for (i = theLP->dim() - 1; i >= 0; --i)
109  {
110  matrix[i] = &theLP->vector(baseId(i));
111  nzCount += matrix[i]->size();
112  }
113  matrixIsSetup = true;
114  factorized = false;
115  if (factor != 0)
116  factor->clear();
117 }
118 
120 {
121  METHOD( "SPxBasis::isDescValid()" );
122 
123  assert(status() > NO_PROBLEM);
124  assert(theLP != 0);
125 
126  int basisdim;
127 
128  if ( ds.nRows() != theLP->nRows() || ds.nCols() != theLP->nCols() )
129  {
130  MSG_DEBUG( spxout << "IBASIS20 Dimension mismatch\n" );
131  return false;
132  }
133 
134  basisdim = 0;
135  for ( int row = ds.nRows()-1; row >= 0; --row )
136  {
137  if ( ds.rowstat[row] >= 0 )
138  {
139  if ( ds.rowstat[row] != dualRowStatus(row) )
140  {
141  MSG_DEBUG( spxout << "IBASIS21 Basic row " << row << " with incorrect dual status " << dualRowStatus(row) << "\n");
142  return false;
143  }
144  }
145  else
146  {
147  basisdim++;
148  if ( (ds.rowstat[row] == Desc::P_FIXED && theLP->SPxLP::lhs(row) != theLP->SPxLP::rhs(row))
149  || (ds.rowstat[row] == Desc::P_ON_UPPER && theLP->SPxLP::rhs(row) >= infinity)
150  || (ds.rowstat[row] == Desc::P_ON_LOWER && theLP->SPxLP::lhs(row) <= -infinity) )
151  {
152  MSG_DEBUG( spxout << "IBASIS22 Nonbasic row with incorrect status: lhs=" << theLP->SPxLP::lhs(row) << ", rhs=" << theLP->SPxLP::rhs(row) << ", stat=" << ds.rowstat[row] << "\n");
153  return false;
154  }
155  }
156  }
157 
158  for ( int col = ds.nCols()-1; col >= 0; --col )
159  {
160  if ( ds.colstat[col] >= 0 )
161  {
162  if ( ds.colstat[col] != dualColStatus(col) )
163  {
164  MSG_DEBUG( spxout << "IBASIS23 Basic column " << col << " with incorrect dual status " << ds.colstat[col] << " != " << dualColStatus(col) << "\n");
165  return false;
166  }
167  }
168  else
169  {
170  basisdim++;
171  if ( (ds.colstat[col] == Desc::P_FIXED && theLP->SPxLP::lower(col) != theLP->SPxLP::upper(col))
172  || (ds.colstat[col] == Desc::P_ON_UPPER && theLP->SPxLP::upper(col) >= infinity)
173  || (ds.colstat[col] == Desc::P_ON_LOWER && theLP->SPxLP::lower(col) <= -infinity) )
174  {
175  MSG_DEBUG( spxout << "IBASIS24 Nonbasic column with incorrect status: lower=" << theLP->SPxLP::lower(col) << ", upper=" << theLP->SPxLP::upper(col) << ", stat=" << ds.colstat[col] << "\n");
176  return false;
177  }
178  }
179  }
180 
181  if ( basisdim != theLP->nCols() )
182  {
183  MSG_DEBUG( spxout << "IBASIS25 Incorrect basis dimension " << basisdim << " != " << theLP->nCols() << "\n" );
184  return false;
185  }
186 
187  // basis descriptor valid
188  return true;
189 }
190 
191 /*
192  Loading a #Desc# into the basis can be done more efficiently, by
193  explicitely programming both cases, for the rowwise and for the columnwise
194  representation. This implementation hides this distingtion in the use of
195  methods #isBasic()# and #vector()#.
196  */
197 void SPxBasis::loadDesc(const Desc& ds)
198 {
199  METHOD( "SPxBasis::loadDesc()" );
200  assert(status() > NO_PROBLEM);
201  assert(theLP != 0);
202  assert(ds.nRows() == theLP->nRows());
203  assert(ds.nCols() == theLP->nCols());
204 
205  SPxId none;
206  int i;
207  int j;
208 
209  MSG_INFO3( spxout << "IBASIS02 loading of Basis invalidates factorization"
210  << std::endl; )
211 
212  lastin = none;
213  lastout = none;
214  lastidx = -1;
215  iterCount = 0;
216  updateCount = 0;
217 
218  if (&ds != &thedesc)
219  {
220  thedesc = ds;
221  setRep();
222  }
223 
224  assert(theLP->dim() == matrix.size());
225 
226  // MSG_DEBUG( dump(); )
227 
228  nzCount = 0;
229  for (j = i = 0; i < theLP->nRows(); ++i)
230  {
231  /* for columns and rows with D_... status, the correct D_... status depends on bounds and sides; if a basis
232  * descriptor is loaded after changing bounds or sides, e.g. in the refine() method, we have to correct them
233  */
234  if (thedesc.rowStatus(i) >= 0)
236  else if (thedesc.rowStatus(i) == SPxBasis::Desc::P_FIXED && theLP->SPxLP::lhs(i) != theLP->SPxLP::rhs(i))
237  {
238  if (theLP->SPxLP::lhs(i) > -infinity)
240  else if (theLP->SPxLP::rhs(i) < infinity)
242  else
244  }
245 
246  if (theLP->isBasic(thedesc.rowStatus(i)))
247  {
248  SPxRowId id = theLP->SPxLP::rId(i);
249  theBaseId[j] = id;
250  matrix[j] = &theLP->vector(id);
251  nzCount += matrix[j++]->size();
252  }
253  }
254 
255  for (i = 0; i < theLP->nCols(); ++i)
256  {
257  /* for columns and rows with D_... status, the correct D_... status depends on bounds and sides; if a basis
258  * descriptor is loaded after changing bounds or sides, e.g. in the refine() method, we have to correct them
259  */
260  if (thedesc.colStatus(i) >= 0)
262  else if (thedesc.colStatus(i) == SPxBasis::Desc::P_FIXED && theLP->SPxLP::lower(i) != theLP->SPxLP::upper(i))
263  {
264  if (theLP->SPxLP::lower(i) <= -infinity && theLP->SPxLP::upper(i) >= infinity)
266  else if (theLP->SPxLP::upper(i) >= infinity || (theLP->SPxLP::lower(i) > -infinity && theLP->SPxLP::maxObj(i) < 0.0))
268  else
270  }
271 
272  if (theLP->isBasic(thedesc.colStatus(i)))
273  {
274  SPxColId id = theLP->SPxLP::cId(i);
275  theBaseId[j] = id;
276  matrix[j] = &theLP->vector(id);
277  nzCount += matrix[j++]->size();
278  }
279  }
280 
281  assert(j == matrix.size());
282  assert(isDescValid(thedesc));
283 
284  matrixIsSetup = true;
285  factorized = false;
286  if (factor != 0)
287  factor->clear();
288 }
289 
291 {
292  METHOD( "SPxBasis::setRep()" );
293  assert(theLP != 0);
294 
295  reDim();
296  minStab = 0.0;
297 
298  if (theLP->rep() == SPxSolver::ROW)
299  {
302  }
303  else
304  {
307  }
308 }
309 
311 {
312  METHOD( "SPxBasis::load()" );
313  assert(lp != 0);
314  theLP = lp;
315 
316  setRep();
317 
318  addedRows(lp->nRows());
319  addedCols(lp->nCols());
320 
322 
323  loadDesc(thedesc);
324 }
325 
326 void SPxBasis::loadSolver(SLinSolver* p_solver, const bool destroy)
327 {
328  METHOD( "SPxBasis::loadSolver()" );
329 
330  assert(!freeSlinSolver || factor != 0);
331 
332  MSG_INFO3( spxout << "IBASIS03 loading of Solver invalidates factorization"
333  << std::endl; )
334 
335 
336  if(freeSlinSolver)
337  {
338  delete factor;
339  factor = 0;
340  }
341 
342  factor = p_solver;
343  factorized = false;
344  factor->clear();
345  freeSlinSolver = destroy;
346 }
347 
348 
349 
350 
351 /**
352  * The specification is taken from the
353  *
354  * ILOG CPLEX 7.0 Reference Manual, Appendix E, Page 543.
355  *
356  * This routine should read valid BAS format files.
357  *
358  * @return true if the file was read correctly.
359  *
360  * Here is a very brief outline of the format:
361  *
362  * The format is in a form similar to an MPS file. The basic assumption is that all (column)
363  * variables are nonbasic at their lower bound and all row (variables) are basic; only the
364  * differences to this rule are given. Each data line contains an indicator, a variable name and
365  * possibly a row/constraint name. The following meaning applies with respect to the indicators:
366  *
367  * - XU: the variable is basic, the row is nonbasic at its upper bound
368  * - XL: the variable is basic, the row is nonbasic at its lower bound
369  * - UL: the variable is nonbasic and at its upper bound
370  * - LL: the variable is nonbasic and at its lower bound
371  *
372  * The CPLEX format contains an additional indicator 'BS', but this is unsupported here.
373  *
374  * Nonbasic variables without lower bound have the following default status for SoPlex:
375  * - at their upper bound if finite,
376  * - at zero if free.
377  */
379  std::istream& is,
380  const NameSet* rowNames,
381  const NameSet* colNames)
382 {
383  METHOD( "SPxBasis::readBasis()" );
384  assert(theLP != 0);
385 
386  /* prepare names */
387  const NameSet* rNames = rowNames;
388  const NameSet* cNames = colNames;
389 
390  if ( colNames == 0 )
391  {
392  int nCols = theLP->nCols();
393  std::stringstream name;
394 
395  NameSet* p_colNames = new NameSet();
396  p_colNames->reMax(nCols);
397  for (int j = 0; j < nCols; ++j)
398  {
399  name << "x" << j;
400  DataKey key = theLP->colId(j);
401  p_colNames->add(key, name.str().c_str());
402  }
403  cNames = p_colNames;
404  }
405 
406  if ( rNames == 0 )
407  {
408  int nRows = theLP->nRows();
409  std::stringstream name;
410 
411  NameSet* p_rowNames = new NameSet();
412  p_rowNames->reMax(nRows);
413  for (int i = 0; i < nRows; ++i)
414  {
415  name << "C" << i;
416  DataKey key = theLP->rowId(i);
417  p_rowNames->add(key, name.str().c_str());
418  }
419  rNames = p_rowNames;
420  }
421 
422  /* load default basis if necessary */
423  if (status() == NO_PROBLEM)
424  load(theLP);
425 
426  /* initialize with standard settings */
427  Desc l_desc(thedesc);
428 
429  for (int i = 0; i < theLP->nRows(); i++)
430  l_desc.rowstat[i] = dualRowStatus(i);
431 
432  for (int i = 0; i < theLP->nCols(); i++)
433  {
434  if (theLP->SPxLP::lower(i) == theLP->SPxLP::upper(i))
435  l_desc.colstat[i] = Desc::P_FIXED;
436  else if (theLP->SPxLP::lower(i) <= -infinity && theLP->SPxLP::upper(i) >= infinity)
437  l_desc.colstat[i] = Desc::P_FREE;
438  else if (theLP->SPxLP::lower(i) <= -infinity)
439  l_desc.colstat[i] = Desc::P_ON_UPPER;
440  else
441  l_desc.colstat[i] = Desc::P_ON_LOWER;
442  }
443 
444  MPSInput mps(is);
445 
446  if (mps.readLine() && (mps.field0() != 0) && !strcmp(mps.field0(), "NAME"))
447  {
448  while (mps.readLine())
449  {
450  int c = -1;
451  int r = -1;
452 
453  if ((mps.field0() != 0) && !strcmp(mps.field0(), "ENDATA"))
454  {
456  break;
457  }
458  if ((mps.field1() == 0) || (mps.field2() == 0))
459  break;
460 
461  if ((c = cNames->number(mps.field2())) < 0)
462  break;
463 
464  if (*mps.field1() == 'X')
465  if (mps.field3() == 0 || (r = rNames->number(mps.field3())) < 0)
466  break;
467 
468  if (!strcmp(mps.field1(), "XU"))
469  {
470  l_desc.colstat[c] = dualColStatus(c);
471  if ( theLP->SPxLP::lhs(r) == theLP->SPxLP::rhs(r) )
472  l_desc.rowstat[r] = Desc::P_FIXED;
473  else
474  l_desc.rowstat[r] = Desc::P_ON_UPPER;
475  }
476  else if (!strcmp(mps.field1(), "XL"))
477  {
478  l_desc.colstat[c] = dualColStatus(c);
479  if ( theLP->SPxLP::lhs(r) == theLP->SPxLP::rhs(r) )
480  l_desc.rowstat[r] = Desc::P_FIXED;
481  else
482  l_desc.rowstat[r] = Desc::P_ON_LOWER;
483  }
484  else if (!strcmp(mps.field1(), "UL"))
485  {
486  l_desc.colstat[c] = Desc::P_ON_UPPER;
487  }
488  else if (!strcmp(mps.field1(), "LL"))
489  {
490  l_desc.colstat[c] = Desc::P_ON_LOWER;
491  }
492  else
493  {
494  mps.syntaxError();
495  break;
496  }
497  }
498  }
499  if (!mps.hasError())
500  {
501  if (mps.section() == MPSInput::ENDATA)
502  {
503  // force basis to be different from NO_PROBLEM
504  // otherwise the basis will be overwritten at later stages.
506  loadDesc(l_desc);
507  }
508  else
509  mps.syntaxError();
510  }
511 
512  if ( rowNames == 0 )
513  delete rNames;
514  if ( colNames == 0 )
515  delete cNames;
516 
517 #ifndef NDEBUG
518  MSG_DEBUG( thedesc.dump() );
519 #endif
520 
521  return !mps.hasError();
522 }
523 
524 
525 /* Get row name - copied from spxmpswrite.cpp
526  *
527  * @todo put this in a common file and unify accross different formats (mps, lp, basis).
528  */
529 static const char* getRowName(
530  const SPxLP* lp,
531  int idx,
532  const NameSet* rnames,
533  char* buf)
534 {
535  assert(buf != 0);
536  assert(idx >= 0);
537  assert(idx < lp->nRows());
538 
539  if (rnames != 0)
540  {
541  DataKey key = lp->rId(idx);
542 
543  if (rnames->has(key))
544  return (*rnames)[key];
545  }
546  std::sprintf(buf, "C%d", idx);
547 
548  return buf;
549 }
550 
551 /* Get column name - copied from spxmpswrite.cpp
552  *
553  * @todo put this in a common file and unify accross different formats (mps, lp, basis).
554  */
555 static const char* getColName(
556  const SPxLP* lp,
557  int idx,
558  const NameSet* cnames,
559  char* buf)
560 {
561  assert(buf != 0);
562  assert(idx >= 0);
563  assert(idx < lp->nCols());
564 
565  if (cnames != 0)
566  {
567  DataKey key = lp->cId(idx);
568 
569  if (cnames->has(key))
570  return (*cnames)[key];
571  }
572  std::sprintf(buf, "x%d", idx);
573 
574  return buf;
575 }
576 
577 /* writes a file in MPS basis format to \p os.
578  *
579  * See SPxBasis::readBasis() for a short description of the format.
580  */
582  std::ostream& os,
583  const NameSet* rowNames,
584  const NameSet* colNames
585  ) const
586 {
587  METHOD( "SPxBasis::writeBasis()" );
588  assert(theLP != 0);
589 
590  os.setf(std::ios::left);
591  os << "NAME soplex.bas\n";
592 
593  /* do not write basis if there is none */
594  if (status() == NO_PROBLEM)
595  {
596  os << "ENDATA" << std::endl;
597  return;
598  }
599 
600  /* start writing */
601  char buf[255];
602  int row = 0;
603  for (int col = 0; col < theLP->nCols(); col++)
604  {
605  if ( thedesc.colStatus(col) > 0 )
606  {
607  /* Find non basic row */
608  for (; row < theLP->nRows(); row++)
609  {
610  if ( thedesc.rowStatus(row) < 0 )
611  break;
612  }
613 
614  assert( row != theLP->nRows() );
615 
616  os << ( thedesc.rowStatus( row ) == Desc::P_ON_UPPER ? " XU " : " XL " )
617  << std::setw(8) << getColName(theLP, col, colNames, buf);
618 
619  /* break in two parts since buf is reused */
620  os << " "
621  << getRowName(theLP, row, rowNames, buf)
622  << std::endl;
623 
624  row++;
625  }
626  else
627  {
628  if ( thedesc.colStatus( col ) == Desc::P_ON_UPPER )
629  {
630  os << " UL "
631  << getColName(theLP, col, colNames, buf)
632  << std::endl;
633  }
634  else
635  {
636  /* Default is all non-basic variables on lower bound (if finite) or at zero (if free).
637  * nothing to do in this case.
638  */
639  assert(theLP->lower(col) <= -infinity || thedesc.colStatus(col) == Desc::P_ON_LOWER || thedesc.colStatus(col) == Desc::P_FIXED);
640  assert(theLP->lower(col) > -infinity || theLP->upper(col) < infinity || thedesc.colStatus(col) == Desc::P_FREE);
641  }
642  }
643  }
644 
645 #ifndef NDEBUG
646  MSG_DEBUG( thedesc.dump() );
647 
648  // Check that we covered all nonbasic rows - the remaining should be basic.
649  for (; row < theLP->nRows(); row++)
650  {
651  if ( thedesc.rowStatus(row) < 0 )
652  break;
653  }
654  assert( row == theLP->nRows() );
655 
656 #endif // NDEBUG
657 
658  os << "ENDATA" << std::endl;
659 }
660 
662 {
663  METHOD( "SPxBasis::printMatrix()" );
664 
665  assert(matrixIsSetup);
666 
667  for( int i = 0; i < matrix.size(); i++ )
668  {
669  spxout << "C" << i << "=" << *matrix[i] << std::endl;
670  }
671 }
672 
673 void SPxBasis::printMatrixMTX(int number)
674 {
675  int dim;
676  int nnz;
677  char filename[30];
678 
679  dim = matrix.size();
680  nnz = nzCount;
681  sprintf(filename, "basis/basis%d.mtx",number);
682  std::cout << "printing basis matrix to file " << filename << "\n";
683  FILE * basisfile;
684  basisfile = fopen (filename,"w");
685  // print marker necessary for reading the file in Matlab
686  fprintf(basisfile, "%%%%MatrixMarket matrix coordinate real general\n");
687  // print matrix information
688  fprintf(basisfile, "%d %d %d\n", dim, dim, nnz );
689  // print matrix data
690  for( int i = 0; i < matrix.size(); ++i )
691  {
692  for( int j = 0; j < baseVec(i).size(); ++j )
693  {
694  int idx = baseVec(i).index(j);
695  Real val = baseVec(i).value(j);
696  fprintf(basisfile, "%d %d %.13e\n",i+1,idx+1,val);
697  }
698  }
699  fclose (basisfile);
700 
701  return;
702 }
703 
705  int i,
706  SPxId& id,
707  const SVector* enterVec,
708  const SSVector* eta)
709 {
710  METHOD( "SPxBasis::change()" );
711 
712  assert(matrixIsSetup);
713  assert(!id.isValid() || (enterVec != 0));
714  assert(factor != 0);
715 
716  lastidx = i;
717  lastin = id;
718 
719  if (id.isValid() && i >= 0)
720  {
721  assert(enterVec != 0);
722 
723  nzCount = nzCount - matrix[i]->size() + enterVec->size();
724  matrix[i] = enterVec;
725  lastout = theBaseId[i];
726  theBaseId[i] = id;
727 
728  ++iterCount;
729  ++updateCount;
730 
731  // never factorize? Just do it !
732  if (!factorized)
733  factorize();
734  // relative fill too high ?
735  else if (Real(factor->memory()) > lastFill * Real(nzCount))
736  {
737  MSG_INFO3( spxout << "IBASIS04 fill factor triggers refactorization"
738  << " memory= " << factor->memory()
739  << " nzCount= " << nzCount
740  << " lastFill= " << lastFill
741  << std::endl; )
742 
743  factorize();
744  }
745  // absolute fill too high ?
746  else if (nzCount > lastNzCount)
747  {
748  MSG_INFO3( spxout << "IBASIS05 nonzero factor triggers refactorization"
749  << " nzCount= " << nzCount
750  << " lastNzCount= " << lastNzCount
751  << " nonzeroFactor= " << nonzeroFactor
752  << std::endl; )
753  factorize();
754  }
755  // too many updates ?
756  else if (updateCount >= maxUpdates)
757  {
758  MSG_INFO3( spxout << "IBASIS06 update count triggers refactorization"
759  << " updateCount= " << updateCount
760  << " maxUpdates= " << maxUpdates
761  << std::endl; )
762  factorize();
763  }
764  else
765  {
766  try
767  {
768 #ifdef MEASUREUPDATETIME
770 #endif
771  factor->change(i, *enterVec, eta);
773 #ifdef MEASUREUPDATETIME
775 #endif
776  }
777  catch( ... )
778  {
779  MSG_INFO3( spxout << "IBASIS13 problems updating factorization; refactorizing basis"
780  << std::endl; )
781 
782 #ifdef MEASUREUPDATETIME
784 #endif
785 
786  // singularity was detected in update; we refactorize
787  factorize();
788 
789  // if factorize() detects singularity, an exception is thrown, hence at this point we have a regular basis
790  // and can try the update again
791  assert(status() >= SPxBasis::REGULAR);
792  try
793  {
794 #ifdef MEASUREUPDATETIME
796 #endif
797  factor->change(i, *enterVec, eta);
799 #ifdef MEASUREUPDATETIME
801 #endif
802  }
803  // with a freshly factorized, regular basis, the update is unlikely to fail; if this happens nevertheless,
804  // we have to invalidate the basis to have the statuses correct
805  catch( SPxException F )
806  {
807  MSG_INFO3( spxout << "IBASIS14 problems updating factorization; invalidating factorization"
808  << std::endl; )
809 
810 #ifdef MEASUREUPDATETIME
812 #endif
813 
814  factorized = false;
815  throw F;
816  }
817  }
818 
819  assert(minStab > 0.0);
820 
822  {
823  MSG_INFO3( spxout << "IBASIS07 stability triggers refactorization"
824  << " stability= " << factor->stability()
825  << " minStab= " << minStab
826  << std::endl; )
827  factorize();
828  }
829  }
830  }
831  else
832  lastout = id;
833 }
834 
836 {
837  METHOD( "SPxBasis::factorize()" );
838 
839  assert(factor != 0);
840 
841  if (!matrixIsSetup)
842  loadDesc(thedesc);
843 
844  assert(matrixIsSetup);
845 
846  updateCount = 0;
847 
848  switch(factor->load(matrix.get_ptr(), matrix.size()))
849  {
850  case SLinSolver::OK :
851  if (status() == SINGULAR)
853 
854  minStab = factor->stability();
855 
856  // This seems allways be about 1e-7
857  if (minStab > 1e-4)
858  minStab *= 0.001;
859  if (minStab > 1e-5)
860  minStab *= 0.01;
861  if (minStab > 1e-6)
862  minStab *= 0.1;
863  break;
864  case SLinSolver::SINGULAR :
866  break;
867  default :
868  MSG_ERROR( spxout << "EBASIS08 error: unknown status of factorization.\n"; )
869  throw SPxInternalCodeException("XBASIS01 This should never happen.");
870  // factorized = false;
871  }
872 
873  lastMem = factor->memory();
874  lastFill = fillFactor * Real(factor->memory()) / Real(nzCount > 0 ? nzCount : 1);
875  lastNzCount = int(nonzeroFactor * Real(nzCount > 0 ? nzCount : 1));
876  factorized = true;
877 }
878 
880 {
881  METHOD( "SPxBasis::multWithBase()" );
882  assert(status() > SINGULAR);
883  assert(theLP->dim() == x.dim());
884 
885  int i;
886  DVector tmp(x);
887 
888  if (!matrixIsSetup)
889  (const_cast<SPxBasis*>(this))->loadDesc(thedesc);
890 
891  assert( matrixIsSetup );
892 
893  for (i = x.dim() - 1; i >= 0; --i)
894  x[i] = *(matrix[i]) * tmp;
895 
896  return x;
897 }
898 
900 {
901  METHOD( "SPxBasis::multBaseWith()" );
902  assert(status() > SINGULAR);
903  assert(theLP->dim() == x.dim());
904 
905  int i;
906  DVector tmp(x);
907 
908  if (!matrixIsSetup)
909  (const_cast<SPxBasis*>(this))->loadDesc(thedesc);
910 
911  assert( matrixIsSetup );
912 
913  x.clear();
914  for (i = x.dim() - 1; i >= 0; --i)
915  {
916  if (tmp[i])
917  x.multAdd(tmp[i], *(matrix[i]));
918  }
919 
920  return x;
921 }
922 
924 {
925  METHOD( "SPxBasis::dump()" );
926  assert(status() > NO_PROBLEM);
927  assert(theLP != 0);
928  assert(thedesc.nRows() == theLP->nRows());
929  assert(thedesc.nCols() == theLP->nCols());
930  assert(theLP->dim() == matrix.size());
931 
932  int i, basesize;
933 
934  // Dump regardless of the verbosity level if this method is called.
935  const SPxOut::Verbosity tmp_verbosity = spxout.getVerbosity();
937 
938  spxout << "DBASIS09 Basis entries:";
939  basesize = 0;
940  for (i = 0; i < theLP->nRows(); ++i)
941  {
942  if (theLP->isBasic(thedesc.rowStatus(i)))
943  {
944  if(basesize % 10 == 0)
945  spxout << std::endl << "DBASIS10 ";
946  SPxRowId id = theLP->SPxLP::rId(i);
947  spxout << "\tR" << theLP->number(id);
948  basesize++;
949  }
950  }
951 
952  for (i = 0; i < theLP->nCols(); ++i)
953  {
954  if (theLP->isBasic(thedesc.colStatus(i)))
955  {
956  if(basesize % 10 == 0)
957  spxout << std::endl << "DBASIS11 ";
958  SPxColId id = theLP->SPxLP::cId(i);
959  spxout << "\tC" << theLP->number(id);
960  basesize++;
961  }
962  }
963  spxout << std::endl;
964 
965  assert(basesize == matrix.size());
966  spxout.setVerbosity( tmp_verbosity );
967 }
968 
969 
971 {
972 #ifdef ENABLE_CONSISTENCY_CHECKS
973  METHOD( "SPxBasis::isConsistent()" );
974  int primals = 0;
975  int i;
976 
977  if (status() > NO_PROBLEM)
978  {
979  if (theLP == 0)
980  return MSGinconsistent("SPxBasis");
981 
982  if (theBaseId.size() != theLP->dim() || matrix.size() != theLP->dim())
983  return MSGinconsistent("SPxBasis");
984 
985  if (thedesc.nCols() != theLP->nCols() || thedesc.nRows() != theLP->nRows())
986  return MSGinconsistent("SPxBasis");
987 
988  for (i = 0; i < thedesc.nRows(); ++i)
989  {
990  if (thedesc.rowStatus(i) >= 0)
991  {
992  if (thedesc.rowStatus(i) != dualRowStatus(i))
993  return MSGinconsistent("SPxBasis");
994  }
995  else
996  ++primals;
997  }
998 
999  for (i = 0; i < thedesc.nCols(); ++i)
1000  {
1001  if (thedesc.colStatus(i) >= 0)
1002  {
1003  if (thedesc.colStatus(i) != dualColStatus(i))
1004  return MSGinconsistent("SPxBasis");
1005  }
1006  else
1007  ++primals;
1008  }
1009  if (primals != thedesc.nCols())
1010  return MSGinconsistent("SPxBasis");
1011  }
1012  return thedesc.isConsistent() && theBaseId.isConsistent()
1013  && matrix.isConsistent() && factor->isConsistent();
1014 #else
1015  return true;
1016 #endif // CONSISTENCY_CHECKS
1017 }
1018 
1020  : theLP (0)
1021  , matrixIsSetup (false)
1022  , factor (0)
1023  , factorized (false)
1024  , maxUpdates (180)
1025  , nonzeroFactor(10.0)
1026  , fillFactor(5.0)
1027  , iterCount (0)
1028  , nzCount (1)
1029  , minStab(0.0)
1030  , thestatus (NO_PROBLEM)
1031  , freeSlinSolver(false)
1032 {
1033  METHOD( "SPxBasis::SPxBasis()" );
1034 
1035  // info: is not consistent at this moment, e.g. because theLP == 0
1036 }
1037 
1038 /**@warning Do not change the LP object.
1039  * Only pointer to that object is copied.
1040  * Hint: no problem, we use this function for copy
1041  * constructor of SPxSolver
1042  */
1044  : theLP(old.theLP)
1045  , theBaseId(old.theBaseId)
1046  , matrix(old.matrix)
1047  , matrixIsSetup(old.matrixIsSetup)
1048  , factor(old.factor)
1049  , factorized(old.factorized)
1050  , maxUpdates(old.maxUpdates)
1051  , nonzeroFactor(old.nonzeroFactor)
1052  , fillFactor(old.fillFactor)
1053  , iterCount(old.iterCount)
1054  , nzCount(old.nzCount)
1055  , lastFill(old.lastFill)
1056  , lastNzCount(old.lastNzCount)
1057  , lastin(old.lastin)
1058  , lastout(old.lastout)
1059  , lastidx(old.lastidx)
1060  , minStab(old.minStab)
1061  , thestatus(old.thestatus)
1062  , thedesc(old.thedesc)
1063 {
1064  METHOD( "SPxBasis::SPxBasis()" );
1065 
1066  this->factor = old.factor->clone();
1067  freeSlinSolver = true;
1068 
1069  assert(SPxBasis::isConsistent());
1070 }
1071 
1073 {
1074  METHOD( "SPxBasis::~SPxBasis()" );
1075 
1076  assert(!freeSlinSolver || factor != 0);
1077 
1078  if(freeSlinSolver)
1079  {
1080  delete factor;
1081  factor = 0;
1082  }
1083 }
1084 
1085 
1086 /**@warning Note that we do not create a deep copy of the corresponding SPxSolver object.
1087  * Only the reference to that object is copied.
1088  */
1090 {
1091  METHOD( "SPxBasis::operator=()" );
1092 
1093  assert(!freeSlinSolver || factor != 0);
1094 
1095  if (this != &rhs)
1096  {
1097  theLP = rhs.theLP;
1098  theBaseId = rhs.theBaseId;
1099  matrix = rhs.matrix;
1101 
1102  if(freeSlinSolver)
1103  {
1104  delete factor;
1105  factor = 0;
1106  }
1107  factor = rhs.factor->clone();
1108  freeSlinSolver = true;
1109 
1110  factorized = rhs.factorized;
1111  maxUpdates = rhs.maxUpdates;
1113  fillFactor = rhs.fillFactor;
1114  iterCount = rhs.iterCount;
1115  nzCount = rhs.nzCount;
1116  lastFill = rhs.lastFill;
1117  lastNzCount = rhs.lastNzCount;
1118  lastin = rhs.lastin;
1119  lastout = rhs.lastout;
1120  lastidx = rhs.lastidx;
1121  minStab = rhs.minStab;
1122  thestatus = rhs.thestatus;
1123  thedesc = rhs.thedesc;
1124 
1125  assert(SPxBasis::isConsistent());
1126  }
1127 
1128  return *this;
1129 }
1130 
1131 
1132 //
1133 // Auxiliary functions.
1134 //
1135 
1136 // Pretty-printing of basis status.
1137 std::ostream& operator<<( std::ostream& os,
1138  const SPxBasis::SPxStatus& status )
1139 {
1140  switch ( status )
1141  {
1142  case SPxBasis::NO_PROBLEM:
1143  os << "NO_PROBLEM";
1144  break;
1145  case SPxBasis::SINGULAR:
1146  os << "SINGULAR";
1147  break;
1148  case SPxBasis::REGULAR:
1149  os << "REGULAR";
1150  break;
1151  case SPxBasis::DUAL:
1152  os << "DUAL";
1153  break;
1154  case SPxBasis::PRIMAL:
1155  os << "PRIMAL";
1156  break;
1157  case SPxBasis::OPTIMAL:
1158  os << "OPTIMAL";
1159  break;
1160  case SPxBasis::UNBOUNDED:
1161  os << "UNBOUNDED";
1162  break;
1163  case SPxBasis::INFEASIBLE:
1164  os << "INFEASIBLE";
1165  break;
1166  default:
1167  os << "?other?";
1168  break;
1169  }
1170  return os;
1171 }
1172 
1173 
1174 } // namespace soplex
1175 
1176 //-----------------------------------------------------------------------------
1177 //Emacs Local Variables:
1178 //Emacs mode:c++
1179 //Emacs c-basic-offset:3
1180 //Emacs tab-width:8
1181 //Emacs indent-tabs-mode:nil
1182 //Emacs End:
1183 //-----------------------------------------------------------------------------