SoPlex Doxygen Documentation
clufactor.h
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 clufactor.h
17  * @brief Implementation of sparse LU factorization.
18  */
19 #ifndef _CLUFACTOR_H_
20 #define _CLUFACTOR_H_
21 
22 #include "spxdefines.h"
23 #include "slinsolver.h"
24 #include "timer.h"
25 #include "svector.h"
26 
27 #define WITH_L_ROWS 1
28 
29 namespace soplex
30 {
31 /**@brief Implementation of sparse LU factorization.
32  * @ingroup Algo
33  *
34  * This class implements a sparse LU factorization with either
35  * FOREST-TOMLIN or ETA updates, using dynamic Markowitz pivoting.
36  */
37 class CLUFactor
38 {
39 public:
40 
41  //----------------------------------------
42  /**@name Public types */
43  //@{
44  /** Doubly linked ring structure for garbage collection of column or
45  * row file in working matrix.
46  */
47  struct Dring
48  {
51  int idx;
52  };
53 
54  /// Pivot Ring
55  class Pring
56  {
57  public:
58  Pring* next; ///<
59  Pring* prev; ///<
60  int idx; ///< index of pivot row
61  int pos; ///< position of pivot column in row
62  int mkwtz; ///< markowitz number of pivot
63 
64  Pring() : next(0), prev(0) ///< constructor
65  {}
66 
67  private:
68  Pring(const Pring&); ///< blocked copy constructor
69  Pring& operator= (const Pring&); ///< blocked assignment operator
70  };
71  //@}
72 
73 protected:
74 
75  //----------------------------------------
76  /**@name Protected types */
77  //@{
78  /// Temporary data structures.
79  class Temp
80  {
81  public:
82  int* s_mark; ///< marker
83  Real* s_max; ///< maximum absolute value per row (or -1)
84  int* s_cact; ///< lengths of columns of active submatrix
85  int stage; ///< stage of the structure
86  Pring pivots; ///< ring of selected pivot rows
87  Pring* pivot_col; ///< column index handlers for Real linked list
88  Pring* pivot_colNZ; ///< lists for columns to number of nonzeros
89  Pring* pivot_row; ///< row index handlers for Real linked list
90  Pring* pivot_rowNZ; ///< lists for rows to number of nonzeros
91 
92  Temp(); ///< constructor
93  ~Temp(); ///< destructor
94  void init(int p_dim); ///< initialization
95  void clear(); ///< clears the structure
96 
97  private:
98  Temp( const Temp& ); ///< blocked copy constructor
99  Temp& operator= ( const Temp& ); ///< blocked assignment operator
100  };
101 
102  /// Data structures for saving the row and column permutations.
103  struct Perm
104  {
105  int* orig; ///< orig[p] original index from p
106  int* perm; ///< perm[i] permuted index from i
107  };
108 
109  /// Data structures for saving the working matrix and U factor.
110  struct U
111  {
112  ///
113  struct Row
114  {
115  Dring list; /*!< \brief Double linked ringlist of vector
116  indices in the order they appear
117  in the row file */
118  Dring* elem; ///< %Array of ring elements.
119  int size; ///< size of arrays val and idx
120  int used; ///< used entries of arrays idx and val
121  Real* val; ///< hold nonzero values
122  int* idx; ///< hold column indices of nonzeros
123  int* start; ///< starting positions in val and idx
124  int* len; ///< used nonzeros per row vectors
125  int* max; /*!< \brief maximum available nonzeros per row:
126  start[i] + max[i] == start[elem[i].next->idx]
127  len[i] <= max[i]. */
128  } row;
129 
130  ///
131  struct Col
132  {
133  Dring list; /*!< \brief Double linked ringlist of vector
134  indices in the order they appear
135  in the column file */
136  Dring *elem; ///< %Array of ring elements.
137  int size; ///< size of array idx
138  int used; ///< used entries of array idx
139  int *idx; ///< hold row indices of nonzeros
140  Real *val; /*!< \brief hold nonzero values: this is only initialized
141  in the end of the factorization with DEFAULT
142  updates. */
143  int *start; ///< starting positions in val and idx
144  int *len; ///< used nonzeros per column vector
145  int *max; /*!< \brief maximum available nonzeros per colunn:
146  start[i] + max[i] == start[elem[i].next->idx]
147  len[i] <= max[i]. */
148  } col;
149  };
150 
151 
152  /// Data structures for saving the working matrix and L factor.
153  struct L
154  {
155  int size; ///< size of arrays val and idx
156  Real *val; ///< values of L vectors
157  int *idx; ///< indices of L vectors
158  int startSize; ///< size of array start
159  int firstUpdate; ///< number of first update L vector
160  int firstUnused; ///< number of first unused L vector
161  int *start; ///< starting positions in val and idx
162  int *row; ///< column indices of L vectors
163  int updateType; ///< type of updates to be used.
164 
165  /* The following arrays have length |firstUpdate|, since they keep
166  * rows of the L-vectors occuring during the factorization (without
167  * updates), only:
168  */
169  Real *rval; ///< values of rows of L
170  int *ridx; ///< indices of rows of L
171  int *rbeg; ///< start of rows in rval and ridx
172  int *rorig; ///< original row permutation
173  int *rperm; ///< original row permutation
174  };
175  //@}
176 
177  //----------------------------------------
178  /**@name Protected data */
179  //@{
180  SLinSolver::Status stat; ///< Status indicator.
181 
182  int thedim; ///< dimension of factorized matrix
183  int nzCnt; ///< number of nonzeros in U
184  Real initMaxabs; ///< maximum abs number in initail Matrix
185  Real maxabs; ///< maximum abs number in L and U
186 
187  Real rowMemMult; ///< factor of minimum Memory * number of nonzeros
188  Real colMemMult; ///< factor of minimum Memory * number of nonzeros
189  Real lMemMult; ///< factor of minimum Memory * number of nonzeros
190 
191  Perm row; ///< row permutation matrices
192  Perm col; ///< column permutation matrices
193 
194  L l; ///< L matrix
195  Real* diag; ///< Array of pivot elements
196  U u; ///< U matrix
197 
198  Real* work; ///< Working array: must always be left as 0!
199 
200  Timer factorTime; ///< Time spent in factorizations
201  int factorCount; ///< Number of factorizations
202  //@}
203 
204 private:
205 
206  //----------------------------------------
207  /**@name Private data */
208  //@{
209  Temp temp; ///< Temporary storage
210  //@}
211 
212  //----------------------------------------
213  /**@name Solving
214  These helper methods are used during the factorization process.
215  The solve*-methods solve lower and upper triangular systems from
216  the left or from the right, respectively The methods with '2' in
217  the end solve two systems at the same time. The methods with
218  "Eps" in the end consider elements smaller then the passed epsilon
219  as zero.
220  */
221  //@{
222  // From solve.cpp
223  ///
224  void solveUright(Real* wrk, Real* vec) const;
225  ///
226  int solveUrightEps(Real* vec, int* nonz, Real eps, Real* rhs);
227  ///
228  void solveUright2(Real* work1, Real* vec1, Real* work2, Real* vec2);
229  ///
230  int solveUright2eps(Real* work1, Real* vec1, Real* work2, Real* vec2, int* nonz, Real eps);
231  ///
232  void solveLright2(Real* vec1, Real* vec2);
233  ///
234  void solveUpdateRight(Real* vec);
235  ///
236  void solveUpdateRight2(Real* vec1, Real* vec2);
237  ///
238  void solveUleft(Real* work, Real* vec);
239  ///
240  void solveUleft2(Real* work1, Real* vec1, Real* work2, Real* vec2);
241  ///
242  int solveLleft2forest(Real* vec1, int* /* nonz */, Real* vec2, Real /* eps */);
243  ///
244  void solveLleft2(Real* vec1, int* /* nonz */, Real* vec2, Real /* eps */);
245  ///
246  int solveLleftForest(Real* vec, int* /* nonz */, Real /* eps */);
247  ///
248  void solveLleft(Real* vec) const;
249  ///
250  int solveLleftEps(Real* vec, int* nonz, Real eps);
251  ///
252  void solveUpdateLeft(Real* vec);
253  ///
254  void solveUpdateLeft2(Real* vec1, Real* vec2);
255 
256  // From vsolve.cpp
257  ///
258  int vSolveLright(Real* vec, int* ridx, int rn, Real eps);
259  ///
260  void vSolveLright2(Real* vec, int* ridx, int* rnptr, Real eps,
261  Real* vec2, int* ridx2, int* rn2ptr, Real eps2);
262  ///
263  void vSolveLright3(Real* vec, int* ridx, int* rnptr, Real eps,
264  Real* vec2, int* ridx2, int* rn2ptr, Real eps2,
265  Real* vec3, int* ridx3, int* rn3ptr, Real eps3);
266  ///
267  int vSolveUright(Real* vec, int* vidx, Real* rhs, int* ridx, int rn, Real eps);
268  ///
269  void vSolveUrightNoNZ(Real* vec, Real* rhs, int* ridx, int rn, Real eps);
270  ///
271  int vSolveUright2(Real* vec, int* vidx, Real* rhs, int* ridx, int rn, Real eps,
272  Real* vec2, Real* rhs2, int* ridx2, int rn2, Real eps2);
273  ///
274  int vSolveUpdateRight(Real* vec, int* ridx, int n, Real eps);
275  ///
276  void vSolveUpdateRightNoNZ(Real* vec, Real /*eps*/);
277  ///
278  int solveUleft(Real eps, Real* vec, int* vecidx, Real* rhs, int* rhsidx, int rhsn);
279  ///
280  void solveUleftNoNZ(Real eps, Real* vec, Real* rhs, int* rhsidx, int rhsn);
281  ///
282  int solveLleftForest(Real eps, Real* vec, int* nonz, int n);
283  ///
284  void solveLleftForestNoNZ(Real* vec);
285  ///
286  int solveLleft(Real eps, Real* vec, int* nonz, int rn);
287  ///
288  void solveLleftNoNZ(Real* vec);
289  ///
290  int solveUpdateLeft(Real eps, Real* vec, int* nonz, int n);
291 
292  // from forest.cpp
293  ///
294  void forestPackColumns();
295  ///
296  void forestMinColMem(int size);
297  ///
298  void forestReMaxCol(int col, int len);
299 
300  // from factor.cpp
301  ///
302  void initPerm();
303  ///
304  void initFactorMatrix(const SVector** vec, const Real eps );
305  ///
306  void minLMem(int size);
307  ///
308  void setPivot(const int p_stage, const int p_col, const int p_row, const Real val);
309  ///
310  void colSingletons();
311  ///
312  void rowSingletons();
313 
314  ///
315  void initFactorRings();
316  ///
317  void freeFactorRings();
318 
319  ///
320  int setupColVals();
321  ///
322  void setupRowVals();
323 
324  ///
325  void eliminateRowSingletons();
326  ///
327  void eliminateColSingletons();
328  ///
329  void selectPivots(Real threshold);
330  ///
331  int updateRow(int r, int lv, int prow, int pcol, Real pval, Real eps);
332 
333  ///
334  void eliminatePivot(int prow, int pos, Real eps);
335  ///
336  void eliminateNucleus(const Real eps, const Real threshold);
337  ///
338  void minRowMem(int size);
339  ///
340  void minColMem(int size);
341  ///
342  void remaxCol(int p_col, int len);
343  ///
344  void packRows();
345  ///
346  void packColumns();
347  ///
348  void remaxRow(int p_row, int len);
349  ///
350  int makeLvec(int p_len, int p_row);
351  //@}
352 
353  //----------------------------------------
354  /**@name Blocked */
355  //@{
356  /// copy construtor.
357  CLUFactor(const CLUFactor&);
358  /// assignment operator.
359  CLUFactor& operator=(const CLUFactor&);
360  //@}
361 
362 protected:
363 
364  //----------------------------------------
365  /**@name Construction / destruction */
366  //@{
367  /// default construtor.
368  /** Since there is no sense in constructing a CLUFactor object
369  * per se, this is protected.
370  */
371 
373  {}
374  //@}
375 
376  //----------------------------------------
377  /**@name Solver methods */
378  //@{
379  // From solve.cpp
380  ///
381  void solveLright(Real* vec);
382  ///
383  int solveRight4update(Real* vec, int* nonz, Real eps, Real* rhs,
384  Real* forest, int* forestNum, int* forestIdx);
385  ///
386  void solveRight(Real* vec, Real* rhs);
387  ///
388  int solveRight2update(Real* vec1, Real* vec2, Real* rhs1,
389  Real* rhs2, int* nonz, Real eps, Real* forest, int* forestNum, int* forestIdx);
390  ///
391  void solveRight2(Real* vec1, Real* vec2, Real* rhs1, Real* rhs2);
392  ///
393  void solveLeft(Real* vec, Real* rhs);
394  ///
395  int solveLeftEps(Real* vec, Real* rhs, int* nonz, Real eps);
396  ///
397  int solveLeft2(Real* vec1, int* nonz, Real* vec2, Real eps, Real* rhs1, Real* rhs2);
398 
399  // From vsolve.cpp: Very sparse solution methods.
400  ///
401  int vSolveRight4update(Real eps,
402  Real* vec, int* idx, /* result */
403  Real* rhs, int* ridx, int rn, /* rhs & Forest */
404  Real* forest, int* forestNum, int* forestIdx);
405  ///
406  int vSolveRight4update2(Real eps,
407  Real* vec, int* idx, /* result1 */
408  Real* rhs, int* ridx, int rn, /* rhs1 */
409  Real* vec2, Real eps2, /* result2 */
410  Real* rhs2, int* ridx2, int rn2, /* rhs2 */
411  Real* forest, int* forestNum, int* forestIdx);
412  ///
413  int vSolveRight4update3(Real eps,
414  Real* vec, int* idx, /* result1 */
415  Real* rhs, int* ridx, int rn, /* rhs1 */
416  Real* vec2, Real eps2, /* result2 */
417  Real* rhs2, int* ridx2, int rn2, /* rhs2 */
418  Real* vec3, Real eps3, /* result3 */
419  Real* rhs3, int* ridx3, int rn3, /* rhs3 */
420  Real* forest, int* forestNum, int* forestIdx);
421  ///
422  void vSolveRightNoNZ(Real* vec2, Real eps2, /* result2 */
423  Real* rhs2, int* ridx2, int rn2); /* rhs2 */
424  ///
425  int vSolveLeft(Real eps,
426  Real* vec, int* idx, /* result */
427  Real* rhs, int* ridx, int rn); /* rhs */
428  ///
429  void vSolveLeftNoNZ(Real eps,
430  Real* vec, /* result */
431  Real* rhs, int* ridx, int rn); /* rhs */
432  ///
433  int vSolveLeft2(Real eps,
434  Real* vec, int* idx, /* result */
435  Real* rhs, int* ridx, int rn, /* rhs */
436  Real* vec2, /* result2 */
437  Real* rhs2, int* ridx2, int rn2); /* rhs2 */
438 
439  // from forest.cpp
440  void forestUpdate(int col, Real* work, int num, int *nonz);
441 
442  // from update.cpp
443  void update(int p_col, Real* p_work, const int* p_idx, int num);
444  void updateNoClear(int p_col, const Real* p_work, const int* p_idx, int num);
445 
446 
447  // from factor.cpp
448  ///
449  void factor(const SVector** vec, ///< Array of column vector pointers
450  Real threshold, ///< pivoting threshold
451  Real eps); ///< epsilon for zero detection
452  //@}
453 
454  //----------------------------------------
455  /**@name Debugging */
456  //@{
457  ///
458  void dump() const;
459 
460  ///
461  bool isConsistent() const;
462  //@}
463 };
464 
465 } // namespace soplex
466 #endif // _CLUFACTOR_H_
467 
468 //-----------------------------------------------------------------------------
469 //Emacs Local Variables:
470 //Emacs mode:c++
471 //Emacs c-basic-offset:3
472 //Emacs tab-width:8
473 //Emacs indent-tabs-mode:nil
474 //Emacs End:
475 //-----------------------------------------------------------------------------