SoPlex Doxygen Documentation
slufactor.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 /**@file slufactor.cpp
17  * @todo SLUfactor seems to be partly an wrapper for CLUFactor (was C).
18  * This should be properly integrated and demangled.
19  * @todo Does is make sense, to call x.clear() when next x.altValues()
20  * is called.
21  *
22  */
23 //#define DEBUGGING 1
24 
25 #include <assert.h>
26 #include <sstream>
27 
28 #include "spxdefines.h"
29 #include "slufactor.h"
30 #include "cring.h"
31 #include "spxalloc.h"
32 #include "spxout.h"
33 #include "exceptions.h"
34 
35 #ifdef DEBUGGING
36 #include <stdio.h>
37 #endif
38 
39 namespace soplex
40 {
41 #define MINSTABILITY REAL(4e-2)
42 
43 void SLUFactor::solveRight(Vector& x, const Vector& b) //const
44 {
45  METHOD( "SLUFactor::solveRight()" );
46 
47  solveTime.start();
48 
49  vec = b;
51 
52  solveCount++;
53  solveTime.stop();
54 }
55 
56 void SLUFactor::solveRight(SSVector& x, const SVector& b) //const
57 {
58  METHOD( "SLUFactor::solveRight()" );
59 
60  solveTime.start();
61 
62  vec.assign(b);
63  x.clear();
65 
66  solveCount++;
67  solveTime.stop();
68 }
69 
71 {
72  METHOD( "SLUFactor::solveRight4update()" );
73 
74  solveTime.start();
75 
76  int m;
77  int n;
78  int f;
79 
80  x.clear();
81  ssvec = b;
82  n = ssvec.size();
83  if (l.updateType == ETA)
84  {
86  ssvec.altValues(), ssvec.altIndexMem(), n, 0, 0, 0);
87  x.setSize(m);
88  //x.forceSetup();
89  x.unSetup();
91  }
92  else
93  {
94  forest.clear();
98  forest.setSize(f);
100  x.setSize(m);
101  x.forceSetup();
102  }
103  usetup = true;
104 
105  solveCount++;
106  solveTime.stop();
107 }
108 
110  SSVector& x,
111  Vector& y,
112  const SVector& b,
113  SSVector& rhs)
114 {
115  METHOD( "SLUFactor::solve2right4update()" );
116 
117  solveTime.start();
118 
119  int m;
120  int n;
121  int f;
122  int* sidx = ssvec.altIndexMem();
123  int rsize = rhs.size();
124  int* ridx = rhs.altIndexMem();
125 
126  x.clear();
127  y.clear();
128  usetup = true;
129  ssvec = b;
130 
131  if (l.updateType == ETA)
132  {
133  n = ssvec.size();
135  ssvec.get_ptr(), sidx, n, y.get_ptr(),
136  rhs.getEpsilon(), rhs.altValues(), ridx, rsize, 0, 0, 0);
137  x.setSize(m);
138  // x.forceSetup();
139  x.unSetup();
141  }
142  else
143  {
144  forest.clear();
145  n = ssvec.size();
147  ssvec.get_ptr(), sidx, n, y.get_ptr(),
148  rhs.getEpsilon(), rhs.altValues(), ridx, rsize,
150  x.setSize(m);
151  x.forceSetup();
152  forest.setSize(f);
153  forest.forceSetup();
154  }
155  solveCount++;
156  solveTime.stop();
157 }
158 
160  SSVector& x,
161  Vector& y,
162  Vector& y2,
163  const SVector& b,
164  SSVector& rhs,
165  SSVector& rhs2)
166 {
167  METHOD( "SLUFactor::solve3right4update()" );
168 
169  solveTime.start();
170 
171  int m;
172  int n;
173  int f;
174  int* sidx = ssvec.altIndexMem();
175  int rsize = rhs.size();
176  int* ridx = rhs.altIndexMem();
177  int rsize2 = rhs2.size();
178  int* ridx2 = rhs2.altIndexMem();
179 
180  x.clear();
181  y.clear();
182  y2.clear();
183  usetup = true;
184  ssvec = b;
185 
186  if (l.updateType == ETA)
187  {
188  n = ssvec.size();
190  x.altValues(), x.altIndexMem(), ssvec.get_ptr(), sidx, n,
191  y.get_ptr(), rhs.getEpsilon(), rhs.altValues(), ridx, rsize,
192  y2.get_ptr(), rhs2.getEpsilon(),rhs2.altValues(), ridx2, rsize2,
193  0, 0, 0);
194  x.setSize(m);
195  // x.forceSetup();
196  x.unSetup();
198  }
199  else
200  {
201  forest.clear();
202  n = ssvec.size();
204  x.altValues(), x.altIndexMem(), ssvec.get_ptr(), sidx, n,
205  y.get_ptr(), rhs.getEpsilon(), rhs.altValues(), ridx, rsize,
206  y2.get_ptr(), rhs2.getEpsilon(),rhs2.altValues(), ridx2, rsize2,
208  x.setSize(m);
209  x.forceSetup();
210  forest.setSize(f);
211  forest.forceSetup();
212  }
213  solveCount++;
214  solveTime.stop();
215 }
216 
217 void SLUFactor::solveLeft(Vector& x, const Vector& b) //const
218 {
219  METHOD( "SLUFactor::solveLeft()" );
220 
221  solveTime.start();
222 
223  vec = b;
224  ///@todo Why is x.clear() here used and not with solveRight() ?
225  x.clear();
227 
228  solveCount++;
229  solveTime.stop();
230 }
231 
232 void SLUFactor::solveLeft(SSVector& x, const SVector& b) //const
233 {
234  METHOD( "SLUFactor::solveLeft()" );
235 
236  solveTime.start();
237 
238  ssvec.assign(b);
239 
240  x.clear();
241  int sz = ssvec.size(); // see .altValues()
242  int n = vSolveLeft(x.getEpsilon(), x.altValues(), x.altIndexMem(),
243  ssvec.altValues(), ssvec.altIndexMem(), sz);
244 
245  if (n > 0)
246  {
247  x.setSize(n);
248  x.forceSetup();
249  }
250  else
251  x.unSetup();
252 
253  ssvec.setSize(0);
254  ssvec.forceSetup();
255 
256  solveCount++;
257  solveTime.stop();
258 }
259 
261  SSVector& x,
262  Vector& y,
263  const SVector& rhs1,
264  SSVector& rhs2) //const
265 {
266  METHOD( "SLUFactor::solveLeft()" );
267 
268  solveTime.start();
269 
270  int n;
271  Real* svec = ssvec.altValues();
272  int* sidx = ssvec.altIndexMem();
273  int rn = rhs2.size();
274  int* ridx = rhs2.altIndexMem();
275 
276  x.clear();
277  y.clear();
278  ssvec.assign(rhs1);
279  n = ssvec.size(); // see altValues();
280  n = vSolveLeft2(x.getEpsilon(), x.altValues(), x.altIndexMem(), svec, sidx, n,
281  y.get_ptr(), rhs2.altValues(), ridx, rn);
282 
283  x.setSize(n);
284 
285  if (n > 0)
286  x.forceSetup();
287  else
288  x.unSetup();
289 
290  rhs2.setSize(0);
291  rhs2.forceSetup();
292  ssvec.setSize(0);
293  ssvec.forceSetup();
294 
295  solveCount++;
296  solveTime.stop();
297 }
298 
299 
301 {
302  METHOD( "SLUFactor::stability()" );
303 
304  if (status() != OK)
305  return 0;
306 
307  if (maxabs < initMaxabs)
308  return 1;
309 
310  return initMaxabs / maxabs;
311 }
312 
313 std::string SLUFactor::statistics() const
314 {
315  std::stringstream s;
316  s << "Factorizations : " << std::setw(10) << getFactorCount() << std::endl
317  << " Time spent : " << std::setw(10) << std::fixed << std::setprecision(2) << getFactorTime() << std::endl
318  << "Solves : " << std::setw(10) << getSolveCount() << std::endl
319  << " Time spent : " << std::setw(10) << getSolveTime() << std::endl;
320 
321  return s.str();
322 }
323 
324 void SLUFactor::changeEta(int idx, SSVector& et)
325 {
326  METHOD( "SLUFactor::changeEta()" );
327 
328  int es = et.size(); // see altValues()
329  update(idx, et.altValues(), et.altIndexMem(), es);
330  et.setSize(0);
331  et.forceSetup();
332 }
333 
335  int idx,
336  const SVector& subst,
337  const SSVector* e)
338 {
339  METHOD( "SLUFactor::change()" );
340 
341  // BH 2005-08-23: The boolean usetup indicates that an "update vector"
342  // has been set up. I suppose that SSVector forest is this
343  // update vector, which is set up by solveRight4update() and
344  // solve2right4update() in order to optimize the basis update.
345 
346  if (usetup)
347  {
348  if (l.updateType == FOREST_TOMLIN) // FOREST_TOMLIN updates
349  {
350  // BH 2005-08-19: The size of a SSVector is the size of the
351  // index set, i.e. the number of nonzeros which is only
352  // defined if the SSVector is set up. Since
353  // SSVector::altValues() calls unSetup() the size needs to be
354  // stored before the following call.
355  int fsize = forest.size(); // see altValues()
356  forestUpdate(idx, forest.altValues(), fsize, forest.altIndexMem());
357  forest.setSize(0);
358  forest.forceSetup();
359  }
360  else
361  {
362  assert(l.updateType == ETA);
363  changeEta(idx, eta);
364  }
365  }
366  else if (e != 0) // ETA updates
367  {
368  l.updateType = ETA;
369  updateNoClear(idx, e->values(), e->indexMem(), e->size());
370  l.updateType = uptype;
371  }
372  else if (l.updateType == FOREST_TOMLIN) // FOREST_TOMLIN updates
373  {
374  assert(0); // probably this part is never called.
375  // forestUpdate() with the last parameter set to NULL should fail.
376  forest = subst;
378  forestUpdate(idx, forest.altValues(), 0, 0);
379  forest.setSize(0);
380  forest.forceSetup();
381  }
382  else // ETA updates
383  {
384  assert(l.updateType == ETA);
385  vec = subst;
386  eta.clear();
388  changeEta(idx, eta);
389  }
390  usetup = false;
391 
392  MSG_DEBUG( spxout << "DSLUFA01\tupdated\t\tstability = " << stability()
393  << std::endl; )
394 
395  return status();
396 }
397 
399 {
400  METHOD( "SLUFactor::clear()" );
401 
402  rowMemMult = 5; /* factor of minimum Memory * #of nonzeros */
403  colMemMult = 5; /* factor of minimum Memory * #of nonzeros */
404  lMemMult = 1; /* factor of minimum Memory * #of nonzeros */
405 
406  l.firstUpdate = 0;
407  l.firstUnused = 0;
408  thedim = 0;
409 
411  usetup = false;
412  maxabs = 1;
413  initMaxabs = 1;
414  minThreshold = REAL(0.01);
417  stat = UNLOADED;
418 
419  vec.clear();
420  eta.clear();
421  ssvec.clear();
422  forest.clear();
423 
424  u.row.size = 100;
425  u.col.size = 100;
426  l.size = 100;
427  l.startSize = 100;
428 
429  if (l.rval)
430  spx_free(l.rval);
431  if(l.ridx)
432  spx_free(l.ridx);
433  if(l.rbeg)
434  spx_free(l.rbeg);
435  if(l.rorig)
436  spx_free(l.rorig);
437  if(l.rperm)
438  spx_free(l.rperm);
439 
440  if (l.val)
441  spx_free(u.row.val);
442  if(u.row.idx)
443  spx_free(u.row.idx);
444  if(u.col.idx)
445  spx_free(u.col.idx);
446  if(l.val)
447  spx_free(l.val);
448  if(l.idx)
449  spx_free(l.idx);
450  if(l.start)
451  spx_free(l.start);
452  if(l.row)
453  spx_free(l.row);
454 
455  // G clear() is used in constructor of SLUFactor so we have to
456  // G clean up if anything goes wrong here
457  try
458  {
459  spx_alloc(u.row.val, u.row.size);
460  spx_alloc(u.row.idx, u.row.size);
461  spx_alloc(u.col.idx, u.col.size);
462 
463  spx_alloc(l.val, l.size);
464  spx_alloc(l.idx, l.size);
467  }
468  catch(SPxMemoryException& x)
469  {
470  freeAll();
471  throw x;
472  }
473 }
474 
475 /** assignment used to implement operator=() and copy constructor.
476  * If this is initialised, freeAll() has to be called before.
477  * Class objects from SLUFactor are not copied here.
478  */
479 void SLUFactor::assign(const SLUFactor& old)
480 {
481  METHOD( "SLUFactor::operator=()" );
482 
483  // slufactor
484  uptype = old.uptype;
487  epsilon = old.epsilon;
489 
490  // clufactor
491  stat = old.stat;
492  thedim = old.thedim;
493  nzCnt = old.nzCnt;
494  initMaxabs = old.initMaxabs;
495  maxabs = old.maxabs;
496  rowMemMult = old.rowMemMult;
497  colMemMult = old.colMemMult;
498  lMemMult = old.lMemMult;
499 
505 
506  memcpy(row.perm, old.row.perm, thedim * sizeof(*row.perm));
507  memcpy(row.orig, old.row.orig, thedim * sizeof(*row.orig));
508  memcpy(col.perm, old.col.perm, thedim * sizeof(*col.perm));
509  memcpy(col.orig, old.col.orig, thedim * sizeof(*col.orig));
510  memcpy(diag, old.diag, thedim * sizeof(*diag));
511 
512  work = vec.get_ptr();
513 
514  /* setup U
515  */
516  u.row.size = old.u.row.size;
517  u.row.used = old.u.row.used;
518 
520  spx_alloc(u.row.val, u.row.size);
521  spx_alloc(u.row.idx, u.row.size);
522  spx_alloc(u.row.start, thedim + 1);
523  spx_alloc(u.row.len, thedim + 1);
524  spx_alloc(u.row.max, thedim + 1);
525 
526  memcpy(u.row.elem, old.u.row.elem, thedim * sizeof(*u.row.elem));
527  memcpy(u.row.val, old.u.row.val, u.row.size * sizeof(*u.row.val));
528  memcpy(u.row.idx, old.u.row.idx, u.row.size * sizeof(*u.row.idx));
529  memcpy(u.row.start, old.u.row.start, (thedim + 1) * sizeof(*u.row.start));
530  memcpy(u.row.len, old.u.row.len, (thedim + 1) * sizeof(*u.row.len));
531  memcpy(u.row.max, old.u.row.max, (thedim + 1) * sizeof(*u.row.max));
532 
533  // need to make row list ok ?
534  if (thedim > 0 && stat == OK)
535  {
536  u.row.list.idx = old.u.row.list.idx; // .idx neu
537 
538  const Dring* oring = &old.u.row.list;
539  Dring* ring = &u.row.list;
540 
541  while(oring->next != &old.u.row.list)
542  {
543  ring->next = &u.row.elem[oring->next->idx];
544  ring->next->prev = ring;
545  oring = oring->next;
546  ring = ring->next;
547  }
548  ring->next = &u.row.list;
549  ring->next->prev = ring;
550  }
551 
552  u.col.size = old.u.col.size;
553  u.col.used = old.u.col.used;
554 
556  spx_alloc(u.col.idx, u.col.size);
557  spx_alloc(u.col.start, thedim + 1);
558  spx_alloc(u.col.len, thedim + 1);
559  spx_alloc(u.col.max, thedim + 1);
560 
561  if (old.u.col.val != 0)
562  {
563  spx_alloc(u.col.val, u.col.size);
564  memcpy(u.col.val, old.u.col.val, u.col.size * sizeof(*u.col.val));
565  }
566  else
567  u.col.val = 0;
568 
569  memcpy(u.col.elem, old.u.col.elem, thedim * sizeof(*u.col.elem));
570  memcpy(u.col.idx, old.u.col.idx, u.col.size * sizeof(*u.col.idx));
571  memcpy(u.col.start, old.u.col.start, (thedim + 1) * sizeof(*u.col.start));
572  memcpy(u.col.len, old.u.col.len, (thedim + 1) * sizeof(*u.col.len));
573  memcpy(u.col.max, old.u.col.max, (thedim + 1) * sizeof(*u.col.max));
574 
575  // need to make col list ok
576  if (thedim > 0 && stat == OK)
577  {
578  u.col.list.idx = old.u.col.list.idx; // .idx neu
579 
580  const Dring* oring = &old.u.col.list;
581  Dring* ring = &u.col.list;
582 
583  while(oring->next != &old.u.col.list)
584  {
585  ring->next = &u.col.elem[oring->next->idx];
586  ring->next->prev = ring;
587  oring = oring->next;
588  ring = ring->next;
589  }
590  ring->next = &u.col.list;
591  ring->next->prev = ring;
592  }
593 
594  /* Setup L
595  */
596  l.size = old.l.size;
597  l.startSize = old.l.startSize;
598  l.firstUpdate = old.l.firstUpdate;
599  l.firstUnused = old.l.firstUnused;
600  l.updateType = old.l.updateType;
601 
602  spx_alloc(l.val, l.size);
603  spx_alloc(l.idx, l.size);
606 
607  memcpy(l.val, old.l.val, l.size * sizeof(*l.val));
608  memcpy(l.idx, old.l.idx, l.size * sizeof(*l.idx));
609  memcpy(l.start, old.l.start, l.startSize * sizeof(*l.start));
610  memcpy(l.row, old.l.row, l.startSize * sizeof(*l.row));
611 
612  if (old.l.rval != 0)
613  {
614  assert(old.l.ridx != 0);
615  assert(old.l.rbeg != 0);
616  assert(old.l.rorig != 0);
617  assert(old.l.rperm != 0);
618 
619  int memsize = l.start[l.firstUpdate];
620 
621  spx_alloc(l.rval, memsize);
622  spx_alloc(l.ridx, memsize);
623  spx_alloc(l.rbeg, thedim + 1);
626 
627  memcpy(l.rval, old.l.rval, memsize * sizeof(*l.rval));
628  memcpy(l.ridx, old.l.ridx, memsize * sizeof(*l.ridx));
629  memcpy(l.rbeg, old.l.rbeg, (thedim + 1) * sizeof(*l.rbeg));
630  memcpy(l.rorig, old.l.rorig, thedim * sizeof(*l.rorig));
631  memcpy(l.rperm, old.l.rperm, thedim * sizeof(*l.rperm));
632  }
633  else
634  {
635  assert(old.l.ridx == 0);
636  assert(old.l.rbeg == 0);
637  assert(old.l.rorig == 0);
638  assert(old.l.rperm == 0);
639 
640  l.rval = 0;
641  l.ridx = 0;
642  l.rbeg = 0;
643  l.rorig = 0;
644  l.rperm = 0;
645  }
646 
647  assert(row.perm != 0);
648  assert(row.orig != 0);
649  assert(col.perm != 0);
650  assert(col.orig != 0);
651  assert(diag != 0);
652 
653  assert(u.row.elem != 0);
654  assert(u.row.val != 0);
655  assert(u.row.idx != 0);
656  assert(u.row.start != 0);
657  assert(u.row.len != 0);
658  assert(u.row.max != 0);
659 
660  assert(u.col.elem != 0);
661  assert(u.col.idx != 0);
662  assert(u.col.start != 0);
663  assert(u.col.len != 0);
664  assert(u.col.max != 0);
665 
666  assert(l.val != 0);
667  assert(l.idx != 0);
668  assert(l.start != 0);
669  assert(l.row != 0);
670 
671 }
672 
674 {
675  METHOD( "SLUFactor::operator=()" );
676 
677  if (this != &old)
678  {
679  // we don't need to copy them, because they are temporary vectors
680  vec.clear();
681  ssvec.clear();
682 
683  eta = old.eta;
684  forest = old.forest;
685 
686  freeAll();
687  try
688  {
689  assign(old);
690  }
691  catch(SPxMemoryException& x)
692  {
693  freeAll();
694  throw x;
695  }
696  assert(isConsistent());
697  }
698  return *this;
699 }
700 
702  : CLUFactor()
703  , vec (1)
704  , ssvec (1)
705  , usetup (false)
706  , uptype (FOREST_TOMLIN)
707  , eta (1)
708  , forest (1)
709 {
710  METHOD( "SLUFactor::SLUFactor()" );
711  row.perm = 0;
712  row.orig = 0;
713  col.perm = 0;
714  col.orig = 0;
715  diag = 0;
716  u.row.elem = 0;
717  u.row.val = 0;
718  u.row.idx = 0;
719  u.row.start = 0;
720  u.row.len = 0;
721  u.row.max = 0;
722  u.col.elem = 0;
723  u.col.idx = 0;
724  u.col.start = 0;
725  u.col.len = 0;
726  u.col.max = 0;
727  u.col.val = 0;
728  l.val = 0;
729  l.idx = 0;
730  l.start = 0;
731  l.row = 0;
732  l.rval = 0;
733  l.ridx = 0;
734  l.rbeg = 0;
735  l.rorig = 0;
736  l.rperm = 0;
737 
738  nzCnt = 0;
739  thedim = 0;
740  try
741  {
747 
748  work = vec.get_ptr();
749 
750  u.row.size = 1;
751  u.row.used = 0;
753  spx_alloc(u.row.val, u.row.size);
754  spx_alloc(u.row.idx, u.row.size);
755  spx_alloc(u.row.start, thedim + 1);
756  spx_alloc(u.row.len, thedim + 1);
757  spx_alloc(u.row.max, thedim + 1);
758 
759  u.row.list.idx = thedim;
760  u.row.start[thedim] = 0;
761  u.row.max [thedim] = 0;
762  u.row.len [thedim] = 0;
763 
764  u.col.size = 1;
765  u.col.used = 0;
767  spx_alloc(u.col.idx, u.col.size);
768  spx_alloc(u.col.start, thedim + 1);
769  spx_alloc(u.col.len, thedim + 1);
770  spx_alloc(u.col.max, thedim + 1);
771  u.col.val = 0;
772 
773  u.col.list.idx = thedim;
774  u.col.start[thedim] = 0;
775  u.col.max[thedim] = 0;
776  u.col.len[thedim] = 0;
777 
778  l.size = 1;
779 
780  spx_alloc(l.val, l.size);
781  spx_alloc(l.idx, l.size);
782 
783  l.startSize = 1;
784  l.firstUpdate = 0;
785  l.firstUnused = 0;
786 
789  }
790  catch(SPxMemoryException& x)
791  {
792  freeAll();
793  throw x;
794  }
795 
796  l.rval = 0;
797  l.ridx = 0;
798  l.rbeg = 0;
799  l.rorig = 0;
800  l.rperm = 0;
801 
802  SLUFactor::clear(); // clear() is virtual
803 
804  factorCount = 0;
805  solveCount = 0;
806 
807  assert(row.perm != 0);
808  assert(row.orig != 0);
809  assert(col.perm != 0);
810  assert(col.orig != 0);
811  assert(diag != 0);
812 
813  assert(u.row.elem != 0);
814  assert(u.row.val != 0);
815  assert(u.row.idx != 0);
816  assert(u.row.start != 0);
817  assert(u.row.len != 0);
818  assert(u.row.max != 0);
819 
820  assert(u.col.elem != 0);
821  assert(u.col.idx != 0);
822  assert(u.col.start != 0);
823  assert(u.col.len != 0);
824  assert(u.col.max != 0);
825 
826  assert(l.val != 0);
827  assert(l.idx != 0);
828  assert(l.start != 0);
829  assert(l.row != 0);
830 
831  assert(SLUFactor::isConsistent());
832 }
833 
835  : SLinSolver( old )
836  , CLUFactor()
837  , vec(1) // we don't need to copy it, because they are temporary vectors
838  , ssvec(1) // we don't need to copy it, because they are temporary vectors
839  , eta (old.eta)
840  , forest(old.forest)
841 {
842  row.perm = 0;
843  row.orig = 0;
844  col.perm = 0;
845  col.orig = 0;
846  diag = 0;
847  u.row.elem = 0;
848  u.row.val = 0;
849  u.row.idx = 0;
850  u.row.start = 0;
851  u.row.len = 0;
852  u.row.max = 0;
853  u.col.elem = 0;
854  u.col.idx = 0;
855  u.col.start = 0;
856  u.col.len = 0;
857  u.col.max = 0;
858  u.col.val = 0;
859  l.val = 0;
860  l.idx = 0;
861  l.start = 0;
862  l.row = 0;
863  l.rval = 0;
864  l.ridx = 0;
865  l.rbeg = 0;
866  l.rorig = 0;
867  l.rperm = 0;
868 
869  try
870  {
871  assign(old);
872  }
873  catch(SPxMemoryException& x)
874  {
875  freeAll();
876  throw x;
877  }
878  assert(SLUFactor::isConsistent());
879 }
880 
882 {
883  METHOD( "SLUFactor::freeAll()" );
884 
885  if(row.perm) spx_free(row.perm);
886  if(row.orig) spx_free(row.orig);
887  if(col.perm) spx_free(col.perm);
888  if(col.orig) spx_free(col.orig);
889  if(u.row.elem) spx_free(u.row.elem);
890  if(u.row.val) spx_free(u.row.val);
891  if(u.row.idx) spx_free(u.row.idx);
892  if(u.row.start) spx_free(u.row.start);
893  if(u.row.len) spx_free(u.row.len);
894  if(u.row.max) spx_free(u.row.max);
895  if(u.col.elem) spx_free(u.col.elem);
896  if(u.col.idx) spx_free(u.col.idx);
897  if(u.col.start) spx_free(u.col.start);
898  if(u.col.len) spx_free(u.col.len);
899  if(u.col.max) spx_free(u.col.max);
900  if(l.val) spx_free(l.val);
901  if(l.idx) spx_free(l.idx);
902  if(l.start) spx_free(l.start);
903  if(l.row) spx_free(l.row);
904 
905  if(diag) spx_free(diag);
906 
907  if (u.col.val) spx_free(u.col.val);
908 
909  if (l.rval) spx_free(l.rval);
910  if(l.ridx) spx_free(l.ridx);
911  if(l.rbeg) spx_free(l.rbeg);
912  if(l.rorig) spx_free(l.rorig);
913  if(l.rperm) spx_free(l.rperm);
914 
915 }
916 
918 {
919  METHOD( "SLUFactor::~SLUFactor()" );
920  freeAll();
921 }
922 
924 {
925  assert(th < REAL(1.0));
926 
927  if (LT(th, REAL(0.1)))
928  th *= REAL(10.0);
929  else if (LT(th, REAL(0.9)))
930  th = (th + REAL(1.0)) / REAL(2.0);
931  else if (LT(th, REAL(0.999)))
932  th = REAL(0.99999);
933 
934  assert(th < REAL(1.0));
935 
936  return th;
937 }
938 
939 SLUFactor::Status SLUFactor::load(const SVector* matrix[], int dm)
940 {
941  METHOD( "SLUFactor::Status()" );
942  assert(dm >= 0);
943  assert(matrix != 0);
944 
945  Real lastStability = stability();
946 
947  initDR(u.row.list);
948  initDR(u.col.list);
949 
950  usetup = false;
951  l.updateType = uptype;
952  l.firstUpdate = 0;
953  l.firstUnused = 0;
954 
955  if (dm != thedim)
956  {
957  clear();
958 
959  thedim = dm;
960  vec.reDim(thedim);
961  ssvec.reDim(thedim);
962  eta.reDim(thedim);
964  work = vec.get_ptr();
965 
971 
973  spx_realloc(u.row.len, thedim + 1);
974  spx_realloc(u.row.max, thedim + 1);
975  spx_realloc(u.row.start, thedim + 1);
976 
978  spx_realloc(u.col.len, thedim + 1);
979  spx_realloc(u.col.max, thedim + 1);
980  spx_realloc(u.col.start, thedim + 1);
981 
983 
986  }
987  else if (lastStability > 2.0 * minStability)
988  {
989  Real last = minThreshold;
990  Real better = betterThreshold(last);
991 
992  while (better < lastThreshold)
993  {
994  last = better;
995  better = betterThreshold(last);
996  }
998  lastThreshold = last;
999  }
1000 
1001  u.row.list.idx = thedim;
1002  u.row.start[thedim] = 0;
1003  u.row.max[thedim] = 0;
1004  u.row.len[thedim] = 0;
1005 
1006  u.col.list.idx = thedim;
1007  u.col.start[thedim] = 0;
1008  u.col.max[thedim] = 0;
1009  u.col.len[thedim] = 0;
1010 
1011  for (;;)
1012  {
1013  stat = OK;
1014  factor(matrix, lastThreshold, epsilon);
1015  if (stability() >= minStability)
1016  break;
1017 
1018  Real x = lastThreshold;
1019 
1021 
1022  if (EQ(x, lastThreshold))
1023  break;
1024 
1025  MSG_INFO3( spxout << "ISLUFA01 refactorizing with increased Markowitz threshold: "
1026  << lastThreshold << std::endl; )
1027 
1028  minStability /= 2.0;
1029  }
1030  MSG_DEBUG( spxout << "DSLUFA02 threshold = " << lastThreshold
1031  << "\tstability = " << stability()
1032  << "\tminStability = " << minStability << std::endl; )
1033  MSG_DEBUG(
1034  int i;
1035  FILE* fl = fopen("dump.lp", "w");
1036  spxout << "DSLUFA03 Basis:\n";
1037  int j = 0;
1038  for (i = 0; i < dim(); ++i)
1039  j += matrix[i]->size();
1040  for (i = 0; i < dim(); ++i)
1041  {
1042  for (j = 0; j < matrix[i]->size(); ++j)
1043  fprintf(fl, "%8d %8d %16g\n",
1044  i + 1, matrix[i]->index(j) + 1, matrix[i]->value(j));
1045  }
1046  fclose(fl);
1047  spxout << "DSLUFA04 LU-Factors:" << std::endl;
1048  dump();
1049 
1050  spxout << "DSLUFA05 threshold = " << lastThreshold
1051  << "\tstability = " << stability() << std::endl;
1052  )
1053 
1054  assert(isConsistent());
1055  return Status(stat);
1056 }
1057 
1058 
1060 {
1061 #ifdef ENABLE_CONSISTENCY_CHECKS
1062  METHOD( "SLUFactor::isConsistent()" );
1063  return CLUFactor::isConsistent();
1064 #else
1065  return true;
1066 #endif
1067 }
1068 
1069 void SLUFactor::dump() const
1070 {
1071  METHOD( "SLUFactor::dump()" );
1072  CLUFactor::dump();
1073 }
1074 } // namespace soplex
1075 
1076 //-----------------------------------------------------------------------------
1077 //Emacs Local Variables:
1078 //Emacs mode:c++
1079 //Emacs c-basic-offset:3
1080 //Emacs tab-width:8
1081 //Emacs indent-tabs-mode:nil
1082 //Emacs End:
1083 //-----------------------------------------------------------------------------