GEOS
HypreUtils.hpp
Go to the documentation of this file.
1 /*
2  * ------------------------------------------------------------------------------------------------------------
3  * SPDX-License-Identifier: LGPL-2.1-only
4  *
5  * Copyright (c) 2016-2024 Lawrence Livermore National Security LLC
6  * Copyright (c) 2018-2024 TotalEnergies
7  * Copyright (c) 2018-2024 The Board of Trustees of the Leland Stanford Junior University
8  * Copyright (c) 2023-2024 Chevron
9  * Copyright (c) 2019- GEOS/GEOSX Contributors
10  * All rights reserved
11  *
12  * See top level LICENSE, COPYRIGHT, CONTRIBUTORS, NOTICE, and ACKNOWLEDGEMENTS files for details.
13  * ------------------------------------------------------------------------------------------------------------
14  */
15 
20 #ifndef GEOS_LINEARALGEBRA_INTERFACES_HYPREUTILS_HPP_
21 #define GEOS_LINEARALGEBRA_INTERFACES_HYPREUTILS_HPP_
22 
23 #include "common/DataTypes.hpp"
24 #include "common/GEOS_RAJA_Interface.hpp"
25 
26 #include "codingUtilities/Utilities.hpp"
28 
29 #include <HYPRE_krylov.h>
30 #include <HYPRE_parcsr_ls.h>
31 
36 #define GEOS_HYPRE_DEVICE GEOS_HOST_DEVICE
37 
38 namespace geos
39 {
40 
48 {
50  using SetupFunc = HYPRE_Int (*)( HYPRE_Solver, HYPRE_ParCSRMatrix, HYPRE_ParVector, HYPRE_ParVector );
51 
53  using SolveFunc = HYPRE_Int (*)( HYPRE_Solver, HYPRE_ParCSRMatrix, HYPRE_ParVector, HYPRE_ParVector );
54 
56  using DestroyFunc = HYPRE_Int (*)( HYPRE_Solver );
57 
58  HYPRE_Solver ptr{};
62 };
63 
67 namespace hypre
68 {
69 
74 constexpr HYPRE_MemoryLocation getMemoryLocation( LvArray::MemorySpace const space )
75 {
76  switch( space )
77  {
78  case hostMemorySpace: return HYPRE_MEMORY_HOST;
79 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA
80  case LvArray::MemorySpace::cuda: return HYPRE_MEMORY_DEVICE;
81 #endif
82 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
83  case LvArray::MemorySpace::hip: return HYPRE_MEMORY_DEVICE;
84 #endif
85  default: return HYPRE_MEMORY_HOST;
86  }
87 }
88 
93 constexpr LvArray::MemorySpace getLvArrayMemorySpace( HYPRE_MemoryLocation const location )
94 {
95  switch( location )
96  {
97  case HYPRE_MEMORY_HOST: return hostMemorySpace;
98 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA
99  case HYPRE_MEMORY_DEVICE: return parallelDeviceMemorySpace;
100 #endif
101 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
102  case HYPRE_MEMORY_DEVICE: return parallelDeviceMemorySpace;
103 #endif
104  default: return hostMemorySpace;
105  }
106 }
107 
108 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA || GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
109 
111 using execPolicy = parallelDevicePolicy<>;
113 constexpr LvArray::MemorySpace memorySpace = parallelDeviceMemorySpace;
115 constexpr HYPRE_MemoryLocation memoryLocation = HYPRE_MEMORY_DEVICE;
116 
117 #else
118 
120 using execPolicy = parallelHostPolicy;
122 constexpr LvArray::MemorySpace memorySpace = hostMemorySpace;
124 constexpr HYPRE_MemoryLocation memoryLocation = HYPRE_MEMORY_HOST;
125 
126 #endif
127 
128 // Check matching requirements on index/value types between GEOS and Hypre
129 
130 // WARNING. We don't have consistent types between HYPRE_Int and localIndex.
131 // Decision needs to be made either to use bigint option, or change
132 // localIndex to int. We are getting away with this because we do not
133 // pass ( localIndex * ) to hypre except when it is on the GPU, in
134 // which case we are using int for localIndex.
135 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA || GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
136 static_assert( sizeof( HYPRE_Int ) == sizeof( geos::localIndex ),
137  "HYPRE_Int and geos::localIndex must have the same size" );
138 static_assert( std::is_signed< HYPRE_Int >::value == std::is_signed< geos::localIndex >::value,
139  "HYPRE_Int and geos::localIndex must both be signed or unsigned" );
140 #endif
141 
148 inline void checkDeviceErrors( char const * msg, char const * file, int const line )
149 {
150 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA
151  cudaError_t const err = cudaGetLastError();
152  GEOS_ERROR_IF( err != cudaSuccess, GEOS_FMT( "Previous CUDA errors found: {} ({} at {}:{})", msg, cudaGetErrorString( err ), file, line ) );
153 #elif GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
154  hipError_t const err = hipGetLastError();
155  GEOS_UNUSED_VAR( msg, file, line ); // on crusher geos_error_if ultimately resolves to an assert, which drops the content on release
156  // builds
157  GEOS_ERROR_IF( err != hipSuccess, GEOS_FMT( "Previous HIP errors found: {} ({} at {}:{})", msg, hipGetErrorString( err ), file, line ) );
158 #else
159  GEOS_UNUSED_VAR( msg, file, line );
160 #endif
161 }
162 
168 #define GEOS_HYPRE_CHECK_DEVICE_ERRORS( msg ) ::geos::hypre::checkDeviceErrors( msg, __FILE__, __LINE__ )
169 
170 static_assert( sizeof( HYPRE_BigInt ) == sizeof( geos::globalIndex ),
171  "HYPRE_BigInt and geos::globalIndex must have the same size" );
172 
173 static_assert( std::is_signed< HYPRE_BigInt >::value == std::is_signed< geos::globalIndex >::value,
174  "HYPRE_BigInt and geos::globalIndex must both be signed or unsigned" );
175 
176 static_assert( std::is_same< HYPRE_Real, geos::real64 >::value,
177  "HYPRE_Real and geos::real64 must be the same type" );
178 
184 inline HYPRE_BigInt * toHypreBigInt( geos::globalIndex * const index )
185 {
186  return reinterpret_cast< HYPRE_BigInt * >(index);
187 }
188 
194 inline HYPRE_BigInt const * toHypreBigInt( geos::globalIndex const * const index )
195 {
196  return reinterpret_cast< HYPRE_BigInt const * >(index);
197 }
198 
208 HYPRE_Vector parVectorToVectorAll( HYPRE_ParVector const vec );
209 
217 HYPRE_Vector parVectorToVector( HYPRE_ParVector const vec, int const targetRank );
218 
225 HYPRE_Int dummySetup( HYPRE_Solver,
226  HYPRE_ParCSRMatrix,
227  HYPRE_ParVector,
228  HYPRE_ParVector );
229 
238 HYPRE_Int SuperLUDistSolve( HYPRE_Solver solver,
239  HYPRE_ParCSRMatrix A,
240  HYPRE_ParVector b,
241  HYPRE_ParVector x );
242 
248 HYPRE_Int SuperLUDistDestroy( HYPRE_Solver solver );
249 
256 HYPRE_Int relaxationCreate( HYPRE_Solver & solver,
257  HYPRE_Int const type );
258 
267 HYPRE_Int relaxationSetup( HYPRE_Solver solver,
268  HYPRE_ParCSRMatrix A,
269  HYPRE_ParVector b,
270  HYPRE_ParVector x );
271 
280 HYPRE_Int relaxationSolve( HYPRE_Solver solver,
281  HYPRE_ParCSRMatrix A,
282  HYPRE_ParVector b,
283  HYPRE_ParVector x );
284 
290 HYPRE_Int relaxationDestroy( HYPRE_Solver solver );
291 
299 HYPRE_Int chebyshevCreate( HYPRE_Solver & solver,
300  HYPRE_Int const order,
301  HYPRE_Int const eigNumIter );
302 
311 HYPRE_Int chebyshevSetup( HYPRE_Solver solver,
312  HYPRE_ParCSRMatrix A,
313  HYPRE_ParVector b,
314  HYPRE_ParVector x );
315 
324 HYPRE_Int chebyshevSolve( HYPRE_Solver solver,
325  HYPRE_ParCSRMatrix A,
326  HYPRE_ParVector b,
327  HYPRE_ParVector x );
328 
334 HYPRE_Int chebyshevDestroy( HYPRE_Solver solver );
335 
342 {
344  {
347  };
348  return findOption( typeMap, type, "multigrid cycle", "HyprePreconditioner" );
349 }
350 
357 {
359  {
368  };
369  return findOption( typeMap, type, "multigrid relaxation", "HyprePreconditioner" );
370 }
371 
378 {
380  {
392  };
393  return findOption( typeMap, type, "multigrid interpolation", "HyprePreconditioner" );
394 }
395 
402 {
404  {
414  };
415  return findOption( typeMap, type, "multigrid aggressive interpolation", "HyprePreconditioner" );
416 }
417 
424 {
426  {
429  };
430  return findOption( typeMap, type, "ILU", "HyprePreconditioner" );
431 }
432 
439 {
441  {
453  };
454  return findOption( typeMap, type, "multigrid coarse solver", "HyprePreconditioner" );
455 }
456 
463 {
465  {
472  };
473  return findOption( typeMap, type, "multigrid coarsening", "HyprePreconditioner" );
474 }
475 
482 {
484  {
492  };
493  return findOption( typeMap, type, "relaxation", "HyprePreconditioner" );
494 }
495 
502 {
504  {
507  };
508  return findOption( typeMap, type, "ILU", "HyprePreconditioner" );
509 }
510 
515 enum class AMGCoarseningType : HYPRE_Int
516 {
517  CLJP = 0,
518  Ruge_Stueben = 3,
519  Falgout = 6,
520  CLJPDebug = 7,
521  PMIS = 8,
522  PMISDebug = 9,
523  HMIS = 10,
524  CGC = 21,
525  CGC_E = 22
526 };
527 
532 enum class MGRInterpolationType : HYPRE_Int
533 {
534  injection = 0,
535  l1jacobi = 1,
536  jacobi = 2,
538  approximateInverse = 4,
539  blockJacobi = 12
540 };
541 
547 enum class MGRRestrictionType : HYPRE_Int
548 {
549  injection = 0,
550  jacobi = 2,
551  approximateInverse = 3,
552  blockJacobi = 12,
553  cprLike = 13,
554  blockColLumped = 14,
555  partialColLumped = 15
556 };
557 
562 enum class MGRCoarseGridMethod : HYPRE_Int
563 {
564  galerkin = 0,
565  nonGalerkin = 1,
567  cprLikeDiag = 2,
569  cprLikeBlockDiag = 3,
571  approximateInverse = 4
573 };
574 
579 enum class MGRFRelaxationType : HYPRE_Int
580 {
581  none = -1,
583  amgVCycle = 2,
587  jacobi = 7,
589  gsElim = 9,
590  l1forwardGaussSeidel = 13,
591  l1backwardGaussSeidel = 14,
592  l1jacobi = 18,
593  gsElimWPivoting = 99,
594  gsElimWInverse = 199
595 };
596 
601 enum class MGRGlobalSmootherType : HYPRE_Int
602 {
603  none = -1,
604  blockJacobi = 0,
605  blockGaussSeidel = 1,
606  jacobi = 2,
607  ilu0 = 16
608 };
609 
610 } // namespace hypre
611 
612 } // namespace geos
613 
614 #endif /*GEOS_LINEARALGEBRA_INTERFACES_HYPREUTILS_HPP_*/
#define GEOS_UNUSED_VAR(...)
Mark an unused variable and silence compiler warnings.
Definition: GeosxMacros.hpp:84
#define GEOS_ERROR_IF(COND,...)
Conditionally raise a hard error and terminate the program.
Definition: Logger.hpp:217
Base template for ordered and unordered maps.
HYPRE_Int getAMGInterpolationType(LinearSolverParameters::AMG::InterpType const &type)
Returns hypre's identifier of the AMG interpolation type.
Definition: HypreUtils.hpp:377
MGRCoarseGridMethod
This enum class specifies the strategy for level coarse grid computation in MGR.
Definition: HypreUtils.hpp:563
@ galerkin
Galerkin coarse grid computation using RAP.
MGRRestrictionType
This enum class specifies the strategy for computing the level restriction operator in MGR.
Definition: HypreUtils.hpp:548
@ blockColLumped
Block column-lumped approximation.
@ partialColLumped
Partial column-lumped approximation.
@ approximateInverse
Approximate inverse.
@ cprLike
CPR-like restriction.
HYPRE_Int chebyshevSetup(HYPRE_Solver solver, HYPRE_ParCSRMatrix A, HYPRE_ParVector b, HYPRE_ParVector x)
Setup a Chebyshev smoother.
HYPRE_Int getAMGAggressiveInterpolationType(LinearSolverParameters::AMG::AggInterpType const &type)
Returns hypre's identifier of the AMG aggressive interpolation type.
Definition: HypreUtils.hpp:401
HYPRE_Vector parVectorToVectorAll(HYPRE_ParVector const vec)
Gather a parallel vector on every rank.
HYPRE_Int getRelaxationType(LinearSolverParameters::PreconditionerType const type)
Returns hypre's identifier of the relaxation preconditioner type.
Definition: HypreUtils.hpp:481
HYPRE_BigInt * toHypreBigInt(geos::globalIndex *const index)
Converts a non-const array from GEOS globalIndex type to HYPRE_BigInt.
Definition: HypreUtils.hpp:184
HYPRE_Int relaxationSolve(HYPRE_Solver solver, HYPRE_ParCSRMatrix A, HYPRE_ParVector b, HYPRE_ParVector x)
Solve with a relaxation-based smoother.
constexpr HYPRE_MemoryLocation memoryLocation
Memory location used by hypre matrix/vector objects.
Definition: HypreUtils.hpp:115
HYPRE_Int getILUType(LinearSolverParameters::AMG::SmootherType const type)
Returns hypre's identifier of the AMG ILU smoother type.
Definition: HypreUtils.hpp:423
MGRFRelaxationType
This enum class specifies the F-relaxation type.
Definition: HypreUtils.hpp:580
@ amgVCycle
Full AMG VCycle solver.
@ none
no F-relaxation if performed
@ l1forwardGaussSeidel
Gauss-Seidel, forward solve
@ forwardHybridGaussSeidel
hybrid Gauss-Seidel or SOR, forward solve
@ gsElimWInverse
Direct Inversion with Gaussian Elimination (OK for larger systems)
@ backwardHybridGaussSeidel
hybrid Gauss-Seidel or SOR, backward solve
@ hybridSymmetricGaussSeidel
hybrid symmetric Gauss-Seidel or SSOR
@ l1hybridSymmetricGaussSeidel
-scaled hybrid symmetric Gauss-Seidel
@ l1backwardGaussSeidel
Gauss-Seidel, backward solve
@ singleVCycleSmoother
V(1,0) cycle smoother.
@ gsElimWPivoting
Gaussian Elimination with pivoting direct solver (for small systems)
@ gsElim
Gaussian Elimination direct solver (for small systems)
HYPRE_Int getAMGRelaxationType(LinearSolverParameters::AMG::SmootherType const &type)
Returns hypre's identifier of the AMG smoother type.
Definition: HypreUtils.hpp:356
HYPRE_Int dummySetup(HYPRE_Solver, HYPRE_ParCSRMatrix, HYPRE_ParVector, HYPRE_ParVector)
Dummy function that does nothing but conform to hypre's signature for preconditioner setup/apply func...
AMGCoarseningType
This enum class specifies the AMG parallel coarsening algorithm.
Definition: HypreUtils.hpp:516
@ CLJP
Parallel coarsening algorithm using independent sets.
@ Falgout
Uses Ruge_Stueben first, followed by CLJP.
@ Ruge_Stueben
Classical Ruge-Stueben coarsening on each processor.
@ PMIS
Parallel coarsening algorithm using independent sets.
@ CGC_E
Coarsening by M. Griebel, B. Metsch and A. Schweitzer.
@ HMIS
Uses one pass Ruge-Stueben on each processor independently, followed by PMIS.
@ CLJPDebug
Using a fixed random vector, for debugging purposes only.
@ PMISDebug
Using a fixed random vector, for debugging purposes only.
@ CGC
Coarsening by M. Griebel, B. Metsch and A. Schweitzer.
HYPRE_Int relaxationSetup(HYPRE_Solver solver, HYPRE_ParCSRMatrix A, HYPRE_ParVector b, HYPRE_ParVector x)
Setup a relaxation-based smoother.
HYPRE_Int relaxationCreate(HYPRE_Solver &solver, HYPRE_Int const type)
Create a relaxation-based smoother.
constexpr HYPRE_MemoryLocation getMemoryLocation(LvArray::MemorySpace const space)
Definition: HypreUtils.hpp:74
HYPRE_Int getAMGCoarseType(LinearSolverParameters::AMG::CoarseType const &type)
Returns hypre's identifier of the AMG coarse solver type.
Definition: HypreUtils.hpp:438
constexpr LvArray::MemorySpace getLvArrayMemorySpace(HYPRE_MemoryLocation const location)
Definition: HypreUtils.hpp:93
HYPRE_Int chebyshevDestroy(HYPRE_Solver solver)
Destroy a Chebyshev smoother.
HYPRE_Int getAMGCycleType(LinearSolverParameters::AMG::CycleType const &type)
Returns hypre's identifier of the AMG cycle type.
Definition: HypreUtils.hpp:341
HYPRE_Vector parVectorToVector(HYPRE_ParVector const vec, int const targetRank)
Gather a parallel vector onto a single rank.
HYPRE_Int relaxationDestroy(HYPRE_Solver solver)
Destroy a relaxation-based smoother.
HYPRE_Int getAMGCoarseningType(LinearSolverParameters::AMG::CoarseningType const &type)
Returns hypre's identifier of the AMG coarsening type.
Definition: HypreUtils.hpp:462
HYPRE_Int SuperLUDistSolve(HYPRE_Solver solver, HYPRE_ParCSRMatrix A, HYPRE_ParVector b, HYPRE_ParVector x)
The missing wrapper compatible with hypre solver solve signature.
HYPRE_Int SuperLUDistDestroy(HYPRE_Solver solver)
The missing wrapper compatible with hypre solver destroy signature.
void checkDeviceErrors(char const *msg, char const *file, int const line)
Definition: HypreUtils.hpp:148
HYPRE_Int chebyshevSolve(HYPRE_Solver solver, HYPRE_ParCSRMatrix A, HYPRE_ParVector b, HYPRE_ParVector x)
Solve with a Chebyshev smoother.
HYPRE_Int chebyshevCreate(HYPRE_Solver &solver, HYPRE_Int const order, HYPRE_Int const eigNumIter)
Create a Chebyshev smoother.
MGRInterpolationType
This enum class specifies the strategy for computing the level intepolation operator in MGR.
Definition: HypreUtils.hpp:533
@ classicalModifiedInterpolation
Classical modified interpolation.
@ approximateInverse
Approximate inverse.
constexpr LvArray::MemorySpace memorySpace
Memory space used by hypre matrix/vector objects.
Definition: HypreUtils.hpp:113
MGRGlobalSmootherType
This enum class specifies the global smoother type.
Definition: HypreUtils.hpp:602
@ none
no global smoothing is performed (default)
@ ilu0
incomplete LU factorization
parallelDevicePolicy<> execPolicy
Execution policy for operations on hypre data.
Definition: HypreUtils.hpp:111
GEOS_GLOBALINDEX_TYPE globalIndex
Global index type (for indexing objects across MPI partitions).
Definition: DataTypes.hpp:87
GEOS_LOCALINDEX_TYPE localIndex
Local index type (for indexing objects within an MPI partition).
Definition: DataTypes.hpp:84
Container for hypre preconditioner function pointers.
Definition: HypreUtils.hpp:48
HYPRE_Int(*)(HYPRE_Solver) DestroyFunc
Alias for destroy function type.
Definition: HypreUtils.hpp:56
HYPRE_Int(*)(HYPRE_Solver, HYPRE_ParCSRMatrix, HYPRE_ParVector, HYPRE_ParVector) SolveFunc
Alias for apply function type.
Definition: HypreUtils.hpp:53
HYPRE_Int(*)(HYPRE_Solver, HYPRE_ParCSRMatrix, HYPRE_ParVector, HYPRE_ParVector) SetupFunc
Alias for setup function type.
Definition: HypreUtils.hpp:50
HYPRE_Solver ptr
pointer to preconditioner
Definition: HypreUtils.hpp:58
SetupFunc setup
pointer to setup function
Definition: HypreUtils.hpp:59
DestroyFunc destroy
pointer to destroy function
Definition: HypreUtils.hpp:61
SolveFunc solve
pointer to apply function
Definition: HypreUtils.hpp:60
InterpType
AMG interpolation type (HYPRE only)
@ modifiedExtendedI
Modularized extended+i (GPU support)
@ modifiedExtendedE
Modularized extended+e (GPU support)
@ modifiedExtended
Modularized extended classical (GPU support)
@ directBAMG
Direct with separation of weights (GPU support)
@ extended
Extended classical (GPU support)
@ chebyshev
Chebyshev polynomial smoothing.
@ iluk
Incomplete LU with k-level of fill.
@ l1sgs
l1-Symmetric Gauss-Seidel smoothing
@ ilut
Incomplete LU with thresholding.
@ sgs
Symmetric Gauss-Seidel smoothing.
@ bgs
Gauss-Seidel smoothing (backward sweep)
@ fgs
Gauss-Seidel smoothing (forward sweep)
CoarseningType
AMG coarsening types (HYPRE only)
@ CLJP
A parallel coarsening algorithm using independent sets.
@ Falgout
Ruge-Stueben followed by CLJP.
@ RugeStueben
Classical Ruge-Stueben on each processor, followed by a third pass.
@ PMIS
Parallel coarsening as CLJP but with lower complexities (GPU support)
AggInterpType
AMG interpolation type for aggressive coarsening levels (HYPRE only)
@ modifiedExtendedI
Modularized Extended+i (GPU support)
@ extendedIStage2
Extended+i 2-stage (GPU support)
@ modifiedExtendedE
Modularized Extended+e (GPU support)
@ modifiedExtended
Modularized Extended (GPU support)
@ modifiedMultipass
Modularized Multipass (GPU support)
@ extendedStage2
Extended 2-stage (GPU support)
@ jacobi
Jacobi (GPU support in hypre)
@ chebyshev
Chebyshev polynomial (GPU support in hypre)
@ l1jacobi
l1-Jacobi (GPU support in hypre)
@ gsElimWInverse
Direct inverse with Gaussian Elimination.
@ direct
Direct solver as preconditioner.
@ gsElimWPivoting
Gaussian Elimination with pivoting direct solver.
@ bgs
Gauss-Seidel smoothing (backward sweep)
@ chebyshev
Chebyshev polynomial smoothing.
@ iluk
Incomplete LU with k-level of fill.
@ l1sgs
l1-Symmetric Gauss-Seidel smoothing
@ ilut
Incomplete LU with thresholding.
@ sgs
Symmetric Gauss-Seidel smoothing.
@ bgs
Gauss-Seidel smoothing (backward sweep)
@ fgs
Gauss-Seidel smoothing (forward sweep)