16#ifndef dealii_linear_operator_h
17#define dealii_linear_operator_h
35 namespace LinearOperatorImplementation
41template <
typename Number>
46template <
typename Range = Vector<
double>,
47 typename Domain = Range,
49 internal::LinearOperatorImplementation::EmptyPayload>
55 typename Domain = Range,
64 typename Domain = Range,
72 typename Domain = Range,
77template <
typename Range,
typename Domain,
typename Payload>
178template <
typename Range,
typename Domain,
typename Payload>
194 vmult = [](Range &,
const Domain &) {
196 ExcMessage(
"Uninitialized LinearOperator<Range, "
197 "Domain>::vmult called"));
200 vmult_add = [](Range &,
const Domain &) {
202 ExcMessage(
"Uninitialized LinearOperator<Range, "
203 "Domain>::vmult_add called"));
206 Tvmult = [](Domain &,
const Range &) {
208 ExcMessage(
"Uninitialized LinearOperator<Range, "
209 "Domain>::Tvmult called"));
214 ExcMessage(
"Uninitialized LinearOperator<Range, "
215 "Domain>::Tvmult_add called"));
220 ExcMessage(
"Uninitialized LinearOperator<Range, "
221 "Domain>::reinit_range_vector method called"));
226 ExcMessage(
"Uninitialized LinearOperator<Range, "
227 "Domain>::reinit_domain_vector method called"));
243 typename = std::enable_if_t<
244 !std::is_base_of<LinearOperator<Range, Domain, Payload>,
Op>::value>>
262 typename = std::enable_if_t<
263 !std::is_base_of<LinearOperator<Range, Domain, Payload>,
Op>::value>>
275 std::function<void(Range &v,
const Domain &
u)>
vmult;
287 std::function<void(Domain &v,
const Range &
u)>
Tvmult;
359 *
this = *
this * number;
388template <
typename Range,
typename Domain,
typename Payload>
404 static_cast<const Payload &
>(
first_op) +
405 static_cast<const Payload &
>(
second_op)};
447template <
typename Range,
typename Domain,
typename Payload>
485template <
typename Range,
typename Domain,
typename Payload>
491 std::is_convertible<
typename Range::value_type,
492 typename Domain::value_type>::value,
493 "Range and Domain must have implicitly convertible 'value_type's");
495 if (op.is_null_operator)
499 else if (number == 0.)
510 return_op.vmult = [number, op](Range &v,
const Domain &
u) {
515 return_op.vmult_add = [number, op](Range &v,
const Domain &
u) {
521 return_op.Tvmult = [number, op](Domain &v,
const Range &
u) {
526 return_op.Tvmult_add = [number, op](Domain &v,
const Range &
u) {
552template <
typename Range,
typename Domain,
typename Payload>
555 typename Domain::value_type number)
558 std::is_convertible<
typename Range::value_type,
559 typename Domain::value_type>::value,
560 "Range and Domain must have implicitly convertible 'value_type's");
582template <
typename Range,
600 static_cast<const Payload &
>(
first_op) *
601 static_cast<const Payload &
>(
second_op)};
631 first_op.reinit_domain_vector(*i,
true);
640 first_op.reinit_domain_vector(*i,
true);
658template <
typename Range,
typename Domain,
typename Payload>
664 return_op.reinit_range_vector = op.reinit_domain_vector;
665 return_op.reinit_domain_vector = op.reinit_range_vector;
695template <
typename Payload,
698 typename Range =
typename Solver::vector_type,
699 typename Domain = Range>
706 op.inverse_payload(solver, preconditioner)};
708 return_op.reinit_range_vector = op.reinit_domain_vector;
709 return_op.reinit_domain_vector = op.reinit_range_vector;
711 return_op.vmult = [op, &solver, &preconditioner](Range &v,
const Domain &
u) {
712 op.reinit_range_vector(v,
false);
713 solver.solve(op, v,
u, preconditioner);
716 return_op.vmult_add = [op, &solver, &preconditioner](Range & v,
721 op.reinit_range_vector(*
v2,
false);
722 solver.solve(op, *
v2,
u, preconditioner);
726 return_op.Tvmult = [op, &solver, &preconditioner](Range &v,
const Domain &
u) {
727 op.reinit_range_vector(v,
false);
731 return_op.Tvmult_add = [op, &solver, &preconditioner](Range & v,
736 op.reinit_range_vector(*
v2,
false);
753template <
typename Payload,
755 typename Range =
typename Solver::vector_type,
756 typename Domain = Range>
763 op.inverse_payload(solver, preconditioner)};
765 return_op.reinit_range_vector = op.reinit_domain_vector;
766 return_op.reinit_domain_vector = op.reinit_range_vector;
768 return_op.vmult = [op, &solver, preconditioner](Range &v,
const Domain &
u) {
769 op.reinit_range_vector(v,
false);
770 solver.solve(op, v,
u, preconditioner);
773 return_op.vmult_add = [op, &solver, preconditioner](Range & v,
778 op.reinit_range_vector(*
v2,
false);
779 solver.solve(op, *
v2,
u, preconditioner);
783 return_op.Tvmult = [op, &solver, preconditioner](Range &v,
const Domain &
u) {
784 op.reinit_range_vector(v,
false);
788 return_op.Tvmult_add = [op, &solver, preconditioner](Range & v,
793 op.reinit_range_vector(*
v2,
false);
811template <
typename Payload,
813 typename Range =
typename Solver::vector_type,
814 typename Domain = Range>
831template <
typename Payload,
833 typename Range =
typename Solver::vector_type,
834 typename Domain = Range>
870 return_op.reinit_range_vector = reinit_vector;
871 return_op.reinit_domain_vector = reinit_vector;
873 return_op.vmult = [](Range &v,
const Range &
u) { v =
u; };
875 return_op.vmult_add = [](Range &v,
const Range &
u) { v +=
u; };
877 return_op.Tvmult = [](Range &v,
const Range &
u) { v =
u; };
879 return_op.Tvmult_add = [](Range &v,
const Range &
u) { v +=
u; };
897template <
typename Range,
typename Domain,
typename Payload>
902 static_cast<Payload &
>(
return_op) = op.identity_payload();
917template <
typename Range,
typename Domain,
typename Payload>
925 return_op.reinit_range_vector = op.reinit_range_vector;
926 return_op.reinit_domain_vector = op.reinit_domain_vector;
928 return_op.vmult = [](Range &v,
const Domain &) { v = 0.; };
930 return_op.vmult_add = [](Range &,
const Domain &) {};
932 return_op.Tvmult = [](Domain &v,
const Range &) { v = 0.; };
934 return_op.Tvmult_add = [](Domain &,
const Range &) {};
960 return_op.reinit_range_vector = reinit_vector;
961 return_op.reinit_domain_vector = reinit_vector;
963 return_op.vmult = [](Range &v,
const Range &
u) {
964 const auto mean =
u.mean_value();
970 return_op.vmult_add = [](Range &v,
const Range &
u) {
971 const auto mean =
u.mean_value();
996template <
typename Range,
typename Domain,
typename Payload>
1001 static_cast<Payload &
>(
return_op) = op.identity_payload();
1009 namespace LinearOperatorImplementation
1022 template <
typename Vector>
1037 template <
typename Matrix>
1057 template <
typename Matrix>
1089 template <
typename...
Args>
1127 template <
typename Solver,
typename Preconditioner>
1159 template <
typename Range,
typename Domain,
typename T>
1162 template <
typename C>
1163 static std::false_type
1166 template <
typename C>
1169 ->
decltype(std::declval<C>().vmult_add(*r, *d),
1170 std::declval<C>().Tvmult_add(*d, *r),
1183 template <
typename Function,
typename Range,
typename Domain>
1206 template <
typename Range,
typename Domain,
typename Payload>
1210 template <
typename Matrix>
1213 const Matrix & matrix)
1215 op.vmult = [&matrix](Range &v,
const Domain &
u) {
1221 [&matrix](Range &b,
const Domain &a) { matrix.vmult(b, a); },
1232 op.vmult_add = [&matrix](Range &v,
const Domain &
u) {
1235 [&matrix](Range &b,
const Domain &a) { matrix.vmult(b, a); },
1241 op.Tvmult = [&matrix](Domain &v,
const Range &
u) {
1247 [&matrix](Domain &b,
const Range &a) { matrix.Tvmult(b, a); },
1254 matrix.Tvmult(v,
u);
1258 op.Tvmult_add = [&matrix](Domain &v,
const Range &
u) {
1261 [&matrix](Domain &b,
const Range &a) { matrix.Tvmult(b, a); },
1271 template <
typename Range,
typename Domain,
typename Payload>
1275 template <
typename Matrix>
1278 const Matrix & matrix)
1287 op.vmult_add = [&matrix](Range &v,
const Domain &
u) {
1291 [&matrix](Range &b,
const Domain &a) { matrix.vmult(b, a); },
1298 matrix.vmult_add(v,
u);
1302 op.Tvmult_add = [&matrix](Domain &v,
const Range &
u) {
1306 [&matrix](Domain &b,
const Range &a) { matrix.Tvmult(b, a); },
1313 matrix.Tvmult_add(v,
u);
1378template <
typename Range,
typename Domain,
typename Payload,
typename Matrix>
1402template <
typename Range,
1432 typename std::conditional<
1460template <
typename Range,
typename Domain,
typename Payload,
typename Matrix>
1463 const Matrix & matrix)
1469 typename std::conditional<
1491 typename Domain = Range,
1495 typename = std::enable_if_t<!std::is_lvalue_reference<Matrix>::value>>
1501 typename Domain = Range,
1506 std::enable_if_t<!std::is_lvalue_reference<OperatorExemplar>::value>,
1507 typename = std::enable_if_t<
1515 typename Domain = Range,
1519 typename = std::enable_if_t<!std::is_lvalue_reference<Matrix>::value>,
1521 std::enable_if_t<!std::is_lvalue_reference<OperatorExemplar>::value>,
1522 typename = std::enable_if_t<
1530 typename Domain = Range,
1533 typename = std::enable_if_t<!std::is_lvalue_reference<Matrix>::value>>
1536 Matrix &&) =
delete;
1540 typename Domain = Range,
1543 typename = std::enable_if_t<!std::is_lvalue_reference<Matrix>::value>>
1551 typename Range =
typename Solver::vector_type,
1552 typename Domain = Range,
1553 typename = std::enable_if_t<!std::is_lvalue_reference<Preconditioner>::value>,
1554 typename = std::enable_if_t<
1555 !std::is_same<Preconditioner, PreconditionIdentity>::value>,
1556 typename = std::enable_if_t<
LinearOperator< Range, Domain, Payload > operator*=(typename Domain::value_type number)
LinearOperator< Range, Domain, Payload > & operator=(const LinearOperator< Range, Domain, Payload > &)=default
std::function< void(Range &v, const Domain &u)> vmult_add
LinearOperator< Range, Domain, Payload > & operator=(const Op &op)
std::function< void(Domain &v, const Range &u)> Tvmult
std::function< void(Domain &v, bool omit_zeroing_entries)> reinit_domain_vector
LinearOperator(const Payload &payload=Payload())
std::function< void(Range &v, const Domain &u)> vmult
std::function< void(Range &v, bool omit_zeroing_entries)> reinit_range_vector
LinearOperator< Range, Domain, Payload > & operator+=(const LinearOperator< Range, Domain, Payload > &second_op)
std::function< void(Domain &v, const Range &u)> Tvmult_add
LinearOperator(const Op &op)
LinearOperator< Range, Domain, Payload > & operator*=(const LinearOperator< Domain, Domain, Payload > &second_op)
LinearOperator(const LinearOperator< Range, Domain, Payload > &)=default
LinearOperator< Range, Domain, Payload > & operator-=(const LinearOperator< Range, Domain, Payload > &second_op)
virtual void reinit(const size_type N, const bool omit_zeroing_entries=false)
EmptyPayload(const Args &...)
EmptyPayload transpose_payload() const
EmptyPayload identity_payload() const
EmptyPayload null_payload() const
EmptyPayload inverse_payload(Solver &, const Preconditioner &) const
void operator()(LinearOperator< Range, Domain, Payload > &op, const Matrix &matrix)
void operator()(LinearOperator< Range, Domain, Payload > &op, const Matrix &matrix)
static void reinit_domain_vector(const Matrix &matrix, Vector &v, bool omit_zeroing_entries)
static void reinit_range_vector(const Matrix &matrix, Vector &v, bool omit_zeroing_entries)
static std::false_type test(...)
static auto test(Range *r, Domain *d) -> decltype(std::declval< C >().vmult_add(*r, *d), std::declval< C >().Tvmult_add(*d, *r), std::true_type())
#define DEAL_II_NAMESPACE_OPEN
#define DEAL_II_NAMESPACE_CLOSE
#define Assert(cond, exc)
static ::ExceptionBase & ExcMessage(std::string arg1)
LinearOperator< Domain, Range, Payload > inverse_operator(const LinearOperator< Range, Domain, Payload > &op, Solver &solver)
LinearOperator< Range, Domain, Payload > operator*(const LinearOperator< Range, Intermediate, Payload > &first_op, const LinearOperator< Intermediate, Domain, Payload > &second_op)
LinearOperator< Range, Domain, Payload > linear_operator(const OperatorExemplar &operator_exemplar, const Matrix &matrix)
LinearOperator< Range, Domain, Payload > operator-(const LinearOperator< Range, Domain, Payload > &first_op, const LinearOperator< Range, Domain, Payload > &second_op)
LinearOperator< Range, Range, Payload > identity_operator(const std::function< void(Range &, bool)> &reinit_vector)
LinearOperator< Range, Domain, Payload > null_operator(const LinearOperator< Range, Domain, Payload > &op)
LinearOperator< Range, Domain, Payload > identity_operator(const LinearOperator< Range, Domain, Payload > &op)
LinearOperator< Domain, Range, Payload > inverse_operator(const LinearOperator< Range, Domain, Payload > &op, Solver &solver, const PreconditionIdentity &)
LinearOperator< Domain, Range, Payload > inverse_operator(const LinearOperator< Range, Domain, Payload > &op, Solver &solver, const LinearOperator< Range, Domain, Payload > &preconditioner)
LinearOperator< Range, Domain, Payload > linear_operator(const OperatorExemplar &, const Matrix &)
LinearOperator< Range, Domain, Payload > null_operator(const LinearOperator< Range, Domain, Payload > &)
LinearOperator< Range, Domain, Payload > linear_operator(const Matrix &matrix)
LinearOperator< Domain, Range, Payload > transpose_operator(const LinearOperator< Range, Domain, Payload > &op)
LinearOperator< Range, Domain, Payload > mean_value_filter(const LinearOperator< Range, Domain, Payload > &op)
LinearOperator< Range, Domain, Payload > operator*(typename Range::value_type number, const LinearOperator< Range, Domain, Payload > &op)
LinearOperator< Domain, Range, Payload > inverse_operator(const LinearOperator< Range, Domain, Payload > &op, Solver &solver, const Preconditioner &preconditioner)
LinearOperator< Range, Domain, Payload > operator*(const LinearOperator< Range, Domain, Payload > &op, typename Domain::value_type number)
LinearOperator< Range, Domain, Payload > linear_operator(const LinearOperator< Range, Domain, Payload > &operator_exemplar, const Matrix &matrix)
LinearOperator< Range, Range, Payload > mean_value_filter(const std::function< void(Range &, bool)> &reinit_vector)
LinearOperator< Range, Domain, Payload > identity_operator(const LinearOperator< Range, Domain, Payload > &)
LinearOperator< Range, Domain, Payload > operator+(const LinearOperator< Range, Domain, Payload > &first_op, const LinearOperator< Range, Domain, Payload > &second_op)
EmptyPayload operator+(const EmptyPayload &, const EmptyPayload &)
void apply_with_intermediate_storage(Function function, Range &v, const Domain &u, bool add)
EmptyPayload operator*(const EmptyPayload &, const EmptyPayload &)
static bool equal(const T *p1, const T *p2)