SoPlex Doxygen Documentation
spxboundflippingrt.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
17 
18 #include <assert.h>
19 #include "spxdefines.h"
20 #include "spxboundflippingrt.h"
21 #include "sorter.h"
22 #include "spxsolver.h"
23 #include "spxout.h"
24 #include "spxid.h"
25 
26 namespace soplex
27 {
28 
29 #define MINSTAB 1e-5
30 #define LOWSTAB 1e-10
31 #define MAX_RELAX_COUNT 2
32 #define LONGSTEP_FREQ 100
33 #define MIN_LONGSTEP 1e-5
34 
35 
36 /** perform necessary bound flips to restore dual feasibility */
38  int& usedBp /**< number of bounds that should be flipped */
39  )
40 {
41  assert(thesolver->rep() == SPxSolver::COLUMN);
42 
43  int skipped;
44 
45  updPrimRhs.setup();
48  updPrimRhs.clear();
49  updPrimVec.clear();
50 
51  skipped = 0;
52  for( int i = 0; i < usedBp; ++i )
53  {
54  int idx;
55  idx = breakpoints[i].idx;
56  if( idx < 0 )
57  {
58  ++skipped;
59  continue;
60  }
61  Real range;
62  Real upper;
63  Real lower;
66  range = 0;
67  if( breakpoints[i].src == PVEC )
68  {
69  stat = ds.status(idx);
70  upper = thesolver->upper(idx);
71  lower = thesolver->lower(idx);
72  switch( stat )
73  {
76  range = lower - upper;
77  assert((*thesolver->theLbound)[idx] == -infinity);
78  (*thesolver->theLbound)[idx] = (*thesolver->theUbound)[idx];
79  (*thesolver->theUbound)[idx] = infinity;
80  break;
83  range = upper - lower;
84  assert((*thesolver->theUbound)[idx] == infinity);
85  (*thesolver->theUbound)[idx] = (*thesolver->theLbound)[idx];
86  (*thesolver->theLbound)[idx] = -infinity;
87  break;
88  default :
89  ++skipped;
90  MSG_WARNING( spxout << "PVEC unexpected status: " << stat
91  << " index: " << idx
92  << " val: " << thesolver->pVec()[idx]
93  << " upd: " << thesolver->pVec().delta()[idx]
94  << " lower: " << lower
95  << " upper: " << upper
96  << " bp.val: " << breakpoints[i].val
97  << std::endl; )
98  }
99  MSG_DEBUG( spxout << "PVEC flipped from: " << stat
100  << " index: " << idx
101  << " val: " << thesolver->pVec()[idx]
102  << " upd: " << thesolver->pVec().delta()[idx]
103  << " lower: " << lower
104  << " upper: " << upper
105  << " bp.val: " << breakpoints[i].val
106  << std::endl; )
107  assert(fabs(range) < 1e20);
108  updPrimRhs.multAdd(range, thesolver->vector(idx));
109  }
110  else if( breakpoints[i].src == COPVEC )
111  {
112  stat = ds.coStatus(idx);
113  upper = thesolver->rhs(idx);
114  lower = thesolver->lhs(idx);
115  switch( stat )
116  {
119  range = lower - upper;
120  assert((*thesolver->theCoUbound)[idx] == infinity);
121  (*thesolver->theCoUbound)[idx] = -(*thesolver->theCoLbound)[idx];
122  (*thesolver->theCoLbound)[idx] = -infinity;
123  break;
126  range = upper - lower;
127  assert((*thesolver->theCoLbound)[idx] == -infinity);
128  (*thesolver->theCoLbound)[idx] = -(*thesolver->theCoUbound)[idx];
129  (*thesolver->theCoUbound)[idx] = infinity;
130  break;
131  default :
132  ++skipped;
133  MSG_WARNING( spxout << "COPVEC unexpected status: " << stat
134  << " index: " << idx
135  << " val: " << thesolver->coPvec()[idx]
136  << " upd: " << thesolver->coPvec().delta()[idx]
137  << " lower: " << lower
138  << " upper: " << upper
139  << " bp.val: " << breakpoints[i].val
140  << std::endl; )
141  }
142  MSG_DEBUG( spxout << "COPVEC flipped from: " << stat
143  << " index: " << idx
144  << " val: " << thesolver->coPvec()[idx]
145  << " upd: " << thesolver->coPvec().delta()[idx]
146  << " lower: " << lower
147  << " upper: " << upper
148  << " bp.val: " << breakpoints[i].val
149  << std::endl; )
150  assert(fabs(range) < 1e20);
151  updPrimRhs.setValue(idx, updPrimRhs[idx] - range);
152  }
153  }
154  usedBp -= skipped;
155  if( usedBp > 0 )
156  {
159  }
160 
161  return;
162 }
163 
164 /** store all available pivots/breakpoints in an array (positive pivot search direction) */
166  int& nBp, /**< number of found breakpoints so far */
167  int& minIdx, /**< index to current minimal breakpoint */
168  const int* idx, /**< pointer to indices of current vector */
169  int nnz, /**< number of nonzeros in current vector */
170  const Real* upd, /**< pointer to update values of current vector */
171  const Real* vec, /**< pointer to values of current vector */
172  const Real* upp, /**< pointer to upper bound/rhs of current vector */
173  const Real* low, /**< pointer to lower bound/lhs of current vector */
174  BreakpointSource src /**< type of vector (pVec or coPvec)*/
175  )
176 {
177  Real minVal;
178  Real curVal;
179  const int* last;
180 
181  minVal = ( nBp == 0 ) ? infinity : breakpoints[minIdx].val;
182 
183  last = idx + nnz;
184 
185  for( ; idx < last; ++idx )
186  {
187  int i = *idx;
188  Real x = upd[i];
189  if( x > epsilon )
190  {
191  if( upp[i] < infinity )
192  {
193  Real y = upp[i] - vec[i];
194  curVal = (y <= 0) ? fastDelta / x : (y + fastDelta) / x;
195  assert(curVal > 0);
196 
197  breakpoints[nBp].idx = i;
198  breakpoints[nBp].src = src;
199  breakpoints[nBp].val = curVal;
200 
201  if( curVal < minVal )
202  {
203  minVal = curVal;
204  minIdx = nBp;
205  }
206 
207  nBp++;
208  }
209  }
210  else if( x < -epsilon )
211  {
212  if (low[i] > -infinity)
213  {
214  Real y = low[i] - vec[i];
215  curVal = (y >= 0) ? -fastDelta / x : (y - fastDelta) / x;
216  assert(curVal > 0);
217 
218  breakpoints[nBp].idx = i;
219  breakpoints[nBp].src = src;
220  breakpoints[nBp].val = curVal;
221 
222  if( curVal < minVal )
223  {
224  minVal = curVal;
225  minIdx = nBp;
226  }
227 
228  nBp++;
229  }
230  }
231  if( nBp >= breakpoints.size() )
232  breakpoints.reSize(nBp * 2);
233  }
234 
235  return;
236 }
237 
238 /** store all available pivots/breakpoints in an array (negative pivot search direction) */
240  int& nBp, /**< number of found breakpoints so far */
241  int& minIdx, /**< index to current minimal breakpoint */
242  const int* idx, /**< pointer to indices of current vector */
243  int nnz, /**< number of nonzeros in current vector */
244  const Real* upd, /**< pointer to update values of current vector */
245  const Real* vec, /**< pointer to values of current vector */
246  const Real* upp, /**< pointer to upper bound/rhs of current vector */
247  const Real* low, /**< pointer to lower bound/lhs of current vector */
248  BreakpointSource src /**< type of vector (pVec or coPvec)*/
249  )
250 {
251  Real minVal;
252  Real curVal;
253  const int* last;
254 
255  minVal = ( nBp == 0 ) ? infinity : breakpoints[minIdx].val;
256 
257  last = idx + nnz;
258 
259  for( ; idx < last; ++idx )
260  {
261  int i = *idx;
262  Real x = upd[i];
263  if( x > epsilon )
264  {
265  if( low[i] > -infinity )
266  {
267  Real y = low[i] - vec[i];
268 
269  curVal = (y >= 0) ? fastDelta / x : (fastDelta - y) / x;
270  assert(curVal > 0);
271 
272  breakpoints[nBp].idx = i;
273  breakpoints[nBp].src = src;
274  breakpoints[nBp].val = curVal;
275 
276  if( curVal < minVal )
277  {
278  minVal = curVal;
279  minIdx = nBp;
280  }
281 
282  nBp++;
283  }
284  }
285  else if( x < -epsilon )
286  {
287  if (upp[i] < infinity)
288  {
289  Real y = upp[i] - vec[i];
290  curVal = (y <= 0) ? -fastDelta / x : -(y + fastDelta) / x;
291  assert(curVal > 0);
292 
293  breakpoints[nBp].idx = i;
294  breakpoints[nBp].src = src;
295  breakpoints[nBp].val = curVal;
296 
297  if( curVal < minVal )
298  {
299  minVal = curVal;
300  minIdx = nBp;
301  }
302 
303  nBp++;
304  }
305  }
306  if( nBp >= breakpoints.size() )
307  breakpoints.reSize(nBp * 2);
308  }
309  return;
310 }
311 
312 /** get values for entering index and perform shifts if necessary */
314  Real& val,
315  SPxId& enterId,
316  int idx,
317  Real stab,
318  Real degeneps,
319  const Real* upd,
320  const Real* vec,
321  const Real* low,
322  const Real* upp,
323  BreakpointSource src,
324  Real max
325  )
326 {
327  if( src == PVEC )
328  {
329  thesolver->pVec()[idx] = thesolver->vector(idx) * thesolver->coPvec();
330  Real x = upd[idx];
331  // skip breakpoint if it is too small
332  if( fabs(x) < stab )
333  {
334  return false;
335  }
336  enterId = thesolver->id(idx);
337  val = (max * x > 0) ? upp[idx] : low[idx];
338  val = (val - vec[idx]) / x;
339  if( upp[idx] == low[idx] )
340  {
341  val = 0.0;
342  if( vec[idx] > upp[idx] )
343  thesolver->theShift += vec[idx] - upp[idx];
344  else
345  thesolver->theShift += low[idx] - vec[idx];
346  thesolver->upBound()[idx] = thesolver->lpBound()[idx] = vec[idx];
347  }
348  else if( (max > 0 && val < -degeneps) || (max < 0 && val > degeneps) )
349  {
350  val = 0.0;
351  if( max * x > 0 )
352  thesolver->shiftUPbound(idx, vec[idx]);
353  else
354  thesolver->shiftLPbound(idx, vec[idx]);
355  }
356  }
357  else // breakpoints[usedBp].src == COPVEC
358  {
359  Real x = upd[idx];
360  if( fabs(x) < stab )
361  {
362  return false;
363  }
364  enterId = thesolver->coId(idx);
365  val = (max * x > 0.0) ? upp[idx] : low[idx];
366  val = (val - vec[idx]) / x;
367  if( upp[idx] == low[idx] )
368  {
369  val = 0.0;
370  if( vec[idx] > upp[idx] )
371  thesolver->theShift += vec[idx] - upp[idx];
372  else
373  thesolver->theShift += low[idx] - vec[idx];
374  thesolver->ucBound()[idx] = thesolver->lcBound()[idx] = vec[idx];
375  }
376  else if( (max > 0 && val < -degeneps) || (max < 0 && val > degeneps) )
377  {
378  val = 0.0;
379  if( max * x > 0 )
380  thesolver->shiftUCbound(idx, vec[idx]);
381  else
382  thesolver->shiftLCbound(idx, vec[idx]);
383  }
384  }
385  return true;
386 }
387 
388 /** determine entering variable */
390  Real& val,
391  int leaveIdx
392  )
393 {
394  assert( m_type == SPxSolver::LEAVE );
395  assert(thesolver->boundflips == 0);
396 
397  // reset the history and try again to do some long steps
398  if( thesolver->leaveCount % LONGSTEP_FREQ == 0 )
399  {
400  MSG_DEBUG( spxout << "ILSTEP06 resetting long step history" << std::endl; )
401  flipPotential = 1;
402  }
403  if( !enableLongsteps || thesolver->rep() == SPxSolver::ROW || flipPotential < 0.01 )
404  {
405  MSG_DEBUG( spxout << "ILSTEP07 switching to fast ratio test" << std::endl; )
406  return SPxFastRT::selectEnter(val, leaveIdx);
407  }
408  const Real* pvec = thesolver->pVec().get_const_ptr();
409  const Real* pupd = thesolver->pVec().delta().values();
410  const int* pidx = thesolver->pVec().delta().indexMem();
411  int pupdnnz = thesolver->pVec().delta().size();
412  const Real* lpb = thesolver->lpBound().get_const_ptr();
413  const Real* upb = thesolver->upBound().get_const_ptr();
414 
415  const Real* cvec = thesolver->coPvec().get_const_ptr();
416  const Real* cupd = thesolver->coPvec().delta().values();
417  const int* cidx = thesolver->coPvec().delta().indexMem();
418  int cupdnnz = thesolver->coPvec().delta().size();
419  const Real* lcb = thesolver->lcBound().get_const_ptr();
420  const Real* ucb = thesolver->ucBound().get_const_ptr();
421 
422  resetTols();
423 
424  Real max;
425 
426  // index in breakpoint array of minimal value (i.e. choice of normal RT)
427  int minIdx;
428 
429  // temporary breakpoint data structure to make swaps possible
430  Breakpoint tmp;
431 
432  // most stable pivot value in candidate set
433  Real moststable;
434 
435  // initialize invalid enterId
436  SPxId enterId;
437 
438  // slope of objective function improvement
439  Real slope;
440 
441  // number of found breakpoints
442  int nBp;
443 
444  // number of latest skipable breakpoint
445  int usedBp;
446 
447  Real degeneps;
448  Real stab;
449  bool instable;
450 
451  max = val;
452  val = 0.0;
453  moststable = 0.0;
454  nBp = 0;
455  minIdx = -1;
456 
457  // get breakpoints and and determine the index of the minimal value
458  if( max > 0 )
459  {
460  collectBreakpointsMax(nBp, minIdx, pidx, pupdnnz, pupd, pvec, upb, lpb, PVEC);
461  collectBreakpointsMax(nBp, minIdx, cidx, cupdnnz, cupd, cvec, ucb, lcb, COPVEC);
462  }
463  else
464  {
465  collectBreakpointsMin(nBp, minIdx, pidx, pupdnnz, pupd, pvec, upb, lpb, PVEC);
466  collectBreakpointsMin(nBp, minIdx, cidx, cupdnnz, cupd, cvec, ucb, lcb, COPVEC);
467  }
468 
469  if( nBp == 0 )
470  return enterId;
471 
472  assert(minIdx >= 0);
473 
474  // swap smallest breakpoint to the front to skip the sorting phase if no bound flip is possible
475  tmp = breakpoints[minIdx];
476  breakpoints[minIdx] = breakpoints[0];
477  breakpoints[0] = tmp;
478 
479  // compute initial slope
480  slope = fabs(thesolver->fTest()[leaveIdx]);
481  if( slope == 0 )
482  {
483  // this may only happen if SoPlex decides to make an instable pivot
484  assert(thesolver->instableLeaveNum >= 0);
485  // restore original slope
486  slope = fabs(thesolver->instableLeaveVal);
487  }
488 
489  // set up structures for the quicksort implementation
490  BreakpointCompare compare;
491  compare.entry = breakpoints.get_const_ptr();
492 
493  // pointer to end of sorted part of breakpoints
494  int sorted = 0;
495  // minimum number of entries that are supposed to be sorted by partial sort
496  int sortsize = 4;
497 
498  // get all skipable breakpoints
499  for( usedBp = 0; usedBp < nBp && slope > 0; ++usedBp)
500  {
501  // sort breakpoints only partially to save time
502  if( usedBp > sorted )
503  {
504  sorted = sorter_qsortPart(breakpoints.get_ptr(), compare, sorted + 1, nBp, sortsize);
505  }
506  int i = breakpoints[usedBp].idx;
507  // compute new slope
508  if( breakpoints[usedBp].src == PVEC )
509  {
510  if( thesolver->isBasic(i) )
511  {
512  // mark basic indices
513  breakpoints[usedBp].idx = -1;
514  thesolver->pVec().delta().clearIdx(i);
515  }
516  else
517  {
518  Real absupd = fabs(pupd[i]);
519  slope -= (thesolver->upper(i) * absupd) - (thesolver->lower(i) * absupd);
520  // get most stable pivot
521  if( absupd > moststable )
522  moststable = absupd;
523  }
524  }
525  else
526  {
527  assert(breakpoints[usedBp].src == COPVEC);
528  if( thesolver->isCoBasic(i) )
529  {
530  // mark basic indices
531  breakpoints[usedBp].idx = -1;
532  thesolver->coPvec().delta().clearIdx(i);
533  }
534  else
535  {
536  Real absupd = fabs(cupd[i]);
537  slope -= (thesolver->rhs(i) * absupd) - (thesolver->lhs(i) * absupd);
538  if( absupd > moststable )
539  moststable = absupd;
540  }
541  }
542  }
543  --usedBp;
544  assert(usedBp >= 0);
545 
546  // check for unboundedness/infeasibility
547  if( slope > delta && usedBp >= nBp - 1 )
548  {
549  MSG_DEBUG( spxout << "DLSTEP02 " << thesolver->basis().iteration()
550  << ": unboundedness in ratio test" << std::endl; )
551  flipPotential *= 0.95;
552  val = max;
553  return SPxFastRT::selectEnter(val, leaveIdx);
554  }
555 
556  MSG_DEBUG( spxout << "DLSTEP01 "
557  << thesolver->basis().iteration()
558  << ": number of flip candidates: "
559  << usedBp
560  << std::endl; )
561 
562  // try to get a more stable pivot by looking at those with similar step length
563  int stableBp; // index to walk over additional breakpoints (after slope change)
564  int bestBp = -1; // breakpoints index with best possible stability
565  Real bestDelta = breakpoints[usedBp].val; // best step length (after bound flips)
566  for( stableBp = usedBp + 1; stableBp < nBp; ++stableBp )
567  {
568  Real stableDelta;
569  // get next breakpoints in increasing order
570  if( stableBp > sorted )
571  {
572  sorted = sorter_qsortPart(breakpoints.get_ptr(), compare, sorted + 1, nBp, sortsize);
573  }
574  int idx = breakpoints[stableBp].idx;
575  if( breakpoints[stableBp].src == PVEC )
576  {
577  if( thesolver->isBasic(idx) )
578  {
579  // mark basic indices
580  // TODO this is probably unnecessary since we do not look at these breakpoints anymore
581  breakpoints[stableBp].idx = -1;
582  thesolver->pVec().delta().clearIdx(idx);
583  continue;
584  }
585  thesolver->pVec()[idx] = thesolver->vector(idx) * thesolver->coPvec();
586  Real x = pupd[idx];
587  stableDelta = (x > 0.0) ? upb[idx] : lpb[idx];
588  stableDelta = (stableDelta - pvec[idx]) / x;
589 
590  if( stableDelta <= bestDelta)
591  {
592  if( fabs(x) > moststable )
593  {
594  moststable = fabs(x);
595  bestBp = stableBp;
596  }
597  }
598  // stop searching if the step length is too big
599  else if( stableDelta > 2 * bestDelta )
600  break;
601  }
602  else
603  {
604  if( thesolver->isCoBasic(idx) )
605  {
606  // mark basic indices
607  breakpoints[usedBp].idx = -1;
608  thesolver->coPvec().delta().clearIdx(idx);
609  continue;
610  }
611  Real x = cupd[idx];
612  stableDelta = (x > 0.0) ? ucb[idx] : lcb[idx];
613  stableDelta = (stableDelta - cvec[idx]) / x;
614 
615  if( stableDelta <= bestDelta)
616  {
617  if( fabs(x) > moststable )
618  {
619  moststable = fabs(x);
620  bestBp = stableBp;
621  }
622  }
623  // TODO check whether 2 is a good factor
624  // stop searching if the step length is too big
625  else if( stableDelta > 2 * bestDelta )
626  break;
627  }
628  }
629 
630  degeneps = fastDelta / moststable; /* as in SPxFastRT */
631  // get stability requirements
632  instable = thesolver->instableLeave;
633  assert(!instable || thesolver->instableLeaveNum >= 0);
634  stab = instable ? LOWSTAB : SPxFastRT::minStability(moststable);
635 
636  bool foundStable= false;
637 
638  if( bestBp > -1 )
639  {
640  // we found a more stable pivot
641  if( moststable > stab )
642  {
643  // stability requirements are satisfied
644  int idx = breakpoints[bestBp].idx;
645  if( breakpoints[bestBp].src == PVEC )
646  foundStable = getData(val, enterId, idx, stab, degeneps, pupd, pvec, lpb, upb, PVEC, max);
647  else
648  foundStable = getData(val, enterId, idx, stab, degeneps, cupd, cvec, lcb, ucb, COPVEC, max);
649  }
650  }
651 
652 #if 0
653  // do not make long steps if the gain in the dual objective is too small, except to avoid degenerate steps
654  if( usedBp > 0 && breakpoints[0].val > epsilon && breakpoints[usedBp].val - breakpoints[0].val < MIN_LONGSTEP * breakpoints[0].val )
655  {
656  MSG_DEBUG( spxout << "DLSTEP03 "
657  << thesolver->basis().iteration()
658  << ": bound flip gain is too small"
659  << std::endl; )
660 
661  // ensure that the first breakpoint is nonbasic
662  while( breakpoints[usedBp].idx < 0 && usedBp < nBp )
663  ++usedBp;
664  // @todo make sure that the selected pivot element is stable
665  }
666 #endif
667 
668  // scan pivot candidates from back to front and stop as soon as a good one is found
669  // @todo select the moststable pivot or one that is comparably stable
670 // stab = (moststable > stab) ? moststable : stab;
671 
672  while( !foundStable && usedBp >= 0 )
673  {
674  int idx = breakpoints[usedBp].idx;
675 
676  // only look for non-basic variables
677  if( idx >= 0 )
678  {
679  if( breakpoints[usedBp].src == PVEC )
680  foundStable = getData(val, enterId, idx, stab, degeneps, pupd, pvec, lpb, upb, PVEC, max);
681  else
682  foundStable = getData(val, enterId, idx, stab, degeneps, cupd, cvec, lcb, ucb, COPVEC, max);
683  }
684  --usedBp;
685  }
686 
687  if( !foundStable )
688  {
689 // assert(usedBp < 0);
690  assert(!enterId.isValid());
692  {
693  MSG_DEBUG( spxout << "DLSTEP04 "
694  << thesolver->basis().iteration()
695  << ": no valid enterId found - relaxing..."
696  << std::endl; )
697  relax();
698  ++relax_count;
699  // restore original value
700  val = max;
701  // try again with relaxed delta
702  return SPxBoundFlippingRT::selectEnter(val, leaveIdx);
703  }
704  else
705  {
706  MSG_DEBUG( spxout << "DLSTEP05 "
707  << thesolver->basis().iteration()
708  << " no valid enterId found - breaking..."
709  << std::endl; )
710  assert(!enterId.isValid());
711  return enterId;
712  }
713  }
714  else
715  {
716 // assert(usedBp >= 0);
717  relax_count = 0;
718  tighten();
719  }
720 
721  // flip bounds of skipped breakpoints only if a nondegenerate step is to be performed
722  if( usedBp > 0 && fabs(breakpoints[usedBp].val) > fastDelta )
723  {
724  flipAndUpdate(usedBp);
725  thesolver->boundflips = usedBp;
726  }
727  else
728  thesolver->boundflips = 0;
729 
730  //TODO incorporate the ratio between usedBp, nBp and dim/coDim
731  // to get an idea of effort and speed
732 
733  // estimate wether long steps may be possible in future iterations
734  flipPotential *= (usedBp + 0.95);
735 
736  MSG_DEBUG( spxout << "DLSTEP06 "
737  << thesolver->basis().iteration()
738  << ": selected Id: "
739  << enterId
740  << " number of candidates: "
741  << nBp
742  << std::endl; )
743  return enterId;
744 }
745 
747  Real& max,
748  SPxId enterId
749  )
750 {
751  return SPxFastRT::selectLeave(max, enterId);
752 }
753 
754 
755 } // namespace soplex
756 
757 //-----------------------------------------------------------------------------
758 //Emacs Local Variables:
759 //Emacs mode:c++
760 //Emacs c-basic-offset:3
761 //Emacs tab-width:8
762 //Emacs indent-tabs-mode:nil
763 //Emacs End:
764 //-----------------------------------------------------------------------------