10#ifndef THYRA_DEFAULT_BLOCKED_LINEAR_OP_DEF_HPP
11#define THYRA_DEFAULT_BLOCKED_LINEAR_OP_DEF_HPP
14#include "Thyra_DefaultBlockedLinearOp_decl.hpp"
15#include "Thyra_DefaultProductVectorSpace.hpp"
16#include "Thyra_DefaultProductVector.hpp"
17#include "Thyra_DefaultProductMultiVector.hpp"
18#include "Thyra_MultiVectorStdOps.hpp"
19#include "Thyra_VectorStdOps.hpp"
20#include "Thyra_AssertOp.hpp"
21#include "Thyra_ScaledAdjointLinearOpBase.hpp"
32 :numRowBlocks_(0), numColBlocks_(0), blockFillIsActive_(false)
42 assertBlockFillIsActive(
false);
50 const int numRowBlocks,
const int numColBlocks
53 assertBlockFillIsActive(
false);
55 resetStorage(numRowBlocks,numColBlocks);
65 using Teuchos::rcp_dynamic_cast;
66 assertBlockFillIsActive(
false);
68 productRange_ = new_productRange.assert_not_null();
69 productDomain_ = new_productDomain.assert_not_null();
70 defaultProductRange_ =
71 rcp_dynamic_cast<const DefaultProductVectorSpace<Scalar> >(productRange_);
72 defaultProductDomain_ =
73 rcp_dynamic_cast<const DefaultProductVectorSpace<Scalar> >(productDomain_);
75 resetStorage(productRange_->numBlocks(), productDomain_->numBlocks());
82 return blockFillIsActive_;
88 const int i,
const int j
91 assertBlockFillIsActive(
true);
92 assertBlockRowCol(i,j);
99 const int i,
const int j
103 setBlockImpl(i, j, block);
107template<
class Scalar>
109 const int i,
const int j
113 setBlockImpl(i, j, block);
117template<
class Scalar>
123 assertBlockFillIsActive(
true);
130 if (nonnull(productRange_)) {
131 numRowBlocks_ = productRange_->numBlocks();
132 numColBlocks_ = productDomain_->numBlocks();
135 numRowBlocks_ = rangeBlocks_.size();
136 numColBlocks_ = domainBlocks_.size();
145 if (is_null(productRange_)) {
146 for (
int i = 0; i < numRowBlocks_; ++i) {
148 !rangeBlocks_[i].get(), std::logic_error
149 ,
"DefaultBlockedLinearOp<Scalar>::endBlockFill():"
150 " Error, no linear operator block for the i="<<i<<
" block row was added"
151 " and we can not complete the block fill!"
154 for(
int j = 0; j < numColBlocks_; ++j) {
156 !domainBlocks_[j].get(), std::logic_error
157 ,
"DefaultBlockedLinearOp<Scalar>::endBlockFill():"
158 " Error, no linear operator block for the j="
159 <<j<<
" block column was added"
160 " and we can not complete the block fill!"
167 if (Ops_stack_.size()) {
168 Ops_.resize(numRowBlocks_*numColBlocks_);
169 for (
int k = 0; k < as<int>(Ops_stack_.size()); ++k ) {
170 const BlockEntry<Scalar> &block_i_j = Ops_stack_[k];
171 Ops_[numRowBlocks_*block_i_j.j + block_i_j.i] = block_i_j.block;
173 Ops_stack_.resize(0);
177 if (is_null(productRange_)) {
179 defaultProductRange_ = productVectorSpace<Scalar>(rangeBlocks_());
180 defaultProductDomain_ = productVectorSpace<Scalar>(domainBlocks_());
181 productRange_ = defaultProductRange_;
182 productDomain_ = defaultProductDomain_;
185 rangeBlocks_.resize(0);
186 domainBlocks_.resize(0);
188 blockFillIsActive_ =
false;
193template<
class Scalar>
196 productRange_ = Teuchos::null;
197 productDomain_ = Teuchos::null;
201 Ops_stack_.resize(0);
202 rangeBlocks_.resize(0);
203 domainBlocks_.resize(0);
204 blockFillIsActive_ =
false;
211template<
class Scalar>
215 return productRange_;
219template<
class Scalar>
223 return productDomain_;
227template<
class Scalar>
229 const int i,
const int j
232 assertBlockFillIsActive(
false);
233 assertBlockRowCol(i,j);
234 return !is_null(Ops_[numRowBlocks_ * j + i]);
238template<
class Scalar>
240 const int i,
const int j
243 if (!blockExists(i, j))
return false;
244 return Ops_[numRowBlocks_*j+i].isConst();
248template<
class Scalar>
252 if (!blockExists(i, j))
return Teuchos::null;
253 return Ops_[numRowBlocks_*j+i].getNonconstObj();
257template<
class Scalar>
261 if (!blockExists(i, j))
return Teuchos::null;
262 return Ops_[numRowBlocks_*j+i];
269template<
class Scalar>
273 return productRange_;
277template<
class Scalar>
281 return productDomain_;
285template<
class Scalar>
289 return Teuchos::null;
296template<
class Scalar>
299 assertBlockFillIsActive(
false);
300 std::ostringstream oss;
303 <<
"numRowBlocks="<<numRowBlocks_
304 <<
",numColBlocks="<<numColBlocks_
310template<
class Scalar>
316 using Teuchos::rcpFromRef;
319 assertBlockFillIsActive(
false);
325 *out << this->description() << std::endl;
333 <<
"rangeDim=" << this->range()->dim()
334 <<
",domainDim=" << this->domain()->dim()
335 <<
",numRowBlocks=" << numRowBlocks_
336 <<
",numColBlocks=" << numColBlocks_
340 <<
"Constituent LinearOpBase objects for M = [ Op[0,0] ..."
341 <<
" ; ... ; ... Op[numRowBlocks-1,numColBlocks-1] ]:\n";
343 for(
int i = 0; i < numRowBlocks_; ++i ) {
344 for(
int j = 0; j < numColBlocks_; ++j ) {
345 *out <<
"Op["<<i<<
","<<j<<
"] = ";
347 block_i_j = getBlock(i,j);
349 *out << Teuchos::describe(*getBlock(i,j),verbLevel);
368template<
class Scalar>
371 bool supported =
true;
372 for(
int i = 0; i < numRowBlocks_; ++i ) {
373 for(
int j = 0; j < numColBlocks_; ++j ) {
375 block_i_j = getBlock(i,j);
376 if( block_i_j.
get() && !Thyra::opSupported(*block_i_j,M_trans) )
384template<
class Scalar>
394 using Teuchos::rcpFromRef;
402 "DefaultBlockedLinearOp<Scalar>::apply(...)", *
this, M_trans, X_in, &*Y_inout
409 opNumRowBlocks = ( !struct_transp ? numRowBlocks_ : numColBlocks_ ),
410 opNumColBlocks = ( !struct_transp ? numColBlocks_ : numRowBlocks_ );
424 ? defaultProductRange_ : defaultProductDomain_ ),
426 ? defaultProductDomain_ : defaultProductRange_ );
429 X = castOrCreateSingleBlockProductMultiVector<Scalar>(
430 defaultProductDomain_op, rcpFromRef(X_in));
432 Y = nonconstCastOrCreateSingleBlockProductMultiVector<Scalar>(
433 defaultProductRange_op, rcpFromPtr(Y_inout));
435 for(
int i = 0; i < opNumRowBlocks; ++i ) {
436 MultiVectorPtr Y_i = Y->getNonconstMultiVectorBlock(i);
437 for(
int j = 0; j < opNumColBlocks; ++j ) {
439 Op_i_j = ( !struct_transp ? getBlock(i,j) : getBlock(j,i) );
441 X_j = X->getMultiVectorBlock(j);
444 Thyra::apply(*Op_i_j, M_trans,* X_j, Y_i.ptr(), alpha, beta);
446 scale(beta, Y_i.ptr());
450 Thyra::apply(*Op_i_j, M_trans, *X_j, Y_i.ptr(), alpha, ST::one());
458template<
class Scalar>
461 const RowStatLinearOpBaseUtils::ERowStat row_stat)
const
463 using Teuchos::rcpFromRef;
464 using Teuchos::rcp_dynamic_cast;
473 RowStatLinearOpBaseUtils::ERowStat
474 subblk_stat = RowStatLinearOpBaseUtils::ROW_STAT_ROW_SUM,
475 subblk_trans_stat = RowStatLinearOpBaseUtils::ROW_STAT_COL_SUM;
477 case RowStatLinearOpBaseUtils::ROW_STAT_INV_ROW_SUM:
478 case RowStatLinearOpBaseUtils::ROW_STAT_ROW_SUM:
479 subblk_stat = RowStatLinearOpBaseUtils::ROW_STAT_ROW_SUM;
480 subblk_trans_stat = RowStatLinearOpBaseUtils::ROW_STAT_COL_SUM;
482 case RowStatLinearOpBaseUtils::ROW_STAT_INV_COL_SUM:
483 case RowStatLinearOpBaseUtils::ROW_STAT_COL_SUM:
484 subblk_stat = RowStatLinearOpBaseUtils::ROW_STAT_COL_SUM;
485 subblk_trans_stat = RowStatLinearOpBaseUtils::ROW_STAT_ROW_SUM;
493 for(
int i = 0; i < numRowBlocks_; ++i ) {
494 for(
int j = 0; j < numColBlocks_; ++j ) {
495 ConstLinearOpPtr Op_i_j = getBlock(i,j);
497 if (nonnull(Op_i_j)) {
499 ConstLinearOp * Orig_i_j = 0;
504 const RowStatOp & row_stat_op = Teuchos::dyn_cast<const RowStatOp>(*Orig_i_j);
507 RowStatLinearOpBaseUtils::ERowStat stat = subblk_stat;
511 stat = subblk_trans_stat;
513 if(!row_stat_op.rowStatIsSupported(stat))
523template<
class Scalar>
526 const RowStatLinearOpBaseUtils::ERowStat row_stat,
529 using Teuchos::rcpFromRef;
530 using Teuchos::rcpFromPtr;
531 using Teuchos::rcp_dynamic_cast;
541 RowStatLinearOpBaseUtils::ERowStat
542 subblk_stat = RowStatLinearOpBaseUtils::ROW_STAT_ROW_SUM,
543 subblk_trans_stat = RowStatLinearOpBaseUtils::ROW_STAT_COL_SUM;
545 case RowStatLinearOpBaseUtils::ROW_STAT_INV_ROW_SUM:
546 case RowStatLinearOpBaseUtils::ROW_STAT_ROW_SUM:
547 subblk_stat = RowStatLinearOpBaseUtils::ROW_STAT_ROW_SUM;
548 subblk_trans_stat = RowStatLinearOpBaseUtils::ROW_STAT_COL_SUM;
550 case RowStatLinearOpBaseUtils::ROW_STAT_INV_COL_SUM:
551 case RowStatLinearOpBaseUtils::ROW_STAT_COL_SUM:
552 subblk_stat = RowStatLinearOpBaseUtils::ROW_STAT_COL_SUM;
553 subblk_trans_stat = RowStatLinearOpBaseUtils::ROW_STAT_ROW_SUM;
560 Y = rcp_dynamic_cast<ProductVectorBase<Scalar> >(rcpFromPtr(rowStatVec));
564 for(
int i = 0; i < numRowBlocks_; ++i ) {
565 VectorPtr blk_vec = Y->getNonconstVectorBlock(i);
566 put_scalar (ST::zero (), blk_vec.ptr ());
568 for(
int j = 0; j < numColBlocks_; ++j ) {
569 ConstLinearOpPtr Op_i_j = getBlock(i,j);
571 VectorPtr tmp_vec = createMember(Op_i_j->range());
573 put_scalar (ST::zero (), tmp_vec.ptr ());
575 if (nonnull(Op_i_j)) {
577 ConstLinearOp * Orig_i_j = 0;
582 const RowStatOp & row_stat_op = Teuchos::dyn_cast<const RowStatOp>(*Orig_i_j);
586 row_stat_op.getRowStat(subblk_stat,tmp_vec.ptr());
588 row_stat_op.getRowStat(subblk_trans_stat,tmp_vec.ptr());
593 Vp_V(blk_vec.ptr(),*tmp_vec);
600 case RowStatLinearOpBaseUtils::ROW_STAT_INV_ROW_SUM:
601 case RowStatLinearOpBaseUtils::ROW_STAT_INV_COL_SUM:
602 reciprocal(*rowStatVec,rowStatVec.ptr());
603 case RowStatLinearOpBaseUtils::ROW_STAT_ROW_SUM:
604 case RowStatLinearOpBaseUtils::ROW_STAT_COL_SUM:
613template<
class Scalar>
617 using Teuchos::rcp_dynamic_cast;
623 bool supported =
true;
624 for(
int i = 0; i < numRowBlocks_; ++i ) {
625 for(
int j = 0; j < numColBlocks_; ++j ) {
626 LinearOpPtr Op_i_j = getBlock(i,j);
630 if (nonnull(Op_i_j)) {
632 LinearOp * Orig_i_j = 0;
636 ScaledOp & scaled_op = Teuchos::dyn_cast<ScaledOp>(*Orig_i_j);
639 supported &= scaled_op.supportsScaleLeft();
641 supported &= scaled_op.supportsScaleRight();
649template<
class Scalar>
653 using Teuchos::rcp_dynamic_cast;
659 bool supported =
true;
660 for(
int i = 0; i < numRowBlocks_; ++i ) {
661 for(
int j = 0; j < numColBlocks_; ++j ) {
662 LinearOpPtr Op_i_j = getBlock(i,j);
664 if (nonnull(Op_i_j)) {
666 LinearOp * Orig_i_j = 0;
671 ScaledOp & scaled_op = Teuchos::dyn_cast<ScaledOp>(*Orig_i_j);
674 supported &= scaled_op.supportsScaleRight();
676 supported &= scaled_op.supportsScaleLeft();
684template<
class Scalar>
690 using Teuchos::rcp_dynamic_cast;
696 Y = dyn_cast<const ProductVectorBase<Scalar> >(row_scaling);
700 for(
int i = 0; i < numRowBlocks_; ++i ) {
703 for(
int j = 0; j < numColBlocks_; ++j ) {
704 LinearOpPtr Op_i_j = getNonconstBlock(i,j);
706 if (nonnull(Op_i_j)) {
708 LinearOpPtr Orig_i_j;
713 = rcp_dynamic_cast<ScaledAdjointLinearOpBase<Scalar> >(Op_i_j);
715 transp = saOp->overallTransp();
716 Orig_i_j = saOp->getNonconstOrigOp();
722 RCP<ScaledOp> scaled_op = rcp_dynamic_cast<ScaledOp>(Orig_i_j);
729 scaled_op->scaleLeft(*blk_vec);
731 scaled_op->scaleRight(*blk_vec);
737template<
class Scalar>
743 using Teuchos::rcp_dynamic_cast;
749 Y = dyn_cast<const ProductVectorBase<Scalar> >(col_scaling);
753 for(
int j = 0; j < numColBlocks_; ++j ) {
756 for(
int i = 0; i < numRowBlocks_; ++i ) {
757 LinearOpPtr Op_i_j = getNonconstBlock(i,j);
759 if (nonnull(Op_i_j)) {
761 LinearOpPtr Orig_i_j;
766 = rcp_dynamic_cast<ScaledAdjointLinearOpBase<Scalar> >(Op_i_j);
768 transp = saOp->overallTransp();
769 Orig_i_j = saOp->getNonconstOrigOp();
775 RCP<ScaledOp> scaled_op = rcp_dynamic_cast<ScaledOp>(Orig_i_j);
782 scaled_op->scaleRight(*blk_vec);
784 scaled_op->scaleLeft(*blk_vec);
793template<
class Scalar>
795 const int numRowBlocks,
const int numColBlocks
798 numRowBlocks_ = numRowBlocks;
799 numColBlocks_ = numColBlocks;
800 Ops_.resize(numRowBlocks_*numColBlocks_);
801 if (is_null(productRange_)) {
802 rangeBlocks_.resize(numRowBlocks);
803 domainBlocks_.resize(numColBlocks);
805 blockFillIsActive_ =
true;
809template<
class Scalar>
810void DefaultBlockedLinearOp<Scalar>::assertBlockFillIsActive(
822template<
class Scalar>
823void DefaultBlockedLinearOp<Scalar>::assertBlockRowCol(
824 const int i,
const int j
829 !( 0 <= i ), std::logic_error
830 ,
"Error, i="<<i<<
" is invalid!"
833 !( 0 <= j ), std::logic_error
834 ,
"Error, j="<<j<<
" is invalid!"
840 !( 0 <= i && i < numRowBlocks_ ), std::logic_error
841 ,
"Error, i="<<i<<
" does not fall in the range [0,"<<numRowBlocks_-1<<
"]!"
844 !( 0 <= j && j < numColBlocks_ ), std::logic_error
845 ,
"Error, j="<<j<<
" does not fall in the range [0,"<<numColBlocks_-1<<
"]!"
855template<
class Scalar>
856void DefaultBlockedLinearOp<Scalar>::setBlockSpaces(
857 const int i,
const int j,
const LinearOpBase<Scalar> &block
861 assertBlockFillIsActive(
true);
862 assertBlockRowCol(i,j);
866 if( i < numRowBlocks_ && j < numColBlocks_ ) {
868 RCP<const VectorSpaceBase<Scalar> >
871 ? productRange_->getBlock(i)
876 ? productDomain_->getBlock(j)
879 if(rangeBlock.get()) {
881 "DefaultBlockedLinearOp<Scalar>::setBlockSpaces(i,j,block):\n\n"
882 "Adding block: " + block.description(),
883 *rangeBlock,(
"(*productRange->getBlock("+
toString(i)+
"))"),
887 if(domainBlock.get()) {
889 "DefaultBlockedLinearOp<Scalar>::setBlockSpaces(i,j,block):\n\n"
890 "Adding block: " + block.description(),
891 *domainBlock,(
"(*productDomain->getBlock("+
toString(j)+
"))"),
900 for(
int k = numRowBlocks_; k <= i; ++k )
901 rangeBlocks_.push_back(Teuchos::null);
902 for(
int k = numColBlocks_; k <= j; ++k )
903 domainBlocks_.push_back(Teuchos::null);
906 if(!productRange_.get()) {
907 if(!rangeBlocks_[i].get())
908 rangeBlocks_[i] = block.range().assert_not_null();
909 if(!domainBlocks_[j].get()) {
910 domainBlocks_[j] = block.domain().assert_not_null();
917 numRowBlocks_ = rangeBlocks_.size();
918 numColBlocks_ = domainBlocks_.size();
924template<
class Scalar>
925template<
class LinearOpType>
926void DefaultBlockedLinearOp<Scalar>::setBlockImpl(
927 const int i,
const int j,
928 const RCP<LinearOpType> &block
931 setBlockSpaces(i, j, *block);
935 Ops_[numRowBlocks_*j+i] = block;
940 bool foundBlock =
false;
941 for(
unsigned int k = 0; k < Ops_stack_.size(); ++k ) {
942 BlockEntry<Scalar> &block_i_j = Ops_stack_[k];
943 if( block_i_j.i == i && block_i_j.j == j ) {
944 block_i_j.block = block;
950 Ops_stack_.push_back(BlockEntry<Scalar>(i,j,block));
955template<
class Scalar>
956void DefaultBlockedLinearOp<Scalar>::adjustBlockSpaces()
975 for (
int i = 0; i < numRowBlocks_; ++i) {
976 for (
int j = 0; j < numColBlocks_; ++j) {
977 const RCP<const LinearOpBase<Scalar> >
978 op_i_j = Ops_[numRowBlocks_*j+i];
981 const RCP<const VectorSpaceBase<Scalar> > range_i_j = op_i_j->range();
982 if (
is_null(productVectorSpaceBase<Scalar>(range_i_j,
false))) {
983 rangeBlocks_[i] = range_i_j;
990 for (
int j = 0; j < numColBlocks_; ++j) {
991 for (
int i = 0; i < numRowBlocks_; ++i) {
992 const RCP<const LinearOpBase<Scalar> >
993 op_i_j = Ops_[numRowBlocks_*j+i];
996 const RCP<const VectorSpaceBase<Scalar> >
997 domain_i_j = op_i_j->domain();
998 if (
is_null(productVectorSpaceBase<Scalar>(domain_i_j,
false))) {
999 domainBlocks_[j] = domain_i_j;
1011template<
class Scalar>
1013Thyra::defaultBlockedLinearOp()
1015 return Teuchos::rcp(
new DefaultBlockedLinearOp<Scalar>());
1019template<
class Scalar>
1022 const RCP<
const LinearOpBase<Scalar> > &A00,
1023 const std::string &label
1026 RCP<PhysicallyBlockedLinearOpBase<Scalar> >
1027 M = defaultBlockedLinearOp<Scalar>();
1028 M->beginBlockFill(1,1);
1029 M->setBlock(0, 0, A00);
1032 M->setObjectLabel(label);
1037template<
class Scalar>
1040 const RCP<
const LinearOpBase<Scalar> > &A00,
1041 const RCP<
const LinearOpBase<Scalar> > &A01,
1042 const std::string &label
1045 RCP<PhysicallyBlockedLinearOpBase<Scalar> >
1046 M = defaultBlockedLinearOp<Scalar>();
1047 M->beginBlockFill(1,2);
1048 if(A00.get()) M->setBlock(0,0,A00);
1049 if(A01.get()) M->setBlock(0,1,A01);
1052 M->setObjectLabel(label);
1057template<
class Scalar>
1060 const RCP<
const LinearOpBase<Scalar> > &A00,
1061 const RCP<
const LinearOpBase<Scalar> > &A10,
1062 const std::string &label
1065 RCP<PhysicallyBlockedLinearOpBase<Scalar> >
1066 M = defaultBlockedLinearOp<Scalar>();
1067 M->beginBlockFill(2,1);
1068 if(A00.get()) M->setBlock(0,0,A00);
1069 if(A10.get()) M->setBlock(1,0,A10);
1072 M->setObjectLabel(label);
1077template<
class Scalar>
1080 const RCP<
const LinearOpBase<Scalar> > &A00,
1081 const RCP<
const LinearOpBase<Scalar> > &A01,
1082 const RCP<
const LinearOpBase<Scalar> > &A10,
1083 const RCP<
const LinearOpBase<Scalar> > &A11,
1084 const std::string &label
1087 RCP<PhysicallyBlockedLinearOpBase<Scalar> >
1088 M = defaultBlockedLinearOp<Scalar>();
1089 M->beginBlockFill(2,2);
1090 if(A00.get()) M->setBlock(0,0,A00);
1091 if(A01.get()) M->setBlock(0,1,A01);
1092 if(A10.get()) M->setBlock(1,0,A10);
1093 if(A11.get()) M->setBlock(1,1,A11);
1096 M->setObjectLabel(label);
1101template<
class Scalar>
1103Thyra::nonconstBlock1x1(
1104 const RCP<LinearOpBase<Scalar> > &A00,
1105 const std::string &label
1108 RCP<PhysicallyBlockedLinearOpBase<Scalar> >
1109 M = defaultBlockedLinearOp<Scalar>();
1110 M->beginBlockFill(1, 1);
1111 M->setNonconstBlock(0, 0, A00);
1114 M->setObjectLabel(label);
1119template<
class Scalar>
1121Thyra::nonconstBlock1x2(
1122 const RCP<LinearOpBase<Scalar> > &A00,
1123 const RCP<LinearOpBase<Scalar> > &A01,
1124 const std::string &label
1127 RCP<PhysicallyBlockedLinearOpBase<Scalar> >
1128 M = defaultBlockedLinearOp<Scalar>();
1129 M->beginBlockFill(1,2);
1130 if(A00.get()) M->setNonconstBlock(0,0,A00);
1131 if(A01.get()) M->setNonconstBlock(0,1,A01);
1134 M->setObjectLabel(label);
1139template<
class Scalar>
1141Thyra::nonconstBlock2x1(
1142 const RCP<LinearOpBase<Scalar> > &A00,
1143 const RCP<LinearOpBase<Scalar> > &A10,
1144 const std::string &label
1147 RCP<PhysicallyBlockedLinearOpBase<Scalar> >
1148 M = defaultBlockedLinearOp<Scalar>();
1149 M->beginBlockFill(2,1);
1150 if(A00.get()) M->setNonconstBlock(0,0,A00);
1151 if(A10.get()) M->setNonconstBlock(1,0,A10);
1154 M->setObjectLabel(label);
1159template<
class Scalar>
1161Thyra::nonconstBlock2x2(
1162 const RCP<LinearOpBase<Scalar> > &A00,
1163 const RCP<LinearOpBase<Scalar> > &A01,
1164 const RCP<LinearOpBase<Scalar> > &A10,
1165 const RCP<LinearOpBase<Scalar> > &A11,
1166 const std::string &label
1169 RCP<PhysicallyBlockedLinearOpBase<Scalar> >
1170 M = defaultBlockedLinearOp<Scalar>();
1171 M->beginBlockFill(2,2);
1172 if(A00.get()) M->setNonconstBlock(0,0,A00);
1173 if(A01.get()) M->setNonconstBlock(0,1,A01);
1174 if(A10.get()) M->setNonconstBlock(1,0,A10);
1175 if(A11.get()) M->setNonconstBlock(1,1,A11);
1178 M->setObjectLabel(label);
1190#define THYRA_DEFAULT_BLOCKED_LINEAR_OP_INSTANT(SCALAR) \
1192 template class DefaultBlockedLinearOp<SCALAR >; \
1194 template RCP<DefaultBlockedLinearOp<SCALAR > > \
1195 defaultBlockedLinearOp<SCALAR >(); \
1197 template RCP<const LinearOpBase<SCALAR > > \
1199 const RCP<const LinearOpBase<SCALAR > > &A00, \
1200 const std::string &label \
1203 template RCP<const LinearOpBase<SCALAR > > \
1205 const RCP<const LinearOpBase<SCALAR > > &A00, \
1206 const RCP<const LinearOpBase<SCALAR > > &A01, \
1207 const std::string &label \
1210 template RCP<const LinearOpBase<SCALAR > > \
1212 const RCP<const LinearOpBase<SCALAR > > &A00, \
1213 const RCP<const LinearOpBase<SCALAR > > &A10, \
1214 const std::string &label \
1217 template RCP<const LinearOpBase<SCALAR > > \
1219 const RCP<const LinearOpBase<SCALAR > > &A00, \
1220 const RCP<const LinearOpBase<SCALAR > > &A01, \
1221 const RCP<const LinearOpBase<SCALAR > > &A10, \
1222 const RCP<const LinearOpBase<SCALAR > > &A11, \
1223 const std::string &label \
1226 template RCP<LinearOpBase<SCALAR > > \
1228 const RCP<LinearOpBase<SCALAR > > &A00, \
1229 const std::string &label \
1232 template RCP<LinearOpBase<SCALAR > > \
1234 const RCP<LinearOpBase<SCALAR > > &A00, \
1235 const RCP<LinearOpBase<SCALAR > > &A01, \
1236 const std::string &label \
1239 template RCP<LinearOpBase<SCALAR > > \
1241 const RCP<LinearOpBase<SCALAR > > &A00, \
1242 const RCP<LinearOpBase<SCALAR > > &A10, \
1243 const std::string &label \
1246 template RCP<LinearOpBase<SCALAR > > \
1248 const RCP<LinearOpBase<SCALAR > > &A00, \
1249 const RCP<LinearOpBase<SCALAR > > &A01, \
1250 const RCP<LinearOpBase<SCALAR > > &A10, \
1251 const RCP<LinearOpBase<SCALAR > > &A11, \
1252 const std::string &label \
virtual std::string description() const
Concrete composite LinearOpBase subclass that creates single linear operator object out of a set of c...
Teuchos::RCP< const VectorSpaceBase< Scalar > > domain() const
bool blockIsConst(const int i, const int j) const
virtual void scaleRightImpl(const VectorBase< Scalar > &col_scaling)
void applyImpl(const EOpTransp M_trans, const MultiVectorBase< Scalar > &X, const Ptr< MultiVectorBase< Scalar > > &Y, const Scalar alpha, const Scalar beta) const
bool acceptsBlock(const int i, const int j) const
Teuchos::RCP< const VectorSpaceBase< Scalar > > range() const
virtual bool rowStatIsSupportedImpl(const RowStatLinearOpBaseUtils::ERowStat rowStat) const
virtual bool supportsScaleRightImpl() const
virtual bool supportsScaleLeftImpl() const
Teuchos::RCP< LinearOpBase< Scalar > > getNonconstBlock(const int i, const int j)
Teuchos::RCP< const LinearOpBase< Scalar > > getBlock(const int i, const int j) const
bool blockExists(const int i, const int j) const
virtual void scaleLeftImpl(const VectorBase< Scalar > &row_scaling)
void setNonconstBlock(const int i, const int j, const Teuchos::RCP< LinearOpBase< Scalar > > &block)
Teuchos::RCP< const ProductVectorSpaceBase< Scalar > > productDomain() const
void describe(Teuchos::FancyOStream &out, const Teuchos::EVerbosityLevel verbLevel) const
Prints the details about the constituent linear operators.
Teuchos::RCP< const ProductVectorSpaceBase< Scalar > > productRange() const
virtual void getRowStatImpl(const RowStatLinearOpBaseUtils::ERowStat rowStat, const Teuchos::Ptr< VectorBase< Scalar > > &rowStatVec) const
void setBlock(const int i, const int j, const Teuchos::RCP< const LinearOpBase< Scalar > > &block)
std::string description() const
Prints just the name DefaultBlockedLinearOp along with the overall dimensions and the number of const...
bool blockFillIsActive() const
Teuchos::RCP< const LinearOpBase< Scalar > > clone() const
bool opSupportedImpl(EOpTransp M_trans) const
Returns true only if all constituent operators support M_trans.
Base class for all linear operators.
Interface for a collection of column vectors called a multi-vector.
Base interface for product vectors.
virtual RCP< const VectorBase< Scalar > > getVectorBlock(const int k) const =0
Returns a non-persisting const view of the (zero-based) kth block vector.
Interface for exxtracting row statistics as a VectorBase from a supporting LinearOpBase object.
Applies left or right sclaing to the linear operator.
Abstract interface for finite-dimensional dense vectors.
#define TEUCHOS_ASSERT(assertion_test)
#define TEUCHOS_TEST_FOR_EXCEPT(throw_exception_test)
#define TEUCHOS_TEST_FOR_EXCEPTION(throw_exception_test, Exception, msg)
#define TEUCHOS_ASSERT_INEQUALITY(val1, comp, val2)
bool is_null(const std::shared_ptr< T > &p)
#define THYRA_ASSERT_LINEAR_OP_MULTIVEC_APPLY_SPACES(FUNC_NAME, M, M_T, X, Y)
This is a very useful macro that should be used to validate that the spaces for the multi-vector vers...
#define THYRA_ASSERT_VEC_SPACES_NAMES(FUNC_NAME, VS1, VS1_NAME, VS2, VS2_NAME)
Helper assertion macro.
void unwrap(const LinearOpBase< Scalar > &Op, Scalar *scalar, EOpTransp *transp, const LinearOpBase< Scalar > **origOp)
Extract the overallScalar, overallTransp and const origOp from a const LinearOpBase object.
EOpTransp
Enumeration for determining how a linear operator is applied. `*.
const char * toString(EConj conj)
Return a string name for a EOpTransp value. `*.
EOpTransp real_trans(EOpTransp transp)
Return NOTRANS or TRANS for real scalar valued operators and this also is used for determining struct...
@ TRANS
Use the transposed operator.
@ NOTRANS
Use the non-transposed operator.
@ CONJTRANS
Use the transposed operator with complex-conjugate clements (same as TRANS for real scalar types).
@ CONJ
Use the non-transposed operator with complex-conjugate elements (same as NOTRANS for real scalar type...
TypeTo as(const TypeFrom &t)
T_To & dyn_cast(T_From &from)
std::string toString(const T &t)
TEUCHOS_DEPRECATED RCP< T > rcp(T *p, Dealloc_T dealloc, bool owns_mem)