10#ifndef THYRA_MUELU_MAXWELL1_PRECONDITIONER_FACTORY_DEF_HPP
11#define THYRA_MUELU_MAXWELL1_PRECONDITIONER_FACTORY_DEF_HPP
15#include <Xpetra_CrsMatrixWrap.hpp>
16#include <Xpetra_CrsMatrix.hpp>
17#include <Xpetra_Matrix.hpp>
18#include <Xpetra_ThyraUtils.hpp>
19#include <MueLu_Maxwell1.hpp>
20#include <Xpetra_TpetraHalfPrecisionOperator.hpp>
22#if defined(HAVE_MUELU_STRATIMIKOS) && defined(HAVE_MUELU_THYRA)
25#if ((defined(HAVE_TPETRA_INST_DOUBLE) && defined(HAVE_TPETRA_INST_FLOAT) && !defined(HAVE_TPETRA_INST_COMPLEX_DOUBLE) && !defined(HAVE_TPETRA_INST_COMPLEX_FLOAT)) || \
26 (!defined(HAVE_TPETRA_INST_DOUBLE) && !defined(HAVE_TPETRA_INST_FLOAT) && defined(HAVE_TPETRA_INST_COMPLEX_DOUBLE) && defined(HAVE_TPETRA_INST_COMPLEX_FLOAT)) || \
27 (defined(HAVE_TPETRA_INST_DOUBLE) && defined(HAVE_TPETRA_INST_FLOAT) && defined(HAVE_TPETRA_INST_COMPLEX_DOUBLE) && defined(HAVE_TPETRA_INST_COMPLEX_FLOAT)))
28#define MUELU_CAN_USE_MIXED_PRECISION
33using Teuchos::ParameterList;
36using Teuchos::rcp_const_cast;
37using Teuchos::rcp_dynamic_cast;
41template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
42MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::MueLuMaxwell1PreconditionerFactory()
43 : paramList_(rcp(new ParameterList())) {}
47template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
48bool MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::isCompatible(
const LinearOpSourceBase<Scalar>& fwdOpSrc)
const {
49 const RCP<const LinearOpBase<Scalar>> fwdOp = fwdOpSrc.getOp();
51 if (Xpetra::ThyraUtils<Scalar, LocalOrdinal, GlobalOrdinal, Node>::isTpetra(fwdOp))
return true;
56template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
57RCP<PreconditionerBase<Scalar>> MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::createPrec()
const {
58 return Teuchos::rcp(
new DefaultPreconditioner<Scalar>);
61template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
62void MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::
63 initializePrec(
const RCP<
const LinearOpSourceBase<Scalar>>& fwdOpSrc, PreconditionerBase<Scalar>* prec,
const ESupportSolveUse supportSolveUse)
const {
65 typedef Xpetra::Operator<Scalar, LocalOrdinal, GlobalOrdinal, Node> XpOp;
66 typedef Xpetra::ThyraUtils<Scalar, LocalOrdinal, GlobalOrdinal, Node> XpThyUtils;
67 typedef Xpetra::Matrix<Scalar, LocalOrdinal, GlobalOrdinal, Node> XpMat;
68 typedef Thyra::LinearOpBase<Scalar> ThyLinOpBase;
70#if defined(MUELU_CAN_USE_MIXED_PRECISION)
71 typedef Xpetra::TpetraHalfPrecisionOperator<Scalar, LocalOrdinal, GlobalOrdinal, Node> XpHalfPrecOp;
72 typedef Xpetra::MultiVector<Scalar, LocalOrdinal, GlobalOrdinal, Node> XpMV;
73 typedef typename XpHalfPrecOp::HalfScalar HalfScalar;
74 typedef typename Teuchos::ScalarTraits<Scalar>::magnitudeType Magnitude;
75 typedef typename Teuchos::ScalarTraits<Magnitude>::halfPrecision HalfMagnitude;
76 typedef Xpetra::MultiVector<HalfScalar, LocalOrdinal, GlobalOrdinal, Node> XphMV;
77 typedef Xpetra::MultiVector<Magnitude, LocalOrdinal, GlobalOrdinal, Node> XpmMV;
78 typedef Xpetra::MultiVector<HalfMagnitude, LocalOrdinal, GlobalOrdinal, Node> XphmMV;
79 typedef Xpetra::Matrix<HalfScalar, LocalOrdinal, GlobalOrdinal, Node> XphMat;
81 Teuchos::TimeMonitor tM(*Teuchos::TimeMonitor::getNewTimer(std::string(
"ThyraMueLuMaxwell1::initializePrec")));
84 TEUCHOS_ASSERT(Teuchos::nonnull(fwdOpSrc));
85 TEUCHOS_ASSERT(this->isCompatible(*fwdOpSrc));
89 ParameterList paramList = *paramList_;
92 const RCP<const ThyLinOpBase> fwdOp = fwdOpSrc->getOp();
93 TEUCHOS_TEST_FOR_EXCEPT(Teuchos::is_null(fwdOp));
96 bool bIsTpetra = XpThyUtils::isTpetra(fwdOp);
97 TEUCHOS_TEST_FOR_EXCEPT((bIsTpetra ==
false));
101 RCP<XpMat> A = XpThyUtils::toXpetra(Teuchos::rcp_const_cast<ThyLinOpBase>(fwdOp));
102 TEUCHOS_TEST_FOR_EXCEPT(Teuchos::is_null(A));
105 const Teuchos::Ptr<DefaultPreconditioner<Scalar>> defaultPrec = Teuchos::ptr(
dynamic_cast<DefaultPreconditioner<Scalar>*
>(prec));
106 TEUCHOS_TEST_FOR_EXCEPT(Teuchos::is_null(defaultPrec));
109 RCP<ThyLinOpBase> thyra_precOp = Teuchos::null;
110 thyra_precOp = rcp_dynamic_cast<Thyra::LinearOpBase<Scalar>>(defaultPrec->getNonconstUnspecifiedPrecOp(),
true);
115 const bool startingOver = (thyra_precOp.is_null() || !paramList.isParameter(
"Maxwell1: enable reuse") || !paramList.get<
bool>(
"Maxwell1: enable reuse"));
116 const bool useHalfPrecision = paramList.get<
bool>(
"half precision",
false) && bIsTpetra;
119 if (startingOver ==
true) {
121 std::list<std::string> convertXpetra = {
"Coordinates",
"Nullspace",
"Kn",
"D0"};
122 for (
auto it = convertXpetra.begin(); it != convertXpetra.end(); ++it)
123 Converters<Scalar, LocalOrdinal, GlobalOrdinal, Node>::replaceWithXpetra(paramList, *it);
125 std::list<std::string> sublists = {
"maxwell1: 11list",
"maxwell1: 22list"};
126 for (
auto itSublist = sublists.begin(); itSublist != sublists.end(); ++itSublist)
127 if (paramList.isSublist(*itSublist)) {
128 ParameterList& sublist = paramList.sublist(*itSublist);
129 for (
int lvlNo = 0; lvlNo < 10; ++lvlNo) {
130 if (sublist.isSublist(
"level " + std::to_string(lvlNo) +
" user data")) {
131 ParameterList& lvlList = sublist.sublist(
"level " + std::to_string(lvlNo) +
" user data");
132 std::list<std::string> convertKeys;
133 for (
auto it = lvlList.begin(); it != lvlList.end(); ++it)
134 convertKeys.push_back(lvlList.name(it));
135 for (
auto it = convertKeys.begin(); it != convertKeys.end(); ++it)
136 Converters<Scalar, LocalOrdinal, GlobalOrdinal, Node>::replaceWithXpetra(lvlList, *it);
141 ParameterList& sublist = paramList.sublist(
"maxwell1: 11list");
142 if (sublist.isParameter(
"D0")) {
143 Converters<Scalar, LocalOrdinal, GlobalOrdinal, Node>::replaceWithXpetra(sublist,
"D0");
146 paramList.set<
bool>(
"Maxwell1: use as preconditioner",
true);
147 if (useHalfPrecision) {
148#if defined(MUELU_CAN_USE_MIXED_PRECISION)
151 RCP<XphMat> halfA = Xpetra::convertToHalfPrecision(A);
152 if (paramList.isType<RCP<XpmMV>>(
"Coordinates")) {
153 RCP<XpmMV> coords = paramList.get<RCP<XpmMV>>(
"Coordinates");
154 paramList.remove(
"Coordinates");
155 RCP<XphmMV> halfCoords = Xpetra::convertToHalfPrecision(coords);
156 paramList.set(
"Coordinates", halfCoords);
158 if (paramList.isType<RCP<XpMV>>(
"Nullspace")) {
159 RCP<XpMV> nullspace = paramList.get<RCP<XpMV>>(
"Nullspace");
160 paramList.remove(
"Nullspace");
161 RCP<XphMV> halfNullspace = Xpetra::convertToHalfPrecision(nullspace);
162 paramList.set(
"Nullspace", halfNullspace);
164 std::list<std::string> convertMat = {
"Kn",
"D0"};
165 for (
auto it = convertMat.begin(); it != convertMat.end(); ++it) {
166 if (paramList.isType<RCP<XpMat>>(*it)) {
167 RCP<XpMat> M = paramList.get<RCP<XpMat>>(*it);
168 paramList.remove(*it);
169 RCP<XphMat> halfM = Xpetra::convertToHalfPrecision(M);
170 paramList.set(*it, halfM);
176 xpPrecOp = rcp(
new XpHalfPrecOp(halfPrec));
178 TEUCHOS_TEST_FOR_EXCEPT(
true);
183 xpPrecOp = rcp_dynamic_cast<XpOp>(preconditioner);
188 RCP<ThyXpOp> thyXpOp = rcp_dynamic_cast<ThyXpOp>(thyra_precOp,
true);
189 RCP<XpOp> xpOp = thyXpOp->getXpetraOperator();
190#if defined(MUELU_CAN_USE_MIXED_PRECISION)
191 RCP<XpHalfPrecOp> xpHalfPrecOp = rcp_dynamic_cast<XpHalfPrecOp>(xpOp);
192 if (!xpHalfPrecOp.is_null()) {
193 RCP<MueLu::Maxwell1<HalfScalar, LocalOrdinal, GlobalOrdinal, Node>> preconditioner = rcp_dynamic_cast<MueLu::Maxwell1<HalfScalar, LocalOrdinal, GlobalOrdinal, Node>>(xpHalfPrecOp->GetHalfPrecisionOperator(),
true);
194 RCP<XphMat> halfA = Xpetra::convertToHalfPrecision(A);
195 preconditioner->resetMatrix(halfA);
196 xpPrecOp = rcp_dynamic_cast<XpOp>(preconditioner);
200 RCP<MueLu::Maxwell1<Scalar, LocalOrdinal, GlobalOrdinal, Node>> preconditioner = rcp_dynamic_cast<MueLu::Maxwell1<Scalar, LocalOrdinal, GlobalOrdinal, Node>>(xpOp,
true);
201 preconditioner->resetMatrix(A);
202 xpPrecOp = rcp_dynamic_cast<XpOp>(preconditioner);
207 RCP<const VectorSpaceBase<Scalar>> thyraRangeSpace = Xpetra::ThyraUtils<Scalar, LocalOrdinal, GlobalOrdinal, Node>::toThyra(xpPrecOp->getRangeMap());
208 RCP<const VectorSpaceBase<Scalar>> thyraDomainSpace = Xpetra::ThyraUtils<Scalar, LocalOrdinal, GlobalOrdinal, Node>::toThyra(xpPrecOp->getDomainMap());
210 RCP<ThyLinOpBase> thyraPrecOp = Thyra::xpetraLinearOp<Scalar, LocalOrdinal, GlobalOrdinal, Node>(thyraRangeSpace, thyraDomainSpace, xpPrecOp);
211 TEUCHOS_TEST_FOR_EXCEPT(Teuchos::is_null(thyraPrecOp));
213 defaultPrec->initializeUnspecified(thyraPrecOp);
216template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
217void MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::
218 uninitializePrec(PreconditionerBase<Scalar>* prec, RCP<
const LinearOpSourceBase<Scalar>>* fwdOp, ESupportSolveUse* supportSolveUse)
const {
219 TEUCHOS_ASSERT(prec);
222 const Teuchos::Ptr<DefaultPreconditioner<Scalar>> defaultPrec = Teuchos::ptr(
dynamic_cast<DefaultPreconditioner<Scalar>*
>(prec));
223 TEUCHOS_TEST_FOR_EXCEPT(Teuchos::is_null(defaultPrec));
227 *fwdOp = Teuchos::null;
230 if (supportSolveUse) {
232 *supportSolveUse = Thyra::SUPPORT_SOLVE_UNSPECIFIED;
235 defaultPrec->uninitialize();
239template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
240void MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::setParameterList(RCP<ParameterList>
const& paramList) {
241 TEUCHOS_TEST_FOR_EXCEPT(Teuchos::is_null(paramList));
242 paramList_ = paramList;
245template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
246RCP<ParameterList> MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::getNonconstParameterList() {
250template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
251RCP<ParameterList> MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::unsetParameterList() {
252 RCP<ParameterList> savedParamList = paramList_;
253 paramList_ = Teuchos::null;
254 return savedParamList;
257template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
258RCP<const ParameterList> MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::getParameterList()
const {
262template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
263RCP<const ParameterList> MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::getValidParameters()
const {
264 static RCP<const ParameterList> validPL;
266 if (Teuchos::is_null(validPL))
267 validPL = rcp(
new ParameterList());
273template <
class Scalar,
class LocalOrdinal,
class GlobalOrdinal,
class Node>
274std::string MueLuMaxwell1PreconditionerFactory<Scalar, LocalOrdinal, GlobalOrdinal, Node>::description()
const {
275 return "Thyra::MueLuMaxwell1PreconditionerFactory";
Preconditioner (wrapped as a Xpetra::Operator) for Maxwell's equations in curl-curl form.
Concrete Thyra::LinearOpBase subclass for Xpetra::Operator.