SoPlex Doxygen Documentation
leave.cpp
Go to the documentation of this file.
1 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2 /* */
3 /* This file is part of the class library */
4 /* SsoPlex --- 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 /* Updating the Basis for Leaving Variables
19  */
20 #include <assert.h>
21 #include <stdio.h>
22 
23 #include "spxdefines.h"
24 #include "spxsolver.h"
25 #include "spxratiotester.h"
26 #include "spxout.h"
27 #include "exceptions.h"
28 
29 namespace soplex
30 {
31 static const Real reject_leave_tol = 1e-10; // = LOWSTAB as defined in spxfastrt.cpp
32 
33 /*
34  Vector |fTest| gives the feasibility test of all basic variables. For its
35  computation |fVec|, |theUBbound| and |theLBbound| must be setup correctly.
36  Values of |fTest| $<0$ represent infeasible variables, which are eligible
37  for leaving the basis in the simplex loop.
38  */
40 {
41  METHOD( "SPxSolver::computeFtest()" );
42 
43  assert(type() == LEAVE);
44 
45  Real theeps = entertol();
47  int ninfeasibilities = 0;
48 
49  for( int i = 0; i < dim(); ++i )
50  {
51  theCoTest[i] = ((*theFvec)[i] > theUBbound[i])
52  ? theUBbound[i] - (*theFvec)[i]
53  : (*theFvec)[i] - theLBbound[i];
54 
55  if( remainingRoundsLeave == 0 )
56  {
57  if( theCoTest[i] < -theeps )
58  {
61  isInfeasible[i] = true;
62  ++ninfeasibilities;
63  }
64  else
65  isInfeasible[i] = false;
66  if( ninfeasibilities > sparsityThresholdLeave )
67  {
68  MSG_INFO2( spxout << "ILEAVE05 too many infeasibilities for sparse pricing"
69  << std::endl; )
71  sparsePricingLeave = false;
72  ninfeasibilities = 0;
73  }
74  }
75  }
76 
77  if( ninfeasibilities == 0 && !sparsePricingLeave )
78  {
80  }
81  else if( ninfeasibilities <= sparsityThresholdLeave && !sparsePricingLeave )
82  {
83  std::streamsize prec = spxout.precision();
84  MSG_INFO2( spxout << "ILEAVE04 sparse pricing active, "
85  << "sparsity: "
86  << std::setw(6) << std::fixed << std::setprecision(4)
87  << (Real) ninfeasibilities/dim()
88  << std::scientific << std::setprecision(int(prec))
89  << std::endl; )
90  sparsePricingLeave = true;
91  }
92 }
93 
95 {
96  METHOD( "SPxSolver::updateFtest()" );
97  const IdxSet& idx = theFvec->idx();
98  Vector& ftest = theCoTest; // |== fTest()|
99  assert(&ftest == &fTest());
100 
101  assert(type() == LEAVE);
102 
103  Real theeps = entertol();
104  for (int j = idx.size() - 1; j >= 0; --j)
105  {
106  int i = idx.index(j);
107 
108  ftest[i] = ((*theFvec)[i] > theUBbound[i])
109  ? theUBbound[i] - (*theFvec)[i]
110  : (*theFvec)[i] - theLBbound[i];
111  if( sparsePricingLeave && ftest[i] < -theeps )
112  {
113  assert(remainingRoundsLeave == 0);
114  if( !isInfeasible[i] )
115  {
117  isInfeasible[i] = true;
118  }
119  }
120  }
121 }
122 
123 
124 /* compute statistics on leaving variable
125  Compute a set of statistical values on the variable selected for leaving the
126  basis.
127  */
129  int leaveIdx,
130  SPxBasis::Desc::Status& leaveStat,
131  SPxId& leaveId,
132  Real& leaveMax,
133  Real& leavebound,
134  int& leaveNum)
135 {
136  METHOD( "SPxSolver::getLeaveVals()" );
137  SPxBasis::Desc& ds = desc();
138  leaveId = baseId(leaveIdx);
139 
140  if (leaveId.isSPxRowId())
141  {
142  leaveNum = number(SPxRowId(leaveId));
143  leaveStat = ds.rowStatus(leaveNum);
144 
145  assert(isBasic(leaveStat));
146  switch (leaveStat)
147  {
149  assert( rep() == ROW );
150  ds.rowStatus(leaveNum) = dualRowStatus(leaveNum);
151  leavebound = 0;
152  leaveMax = -infinity;
153  break;
155  assert( rep() == ROW );
156  ds.rowStatus(leaveNum) = dualRowStatus(leaveNum);
157  leavebound = 0;
158  leaveMax = infinity;
159  break;
161  assert( rep() == ROW );
162  throw SPxInternalCodeException("XLEAVE01 This should never happen.");
164  assert( rep() == COLUMN );
165  ds.rowStatus(leaveNum) = SPxBasis::Desc::P_FIXED;
166  assert(lhs(leaveNum) == rhs(leaveNum));
167  leavebound = -rhs(leaveNum);
168  if ((*theFvec)[leaveIdx] < theLBbound[leaveIdx])
169  leaveMax = infinity;
170  else
171  leaveMax = -infinity;
172  break;
174  assert( rep() == COLUMN );
175  ds.rowStatus(leaveNum) = SPxBasis::Desc::P_ON_UPPER;
176  leavebound = -rhs(leaveNum); // slack !!
177  leaveMax = infinity;
178  break;
180  assert( rep() == COLUMN );
181  ds.rowStatus(leaveNum) = SPxBasis::Desc::P_ON_LOWER;
182  leavebound = -lhs(leaveNum); // slack !!
183  leaveMax = -infinity;
184  break;
186  assert( rep() == COLUMN );
187  if ((*theFvec)[leaveIdx] > theLBbound[leaveIdx])
188  {
189  ds.rowStatus(leaveNum) = SPxBasis::Desc::P_ON_LOWER;
190  theLRbound[leaveNum] = -infinity;
191  leavebound = -lhs(leaveNum); // slack !!
192  leaveMax = -infinity;
193  }
194  else
195  {
196  ds.rowStatus(leaveNum) = SPxBasis::Desc::P_ON_UPPER;
197  theURbound[leaveNum] = infinity;
198  leavebound = -rhs(leaveNum); // slack !!
199  leaveMax = infinity;
200  }
201  break;
202 
203  default:
204  throw SPxInternalCodeException("XLEAVE02 This should never happen.");
205  }
206  MSG_DEBUG( spxout << "DLEAVE51 SPxSolver::getLeaveVals() : row " << leaveNum
207  << ": " << leaveStat
208  << " -> " << ds.rowStatus(leaveNum)
209  << std::endl; )
210  }
211 
212  else
213  {
214  assert(leaveId.isSPxColId());
215  leaveNum = number(SPxColId(leaveId));
216  leaveStat = ds.colStatus(leaveNum);
217 
218  assert(isBasic(leaveStat));
219  switch (leaveStat)
220  {
222  assert( rep() == ROW );
223  ds.colStatus(leaveNum) = dualColStatus(leaveNum);
224  leavebound = 0;
225  leaveMax = -infinity;
226  break;
228  assert( rep() == ROW );
229  ds.colStatus(leaveNum) = dualColStatus(leaveNum);
230  leavebound = 0;
231  leaveMax = infinity;
232  break;
234  assert( rep() == ROW );
235  ds.colStatus(leaveNum) = dualColStatus(leaveNum);
236  if ((*theFvec)[leaveIdx] < theLBbound[leaveIdx])
237  {
238  leavebound = theLBbound[leaveIdx];
239  leaveMax = -infinity;
240  }
241  else
242  {
243  leavebound = theUBbound[leaveIdx];
244  leaveMax = infinity;
245  }
246  break;
247 
249  assert( rep() == COLUMN );
250  assert(SPxLP::upper(leaveNum) == SPxLP::lower(leaveNum));
251  ds.colStatus(leaveNum) = SPxBasis::Desc::P_FIXED;
252  leavebound = SPxLP::upper(leaveNum);
253  if ((*theFvec)[leaveIdx] < theLBbound[leaveIdx])
254  leaveMax = infinity;
255  else
256  leaveMax = -infinity;
257  break;
259  assert( rep() == COLUMN );
260  ds.colStatus(leaveNum) = SPxBasis::Desc::P_ON_UPPER;
261  leavebound = SPxLP::upper(leaveNum);
262  leaveMax = -infinity;
263  break;
265  assert( rep() == COLUMN );
266  ds.colStatus(leaveNum) = SPxBasis::Desc::P_ON_LOWER;
267  leavebound = SPxLP::lower(leaveNum);
268  leaveMax = infinity;
269  break;
271  assert( rep() == COLUMN );
272  if ((*theFvec)[leaveIdx] > theUBbound[leaveIdx])
273  {
274  leaveMax = -infinity;
275  leavebound = SPxLP::upper(leaveNum);
276  theLCbound[leaveNum] = -infinity;
277  ds.colStatus(leaveNum) = SPxBasis::Desc::P_ON_UPPER;
278  }
279  else
280  {
281  leaveMax = infinity;
282  leavebound = SPxLP::lower(leaveNum);
283  theUCbound[leaveNum] = infinity;
284  ds.colStatus(leaveNum) = SPxBasis::Desc::P_ON_LOWER;
285  }
286  break;
287  default:
288  throw SPxInternalCodeException("XLEAVE03 This should never happen.");
289  }
290  MSG_DEBUG( spxout << "DLEAVE52 SPxSolver::getLeaveVals() : col " << leaveNum
291  << ": " << leaveStat
292  << " -> " << ds.colStatus(leaveNum)
293  << std::endl; )
294  }
295 }
296 
298  Real leaveMax,
299  SPxId enterId,
300  Real& enterBound,
301  Real& newUBbound,
302  Real& newLBbound,
303  Real& newCoPrhs
304 )
305 {
306  METHOD( "SPxSolver::getLeaveVals2()" );
307  SPxBasis::Desc& ds = desc();
308 
309  enterBound = 0;
310  if (enterId.isSPxRowId())
311  {
312  int idx = number(SPxRowId(enterId));
313  SPxBasis::Desc::Status enterStat = ds.rowStatus(idx);
314 
315  switch (enterStat)
316  {
318  assert(rep() == ROW);
319  if (thePvec->delta()[idx] * leaveMax < 0)
320  newCoPrhs = theLRbound[idx];
321  else
322  newCoPrhs = theURbound[idx];
323  newUBbound = infinity;
324  newLBbound = -infinity;
326  break;
328  assert(rep() == ROW);
329  newUBbound = 0;
330  newLBbound = -infinity;
332  newCoPrhs = theLRbound[idx];
333  break;
335  assert(rep() == ROW);
336  newUBbound = infinity;
337  newLBbound = 0;
339  newCoPrhs = theURbound[idx];
340  break;
342  assert(rep() == ROW);
343  if (leaveMax * thePvec->delta()[idx] < 0)
344  {
345  newUBbound = 0;
346  newLBbound = -infinity;
348  newCoPrhs = theLRbound[idx];
349  }
350  else
351  {
352  newUBbound = infinity;
353  newLBbound = 0;
355  newCoPrhs = theURbound[idx];
356  }
357  break;
358 
360  assert(rep() == COLUMN);
361  ds.rowStatus(idx) = dualRowStatus(idx);
362  if (lhs(idx) > -infinity)
363  theURbound[idx] = theLRbound[idx];
364  newCoPrhs = theLRbound[idx]; // slack !!
365  newUBbound = -lhs(idx);
366  newLBbound = -rhs(idx);
367  enterBound = -rhs(idx);
368  break;
370  assert(rep() == COLUMN);
371  ds.rowStatus(idx) = dualRowStatus(idx);
372  if (rhs(idx) < infinity)
373  theLRbound[idx] = theURbound[idx];
374  newCoPrhs = theURbound[idx]; // slack !!
375  newLBbound = -rhs(idx);
376  newUBbound = -lhs(idx);
377  enterBound = -lhs(idx);
378  break;
380  assert(rep() == COLUMN);
381 #if 1
382  throw SPxInternalCodeException("XLEAVE04 This should never happen.");
383 #else
384  MSG_ERROR( spxout << "ELEAVE53 ERROR: not yet debugged!" << std::endl; )
385  ds.rowStatus(idx) = dualRowStatus(idx);
386  newCoPrhs = theURbound[idx]; // slack !!
387  newUBbound = infinity;
388  newLBbound = -infinity;
389  enterBound = 0;
390 #endif
391  break;
393  assert(rep() == COLUMN);
394  MSG_ERROR( spxout << "ELEAVE54 "
395  << "ERROR! Tried to put a fixed row variable into the basis: "
396  << "idx=" << idx
397  << ", lhs=" << lhs(idx)
398  << ", rhs=" << rhs(idx) << std::endl; )
399  throw SPxInternalCodeException("XLEAVE05 This should never happen.");
400 
401  default:
402  throw SPxInternalCodeException("XLEAVE06 This should never happen.");
403  }
404  MSG_DEBUG( spxout << "DLEAVE55 SPxSolver::getLeaveVals2(): row " << idx
405  << ": " << enterStat
406  << " -> " << ds.rowStatus(idx)
407  << std::endl; )
408  }
409 
410  else
411  {
412  assert(enterId.isSPxColId());
413  int idx = number(SPxColId(enterId));
414  SPxBasis::Desc::Status enterStat = ds.colStatus(idx);
415 
416  switch (enterStat)
417  {
419  assert(rep() == ROW);
420  newUBbound = 0;
421  newLBbound = -infinity;
423  newCoPrhs = theLCbound[idx];
424  break;
426  assert(rep() == ROW);
427  newUBbound = infinity;
428  newLBbound = 0;
430  newCoPrhs = theUCbound[idx];
431  break;
433  assert(rep() == ROW);
434  newUBbound = infinity;
435  newLBbound = -infinity;
436  newCoPrhs = theLCbound[idx];
438  break;
440  assert(rep() == ROW);
441  if (leaveMax * theCoPvec->delta()[idx] < 0)
442  {
443  newUBbound = 0;
444  newLBbound = -infinity;
446  newCoPrhs = theLCbound[idx];
447  }
448  else
449  {
450  newUBbound = infinity;
451  newLBbound = 0;
453  newCoPrhs = theUCbound[idx];
454  }
455  break;
456 
458  assert(rep() == COLUMN);
459  ds.colStatus(idx) = dualColStatus(idx);
460  if (SPxLP::lower(idx) > -infinity)
461  theLCbound[idx] = theUCbound[idx];
462  newCoPrhs = theUCbound[idx];
463  newUBbound = SPxLP::upper(idx);
464  newLBbound = SPxLP::lower(idx);
465  enterBound = SPxLP::upper(idx);
466  break;
468  assert(rep() == COLUMN);
469  ds.colStatus(idx) = dualColStatus(idx);
470  if (SPxLP::upper(idx) < infinity)
471  theUCbound[idx] = theLCbound[idx];
472  newCoPrhs = theLCbound[idx];
473  newUBbound = SPxLP::upper(idx);
474  newLBbound = SPxLP::lower(idx);
475  enterBound = SPxLP::lower(idx);
476  break;
478  assert(rep() == COLUMN);
479  ds.colStatus(idx) = dualColStatus(idx);
480  if (thePvec->delta()[idx] * leaveMax > 0)
481  newCoPrhs = theUCbound[idx];
482  else
483  newCoPrhs = theLCbound[idx];
484  newUBbound = SPxLP::upper(idx);
485  newLBbound = SPxLP::lower(idx);
486  enterBound = 0;
487  break;
489  assert(rep() == COLUMN);
490  MSG_ERROR( spxout << "ELEAVE56 "
491  << "ERROR! Tried to put a fixed column variable into the basis. "
492  << "idx=" << idx
493  << ", lower=" << lower(idx)
494  << ", upper=" << upper(idx) << std::endl; )
495  throw SPxInternalCodeException("XLEAVE07 This should never happen.");
496  default:
497  throw SPxInternalCodeException("XLEAVE08 This should never happen.");
498  }
499 
500  MSG_DEBUG( spxout << "DLEAVE57 SPxSolver::getLeaveVals2(): col " << idx
501  << ": " << enterStat
502  << " -> " << ds.colStatus(idx)
503  << std::endl; )
504  }
505 
506 }
507 
509  int leaveNum,
510  SPxId leaveId,
511  SPxBasis::Desc::Status leaveStat,
512  const SVector* //newVec
513 )
514 {
515  METHOD( "SPxSolver::rejectLeave()" );
516  SPxBasis::Desc& ds = desc();
517  if (leaveId.isSPxRowId())
518  {
519  MSG_DEBUG( spxout << "DLEAVE58 rejectLeave() : row " << leaveNum
520  << ": " << ds.rowStatus(leaveNum)
521  << " -> " << leaveStat << std::endl; )
522 
523  if (leaveStat == SPxBasis::Desc::D_ON_BOTH)
524  {
525  if (ds.rowStatus(leaveNum) == SPxBasis::Desc::P_ON_LOWER)
526  theLRbound[leaveNum] = theURbound[leaveNum];
527  else
528  theURbound[leaveNum] = theLRbound[leaveNum];
529  }
530  ds.rowStatus(leaveNum) = leaveStat;
531  }
532  else
533  {
534  MSG_DEBUG( spxout << "DLEAVE59 rejectLeave() : col " << leaveNum
535  << ": " << ds.colStatus(leaveNum)
536  << " -> " << leaveStat << std::endl; )
537 
538  if (leaveStat == SPxBasis::Desc::D_ON_BOTH)
539  {
540  if (ds.colStatus(leaveNum) == SPxBasis::Desc::P_ON_UPPER)
541  theLCbound[leaveNum] = theUCbound[leaveNum];
542  else
543  theUCbound[leaveNum] = theLCbound[leaveNum];
544  }
545  ds.colStatus(leaveNum) = leaveStat;
546  }
547 }
548 
549 
550 bool SPxSolver::leave(int leaveIdx)
551 {
552  METHOD( "SPxSolver::leave()" );
553  assert(leaveIdx < dim() && leaveIdx >= 0);
554  assert(type() == LEAVE);
555  assert(initialized);
556 
557  bool instable = instableLeave;
558  assert(!instable || instableLeaveNum >= 0);
559 
560  /*
561  Before performing the actual basis update, we must determine, how this
562  is to be accomplished.
563  */
564  if (theCoPvec->delta().isSetup() && theCoPvec->delta().size() == 0)
565  {
566  coSolve(theCoPvec->delta(), unitVecs[leaveIdx]);
567  }
568 #ifdef ENABLE_ADDITIONAL_CHECKS
569  else
570  {
571  SSVector tmp(dim(), epsilon());
572  tmp.clear();
573  coSolve(tmp, unitVecs[leaveIdx]);
574  tmp -= theCoPvec->delta();
575  if (tmp.length() > leavetol()) {
576  // This happens very frequently and does usually not hurt, so print
577  // these warnings only with verbose level INFO2 and higher.
578  MSG_INFO2( spxout << "WLEAVE60 iteration=" << basis().iteration()
579  << ": coPvec.delta error = " << tmp.length()
580  << std::endl; )
581  }
582  }
583 #endif // ENABLE_ADDITIONAL_CHECKS
584 
585  setupPupdate();
586 
587  assert(thePvec->isConsistent());
588  assert(theCoPvec->isConsistent());
589 
590  SPxBasis::Desc::Status leaveStat; // status of leaving var
591  SPxId leaveId; // id of leaving var
592  SPxId none; // invalid id used if leave fails
593  Real leaveMax; // maximium lambda of leaving var
594  Real leavebound; // current fVec value of leaving var
595  int leaveNum; // number of leaveId in bounds
596 
597  getLeaveVals(leaveIdx, leaveStat, leaveId, leaveMax, leavebound, leaveNum);
598 
599  if (m_numCycle > m_maxCycle)
600  {
601  if (leaveMax > 0)
602  perturbMaxLeave();
603  else
604  perturbMinLeave();
605  //@ m_numCycle /= 2;
606  }
607  //@ testBounds();
608 
609  for(;;)
610  {
611  Real enterVal = leaveMax;
612  SPxId enterId = theratiotester->selectEnter(enterVal, leaveIdx);
613 
614  assert(!enterId.isValid() || !isBasic(enterId));
615 
616  instableLeaveNum = -1;
617  instableLeave = false;
618 
619  /*
620  No variable could be selected to enter the basis and even the leaving
621  variable is unbounded.
622  */
623  if (!enterId.isValid())
624  {
625  /* the following line originally was below in "rejecting leave" case;
626  we need it in the unbounded/infeasible case, too, to have the
627  correct basis size */
628  rejectLeave(leaveNum, leaveId, leaveStat);
629  change(-1, none, 0);
630 
631  if (enterVal != leaveMax)
632  {
633  MSG_DEBUG( spxout << "DLEAVE61 rejecting leave A (leaveIdx=" << leaveIdx
634  << ", theCoTest=" << theCoTest[leaveIdx] << ")"
635  << std::endl; )
636 
637  /* In the LEAVE algorithm, when for a selected leaving variable we find only
638  an instable entering variable, then the basis change is not conducted.
639  Instead, we save the leaving variable's index in instableLeaveNum and scale
640  theCoTest[leaveIdx] down by some factor, hoping to find a different leaving
641  variable with a stable entering variable.
642  If this fails, however, and no more leaving variable is found, we have to
643  perform the instable basis change using instableLeaveNum. In this (and only
644  in this) case, the flag instableLeave is set to true.
645 
646  enterVal != leaveMax is the case that selectEnter has found only an instable entering
647  variable. We store this leaving variable for later -- if we are not already in the
648  instable case: then we continue and conclude unboundness/infeasiblity */
649  if (!instable)
650  {
651  instableLeaveNum = leaveIdx;
652 
653  // Note: These changes do not survive a refactorization
654  instableLeaveVal = theCoTest[leaveIdx];
655  theCoTest[leaveIdx] = 0.0;
656 
657  return true;
658  }
659  }
660 
661  if (lastUpdate() > 1)
662  {
663  MSG_INFO3( spxout << "ILEAVE01 factorization triggered in "
664  << "leave() for feasibility test" << std::endl; )
665  factorize();
666 
667  /* after a factorization, the leaving column/row might not be infeasible or suboptimal anymore, hence we do
668  * not try to call leave(leaveIdx), but rather return to the main solving loop and call the pricer again
669  */
670  return true;
671  }
672 
673  MSG_INFO3( spxout << "ILEAVE02 unboundness/infeasiblity found "
674  << "in leave()" << std::endl; )
675 
676  if (rep() != COLUMN)
678  else
679  {
680  int sign;
681  int i;
682 
683  dualFarkas.clear();
685  sign = (enterVal > 0 ? -1 : +1);
686  for( i = 0; i < coPvec().delta().size(); ++i )
687  dualFarkas.add(coPvec().delta().index(i), sign * coPvec().delta().value(i));
688 
690  }
691  return false;
692  }
693  else
694  {
695  /*
696  If an entering variable has been found, a regular basis update is to
697  be performed.
698  */
699  if (enterId != baseId(leaveIdx))
700  {
701  const SVector& newVector = *enterVector(enterId);
702 
703  // update feasibility vectors
704  if( solveVector2 != NULL && solveVector3 != NULL )
705  {
706  assert(solveVector2->isConsistent());
707  assert(solveVector2rhs->isSetup());
708  assert(solveVector3->isConsistent());
709  assert(solveVector3rhs->isSetup());
710  assert(boundflips > 0);
712  *solveVector2,
713  *solveVector3,
714  newVector,
716  *solveVector3rhs);
717 
718  // perform update of basic solution
719  primVec -= (*solveVector3);
720  MSG_INFO3( spxout << "ILSTEP02 "
721  << "breakpoints passed / bounds flipped = " << boundflips
722  << std::endl; )
724  boundflips = 0;
725  }
726  else if( solveVector2 != NULL )
727  {
728  assert(solveVector2->isConsistent());
729  assert(solveVector2rhs->isSetup());
730 
732  *solveVector2,
733  newVector,
734  *solveVector2rhs);
735  }
736  else if( solveVector3 != NULL )
737  {
738  assert(solveVector3->isConsistent());
739  assert(solveVector3rhs->isSetup());
740  assert(boundflips > 0);
742  *solveVector3,
743  newVector,
744  *solveVector3rhs);
745 
746  // perform update of basic solution
747  primVec -= (*solveVector3);
748  MSG_INFO3( spxout << "ILSTEP02 "
749  << "breakpoints passed / bounds flipped = " << boundflips
750  << std::endl; )
752  boundflips = 0;
753  }
754  else
755  SPxBasis::solve4update (theFvec->delta(), newVector);
756 
757 #ifdef ENABLE_ADDITIONAL_CHECKS
758  {
759  SSVector tmp(dim(), epsilon());
760  SPxBasis::solve(tmp, newVector);
761  tmp -= fVec().delta();
762  if (tmp.length() > entertol()) {
763  // This happens very frequently and does usually not hurt, so print
764  // these warnings only with verbose level INFO2 and higher.
765  MSG_INFO2( spxout << "WLEAVE62\t(" << tmp.length() << ")\n"; )
766  }
767  }
768 #endif // ENABLE_ADDITIONAL_CHECKS
769 
770 
771  if (fabs(theFvec->delta()[leaveIdx]) < reject_leave_tol)
772  {
773  if (instable)
774  {
775  /* We are in the case that for all leaving variables only instable entering
776  variables were found: Thus, above we already accepted such an instable
777  entering variable. Now even this seems to be impossible, thus we conclude
778  unboundedness/infeasibility. */
779  MSG_INFO3( spxout << "ILEAVE03 unboundness/infeasiblity found "
780  << "in leave()" << std::endl; )
781 
782  rejectLeave(leaveNum, leaveId, leaveStat);
783  change(-1, none, 0);
784 
785  /**@todo if shift() is not zero we must not conclude unboundedness */
786  if (rep() == ROW)
787  {
788  int sign;
789 
790  primalRay.clear();
792  sign = (enterVal > 0 ? 1.0 : -1.0);
793 
794  for( int i = 0; i < coPvec().delta().size(); ++i )
795  primalRay.add(coPvec().delta().index(i), sign * coPvec().delta().value(i));
796 
798  }
799  else
800  {
801  int sign;
802  int i;
803 
804  dualFarkas.clear();
806  sign = (enterVal > 0 ? -1.0 : +1.0);
807 
808  for( i = 0; i < coPvec().delta().size(); ++i )
809  dualFarkas.add(coPvec().delta().index(i), sign * coPvec().delta().value(i));
810 
812  }
813 
814  return false;
815  }
816  else
817  {
818  theFvec->delta().clear();
819  rejectLeave(leaveNum, leaveId, leaveStat, &newVector);
820  change(-1, none, 0);
821 
822  MSG_DEBUG( spxout << "DLEAVE63 rejecting leave B (leaveIdx=" << leaveIdx
823  << ", theCoTest=" << theCoTest[leaveIdx]
824  << ")" << std::endl; )
825 
826  // Note: These changes do not survive a refactorization
827  theCoTest[leaveIdx] *= 0.01;
828 
829  return true;
830  }
831  }
832 
833  // process leaving variable
834  if (leavebound > epsilon() || leavebound < -epsilon())
835  theFrhs->multAdd(-leavebound, baseVec(leaveIdx));
836 
837  // process entering variable
838  Real enterBound;
839  Real newUBbound;
840  Real newLBbound;
841  Real newCoPrhs;
842 
843  try
844  {
845  getLeaveVals2(leaveMax, enterId, enterBound, newUBbound, newLBbound, newCoPrhs);
846  }
847  catch( SPxException F )
848  {
849  rejectLeave(leaveNum, leaveId, leaveStat);
850  change(-1, none, 0);
851  throw F;
852  }
853 
854  theUBbound[leaveIdx] = newUBbound;
855  theLBbound[leaveIdx] = newLBbound;
856  (*theCoPrhs)[leaveIdx] = newCoPrhs;
857 
858  if (enterBound > epsilon() || enterBound < -epsilon())
859  theFrhs->multAdd(enterBound, newVector);
860 
861  // update pricing vectors
862  theCoPvec->value() = enterVal;
863  thePvec->value() = enterVal;
864  if (enterVal > epsilon() || enterVal < -epsilon())
865  doPupdate();
866 
867  // update feasibility vector
868  theFvec->value() = -((*theFvec)[leaveIdx] - leavebound)
869  / theFvec->delta()[leaveIdx];
870  theFvec->update();
871  (*theFvec)[leaveIdx] = enterBound - theFvec->value();
872  updateFtest();
873 
874  // change basis matrix
875  change(leaveIdx, enterId, &newVector, &(theFvec->delta()));
876  }
877 
878  /*
879  No entering vector has been selected from the basis. However, if the
880  shift amount for |coPvec| is bounded, we are in the case, that the
881  entering variable is moved from one bound to its other, before any of
882  the basis feasibility variables reaches their bound. This may only
883  happen in primal/columnwise case with upper and lower bounds on
884  variables.
885  */
886  else
887  {
888  assert(rep() == ROW);
889  SPxBasis::Desc& ds = desc();
890 
891  change(leaveIdx, none, 0);
892 
893  if (leaveStat == SPxBasis::Desc::P_ON_UPPER)
894  {
895  if (leaveId.isSPxRowId())
896  {
897  ds.rowStatus(leaveNum) = SPxBasis::Desc::P_ON_LOWER;
898  (*theCoPrhs)[leaveIdx] = theLRbound[leaveNum];
899  }
900  else
901  {
902  ds.colStatus(leaveNum) = SPxBasis::Desc::P_ON_LOWER;
903  (*theCoPrhs)[leaveIdx] = theLCbound[leaveNum];
904  }
905  theUBbound[leaveIdx] = 0;
906  theLBbound[leaveIdx] = -infinity;
907  }
908  else
909  {
910  assert( leaveStat == SPxBasis::Desc::P_ON_LOWER );
911  if (leaveId.isSPxRowId())
912  {
913  ds.rowStatus(leaveNum) = SPxBasis::Desc::P_ON_UPPER;
914  (*theCoPrhs)[leaveIdx] = theURbound[leaveNum];
915  }
916  else
917  {
918  ds.colStatus(leaveNum) = SPxBasis::Desc::P_ON_UPPER;
919  (*theCoPrhs)[leaveIdx] = theUCbound[leaveNum];
920  }
921  theUBbound[leaveIdx] = infinity;
922  theLBbound[leaveIdx] = 0;
923  }
924 
925  // update copricing vector
926  theCoPvec->value() = enterVal;
927  thePvec->value() = enterVal;
928  if (enterVal > epsilon() || enterVal < -epsilon())
929  doPupdate();
930 
931  // update feasibility vectors
932  theFvec->value() = 0;
933  theCoTest[leaveIdx] *= -1;
934  }
935 
936  if ((leaveMax > entertol() && enterVal <= entertol()) || (leaveMax < -entertol() && enterVal >= -entertol()))
937  {
938  if ((theUBbound[leaveIdx] < infinity || theLBbound[leaveIdx] > -infinity)
939  && leaveStat != SPxBasis::Desc::P_FREE
940  && leaveStat != SPxBasis::Desc::D_FREE)
941  m_numCycle++;
942  }
943  else
944  m_numCycle /= 2;
945 
946 #ifdef ENABLE_ADDITIONAL_CHECKS
947  {
948  DVector tmp = fVec();
949  multBaseWith(tmp);
950  tmp -= fRhs();
951  if (tmp.length() > entertol())
952  {
953  // This happens very frequently and does usually not hurt, so print
954  // these warnings only with verbose level INFO2 and higher.
955  MSG_INFO2( spxout << "WLEAVE64\t" << basis().iteration()
956  << ": fVec error = " << tmp.length() << std::endl; )
957  SPxBasis::solve(tmp, fRhs());
958  tmp -= fVec();
959  MSG_INFO2( spxout << "WLEAVE65\t(" << tmp.length() << ")\n"; )
960  }
961  }
962 #endif // ENABLE_ADDITIONAL_CHECKS
963 
964  return true;
965  }
966  }
967 }
968 } // namespace soplex
969 
970 //-----------------------------------------------------------------------------
971 //Emacs Local Variables:
972 //Emacs mode:c++
973 //Emacs c-basic-offset:3
974 //Emacs tab-width:8
975 //Emacs indent-tabs-mode:nil
976 //Emacs End:
977 //-----------------------------------------------------------------------------