10#ifndef IFPACK2_HYPRE_DEF_HPP
11#define IFPACK2_HYPRE_DEF_HPP
13#include "Ifpack2_Hypre_decl.hpp"
14#if defined(HAVE_IFPACK2_HYPRE) && defined(HAVE_IFPACK2_MPI)
17#include "Tpetra_Import.hpp"
18#include "Teuchos_ParameterList.hpp"
19#include "Teuchos_RCP.hpp"
20#include "Teuchos_DefaultMpiComm.hpp"
21#include "HYPRE_IJ_mv.h"
22#include "HYPRE_parcsr_ls.h"
24#include "_hypre_parcsr_mv.h"
25#include "_hypre_IJ_mv.h"
26#include "HYPRE_parcsr_mv.h"
31using Teuchos::rcpFromRef;
35template <
class LocalOrdinal,
class Node>
36Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::
37 Hypre(
const Teuchos::RCP<const row_matrix_type> &A)
39 , IsInitialized_(false)
44 , InitializeTime_(0.0)
52 , IsSolverCreated_(false)
53 , IsPrecondCreated_(false)
54 , SolveOrPrec_(Hypre_Is_Solver)
57 , PrecondType_(Euclid)
58 , UsePreconditioner_(false)
62template <
class LocalOrdinal,
class Node>
63Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::~Hypre() {
68template <
class LocalOrdinal,
class Node>
69void Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Destroy() {
70 if (isInitialized()) {
71 IFPACK2_CHK_ERRV(HYPRE_IJMatrixDestroy(HypreA_));
72 IFPACK2_CHK_ERRV(HYPRE_IJVectorDestroy(XHypre_));
73 IFPACK2_CHK_ERRV(HYPRE_IJVectorDestroy(YHypre_));
75 if (IsSolverCreated_) {
76 IFPACK2_CHK_ERRV(SolverDestroyPtr_(Solver_));
78 if (IsPrecondCreated_) {
79 IFPACK2_CHK_ERRV(PrecondDestroyPtr_(Preconditioner_));
84 IFPACK2_CHK_ERRV(HYPRE_IJMatrixDestroy(HypreG_));
87 IFPACK2_CHK_ERRV(HYPRE_IJVectorDestroy(xHypre_));
90 IFPACK2_CHK_ERRV(HYPRE_IJVectorDestroy(yHypre_));
93 IFPACK2_CHK_ERRV(HYPRE_IJVectorDestroy(zHypre_));
98template <
class LocalOrdinal,
class Node>
99void Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::initialize() {
100 const std::string timerName(
"Ifpack2::Hypre::initialize");
101 Teuchos::RCP<Teuchos::Time> timer = Teuchos::TimeMonitor::lookupCounter(timerName);
102 if (timer.is_null()) timer = Teuchos::TimeMonitor::getNewCounter(timerName);
104 if (IsInitialized_)
return;
105 double startTime = timer->wallTime();
107 Teuchos::TimeMonitor timeMon(*timer);
109 MPI_Comm comm = *(Teuchos::rcp_dynamic_cast<const Teuchos::MpiComm<int> >(A_->getRowMap()->getComm())->getRawMpiComm());
114 if (!A_->getRowMap()->isSameAs(*A_->getRangeMap())) {
115 IFPACK2_CHK_ERRV(-1);
118 if (A_->getRowMap()->isContiguous()) {
119 GloballyContiguousRowMap_ = A_->getRowMap();
120 GloballyContiguousColMap_ = A_->getColMap();
123 if (A_->getDomainMap()->isSameAs(*A_->getRowMap())) {
124 Teuchos::RCP<const crs_matrix_type> Aconst = Teuchos::rcp_dynamic_cast<const crs_matrix_type>(A_);
125 GloballyContiguousColMap_ = MakeContiguousColumnMap(Aconst);
126 GloballyContiguousRowMap_ = rcp(
new map_type(A_->getRowMap()->getGlobalNumElements(),
127 A_->getRowMap()->getLocalNumElements(), 0, A_->getRowMap()->getComm()));
129 throw std::runtime_error(
"Ifpack_Hypre: Unsupported map configuration: Row/Domain maps do not match");
133 HYPRE_Int ilower = GloballyContiguousRowMap_->getMinGlobalIndex();
134 HYPRE_Int iupper = GloballyContiguousRowMap_->getMaxGlobalIndex();
136 IFPACK2_CHK_ERRV(HYPRE_IJVectorCreate(comm, ilower, iupper, &XHypre_));
137 IFPACK2_CHK_ERRV(HYPRE_IJVectorSetObjectType(XHypre_, HYPRE_PARCSR));
138 IFPACK2_CHK_ERRV(HYPRE_IJVectorInitialize(XHypre_));
139 IFPACK2_CHK_ERRV(HYPRE_IJVectorAssemble(XHypre_));
140 IFPACK2_CHK_ERRV(HYPRE_IJVectorGetObject(XHypre_, (
void **)&ParX_));
141 XVec_ = Teuchos::rcp((hypre_ParVector *)hypre_IJVectorObject(((hypre_IJVector *)XHypre_)),
false);
144 IFPACK2_CHK_ERRV(HYPRE_IJVectorCreate(comm, ilower, iupper, &YHypre_));
145 IFPACK2_CHK_ERRV(HYPRE_IJVectorSetObjectType(YHypre_, HYPRE_PARCSR));
146 IFPACK2_CHK_ERRV(HYPRE_IJVectorInitialize(YHypre_));
147 IFPACK2_CHK_ERRV(HYPRE_IJVectorAssemble(YHypre_));
148 IFPACK2_CHK_ERRV(HYPRE_IJVectorGetObject(YHypre_, (
void **)&ParY_));
149 YVec_ = Teuchos::rcp((hypre_ParVector *)hypre_IJVectorObject(((hypre_IJVector *)YHypre_)),
false);
152 VectorCache_.resize(A_->getRowMap()->getLocalNumElements());
155 IsInitialized_ =
true;
158 InitializeTime_ += (timer->wallTime() - startTime);
162template <
class LocalOrdinal,
class Node>
163void Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::setParameters(
const Teuchos::ParameterList &list) {
164 std::map<std::string, Hypre_Solver> solverMap;
165 solverMap[
"BoomerAMG"] = BoomerAMG;
166 solverMap[
"ParaSails"] = ParaSails;
167 solverMap[
"Euclid"] = Euclid;
168 solverMap[
"AMS"] = AMS;
169 solverMap[
"Hybrid"] = Hybrid;
170 solverMap[
"PCG"] = PCG;
171 solverMap[
"GMRES"] =
GMRES;
172 solverMap[
"FlexGMRES"] = FlexGMRES;
173 solverMap[
"LGMRES"] = LGMRES;
174 solverMap[
"BiCGSTAB"] = BiCGSTAB;
176 std::map<std::string, Hypre_Chooser> chooserMap;
177 chooserMap[
"Solver"] = Hypre_Is_Solver;
178 chooserMap[
"Preconditioner"] = Hypre_Is_Preconditioner;
181 Hypre_Solver solType;
182 if (list.isType<std::string>(
"hypre: Solver"))
183 solType = solverMap[list.get<std::string>(
"hypre: Solver")];
184 else if (list.isParameter(
"hypre: Solver"))
185 solType = (Hypre_Solver)list.get<
int>(
"hypre: Solver");
188 SolverType_ = solType;
189 Hypre_Solver precType;
190 if (list.isType<std::string>(
"hypre: Preconditioner"))
191 precType = solverMap[list.get<std::string>(
"hypre: Preconditioner")];
192 else if (list.isParameter(
"hypre: Preconditioner"))
193 precType = (Hypre_Solver)list.get<
int>(
"hypre: Preconditioner");
196 PrecondType_ = precType;
197 Hypre_Chooser chooser;
198 if (list.isType<std::string>(
"hypre: SolveOrPrecondition"))
199 chooser = chooserMap[list.get<std::string>(
"hypre: SolveOrPrecondition")];
200 else if (list.isParameter(
"hypre: SolveOrPrecondition"))
201 chooser = (Hypre_Chooser)list.get<
int>(
"hypre: SolveOrPrecondition");
203 chooser = Hypre_Is_Solver;
204 SolveOrPrec_ = chooser;
205 bool SetPrecond = list.isParameter(
"hypre: SetPreconditioner") ? list.get<
bool>(
"hypre: SetPreconditioner") : false;
206 IFPACK2_CHK_ERR(SetParameter(SetPrecond));
207 int NumFunctions = list.isParameter(
"hypre: NumFunctions") ? list.get<
int>(
"hypre: NumFunctions") : 0;
210 if (NumFunctions > 0) {
211 RCP<FunctionParameter> *params = list.get<RCP<FunctionParameter> *>(
"hypre: Functions");
212 for (
int i = 0; i < NumFunctions; i++) {
213 IFPACK2_CHK_ERR(AddFunToList(params[i]));
217 if (list.isSublist(
"hypre: Solver functions")) {
218 Teuchos::ParameterList solverList = list.sublist(
"hypre: Solver functions");
219 for (
auto it = solverList.begin(); it != solverList.end(); ++it) {
220 std::string funct_name = it->first;
221 if (it->second.isType<HYPRE_Int>()) {
222 IFPACK2_CHK_ERR(AddFunToList(rcp(
new FunctionParameter(Hypre_Is_Solver, funct_name, Teuchos::getValue<HYPRE_Int>(it->second)))));
223 }
else if (!std::is_same<HYPRE_Int, int>::value && it->second.isType<
int>()) {
224 IFPACK2_CHK_ERR(AddFunToList(rcp(
new FunctionParameter(Hypre_Is_Solver, funct_name, Teuchos::as<HYPRE_Int>(Teuchos::getValue<int>(it->second))))));
225 }
else if (it->second.isType<HYPRE_Real>()) {
226 IFPACK2_CHK_ERR(AddFunToList(rcp(
new FunctionParameter(Hypre_Is_Solver, funct_name, Teuchos::getValue<HYPRE_Real>(it->second)))));
233 if (list.isSublist(
"hypre: Preconditioner functions")) {
234 Teuchos::ParameterList precList = list.sublist(
"hypre: Preconditioner functions");
235 for (
auto it = precList.begin(); it != precList.end(); ++it) {
236 std::string funct_name = it->first;
237 if (it->second.isType<HYPRE_Int>()) {
238 IFPACK2_CHK_ERR(AddFunToList(rcp(
new FunctionParameter(Hypre_Is_Preconditioner, funct_name, Teuchos::getValue<HYPRE_Int>(it->second)))));
239 }
else if (!std::is_same<HYPRE_Int, int>::value && it->second.isType<
int>()) {
240 IFPACK2_CHK_ERR(AddFunToList(rcp(
new FunctionParameter(Hypre_Is_Preconditioner, funct_name, Teuchos::as<HYPRE_Int>(Teuchos::getValue<int>(it->second))))));
241 }
else if (it->second.isType<HYPRE_Real>()) {
242 IFPACK2_CHK_ERR(AddFunToList(rcp(
new FunctionParameter(Hypre_Is_Preconditioner, funct_name, Teuchos::getValue<HYPRE_Real>(it->second)))));
243 }
else if (it->second.isList()) {
244 Teuchos::ParameterList pl = Teuchos::getValue<Teuchos::ParameterList>(it->second);
245 if (FunctionParameter::isFuncIntInt(funct_name)) {
246 HYPRE_Int arg0 = pl.get<HYPRE_Int>(
"arg 0");
247 HYPRE_Int arg1 = pl.get<HYPRE_Int>(
"arg 1");
248 IFPACK2_CHK_ERR(AddFunToList(rcp(
new FunctionParameter(Hypre_Is_Preconditioner, funct_name, arg0, arg1))));
249 }
else if (FunctionParameter::isFuncIntIntDoubleDouble(funct_name)) {
250 HYPRE_Int arg0 = pl.get<HYPRE_Int>(
"arg 0");
251 HYPRE_Int arg1 = pl.get<HYPRE_Int>(
"arg 1");
252 HYPRE_Real arg2 = pl.get<HYPRE_Real>(
"arg 2");
253 HYPRE_Real arg3 = pl.get<HYPRE_Real>(
"arg 3");
254 IFPACK2_CHK_ERR(AddFunToList(rcp(
new FunctionParameter(Hypre_Is_Preconditioner, funct_name, arg0, arg1, arg2, arg3))));
255 }
else if (FunctionParameter::isFuncIntIntIntDoubleIntInt(funct_name)) {
256 HYPRE_Int arg0 = pl.get<HYPRE_Int>(
"arg 0");
257 HYPRE_Int arg1 = pl.get<HYPRE_Int>(
"arg 1");
258 HYPRE_Int arg2 = pl.get<HYPRE_Int>(
"arg 2");
259 HYPRE_Real arg3 = pl.get<HYPRE_Real>(
"arg 3");
260 HYPRE_Int arg4 = pl.get<HYPRE_Int>(
"arg 4");
261 HYPRE_Int arg5 = pl.get<HYPRE_Int>(
"arg 5");
262 IFPACK2_CHK_ERR(AddFunToList(rcp(
new FunctionParameter(Hypre_Is_Preconditioner, funct_name, arg0, arg1, arg2, arg3, arg4, arg5))));
270 if (list.isSublist(
"Coordinates") && list.sublist(
"Coordinates").isType<Teuchos::RCP<multivector_type> >(
"Coordinates"))
271 Coords_ = list.sublist(
"Coordinates").get<Teuchos::RCP<multivector_type> >(
"Coordinates");
272 if (list.isSublist(
"Operators") && list.sublist(
"Operators").isType<Teuchos::RCP<const crs_matrix_type> >(
"G"))
273 G_ = list.sublist(
"Operators").get<Teuchos::RCP<const crs_matrix_type> >(
"G");
275 Dump_ = list.isParameter(
"hypre: Dump") ? list.get<
bool>(
"hypre: Dump") : false;
279template <
class LocalOrdinal,
class Node>
280int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::AddFunToList(RCP<FunctionParameter> NewFun) {
281 NumFunsToCall_ = NumFunsToCall_ + 1;
282 FunsToCall_.resize(NumFunsToCall_);
283 FunsToCall_[NumFunsToCall_ - 1] = NewFun;
288template <
class LocalOrdinal,
class Node>
289int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetParameter(Hypre_Chooser chooser, HYPRE_Int (*pt2Func)(HYPRE_Solver, HYPRE_Int), HYPRE_Int parameter) {
290 RCP<FunctionParameter> temp = rcp(
new FunctionParameter(chooser, pt2Func, parameter));
291 IFPACK2_CHK_ERR(AddFunToList(temp));
296template <
class LocalOrdinal,
class Node>
297int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetParameter(Hypre_Chooser chooser, HYPRE_Int (*pt2Func)(HYPRE_Solver, HYPRE_Real), HYPRE_Real parameter) {
298 RCP<FunctionParameter> temp = rcp(
new FunctionParameter(chooser, pt2Func, parameter));
299 IFPACK2_CHK_ERR(AddFunToList(temp));
304template <
class LocalOrdinal,
class Node>
305int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetParameter(Hypre_Chooser chooser, HYPRE_Int (*pt2Func)(HYPRE_Solver, HYPRE_Real, HYPRE_Int), HYPRE_Real parameter1, HYPRE_Int parameter2) {
306 RCP<FunctionParameter> temp = rcp(
new FunctionParameter(chooser, pt2Func, parameter1, parameter2));
307 IFPACK2_CHK_ERR(AddFunToList(temp));
312template <
class LocalOrdinal,
class Node>
313int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetParameter(Hypre_Chooser chooser, HYPRE_Int (*pt2Func)(HYPRE_Solver, HYPRE_Int, HYPRE_Real), HYPRE_Int parameter1, HYPRE_Real parameter2) {
314 RCP<FunctionParameter> temp = rcp(
new FunctionParameter(chooser, pt2Func, parameter1, parameter2));
315 IFPACK2_CHK_ERR(AddFunToList(temp));
320template <
class LocalOrdinal,
class Node>
321int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetParameter(Hypre_Chooser chooser, HYPRE_Int (*pt2Func)(HYPRE_Solver, HYPRE_Int, HYPRE_Int), HYPRE_Int parameter1, HYPRE_Int parameter2) {
322 RCP<FunctionParameter> temp = rcp(
new FunctionParameter(chooser, pt2Func, parameter1, parameter2));
323 IFPACK2_CHK_ERR(AddFunToList(temp));
328template <
class LocalOrdinal,
class Node>
329int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetParameter(Hypre_Chooser chooser, HYPRE_Int (*pt2Func)(HYPRE_Solver, HYPRE_Real *), HYPRE_Real *parameter) {
330 RCP<FunctionParameter> temp = rcp(
new FunctionParameter(chooser, pt2Func, parameter));
331 IFPACK2_CHK_ERR(AddFunToList(temp));
336template <
class LocalOrdinal,
class Node>
337int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetParameter(Hypre_Chooser chooser, HYPRE_Int (*pt2Func)(HYPRE_Solver, HYPRE_Int *), HYPRE_Int *parameter) {
338 RCP<FunctionParameter> temp = rcp(
new FunctionParameter(chooser, pt2Func, parameter));
339 IFPACK2_CHK_ERR(AddFunToList(temp));
344template <
class LocalOrdinal,
class Node>
345int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetParameter(Hypre_Chooser chooser, HYPRE_Int (*pt2Func)(HYPRE_Solver, HYPRE_Int **), HYPRE_Int **parameter) {
346 RCP<FunctionParameter> temp = rcp(
new FunctionParameter(chooser, pt2Func, parameter));
347 IFPACK2_CHK_ERR(AddFunToList(temp));
352template <
class LocalOrdinal,
class Node>
353int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetParameter(Hypre_Chooser chooser, Hypre_Solver solver) {
354 if (chooser == Hypre_Is_Solver) {
355 SolverType_ = solver;
357 PrecondType_ = solver;
363template <
class LocalOrdinal,
class Node>
364int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetDiscreteGradient(Teuchos::RCP<const crs_matrix_type> G) {
365 using LO = local_ordinal_type;
366 using GO = global_ordinal_type;
370 if (!A_->getRowMap()->isSameAs(*G->getRowMap()))
371 throw std::runtime_error(
"Hypre<Tpetra::RowMatrix<double, HYPRE_Int, long long, Node>: Edge map mismatch: A and discrete gradient");
374 GloballyContiguousNodeRowMap_ = rcp(
new map_type(G->getDomainMap()->getGlobalNumElements(),
375 G->getDomainMap()->getLocalNumElements(), 0, A_->getRowMap()->getComm()));
376 GloballyContiguousNodeColMap_ = MakeContiguousColumnMap(G);
379 MPI_Comm comm = *(Teuchos::rcp_dynamic_cast<const Teuchos::MpiComm<int> >(A_->getRowMap()->getComm())->getRawMpiComm());
380 GO ilower = GloballyContiguousRowMap_->getMinGlobalIndex();
381 GO iupper = GloballyContiguousRowMap_->getMaxGlobalIndex();
382 GO jlower = GloballyContiguousNodeRowMap_->getMinGlobalIndex();
383 GO jupper = GloballyContiguousNodeRowMap_->getMaxGlobalIndex();
384 IFPACK2_CHK_ERR(HYPRE_IJMatrixCreate(comm, ilower, iupper, jlower, jupper, &HypreG_));
385 IFPACK2_CHK_ERR(HYPRE_IJMatrixSetObjectType(HypreG_, HYPRE_PARCSR));
386 IFPACK2_CHK_ERR(HYPRE_IJMatrixInitialize(HypreG_));
388 std::vector<GO> new_indices(G->getLocalMaxNumRowEntries());
389 for (LO i = 0; i < (LO)G->getLocalNumRows(); i++) {
390 typename crs_matrix_type::values_host_view_type values;
391 typename crs_matrix_type::local_inds_host_view_type indices;
392 G->getLocalRowView(i, indices, values);
393 for (LO j = 0; j < (LO)indices.extent(0); j++) {
394 new_indices[j] = GloballyContiguousNodeColMap_->getGlobalElement(indices(j));
397 GO numEntries = (GO)indices.extent(0);
398 GlobalRow[0] = GloballyContiguousRowMap_->getGlobalElement(i);
399 IFPACK2_CHK_ERR(HYPRE_IJMatrixSetValues(HypreG_, 1, &numEntries, GlobalRow, new_indices.data(), values.data()));
401 IFPACK2_CHK_ERR(HYPRE_IJMatrixAssemble(HypreG_));
402 IFPACK2_CHK_ERR(HYPRE_IJMatrixGetObject(HypreG_, (
void **)&ParMatrixG_));
405 HYPRE_ParCSRMatrixPrint(ParMatrixG_,
"G.mat");
407 if (SolverType_ == AMS)
408 HYPRE_AMSSetDiscreteGradient(Solver_, ParMatrixG_);
409 if (PrecondType_ == AMS)
410 HYPRE_AMSSetDiscreteGradient(Preconditioner_, ParMatrixG_);
415template <
class LocalOrdinal,
class Node>
416int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetCoordinates(Teuchos::RCP<multivector_type> coords) {
417 if (!G_.is_null() && !G_->getDomainMap()->isSameAs(*coords->getMap()))
418 throw std::runtime_error(
"Hypre<Tpetra::RowMatrix<double, HYPRE_Int, long long, Node>: Node map mismatch: G->DomainMap() and coords");
420 if (SolverType_ != AMS && PrecondType_ != AMS)
423 scalar_type *xPtr = coords->getDataNonConst(0).getRawPtr();
424 scalar_type *yPtr = coords->getDataNonConst(1).getRawPtr();
425 scalar_type *zPtr = coords->getDataNonConst(2).getRawPtr();
427 MPI_Comm comm = *(Teuchos::rcp_dynamic_cast<const Teuchos::MpiComm<int> >(A_->getRowMap()->getComm())->getRawMpiComm());
428 local_ordinal_type NumEntries = coords->getLocalLength();
429 global_ordinal_type *indices =
const_cast<global_ordinal_type *
>(GloballyContiguousNodeRowMap_->getLocalElementList().getRawPtr());
431 global_ordinal_type ilower = GloballyContiguousNodeRowMap_->getMinGlobalIndex();
432 global_ordinal_type iupper = GloballyContiguousNodeRowMap_->getMaxGlobalIndex();
434 if (NumEntries != iupper - ilower + 1) {
435 std::cout <<
"Ifpack2::Hypre::SetCoordinates(): Error on rank " << A_->getRowMap()->getComm()->getRank() <<
": MyLength = " << coords->getLocalLength() <<
" GID range = [" << ilower <<
"," << iupper <<
"]" << std::endl;
436 throw std::runtime_error(
"Hypre<Tpetra::RowMatrix<double, HYPRE_Int, long long, Node>: SetCoordinates: Length mismatch");
439 IFPACK2_CHK_ERR(HYPRE_IJVectorCreate(comm, ilower, iupper, &xHypre_));
440 IFPACK2_CHK_ERR(HYPRE_IJVectorSetObjectType(xHypre_, HYPRE_PARCSR));
441 IFPACK2_CHK_ERR(HYPRE_IJVectorInitialize(xHypre_));
442 IFPACK2_CHK_ERR(HYPRE_IJVectorSetValues(xHypre_, NumEntries, indices, xPtr));
443 IFPACK2_CHK_ERR(HYPRE_IJVectorAssemble(xHypre_));
444 IFPACK2_CHK_ERR(HYPRE_IJVectorGetObject(xHypre_, (
void **)&xPar_));
446 IFPACK2_CHK_ERR(HYPRE_IJVectorCreate(comm, ilower, iupper, &yHypre_));
447 IFPACK2_CHK_ERR(HYPRE_IJVectorSetObjectType(yHypre_, HYPRE_PARCSR));
448 IFPACK2_CHK_ERR(HYPRE_IJVectorInitialize(yHypre_));
449 IFPACK2_CHK_ERR(HYPRE_IJVectorSetValues(yHypre_, NumEntries, indices, yPtr));
450 IFPACK2_CHK_ERR(HYPRE_IJVectorAssemble(yHypre_));
451 IFPACK2_CHK_ERR(HYPRE_IJVectorGetObject(yHypre_, (
void **)&yPar_));
453 IFPACK2_CHK_ERR(HYPRE_IJVectorCreate(comm, ilower, iupper, &zHypre_));
454 IFPACK2_CHK_ERR(HYPRE_IJVectorSetObjectType(zHypre_, HYPRE_PARCSR));
455 IFPACK2_CHK_ERR(HYPRE_IJVectorInitialize(zHypre_));
456 IFPACK2_CHK_ERR(HYPRE_IJVectorSetValues(zHypre_, NumEntries, indices, zPtr));
457 IFPACK2_CHK_ERR(HYPRE_IJVectorAssemble(zHypre_));
458 IFPACK2_CHK_ERR(HYPRE_IJVectorGetObject(zHypre_, (
void **)&zPar_));
461 HYPRE_ParVectorPrint(xPar_,
"coordX.dat");
462 HYPRE_ParVectorPrint(yPar_,
"coordY.dat");
463 HYPRE_ParVectorPrint(zPar_,
"coordZ.dat");
466 if (SolverType_ == AMS)
467 HYPRE_AMSSetCoordinateVectors(Solver_, xPar_, yPar_, zPar_);
468 if (PrecondType_ == AMS)
469 HYPRE_AMSSetCoordinateVectors(Preconditioner_, xPar_, yPar_, zPar_);
476template <
class LocalOrdinal,
class Node>
477void Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::compute() {
478 const std::string timerName(
"Ifpack2::Hypre::compute");
479 Teuchos::RCP<Teuchos::Time> timer = Teuchos::TimeMonitor::lookupCounter(timerName);
480 if (timer.is_null()) timer = Teuchos::TimeMonitor::getNewCounter(timerName);
481 double startTime = timer->wallTime();
484 Teuchos::TimeMonitor timeMon(*timer);
486 if (isInitialized() ==
false) {
495 MPI_Comm comm = *(Teuchos::rcp_dynamic_cast<const Teuchos::MpiComm<int> >(A_->getRowMap()->getComm())->getRawMpiComm());
496 global_ordinal_type ilower = GloballyContiguousRowMap_->getMinGlobalIndex();
497 global_ordinal_type iupper = GloballyContiguousRowMap_->getMaxGlobalIndex();
498 IFPACK2_CHK_ERR(HYPRE_IJMatrixCreate(comm, ilower, iupper, ilower, iupper, &HypreA_));
499 IFPACK2_CHK_ERR(HYPRE_IJMatrixSetObjectType(HypreA_, HYPRE_PARCSR));
500 IFPACK2_CHK_ERR(HYPRE_IJMatrixInitialize(HypreA_));
502 if (SolveOrPrec_ == Hypre_Is_Solver) {
503 IFPACK2_CHK_ERR(SetSolverType(SolverType_));
504 if (SolverPrecondPtr_ != NULL && UsePreconditioner_) {
506 IFPACK2_CHK_ERR(SetPrecondType(PrecondType_));
508 IFPACK2_CHK_ERR(SolverPrecondPtr_(Solver_, PrecondSolvePtr_, PrecondSetupPtr_, Preconditioner_));
513 IFPACK2_CHK_ERR(SetPrecondType(PrecondType_));
518 SetDiscreteGradient(G_);
521 if (!Coords_.is_null()) {
522 SetCoordinates(Coords_);
526 if (SolveOrPrec_ == Hypre_Is_Solver) {
527 IFPACK2_CHK_ERR(SolverSetupPtr_(Solver_, ParMatrix_, ParX_, ParY_));
529 IFPACK2_CHK_ERR(PrecondSetupPtr_(Preconditioner_, ParMatrix_, ParX_, ParY_));
536 ComputeTime_ += (timer->wallTime() - startTime);
540template <
class LocalOrdinal,
class Node>
541int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::CallFunctions()
const {
542 for (
int i = 0; i < NumFunsToCall_; i++) {
543 IFPACK2_CHK_ERR(FunsToCall_[i]->CallFunction(Solver_, Preconditioner_));
549template <
class LocalOrdinal,
class Node>
550void Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::apply(
const Tpetra::MultiVector<scalar_type, local_ordinal_type, global_ordinal_type, node_type> &X,
551 Tpetra::MultiVector<scalar_type, local_ordinal_type, global_ordinal_type, node_type> &Y,
552 Teuchos::ETransp mode,
554 scalar_type beta)
const {
555 using LO = local_ordinal_type;
556 using SC = scalar_type;
557 const std::string timerName(
"Ifpack2::Hypre::apply");
558 Teuchos::RCP<Teuchos::Time> timer = Teuchos::TimeMonitor::lookupCounter(timerName);
559 if (timer.is_null()) timer = Teuchos::TimeMonitor::getNewCounter(timerName);
560 double startTime = timer->wallTime();
563 Teuchos::TimeMonitor timeMon(*timer);
565 if (isComputed() ==
false) {
568 hypre_Vector *XLocal_ = hypre_ParVectorLocalVector(XVec_);
569 hypre_Vector *YLocal_ = hypre_ParVectorLocalVector(YVec_);
570 bool SameVectors =
false;
571 size_t NumVectors = X.getNumVectors();
572 if (NumVectors != Y.getNumVectors()) IFPACK2_CHK_ERR(-1);
581 for (
int VecNum = 0; VecNum < (int)NumVectors; VecNum++) {
584 SC *XValues =
const_cast<SC *
>(X.getData(VecNum).getRawPtr());
587 YValues =
const_cast<SC *
>(Y.getData(VecNum).getRawPtr());
589 YValues = VectorCache_.getRawPtr();
592 SC *XTemp = XLocal_->data;
594 XLocal_->data = XValues;
595 SC *YTemp = YLocal_->data;
596 YLocal_->data = YValues;
598 IFPACK2_CHK_ERR(HYPRE_ParVectorSetConstantValues(ParY_, 0.0));
599 if (SolveOrPrec_ == Hypre_Is_Solver) {
601 IFPACK2_CHK_ERR(SolverSolvePtr_(Solver_, ParMatrix_, ParX_, ParY_));
604 IFPACK2_CHK_ERR(PrecondSolvePtr_(Preconditioner_, ParMatrix_, ParX_, ParY_));
608 Teuchos::ArrayView<SC> Yv = Y.getDataNonConst(VecNum)();
609 LO NumEntries = Y.getLocalLength();
610 for (LO i = 0; i < NumEntries; i++)
613 XLocal_->data = XTemp;
614 YLocal_->data = YTemp;
618 ApplyTime_ += (timer->wallTime() - startTime);
622template <
class LocalOrdinal,
class Node>
623void Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::applyMat(
const Tpetra::MultiVector<scalar_type, local_ordinal_type, global_ordinal_type, node_type> &X,
624 Tpetra::MultiVector<scalar_type, local_ordinal_type, global_ordinal_type, node_type> &Y,
625 Teuchos::ETransp mode)
const {
626 A_->apply(X, Y, mode);
630template <
class LocalOrdinal,
class Node>
631std::string Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::description()
const {
632 std::ostringstream out;
637 out <<
"\"Ifpack2::Hypre\": {";
638 out <<
"Initialized: " << (isInitialized() ?
"true" :
"false") <<
", "
639 <<
"Computed: " << (isComputed() ?
"true" :
"false") <<
", ";
642 out <<
"Matrix: null";
644 out <<
"Global matrix dimensions: ["
645 << A_->getGlobalNumRows() <<
", "
646 << A_->getGlobalNumCols() <<
"]"
647 <<
", Global nnz: " << A_->getGlobalNumEntries();
655template <
class LocalOrdinal,
class Node>
656void Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::describe(Teuchos::FancyOStream &os,
const Teuchos::EVerbosityLevel verbLevel)
const {
659 os <<
"================================================================================" << endl;
660 os <<
"Ifpack2::Hypre: " << endl
662 os <<
"Using " << A_->getComm()->getSize() <<
" processors." << endl;
663 os <<
"Global number of rows = " << A_->getGlobalNumRows() << endl;
664 os <<
"Global number of nonzeros = " << A_->getGlobalNumEntries() << endl;
667 os <<
"Phase # calls Total Time (s)" << endl;
668 os <<
"----- ------- --------------" << endl;
669 os <<
"Initialize() " << std::setw(5) << NumInitialize_
670 <<
" " << std::setw(15) << InitializeTime_ << endl;
671 os <<
"Compute() " << std::setw(5) << NumCompute_
672 <<
" " << std::setw(15) << ComputeTime_ << endl;
673 os <<
"ApplyInverse() " << std::setw(5) << NumApply_
674 <<
" " << std::setw(15) << ApplyTime_ << endl;
675 os <<
"================================================================================" << endl;
680template <
class LocalOrdinal,
class Node>
681int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetSolverType(Hypre_Solver Solver) {
684 if (IsSolverCreated_) {
685 SolverDestroyPtr_(Solver_);
686 IsSolverCreated_ =
false;
688 SolverCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_BoomerAMGCreate;
689 SolverDestroyPtr_ = &HYPRE_BoomerAMGDestroy;
690 SolverSetupPtr_ = &HYPRE_BoomerAMGSetup;
691 SolverPrecondPtr_ = NULL;
692 SolverSolvePtr_ = &HYPRE_BoomerAMGSolve;
695 if (IsSolverCreated_) {
696 SolverDestroyPtr_(Solver_);
697 IsSolverCreated_ =
false;
699 SolverCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_AMSCreate;
700 SolverDestroyPtr_ = &HYPRE_AMSDestroy;
701 SolverSetupPtr_ = &HYPRE_AMSSetup;
702 SolverSolvePtr_ = &HYPRE_AMSSolve;
703 SolverPrecondPtr_ = NULL;
706 if (IsSolverCreated_) {
707 SolverDestroyPtr_(Solver_);
708 IsSolverCreated_ =
false;
710 SolverCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRHybridCreate;
711 SolverDestroyPtr_ = &HYPRE_ParCSRHybridDestroy;
712 SolverSetupPtr_ = &HYPRE_ParCSRHybridSetup;
713 SolverSolvePtr_ = &HYPRE_ParCSRHybridSolve;
714 SolverPrecondPtr_ = &HYPRE_ParCSRHybridSetPrecond;
717 if (IsSolverCreated_) {
718 SolverDestroyPtr_(Solver_);
719 IsSolverCreated_ =
false;
721 SolverCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRPCGCreate;
722 SolverDestroyPtr_ = &HYPRE_ParCSRPCGDestroy;
723 SolverSetupPtr_ = &HYPRE_ParCSRPCGSetup;
724 SolverSolvePtr_ = &HYPRE_ParCSRPCGSolve;
725 SolverPrecondPtr_ = &HYPRE_ParCSRPCGSetPrecond;
728 if (IsSolverCreated_) {
729 SolverDestroyPtr_(Solver_);
730 IsSolverCreated_ =
false;
732 SolverCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRGMRESCreate;
733 SolverDestroyPtr_ = &HYPRE_ParCSRGMRESDestroy;
734 SolverSetupPtr_ = &HYPRE_ParCSRGMRESSetup;
735 SolverPrecondPtr_ = &HYPRE_ParCSRGMRESSetPrecond;
738 if (IsSolverCreated_) {
739 SolverDestroyPtr_(Solver_);
740 IsSolverCreated_ =
false;
742 SolverCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRFlexGMRESCreate;
743 SolverDestroyPtr_ = &HYPRE_ParCSRFlexGMRESDestroy;
744 SolverSetupPtr_ = &HYPRE_ParCSRFlexGMRESSetup;
745 SolverSolvePtr_ = &HYPRE_ParCSRFlexGMRESSolve;
746 SolverPrecondPtr_ = &HYPRE_ParCSRFlexGMRESSetPrecond;
749 if (IsSolverCreated_) {
750 SolverDestroyPtr_(Solver_);
751 IsSolverCreated_ =
false;
753 SolverCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRLGMRESCreate;
754 SolverDestroyPtr_ = &HYPRE_ParCSRLGMRESDestroy;
755 SolverSetupPtr_ = &HYPRE_ParCSRLGMRESSetup;
756 SolverSolvePtr_ = &HYPRE_ParCSRLGMRESSolve;
757 SolverPrecondPtr_ = &HYPRE_ParCSRLGMRESSetPrecond;
760 if (IsSolverCreated_) {
761 SolverDestroyPtr_(Solver_);
762 IsSolverCreated_ =
false;
764 SolverCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRBiCGSTABCreate;
765 SolverDestroyPtr_ = &HYPRE_ParCSRBiCGSTABDestroy;
766 SolverSetupPtr_ = &HYPRE_ParCSRBiCGSTABSetup;
767 SolverSolvePtr_ = &HYPRE_ParCSRBiCGSTABSolve;
768 SolverPrecondPtr_ = &HYPRE_ParCSRBiCGSTABSetPrecond;
778template <
class LocalOrdinal,
class Node>
779int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::SetPrecondType(Hypre_Solver Precond) {
782 if (IsPrecondCreated_) {
783 PrecondDestroyPtr_(Preconditioner_);
784 IsPrecondCreated_ =
false;
786 PrecondCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_BoomerAMGCreate;
787 PrecondDestroyPtr_ = &HYPRE_BoomerAMGDestroy;
788 PrecondSetupPtr_ = &HYPRE_BoomerAMGSetup;
789 PrecondSolvePtr_ = &HYPRE_BoomerAMGSolve;
792 if (IsPrecondCreated_) {
793 PrecondDestroyPtr_(Preconditioner_);
794 IsPrecondCreated_ =
false;
796 PrecondCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParaSailsCreate;
797 PrecondDestroyPtr_ = &HYPRE_ParaSailsDestroy;
798 PrecondSetupPtr_ = &HYPRE_ParaSailsSetup;
799 PrecondSolvePtr_ = &HYPRE_ParaSailsSolve;
802 if (IsPrecondCreated_) {
803 PrecondDestroyPtr_(Preconditioner_);
804 IsPrecondCreated_ =
false;
806 PrecondCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_EuclidCreate;
807 PrecondDestroyPtr_ = &HYPRE_EuclidDestroy;
808 PrecondSetupPtr_ = &HYPRE_EuclidSetup;
809 PrecondSolvePtr_ = &HYPRE_EuclidSolve;
812 if (IsPrecondCreated_) {
813 PrecondDestroyPtr_(Preconditioner_);
814 IsPrecondCreated_ =
false;
816 PrecondCreatePtr_ = &Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_AMSCreate;
817 PrecondDestroyPtr_ = &HYPRE_AMSDestroy;
818 PrecondSetupPtr_ = &HYPRE_AMSSetup;
819 PrecondSolvePtr_ = &HYPRE_AMSSolve;
830template <
class LocalOrdinal,
class Node>
831int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::CreateSolver() {
833 HYPRE_ParCSRMatrixGetComm(ParMatrix_, &comm);
834 int ierr = (this->*SolverCreatePtr_)(comm, &Solver_);
835 IsSolverCreated_ =
true;
840template <
class LocalOrdinal,
class Node>
841int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::CreatePrecond() {
843 HYPRE_ParCSRMatrixGetComm(ParMatrix_, &comm);
844 int ierr = (this->*PrecondCreatePtr_)(comm, &Preconditioner_);
845 IsPrecondCreated_ =
true;
850template <
class LocalOrdinal,
class Node>
851int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::CopyTpetraToHypre() {
852 using LO = local_ordinal_type;
853 using GO = global_ordinal_type;
856 Teuchos::RCP<const crs_matrix_type> Matrix = Teuchos::rcp_dynamic_cast<const crs_matrix_type>(A_);
857 if (Matrix.is_null())
858 throw std::runtime_error(
"Hypre<Tpetra::RowMatrix<double, LocalOrdinal, HYPRE_Int, Node>: Unsupported matrix configuration: Tpetra::CrsMatrix required");
860 std::vector<HYPRE_Int> new_indices(Matrix->getLocalMaxNumRowEntries());
861 for (LO i = 0; i < (LO)Matrix->getLocalNumRows(); i++) {
862 typename crs_matrix_type::values_host_view_type values;
863 typename crs_matrix_type::local_inds_host_view_type indices;
864 Matrix->getLocalRowView(i, indices, values);
865 for (LO j = 0; j < (LO)indices.extent(0); j++) {
866 new_indices[j] = GloballyContiguousColMap_->getGlobalElement(indices(j));
868 HYPRE_Int GlobalRow[1];
869 HYPRE_Int numEntries = (GO)indices.extent(0);
870 GlobalRow[0] = GloballyContiguousRowMap_->getGlobalElement(i);
871 IFPACK2_CHK_ERR(HYPRE_IJMatrixSetValues(HypreA_, 1, &numEntries, GlobalRow, new_indices.data(), values.data()));
873 IFPACK2_CHK_ERR(HYPRE_IJMatrixAssemble(HypreA_));
874 IFPACK2_CHK_ERR(HYPRE_IJMatrixGetObject(HypreA_, (
void **)&ParMatrix_));
876 HYPRE_ParCSRMatrixPrint(ParMatrix_,
"A.mat");
881template <
class LocalOrdinal,
class Node>
882HYPRE_Int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_BoomerAMGCreate(MPI_Comm , HYPRE_Solver *solver) {
return HYPRE_BoomerAMGCreate(solver); }
885template <
class LocalOrdinal,
class Node>
886HYPRE_Int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParaSailsCreate(MPI_Comm comm, HYPRE_Solver *solver) {
return HYPRE_ParaSailsCreate(comm, solver); }
889template <
class LocalOrdinal,
class Node>
890HYPRE_Int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_EuclidCreate(MPI_Comm comm, HYPRE_Solver *solver) {
return HYPRE_EuclidCreate(comm, solver); }
893template <
class LocalOrdinal,
class Node>
894HYPRE_Int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_AMSCreate(MPI_Comm , HYPRE_Solver *solver) {
return HYPRE_AMSCreate(solver); }
897template <
class LocalOrdinal,
class Node>
898HYPRE_Int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRHybridCreate(MPI_Comm , HYPRE_Solver *solver) {
return HYPRE_ParCSRHybridCreate(solver); }
901template <
class LocalOrdinal,
class Node>
902HYPRE_Int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRPCGCreate(MPI_Comm comm, HYPRE_Solver *solver) {
return HYPRE_ParCSRPCGCreate(comm, solver); }
905template <
class LocalOrdinal,
class Node>
906HYPRE_Int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRGMRESCreate(MPI_Comm comm, HYPRE_Solver *solver) {
return HYPRE_ParCSRGMRESCreate(comm, solver); }
909template <
class LocalOrdinal,
class Node>
910HYPRE_Int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRFlexGMRESCreate(MPI_Comm comm, HYPRE_Solver *solver) {
return HYPRE_ParCSRFlexGMRESCreate(comm, solver); }
913template <
class LocalOrdinal,
class Node>
914HYPRE_Int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRLGMRESCreate(MPI_Comm comm, HYPRE_Solver *solver) {
return HYPRE_ParCSRLGMRESCreate(comm, solver); }
917template <
class LocalOrdinal,
class Node>
918HYPRE_Int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::Hypre_ParCSRBiCGSTABCreate(MPI_Comm comm, HYPRE_Solver *solver) {
return HYPRE_ParCSRBiCGSTABCreate(comm, solver); }
921template <
class LocalOrdinal,
class Node>
922Teuchos::RCP<const Tpetra::Map<LocalOrdinal, HYPRE_Int, Node> >
923Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::MakeContiguousColumnMap(Teuchos::RCP<const crs_matrix_type> &Matrix)
const {
924 using import_type = Tpetra::Import<local_ordinal_type, global_ordinal_type, node_type>;
925 using go_vector_type = Tpetra::Vector<global_ordinal_type, local_ordinal_type, global_ordinal_type, node_type>;
931 if (Matrix.is_null())
932 throw std::runtime_error(
"Hypre<Tpetra::RowMatrix<HYPRE_Real, HYPRE_Int, long long, Node>: Unsupported matrix configuration: Tpetra::CrsMatrix required");
933 RCP<const map_type> DomainMap = Matrix->getDomainMap();
934 RCP<const map_type> ColumnMap = Matrix->getColMap();
935 RCP<const import_type> importer = Matrix->getGraph()->getImporter();
937 if (DomainMap->isContiguous()) {
942 Teuchos::RCP<map_type> ContiguousDomainMap = rcp(
new map_type(DomainMap->getGlobalNumElements(),
943 DomainMap->getLocalNumElements(), 0, DomainMap->getComm()));
946 go_vector_type MyGIDsHYPRE(DomainMap, ContiguousDomainMap->getLocalElementList());
949 go_vector_type ColGIDsHYPRE(ColumnMap);
950 ColGIDsHYPRE.doImport(MyGIDsHYPRE, *importer, Tpetra::INSERT);
953 return Teuchos::rcp(
new map_type(ColumnMap->getGlobalNumElements(), ColGIDsHYPRE.getDataNonConst()(), 0, ColumnMap->getComm()));
956 return Teuchos::rcp(
new map_type(ColumnMap->getGlobalNumElements(), ContiguousDomainMap->getLocalElementList(), 0, ColumnMap->getComm()));
961template <
class LocalOrdinal,
class Node>
962int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::getNumInitialize()
const {
963 return NumInitialize_;
966template <
class LocalOrdinal,
class Node>
967int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::getNumCompute()
const {
971template <
class LocalOrdinal,
class Node>
972int Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::getNumApply()
const {
976template <
class LocalOrdinal,
class Node>
977double Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::getInitializeTime()
const {
978 return InitializeTime_;
981template <
class LocalOrdinal,
class Node>
982double Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::getComputeTime()
const {
986template <
class LocalOrdinal,
class Node>
987double Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::getApplyTime()
const {
991template <
class LocalOrdinal,
class Node>
992Teuchos::RCP<const typename Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::map_type>
993Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::
994 getDomainMap()
const {
995 Teuchos::RCP<const row_matrix_type> A = getMatrix();
996 TEUCHOS_TEST_FOR_EXCEPTION(
997 A.is_null(), std::runtime_error,
998 "Ifpack2::Hypre::getDomainMap: The "
999 "input matrix A is null. Please call setMatrix() with a nonnull input "
1000 "matrix before calling this method.");
1001 return A->getDomainMap();
1004template <
class LocalOrdinal,
class Node>
1005Teuchos::RCP<const typename Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::map_type>
1006Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::
1007 getRangeMap()
const {
1008 Teuchos::RCP<const row_matrix_type> A = getMatrix();
1009 TEUCHOS_TEST_FOR_EXCEPTION(
1010 A.is_null(), std::runtime_error,
1011 "Ifpack2::Hypre::getRangeMap: The "
1012 "input matrix A is null. Please call setMatrix() with a nonnull input "
1013 "matrix before calling this method.");
1014 return A->getRangeMap();
1017template <
class LocalOrdinal,
class Node>
1018void Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::setMatrix(
const Teuchos::RCP<const row_matrix_type> &A) {
1019 if (A.getRawPtr() != getMatrix().getRawPtr()) {
1020 IsInitialized_ =
false;
1021 IsComputed_ =
false;
1026template <
class LocalOrdinal,
class Node>
1027Teuchos::RCP<const typename Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::row_matrix_type>
1028Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::
1033template <
class LocalOrdinal,
class Node>
1034bool Hypre<Tpetra::RowMatrix<HYPRE_Real, LocalOrdinal, HYPRE_Int, Node> >::hasTransposeApply()
const {
1040#define IFPACK2_HYPRE_INSTANT(S, LO, GO, N) \
1041 template class Ifpack2::Hypre<Tpetra::RowMatrix<S, LO, GO, N> >;
Preconditioners and smoothers for Tpetra sparse matrices.
Definition Ifpack2_AdditiveSchwarz_decl.hpp:40
@ GMRES
Uses AztecOO's GMRES.
Definition Ifpack2_CondestType.hpp:20