Flow123d  JS_before_hm-2-g912b55d
linsys.hh
Go to the documentation of this file.
1 /*!
2  *
3  * Copyright (C) 2015 Technical University of Liberec. All rights reserved.
4  *
5  * This program is free software; you can redistribute it and/or modify it under
6  * the terms of the GNU General Public License version 3 as published by the
7  * Free Software Foundation. (http://www.gnu.org/licenses/gpl-3.0.en.html)
8  *
9  * This program is distributed in the hope that it will be useful, but WITHOUT
10  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
11  * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
12  *
13  *
14  * @file linsys.hh
15  * @brief Wrappers for linear systems based on MPIAIJ and MATIS format.
16  * @author Jan Brezina
17  *
18  * Linear system only keep together matrix, RHS and the solution vector.
19  *
20  */
21 
22 //=============================================================================
23 //
24 // LINER SYSTEM ROUTINES - linear system use wrapers of PETSc assembling routins
25 // in order to allow counting of allocation and filling of matrix to be done by the same code
26 //
27 // we want to allow allocations of nonlocal rows, to this end we count on- and off-processor
28 // members in an parallel Vector
29 //
30 //=============================================================================
31 
32 #ifndef LA_LINSYS_HH_
33 #define LA_LINSYS_HH_
34 
35 /**
36  * @brief Abstract linear system class.
37  *
38  * Linear system consists of Matrix, RHS and solution.
39  * It provides general methods for:
40  * - matrix preallocation
41  * - assembling matrix and RHS
42  * - application of linear constrains (Dirichlet conditions) either during assembly or
43  * on assembled system
44  * - solution of the system
45  * - output in Matlab format
46  *
47  * Method operates on the system as single object. But some methods for matrix only manipulation
48  * can be provided until we have matrix as separate class.
49  *
50  * TODO:
51  * - simplify constructors
52  * - introduce set_solution_vector() and set_rhs() in order to solve multiple systems with same matrix
53  * - simplify constructor, make common interface, rather introduce particular seters for particular solvers
54  * - which parameters expose through Input::Record
55  * - why we need lsize in constructors if we have Distribution
56  */
57 
58 #include <mpi.h> // for MPI_Comm, MPI_...
59 #include <stdlib.h> // for NULL, abs
60 #include <string.h> // for memcpy
61 #include <boost/exception/detail/error_info_impl.hpp> // for error_info
62 #include <boost/exception/info.hpp> // for operator<<
63 #include <cmath> // for abs, fabs
64 #include <new> // for operator new[]
65 #include <string> // for basic_string
66 #include <typeinfo> // for type_info
67 #include <utility> // for swap, pair
68 #include <vector> // for vector, allocator
69 
70 #include <armadillo>
71 
72 // PETSc includes
73 #include "petsclog.h" // for MPI_Allreduce
74 #include "petscmat.h" // for Mat, MatStructure
75 #include "petscmath.h" // for PetscScalar
76 #include "petscsys.h" // for PetscErrorCode
77 #include "petscvec.h" // for Vec, VecCreate...
78 
79 #include "input/accessors.hh" // for Record
80 #include "la/local_system.hh"
81 #include "la/distribution.hh" // for Distribution
82 #include "system/exc_common.hh" // for ExcAssertMsg
83 #include "system/exceptions.hh" // for ExcAssertMsg::...
84 #include "system/system.hh" // for chkerr
85 #include "system/global_defs.h" // for OLD_ASSERT, msg
86 
87 namespace Input { namespace Type { class Abstract; } }
88 
89 
90 class LinSys
91 {
92 friend class SchurComplement;
93 public:
94  // Abstract Input Record for LinSys initialization
96 
97  typedef enum {
98  INSERT=INSERT_VALUES,
99  ADD=ADD_VALUES,
103  } SetValuesMode;
104 
105  struct SolveInfo {
106  SolveInfo(int cr, int nits)
107  : converged_reason(cr), n_iterations(nits)
108  {}
111  };
112 
113 protected:
114  typedef std::pair<unsigned,double> Constraint_;
116 
117 public:
118  /**
119  * Constructor.
120  * Constructor of abstract class should not be called directly, but is used for initialization of member common
121  * to all particular solvers.
122  *
123  * @param comm - MPI communicator
124  *
125  * TODO: Vector solution_ is now initialized to NULL, but it should be rather allocated
126  * in the constructor instead of the method set_solution().
127  */
128  LinSys(const Distribution *rows_ds)
129  : comm_( rows_ds->get_comm() ), status_( NONE ), lsize_( rows_ds->lsize() ), rows_ds_(rows_ds),
130  symmetric_( false ), positive_definite_( false ), negative_definite_( false ),
131  spd_via_symmetric_general_( false ), solution_(NULL), v_solution_(NULL),
132  own_vec_(false), own_solution_(false)
133  {
134  int lsizeInt = static_cast<int>( rows_ds->lsize() );
135  int sizeInt;
136  MPI_Allreduce ( &lsizeInt, &sizeInt, 1, MPI_INT, MPI_SUM, comm_ );
137  size_ = static_cast<unsigned>( sizeInt );
138 
139  r_tol_ = default_r_tol_;
140  a_tol_ = default_a_tol_;
141  max_it_ = default_max_it_;
142  };
143 
144  /**
145  * Copy constructor.
146  */
147  LinSys(LinSys &other)
148  : r_tol_(other.r_tol_), a_tol_(other.a_tol_), max_it_(other.max_it_), comm_(other.comm_), status_(other.status_),
149  lsize_( other.rows_ds_->lsize() ), size_(other.size_), rows_ds_(other.rows_ds_), symmetric_(other.symmetric_),
150  positive_definite_(other.positive_definite_), negative_definite_( other.negative_definite_ ),
151  spd_via_symmetric_general_(other.spd_via_symmetric_general_), matrix_changed_(other.matrix_changed_),
152  rhs_changed_(other.rhs_changed_), residual_norm_(other.residual_norm_), constraints_(other.constraints_),
153  globalSolution_(other.globalSolution_), in_rec_(other.in_rec_)
154 
155  {
156  OLD_ASSERT( false, "Using copy constructor of LinSys is not allowed!");
157  set_solution(other.v_solution_);
158  };
159 
160  /**
161  * Returns global system size.
162  */
163  inline unsigned int size()
164  {
165  return size_;
166  }
167 
168  /**
169  * Returns local system size. (for partitioning of solution vectors)
170  * for PETSC_MPIAIJ it is also partitioning of the matrix
171  */
172  inline unsigned int vec_lsize()
173  {
174  return lsize_;
175  }
176 
177  /**
178  * Returns PETSC matrix (only for PETSC solvers)
179  *
180  * If matrix is changed, method set_matrix_changed() must be called.
181  * Example:
182  * @CODE
183  * MatDiagonalSet(schur->get_matrix(), new_diagonal, ADD_VALUES);
184  * schur->set_matrix_changed();
185  * @ENDCODE
186  */
187  virtual const Mat *get_matrix()
188  {
189  OLD_ASSERT( false, "Function get_matrix is not implemented for linsys type %s \n.", typeid(*this).name() );
190  return NULL;
191  }
192 
193  /**
194  * Returns RHS vector (only for PETSC solvers)
195  *
196  * If vector is changed, method set_rhs_changed() must be called.
197  * Example:
198  * @CODE
199  * VecScale(schur->get_rhs(), -1.0);
200  * schur->set_rhs_changed();
201  * @ENDCODE
202  */
203  virtual const Vec *get_rhs()
204  {
205  OLD_ASSERT( false, "Function get_rhs is not implemented for linsys type %s \n.", typeid(*this).name() );
206  return NULL;
207  }
208 
209  /**
210  * Sets matrix changed flag.
211  */
213  { matrix_changed_ = true;}
214 
215  /**
216  * Sets rhs changed flag (only for PETSC solvers)
217  */
219  { rhs_changed_ = true; }
220 
221  /**
222  * Set relative tolerance, absolute tolerance, and maximum number of iterations of the linear solver.
223  *
224  * For each of these three parameters we first look for the value at user input
225  * if not set we use the value provided to this method and finally the default values are
226  * set by the call of this method in the constructor.
227  */
228  virtual void set_tolerances(double r_tol, double a_tol, unsigned int max_it) = 0;
229 
230  /**
231  * Returns true if the system matrix has changed since the last solve.
232  */
234  { return matrix_changed_;}
235 
236  /**
237  * Returns true if the system RHS has changed since the last solve.
238  */
240  { return rhs_changed_;}
241 
242 
243  /**
244  * Sets PETSC matrix (only for PETSC solvers)
245  */
246  virtual PetscErrorCode set_matrix(Mat&, MatStructure)
247  {
248  OLD_ASSERT( false, "Function set_matrix is not implemented for linsys type %s \n.", typeid(*this).name() );
249  return 0;
250  }
251 
252  /**
253  * Sets RHS vector (only for PETSC solvers)
254  */
255  virtual PetscErrorCode set_rhs(Vec&)
256  {
257  OLD_ASSERT( false, "Function set_rhs is not implemented for linsys type %s \n.", typeid(*this).name() );
258  return 0;
259  }
260 
261  /**
262  * Clears entries of the matrix
263  */
264  virtual PetscErrorCode mat_zero_entries()
265  {
266  OLD_ASSERT( false, "Function mat_zero_entries is not implemented for linsys type %s \n.", typeid(*this).name() );
267  return 0;
268  }
269 
270  /**
271  * Clears entries of the right-hand side
272  */
273  virtual PetscErrorCode rhs_zero_entries()
274  {
275  OLD_ASSERT( false, "Function vec_zero_entries is not implemented for linsys type %s \n.", typeid(*this).name() );
276  return 0;
277  }
278 
279  /**
280  * Returns PETSC vector with solution. Underlying array can be provided on construction.
281  */
282  const Vec &get_solution()
283  {
284  return solution_;
285  }
286 
287  /**
288  * Set PETSc solution
289  */
290  void set_solution(Vec sol_vec) {
291  solution_ = sol_vec;
292  own_vec_ = false;
293  own_solution_ = false;
294  double *out_array;
295  VecGetArray( solution_, &out_array );
296  v_solution_ = out_array;
297  VecRestoreArray( solution_, &out_array );
298  }
299 
300  /**
301  * Create PETSc solution
302  */
303  void set_solution(double *sol_array) {
304  v_solution_ = sol_array;
305  own_vec_ = true;
306  own_solution_ = false;
307  PetscErrorCode ierr;
308  ierr = VecCreateMPIWithArray( comm_,1, rows_ds_->lsize(), PETSC_DECIDE, v_solution_, &solution_ ); CHKERRV( ierr );
309  }
310 
311  /**
312  * Create PETSc solution
313  */
314  void set_solution() {
315  v_solution_ = new double[ rows_ds_->lsize() + 1 ];
316  own_vec_ = true;
317  own_solution_ = true;
318  PetscErrorCode ierr;
319  ierr = VecCreateMPIWithArray( comm_,1, rows_ds_->lsize(), PETSC_DECIDE, v_solution_, &solution_ ); CHKERRV( ierr );
320  }
321 
322  /**
323  * Returns PETSC subarray with solution. Underlying array can be provided on construction.
324  */
326  {
327  return v_solution_;
328  }
329 
330  /**
331  * Switch linear system into allocating assembly. (only for PETSC_MPIAIJ_preallocate_by_assembly)
332  */
333  virtual void start_allocation()
334  {
335  OLD_ASSERT( false, "Function start_allocation is not implemented for linsys type %s \n.", typeid(*this).name() );
336  }
337 
338  /**
339  * Switch linear system into adding assembly. (the only one supported by triplets ??)
340  */
341  virtual void start_add_assembly()
342  {
343  OLD_ASSERT( false, "Function start_add_assembly is not implemented for linsys type %s \n.", typeid(*this).name() );
344  }
345 
346  /**
347  * Switch linear system into insert assembly. (not currently used)
348  */
349  virtual void start_insert_assembly()
350  {
351  OLD_ASSERT( false, "Function start_insert_assembly is not implemented for linsys type %s \n.", typeid(*this).name() );
352  }
353 
354  /**
355  * Finish assembly of the whole system. For PETSC this should call MatEndAssembly with MAT_FINAL_ASSEMBLY
356  */
357  virtual void finish_assembly( )=0;
358 
359  /**
360  * Assembly full rectangular submatrix into the system matrix.
361  * Should be virtual, implemented differently in particular solvers.
362  */
363  virtual void mat_set_values(int nrow,int *rows,int ncol,int *cols,double *vals)=0;
364 
365  /**
366  * Shortcut for assembling just one element into the matrix.
367  * Similarly we can provide method accepting armadillo matrices.
368  */
369  void mat_set_value(int row,int col,double val)
370  { mat_set_values(1,&row,1,&col,&val); }
371 
372  /**
373  * Set values of the system right-hand side.
374  * Should be virtual, implemented differently in particular solvers.
375  */
376  virtual void rhs_set_values(int nrow,int *rows,double *vals)=0;
377 
378  /**
379  * Shorcut for assembling just one element into RHS vector.
380  */
381  void rhs_set_value(int row,double val)
382  { rhs_set_values(1,&row,&val); }
383 
384 
385  /// Set values in the system matrix and values in the right-hand side vector on corresponding rows.
386  inline void set_values(int nrow,int *rows,int ncol,int *cols,PetscScalar *mat_vals, PetscScalar *rhs_vals)
387  {
388  mat_set_values(nrow, rows, ncol, cols, mat_vals);
389  rhs_set_values(nrow, rows, rhs_vals);
390  }
391 
393  local.eliminate_solution();
394  arma::mat tmp = local.matrix.t();
395 // DBGCOUT(<< "\n" << tmp);
396 // DBGCOUT(<< "row dofs: \n");
397 // for(unsigned int i=0; i< local.row_dofs.size(); i++)
398 // cout << local.row_dofs[i] << " ";
399 // cout <<endl;
400 
401  // This is always done only once, see implementation.
402  mat_set_values(local.matrix.n_rows, (int *)(local.row_dofs.memptr()),
403  local.matrix.n_cols, (int *)(local.col_dofs.memptr()),
404  tmp.memptr());
405 
406  rhs_set_values(local.matrix.n_rows, (int *)(local.row_dofs.memptr()),
407  local.rhs.memptr());
408  }
409 
410  /**
411  * Add given dense matrix to preallocation.
412  */
413  //virtual void allocate(std::vector<int> rows, std::vector<int> cols);
414 
415  /**
416  * Shortcut to assembly into matrix and RHS in one call, possibly apply Dirichlet boundary conditions.
417  * @p row_dofs - are global indices of rows of dense @p matrix and rows of dense vector @rhs in global system
418  * @p col_dofs - are global indices of columns of the matrix, and possibly
419  *
420  * Application of Dirichlet conditions:
421  * 1) Rows with negative dofs are set to zero.
422  * 2) Cols with negative dofs are eliminated.
423  * 3) If there are entries on global diagonal. We determine value K either from diagonal of local matrix, or (if it is zero) from
424  * diagonal average.
425  *
426  * Caveats:
427  * - can not set dirichlet condition on zero dof
428  * - Armadillo stores matrix in column first form (Fortran like) which makes it not well suited
429  * for passing local matrices.
430  *
431  */
432  void set_values(std::vector<int> &row_dofs, std::vector<int> &col_dofs,
433  const arma::mat &matrix, const arma::vec &rhs,
434  const arma::vec &row_solution, const arma::vec &col_solution)
435 
436  {
437  arma::mat tmp = matrix.t();
438  arma::vec tmp_rhs = rhs;
439  bool negative_row = false;
440  bool negative_col = false;
441 
442  for(unsigned int l_row = 0; l_row < row_dofs.size(); l_row++)
443  if (row_dofs[l_row] < 0) {
444  tmp_rhs(l_row)=0.0;
445  tmp.col(l_row).zeros();
446  negative_row=true;
447  }
448 
449  for(unsigned int l_col = 0; l_col < col_dofs.size(); l_col++)
450  if (col_dofs[l_col] < 0) {
451  tmp_rhs -= matrix.col(l_col) * col_solution[l_col];
452  tmp.row(l_col).zeros();
453  negative_col=true;
454  }
455 
456 
457  if (negative_row && negative_col) {
458  // look for diagonal entry
459  for(unsigned int l_row = 0; l_row < row_dofs.size(); l_row++)
460  if (row_dofs[l_row] < 0)
461  for(unsigned int l_col = 0; l_col < col_dofs.size(); l_col++)
462  if (col_dofs[l_col] < 0 && row_dofs[l_row] == col_dofs[l_col]) {
463  double new_diagonal = fabs(matrix.at(l_row,l_col));
464  if (new_diagonal == 0.0) {
465  if (matrix.is_square()) {
466  new_diagonal = arma::sum( abs(matrix.diag())) / matrix.n_rows;
467  } else {
468  new_diagonal = arma::accu( abs(matrix) ) / matrix.n_elem;
469  }
470  }
471  tmp.at(l_col, l_row) = new_diagonal;
472  tmp_rhs(l_row) = new_diagonal * row_solution[l_row];
473  }
474 
475  }
476 
477  if (negative_row)
478  for(int &row : row_dofs) row=abs(row);
479 
480  if (negative_col)
481  for(int &col : col_dofs) col=abs(col);
482 
483 
484  mat_set_values(row_dofs.size(), const_cast<int *>(&(row_dofs[0])),
485  col_dofs.size(), const_cast<int *>(&(col_dofs[0])), tmp.memptr() );
486  rhs_set_values(row_dofs.size(), const_cast<int *>(&(row_dofs[0])), tmp_rhs.memptr() );
487  }
488 
489  /**
490  * Adds Dirichlet constrain.
491  * @param row - global number of row that should be eliminated.
492  * @param value - solution value at the given row
493  */
494  void add_constraint(int row, double value) {
495 
496  constraints_.push_back( Constraint_( static_cast<unsigned>( row ), value ) );
497  }
498 
499  /**
500  * Apply constrains to assembled matrix. Constrains are given by pairs: global row index, value.
501  * i.e. typedef pair<unsigned int, double> Constrain;
502  *
503  * What is th meaning of ( const double factor ) form Cambridge code?
504  */
505  virtual void apply_constrains( double scalar )=0;
506 
507  /**
508  * Solve the system and return convergence reason.
509  */
510  virtual SolveInfo solve()=0;
511 
512  /**
513  * Returns norm of the initial right hand side
514  */
516  return residual_norm_;
517  };
518 
519  /**
520  * Explicitly compute residual and its norm for current solution.
521  */
522  virtual double compute_residual() =0;
523 
524  /**
525  * Returns information on relative solver accuracy
526  */
528  return r_tol_;
529  };
530 
531  /**
532  * Returns information on absolute solver accuracy
533  */
534  virtual double get_absolute_accuracy(){
535  return 0.0;
536  };
537 
538  /**
539  * Provides user knowledge about symmetry.
540  */
541  inline void set_symmetric(bool flag = true)
542  {
543  symmetric_ = flag;
544  if (!flag) {
545  set_positive_definite(false);
546  set_negative_definite(false);
547  }
548  }
549 
550  inline bool is_symmetric()
551  { return symmetric_; }
552 
553  /**
554  * Provides user knowledge about positive definiteness.
555  */
556  inline void set_positive_definite(bool flag = true)
557  {
558  positive_definite_ = flag;
559  if (flag) {
560  set_symmetric();
561  set_negative_definite(false);
562  }
563  }
564 
565  /**
566  * Provides user knowledge about negative definiteness.
567  */
568  inline void set_negative_definite(bool flag = true)
569  {
570  negative_definite_ = flag;
571  if (flag) {
572  set_symmetric();
573  set_positive_definite(false);
574  }
575  }
576 
577  inline bool is_positive_definite()
578  { return positive_definite_; }
579 
580  inline bool is_negative_definite()
581  { return negative_definite_; }
582 
583  /// TODO: In fact we want to know if the matrix is already preallocated
584  /// However to do this we need explicit finalisation of preallocating cycle.
585  inline bool is_new() {
586  return ( status_ == NONE );
587  }
588 
589  inline bool is_preallocated() {
590  return ( status_ == INSERT || status_ == ADD);
591  }
592 
593  /**
594  * Provides user knowledge about positive definiteness via symmetric general approach.
595  * This is useful for solving Darcy flow by mixed hybrid method, where blocks on subdomains are saddle point but
596  * interface among subdomains is only at the block of Lagrange multipliers and is symmetric positive definite.
597  * Problem condensed to interface can thus be solved by PCG method, although original problem is saddle point.
598  */
599  inline void set_spd_via_symmetric_general(bool flag = true)
600  {
601  spd_via_symmetric_general_ = flag;
602  if (flag) set_symmetric();
603  }
604 
606  { return spd_via_symmetric_general_; }
607 
608 
609  /**
610  * Output the system in the Matlab format possibly with given ordering.
611  * Rather we shoud provide output operator <<, since it is more flexible.
612  */
613  virtual void view()
614  {
615  OLD_ASSERT( false, "Function view is not implemented for linsys type %s \n.", typeid(*this).name() );
616  }
617 
618  /**
619  * Sets basic parameters of LinSys defined by user in input file and used to calculate
620  */
621  virtual void set_from_input(const Input::Record in_rec)
622  {
623  in_rec_ = in_rec;
624  set_tolerances(default_r_tol_, default_a_tol_, default_max_it_);
625  }
626 
627  /**
628  * Get precision of solving
629  */
630  virtual double get_solution_precision() = 0;
631 
632  virtual ~LinSys()
633  {
634  if ( own_vec_ && solution_ ) { chkerr(VecDestroy(&solution_)); }
635  if ( own_solution_ ) delete[] v_solution_;
636  }
637 
638 protected:
639  // Default values initialized in constructor
640  static constexpr double default_r_tol_ = 1e-7;
641  static constexpr double default_a_tol_ = 1e-11;
642  static constexpr unsigned int default_max_it_ = 1000;
643 
644  double r_tol_; ///< relative tolerance of linear solver
645  double a_tol_; ///< absolute tolerance of linear solver
646  unsigned int max_it_; ///< maximum number of iterations of linear solver
647 
649  SetValuesMode status_; //!< Set value status of the linear system.
650 
651  const unsigned lsize_; //!< local number of matrix rows (non-overlapping division of rows)
652  unsigned size_; //!< global number of matrix rows, i.e. problem size
653 
654  const Distribution * rows_ds_; //!< final distribution of rows of MH matrix
655 
660 
661  bool matrix_changed_; //!< true if the matrix was changed since the last solve
662  bool rhs_changed_; //!< true if the right hand side was changed since the last solve
663 
664  Vec solution_; //!< PETSc vector constructed with vb array.
665  double *v_solution_; //!< local solution array pointing into Vec solution_
666  bool own_vec_; //!< Indicates if the solution vector has been allocated by this class
667  bool own_solution_; //!< Indicates if the solution array has been allocated by this class
668 
669  double residual_norm_; //!< local solution array pointing into Vec solution_
670 
671  ConstraintVec_ constraints_;
672 
673  std::vector<double> globalSolution_; //!< global solution in numbering for linear system
674 
675  Input::Record in_rec_; // structure contained parameters of LinSys defined in input file
676 
677 };
678 
679 #endif /* LA_LINSYS_HH_ */
unsigned int size()
Definition: linsys.hh:163
std::vector< double > globalSolution_
global solution in numbering for linear system
Definition: linsys.hh:673
SetValuesMode status_
Set value status of the linear system.
Definition: linsys.hh:649
bool is_new()
Definition: linsys.hh:585
virtual void start_insert_assembly()
Definition: linsys.hh:349
bool is_negative_definite()
Definition: linsys.hh:580
bool spd_via_symmetric_general_
Definition: linsys.hh:659
std::pair< unsigned, double > Constraint_
Definition: linsys.hh:114
bool own_vec_
Indicates if the solution vector has been allocated by this class.
Definition: linsys.hh:666
Mat< double, N, M > mat
Definition: armor.hh:214
double get_residual_norm()
Definition: linsys.hh:515
void set_symmetric(bool flag=true)
Definition: linsys.hh:541
double * v_solution_
local solution array pointing into Vec solution_
Definition: linsys.hh:665
virtual PetscErrorCode set_matrix(Mat &, MatStructure)
Definition: linsys.hh:246
int MPI_Comm
Definition: mpi.h:141
virtual void view()
Definition: linsys.hh:613
virtual void set_from_input(const Input::Record in_rec)
Definition: linsys.hh:621
bool matrix_changed_
true if the matrix was changed since the last solve
Definition: linsys.hh:661
virtual void start_add_assembly()
Definition: linsys.hh:341
virtual PetscErrorCode mat_zero_entries()
Definition: linsys.hh:264
Abstract linear system class.
Definition: balance.hh:37
void set_values(std::vector< int > &row_dofs, std::vector< int > &col_dofs, const arma::mat &matrix, const arma::vec &rhs, const arma::vec &row_solution, const arma::vec &col_solution)
Definition: linsys.hh:432
const unsigned lsize_
local number of matrix rows (non-overlapping division of rows)
Definition: linsys.hh:651
void set_rhs_changed()
Definition: linsys.hh:218
bool symmetric_
Definition: linsys.hh:656
void set_solution()
Definition: linsys.hh:314
unsigned int vec_lsize()
Definition: linsys.hh:172
ConstraintVec_ constraints_
Definition: linsys.hh:671
#define MPI_SUM
Definition: mpi.h:196
bool is_matrix_changed()
Definition: linsys.hh:233
void set_negative_definite(bool flag=true)
Definition: linsys.hh:568
virtual void start_allocation()
Definition: linsys.hh:333
void chkerr(unsigned int ierr)
Replacement of new/delete operator in the spirit of xmalloc.
Definition: system.hh:148
Mat< double, N, 1 > vec
Definition: armor.hh:211
DofVec col_dofs
Definition: local_system.hh:37
void add_constraint(int row, double value)
Definition: linsys.hh:494
void set_spd_via_symmetric_general(bool flag=true)
Definition: linsys.hh:599
unsigned size_
global number of matrix rows, i.e. problem size
Definition: linsys.hh:652
arma::vec rhs
local system RHS
Input::Record in_rec_
Definition: linsys.hh:675
bool rhs_changed_
true if the right hand side was changed since the last solve
Definition: linsys.hh:662
LinSys(const Distribution *rows_ds)
Definition: linsys.hh:128
void set_local_system(LocalSystem &local)
Definition: linsys.hh:392
virtual ~LinSys()
Definition: linsys.hh:632
static constexpr bool value
Definition: json.hpp:87
bool is_preallocated()
Definition: linsys.hh:589
void mat_set_value(int row, int col, double val)
Definition: linsys.hh:369
#define OLD_ASSERT(...)
Definition: global_defs.h:131
Global macros to enhance readability and debugging, general constants.
MPI_Comm comm_
Definition: linsys.hh:648
double get_relative_accuracy()
Definition: linsys.hh:527
unsigned int max_it_
maximum number of iterations of linear solver
Definition: linsys.hh:646
double * get_solution_array()
Definition: linsys.hh:325
const Vec & get_solution()
Definition: linsys.hh:282
Accessor to the data with type Type::Record.
Definition: accessors.hh:292
int converged_reason
Definition: linsys.hh:109
const Distribution * rows_ds_
final distribution of rows of MH matrix
Definition: linsys.hh:654
double a_tol_
absolute tolerance of linear solver
Definition: linsys.hh:645
SetValuesMode
Definition: linsys.hh:97
double residual_norm_
local solution array pointing into Vec solution_
Definition: linsys.hh:669
bool is_rhs_changed()
Definition: linsys.hh:239
bool negative_definite_
Definition: linsys.hh:658
void set_solution(double *sol_array)
Definition: linsys.hh:303
Class for declaration of polymorphic Record.
virtual const Vec * get_rhs()
Definition: linsys.hh:203
virtual PetscErrorCode rhs_zero_entries()
Definition: linsys.hh:273
void set_solution(Vec sol_vec)
Definition: linsys.hh:290
virtual PetscErrorCode set_rhs(Vec &)
Definition: linsys.hh:255
bool is_symmetric()
Definition: linsys.hh:550
Support classes for parallel programing.
#define MPI_Allreduce(sendbuf, recvbuf, count, datatype, op, comm)
Definition: mpi.h:612
std::vector< Constraint_ > ConstraintVec_
Definition: linsys.hh:115
arma::mat matrix
local system matrix
void set_values(int nrow, int *rows, int ncol, int *cols, PetscScalar *mat_vals, PetscScalar *rhs_vals)
Set values in the system matrix and values in the right-hand side vector on corresponding rows...
Definition: linsys.hh:386
double r_tol_
relative tolerance of linear solver
Definition: linsys.hh:644
#define MPI_INT
Definition: mpi.h:160
bool is_positive_definite()
Definition: linsys.hh:577
void set_matrix_changed()
Definition: linsys.hh:212
static const Input::Type::Record & get_input_type()
Definition: linsys_PETSC.cc:32
void set_positive_definite(bool flag=true)
Definition: linsys.hh:556
bool positive_definite_
Definition: linsys.hh:657
bool own_solution_
Indicates if the solution array has been allocated by this class.
Definition: linsys.hh:667
virtual const Mat * get_matrix()
Definition: linsys.hh:187
virtual double get_absolute_accuracy()
Definition: linsys.hh:534
Vec solution_
PETSc vector constructed with vb array.
Definition: linsys.hh:664
SolveInfo(int cr, int nits)
Definition: linsys.hh:106
LinSys(LinSys &other)
Definition: linsys.hh:147
bool is_spd_via_symmetric_general()
Definition: linsys.hh:605
void eliminate_solution()
void rhs_set_value(int row, double val)
Definition: linsys.hh:381
DofVec row_dofs
Definition: local_system.hh:37
unsigned int lsize(int proc) const
get local size