16#ifndef dealii_petsc_matrix_base_h
17#define dealii_petsc_matrix_base_h
22#ifdef DEAL_II_WITH_PETSC
42template <
typename Matrix>
52 namespace MatrixIterators
124 <<
"You tried to access row " <<
arg1
125 <<
" of a distributed matrix, but only rows " <<
arg2
126 <<
" through " <<
arg3
127 <<
" are stored locally and can be accessed.");
240 <<
"Attempt to access element " <<
arg2 <<
" of row "
241 <<
arg1 <<
" which doesn't have that many elements.");
407 set(
const std::vector<size_type> & indices,
417 set(
const std::vector<size_type> & row_indices,
439 const std::vector<PetscScalar> &values,
460 const PetscScalar *values,
495 add(
const std::vector<size_type> & indices,
505 add(
const std::vector<size_type> & row_indices,
527 const std::vector<PetscScalar> &values,
548 const PetscScalar *values,
580 clear_rows(
const std::vector<size_type> &rows,
671 std::pair<size_type, size_type>
696 std::pair<size_type, size_type>
931 operator Mat()
const;
991 "You are attempting an operation on two vectors that "
992 "are the same object, but the operation requires that the "
993 "two objects are in fact different.");
1001 <<
"You tried to do a "
1002 << (
arg1 == 1 ?
"'set'" : (
arg1 == 2 ?
"'add'" :
"???"))
1003 <<
" operation but the matrix is currently in "
1004 << (
arg2 == 1 ?
"'set'" : (
arg2 == 2 ?
"'add'" :
"???"))
1005 <<
" mode. You first have to call 'compress()'.");
1122 friend class ::BlockMatrixBase;
1131 namespace MatrixIterators
1134 const size_type row,
1135 const size_type index)
1140 visit_present_row();
1157 return (*colnum_cache)[a_index];
1173 return (*value_cache)[a_index];
1185 inline const_iterator &
1186 const_iterator::operator++()
1194 if (accessor.a_index >= accessor.colnum_cache->size())
1196 accessor.a_index = 0;
1199 while ((accessor.a_row < accessor.matrix->m()) &&
1200 (accessor.a_row < accessor.matrix->local_range().second) &&
1201 (accessor.matrix->row_length(accessor.a_row) == 0))
1204 accessor.visit_present_row();
1210 inline const_iterator
1211 const_iterator::operator++(
int)
1219 inline const const_iterator::Accessor &
1220 const_iterator::operator*()
const
1226 inline const const_iterator::Accessor *
1227 const_iterator::operator->()
const
1234 const_iterator::operator==(
const const_iterator &
other)
const
1236 return (accessor.a_row ==
other.accessor.a_row &&
1237 accessor.a_index ==
other.accessor.a_index);
1242 const_iterator::operator!=(
const const_iterator &
other)
const
1244 return !(*
this ==
other);
1249 const_iterator::operator<(
const const_iterator &
other)
const
1251 return (accessor.row() <
other.accessor.row() ||
1252 (accessor.row() ==
other.accessor.row() &&
1253 accessor.index() <
other.accessor.index()));
1268 MatrixBase::set(
const size_type i,
const size_type
j,
const PetscScalar value)
1272 set(i, 1, &
j, &value,
false);
1278 MatrixBase::set(
const std::vector<size_type> & indices,
1286 for (size_type i = 0; i < indices.size(); ++i)
1297 MatrixBase::set(
const std::vector<size_type> & row_indices,
1307 for (size_type i = 0; i < row_indices.size(); ++i)
1318 MatrixBase::set(
const size_type row,
1320 const std::vector<PetscScalar> &values,
1336 MatrixBase::set(
const size_type row,
1337 const size_type n_cols,
1339 const PetscScalar *values,
1361 if (column_indices.size() < n_cols)
1363 column_indices.resize(n_cols);
1364 column_values.resize(n_cols);
1368 for (size_type
j = 0;
j < n_cols; ++
j)
1372 if (value != PetscScalar())
1375 column_values[n_columns] =
value;
1398 MatrixBase::add(
const size_type i,
const size_type
j,
const PetscScalar value)
1402 if (value == PetscScalar())
1413 add(i, 1, &
j, &value,
false);
1419 MatrixBase::add(
const std::vector<size_type> & indices,
1427 for (size_type i = 0; i < indices.size(); ++i)
1438 MatrixBase::add(
const std::vector<size_type> & row_indices,
1448 for (size_type i = 0; i < row_indices.size(); ++i)
1459 MatrixBase::add(
const size_type row,
1461 const std::vector<PetscScalar> &values,
1477 MatrixBase::add(
const size_type row,
1478 const size_type n_cols,
1480 const PetscScalar *values,
1505 if (column_indices.size() < n_cols)
1507 column_indices.resize(n_cols);
1508 column_values.resize(n_cols);
1512 for (size_type
j = 0;
j < n_cols; ++
j)
1516 if (value != PetscScalar())
1519 column_values[n_columns] =
value;
1537 MatrixBase::operator()(
const size_type i,
const size_type
j)
const
1544 inline MatrixBase::const_iterator
1545 MatrixBase::begin()
const
1548 (in_local_range(0) && in_local_range(m() - 1)),
1550 "begin() and end() can only be called on a processor owning the entire matrix. If this is a distributed matrix, use begin(row) and end(row) instead."));
1562 inline MatrixBase::const_iterator
1563 MatrixBase::end()
const
1566 (in_local_range(0) && in_local_range(m() - 1)),
1568 "begin() and end() can only be called on a processor owning the entire matrix. If this is a distributed matrix, use begin(row) and end(row) instead."));
1570 return const_iterator(
this, m(), 0);
1574 inline MatrixBase::const_iterator
1575 MatrixBase::begin(
const size_type r)
const
1577 Assert(in_local_range(r),
1580 if (row_length(r) > 0)
1581 return const_iterator(
this, r, 0);
1587 inline MatrixBase::const_iterator
1588 MatrixBase::end(
const size_type r)
const
1590 Assert(in_local_range(r),
1603 for (size_type i = r + 1; i < m(); ++i)
1604 if (i == local_range().second || (row_length(i) > 0))
1605 return const_iterator(
this, i, 0);
1611 return {
this, m(), 0};
1617 MatrixBase::in_local_range(
const size_type index)
const
1625 return ((index >=
static_cast<size_type>(begin)) &&
1643 MatrixBase::assert_is_compressed()
1648 ExcMessage(
"Error: missing compress() call."));
1654 MatrixBase::prepare_add()
1662 MatrixBase::prepare_set()
1668 MatrixBase::get_mpi_communicator()
const
value_type * data() const noexcept
void add(const size_type i, const size_type j, const PetscScalar value)
void add(const std::vector< size_type > &row_indices, const std::vector< size_type > &col_indices, const FullMatrix< PetscScalar > &full_matrix, const bool elide_zero_values=true)
const_iterator end(const size_type r) const
size_type row_length(const size_type row) const
std::size_t memory_consumption() const
PetscReal l1_norm() const
VectorOperation::values last_action
void vmult(VectorBase &dst, const VectorBase &src) const
void clear_rows(const std::vector< size_type > &rows, const PetscScalar new_diag_value=0)
std::vector< PetscScalar > column_values
MPI_Comm get_mpi_communicator() const
PetscScalar diag_element(const size_type i) const
MatrixBase(const MatrixBase &)=delete
size_type local_domain_size() const
const_iterator begin() const
MatrixBase & operator/=(const PetscScalar factor)
void mmult(MatrixBase &C, const MatrixBase &B, const VectorBase &V) const
PetscBool is_symmetric(const double tolerance=1.e-12)
PetscReal frobenius_norm() const
virtual ~MatrixBase() override
std::pair< size_type, size_type > local_domain() const
const_iterator end() const
void Tvmult_add(VectorBase &dst, const VectorBase &src) const
void print(std::ostream &out, const bool alternative_output=false) const
const_iterator begin(const size_type r) const
void add(const size_type row, const std::vector< size_type > &col_indices, const std::vector< PetscScalar > &values, const bool elide_zero_values=true)
PetscScalar el(const size_type i, const size_type j) const
size_type local_size() const
PetscScalar operator()(const size_type i, const size_type j) const
MatrixBase & operator=(const MatrixBase &)=delete
void set(const std::vector< size_type > &row_indices, const std::vector< size_type > &col_indices, const FullMatrix< PetscScalar > &full_matrix, const bool elide_zero_values=false)
PetscScalar trace() const
void prepare_action(const VectorOperation::values new_action)
void add(const std::vector< size_type > &indices, const FullMatrix< PetscScalar > &full_matrix, const bool elide_zero_values=true)
bool in_local_range(const size_type index) const
PetscBool is_hermitian(const double tolerance=1.e-12)
PetscScalar matrix_scalar_product(const VectorBase &u, const VectorBase &v) const
void assert_is_compressed()
std::vector< PetscInt > column_indices
void Tvmult(VectorBase &dst, const VectorBase &src) const
void clear_rows_columns(const std::vector< size_type > &row_and_column_indices, const PetscScalar new_diag_value=0)
MatrixBase & operator*=(const PetscScalar factor)
PetscScalar residual(VectorBase &dst, const VectorBase &x, const VectorBase &b) const
void Tmmult(MatrixBase &C, const MatrixBase &B, const VectorBase &V) const
void set(const size_type row, const std::vector< size_type > &col_indices, const std::vector< PetscScalar > &values, const bool elide_zero_values=false)
void add(const size_type row, const size_type n_cols, const size_type *col_indices, const PetscScalar *values, const bool elide_zero_values=true, const bool col_indices_are_sorted=false)
void write_ascii(const PetscViewerFormat format=PETSC_VIEWER_DEFAULT)
void vmult_add(VectorBase &dst, const VectorBase &src) const
std::pair< size_type, size_type > local_range() const
void compress(const VectorOperation::values operation)
PetscScalar matrix_norm_square(const VectorBase &v) const
void set(const size_type row, const size_type n_cols, const size_type *col_indices, const PetscScalar *values, const bool elide_zero_values=false)
void set(const std::vector< size_type > &indices, const FullMatrix< PetscScalar > &full_matrix, const bool elide_zero_values=false)
void set(const size_type i, const size_type j, const PetscScalar value)
std::uint64_t n_nonzero_elements() const
void clear_row(const size_type row, const PetscScalar new_diag_value=0)
PetscReal linfty_norm() const
std::shared_ptr< const std::vector< PetscScalar > > value_cache
PetscScalar value() const
types::global_dof_index size_type
std::shared_ptr< const std::vector< size_type > > colnum_cache
Accessor(const MatrixBase *matrix, const size_type row, const size_type index)
const_iterator operator++(int)
bool operator!=(const const_iterator &) const
const_iterator(const MatrixBase *matrix, const size_type row, const size_type index)
types::global_dof_index size_type
bool operator==(const const_iterator &) const
bool operator<(const const_iterator &) const
const_iterator & operator++()
const Accessor * operator->() const
const Accessor & operator*() const
#define DEAL_II_NAMESPACE_OPEN
#define DEAL_II_NAMESPACE_CLOSE
__global__ void set(Number *val, const Number s, const size_type N)
#define DeclException0(Exception0)
static ::ExceptionBase & ExcAccessToNonlocalRow(int arg1, int arg2, int arg3)
#define Assert(cond, exc)
static ::ExceptionBase & ExcIteratorPastEnd()
static ::ExceptionBase & ExcInvalidIndexWithinRow(int arg1, int arg2)
#define AssertIsFinite(number)
#define DeclException2(Exception2, type1, type2, outsequence)
static ::ExceptionBase & ExcBeyondEndOfMatrix()
#define AssertIndexRange(index, range)
#define DeclExceptionMsg(Exception, defaulttext)
static ::ExceptionBase & ExcSourceEqualsDestination()
static ::ExceptionBase & ExcWrongMode(int arg1, int arg2)
#define DeclException3(Exception3, type1, type2, type3, outsequence)
static ::ExceptionBase & ExcIndexRange(std::size_t arg1, std::size_t arg2, std::size_t arg3)
static ::ExceptionBase & ExcDimensionMismatch(std::size_t arg1, std::size_t arg2)
static ::ExceptionBase & ExcNotQuadratic()
static ::ExceptionBase & ExcMessage(std::string arg1)
#define AssertThrow(cond, exc)
@ matrix
Contents is actually a matrix.
types::global_dof_index size_type
VectorType::value_type * end(VectorType &V)
VectorType::value_type * begin(VectorType &V)
unsigned int global_dof_index