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 
32 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA || GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
34 #define GEOS_HYPRE_DEVICE GEOS_DEVICE
36 #define GEOS_HYPRE_HOST_DEVICE GEOS_HOST_DEVICE
37 #else
39 #define GEOS_HYPRE_DEVICE
41 #define GEOS_HYPRE_HOST_DEVICE
42 #endif
43 
44 namespace geos
45 {
46 
54 {
56  using SetupFunc = HYPRE_Int (*)( HYPRE_Solver, HYPRE_ParCSRMatrix, HYPRE_ParVector, HYPRE_ParVector );
57 
59  using SolveFunc = HYPRE_Int (*)( HYPRE_Solver, HYPRE_ParCSRMatrix, HYPRE_ParVector, HYPRE_ParVector );
60 
62  using DestroyFunc = HYPRE_Int (*)( HYPRE_Solver );
63 
64  HYPRE_Solver ptr{};
68 };
69 
73 namespace hypre
74 {
75 
80 constexpr HYPRE_MemoryLocation getMemoryLocation( LvArray::MemorySpace const space )
81 {
82  switch( space )
83  {
84  case hostMemorySpace: return HYPRE_MEMORY_HOST;
85 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA
86  case LvArray::MemorySpace::cuda: return HYPRE_MEMORY_DEVICE;
87 #endif
88 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
89  case LvArray::MemorySpace::hip: return HYPRE_MEMORY_DEVICE;
90 #endif
91  default: return HYPRE_MEMORY_HOST;
92  }
93 }
94 
99 constexpr LvArray::MemorySpace getLvArrayMemorySpace( HYPRE_MemoryLocation const location )
100 {
101  switch( location )
102  {
103  case HYPRE_MEMORY_HOST: return hostMemorySpace;
104 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA
105  case HYPRE_MEMORY_DEVICE: return parallelDeviceMemorySpace;
106 #endif
107 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
108  case HYPRE_MEMORY_DEVICE: return parallelDeviceMemorySpace;
109 #endif
110  default: return hostMemorySpace;
111  }
112 }
113 
114 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA || GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
115 
117 using execPolicy = parallelDevicePolicy<>;
119 constexpr LvArray::MemorySpace memorySpace = parallelDeviceMemorySpace;
121 constexpr HYPRE_MemoryLocation memoryLocation = HYPRE_MEMORY_DEVICE;
122 
123 #else
124 
126 using execPolicy = parallelHostPolicy;
128 constexpr LvArray::MemorySpace memorySpace = hostMemorySpace;
130 constexpr HYPRE_MemoryLocation memoryLocation = HYPRE_MEMORY_HOST;
131 
132 #endif
133 
134 // Check matching requirements on index/value types between GEOSX and Hypre
135 
136 // WARNING. We don't have consistent types between HYPRE_Int and localIndex.
137 // Decision needs to be made either to use bigint option, or change
138 // localIndex to int. We are getting away with this because we do not
139 // pass ( localIndex * ) to hypre except when it is on the GPU, in
140 // which case we are using int for localIndex.
141 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA || GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
142 static_assert( sizeof( HYPRE_Int ) == sizeof( geos::localIndex ),
143  "HYPRE_Int and geos::localIndex must have the same size" );
144 static_assert( std::is_signed< HYPRE_Int >::value == std::is_signed< geos::localIndex >::value,
145  "HYPRE_Int and geos::localIndex must both be signed or unsigned" );
146 #endif
147 
154 inline void checkDeviceErrors( char const * msg, char const * file, int const line )
155 {
156 #if GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_CUDA
157  cudaError_t const err = cudaGetLastError();
158  GEOS_ERROR_IF( err != cudaSuccess, GEOS_FMT( "Previous CUDA errors found: {} ({} at {}:{})", msg, cudaGetErrorString( err ), file, line ) );
159 #elif GEOS_USE_HYPRE_DEVICE == GEOS_USE_HYPRE_HIP
160  hipError_t const err = hipGetLastError();
161  GEOS_UNUSED_VAR( msg, file, line ); // on crusher geos_error_if ultimately resolves to an assert, which drops the content on release
162  // builds
163  GEOS_ERROR_IF( err != hipSuccess, GEOS_FMT( "Previous HIP errors found: {} ({} at {}:{})", msg, hipGetErrorString( err ), file, line ) );
164 #else
165  GEOS_UNUSED_VAR( msg, file, line );
166 #endif
167 }
168 
173 #define GEOS_HYPRE_CHECK_DEVICE_ERRORS( msg ) ::geos::hypre::checkDeviceErrors( msg, __FILE__, __LINE__ )
174 
175 static_assert( sizeof( HYPRE_BigInt ) == sizeof( geos::globalIndex ),
176  "HYPRE_BigInt and geos::globalIndex must have the same size" );
177 
178 static_assert( std::is_signed< HYPRE_BigInt >::value == std::is_signed< geos::globalIndex >::value,
179  "HYPRE_BigInt and geos::globalIndex must both be signed or unsigned" );
180 
181 static_assert( std::is_same< HYPRE_Real, geos::real64 >::value,
182  "HYPRE_Real and geos::real64 must be the same type" );
183 
189 inline HYPRE_BigInt * toHypreBigInt( geos::globalIndex * const index )
190 {
191  return reinterpret_cast< HYPRE_BigInt * >(index);
192 }
193 
199 inline HYPRE_BigInt const * toHypreBigInt( geos::globalIndex const * const index )
200 {
201  return reinterpret_cast< HYPRE_BigInt const * >(index);
202 }
203 
213 HYPRE_Vector parVectorToVectorAll( HYPRE_ParVector const vec );
214 
222 HYPRE_Vector parVectorToVector( HYPRE_ParVector const vec, int const targetRank );
223 
230 HYPRE_Int dummySetup( HYPRE_Solver,
231  HYPRE_ParCSRMatrix,
232  HYPRE_ParVector,
233  HYPRE_ParVector );
234 
243 HYPRE_Int SuperLUDistSolve( HYPRE_Solver solver,
244  HYPRE_ParCSRMatrix A,
245  HYPRE_ParVector b,
246  HYPRE_ParVector x );
247 
253 HYPRE_Int SuperLUDistDestroy( HYPRE_Solver solver );
254 
261 HYPRE_Int relaxationCreate( HYPRE_Solver & solver,
262  HYPRE_Int const type );
263 
272 HYPRE_Int relaxationSetup( HYPRE_Solver solver,
273  HYPRE_ParCSRMatrix A,
274  HYPRE_ParVector b,
275  HYPRE_ParVector x );
276 
285 HYPRE_Int relaxationSolve( HYPRE_Solver solver,
286  HYPRE_ParCSRMatrix A,
287  HYPRE_ParVector b,
288  HYPRE_ParVector x );
289 
295 HYPRE_Int relaxationDestroy( HYPRE_Solver solver );
296 
304 HYPRE_Int chebyshevCreate( HYPRE_Solver & solver,
305  HYPRE_Int const order,
306  HYPRE_Int const eigNumIter );
307 
316 HYPRE_Int chebyshevSetup( HYPRE_Solver solver,
317  HYPRE_ParCSRMatrix A,
318  HYPRE_ParVector b,
319  HYPRE_ParVector x );
320 
329 HYPRE_Int chebyshevSolve( HYPRE_Solver solver,
330  HYPRE_ParCSRMatrix A,
331  HYPRE_ParVector b,
332  HYPRE_ParVector x );
333 
339 HYPRE_Int chebyshevDestroy( HYPRE_Solver solver );
340 
347 {
349  {
352  };
353  return findOption( typeMap, type, "multigrid cycle", "HyprePreconditioner" );
354 }
355 
362 {
364  {
373  };
374  return findOption( typeMap, type, "multigrid relaxation", "HyprePreconditioner" );
375 }
376 
383 {
385  {
397  };
398  return findOption( typeMap, type, "multigrid interpolation", "HyprePreconditioner" );
399 }
400 
407 {
409  {
419  };
420  return findOption( typeMap, type, "multigrid aggressive interpolation", "HyprePreconditioner" );
421 }
422 
429 {
431  {
434  };
435  return findOption( typeMap, type, "ILU", "HyprePreconditioner" );
436 }
437 
444 {
446  {
458  };
459  return findOption( typeMap, type, "multigrid coarse solver", "HyprePreconditioner" );
460 }
461 
468 {
470  {
477  };
478  return findOption( typeMap, type, "multigrid coarsening", "HyprePreconditioner" );
479 }
480 
487 {
489  {
497  };
498  return findOption( typeMap, type, "relaxation", "HyprePreconditioner" );
499 }
500 
507 {
509  {
512  };
513  return findOption( typeMap, type, "ILU", "HyprePreconditioner" );
514 }
515 
520 enum class AMGCoarseningType : HYPRE_Int
521 {
522  CLJP = 0,
523  Ruge_Stueben = 3,
524  Falgout = 6,
525  CLJPDebug = 7,
526  PMIS = 8,
527  PMISDebug = 9,
528  HMIS = 10,
529  CGC = 21,
530  CGC_E = 22
531 };
532 
537 enum class MGRInterpolationType : HYPRE_Int
538 {
539  injection = 0,
540  l1jacobi = 1,
541  jacobi = 2,
543  approximateInverse = 4,
544  blockJacobi = 12
545 };
546 
552 enum class MGRRestrictionType : HYPRE_Int
553 {
554  injection = 0,
555  jacobi = 2,
556  approximateInverse = 3,
557  blockJacobi = 12,
558  cprLike = 13,
559  blockColLumped = 14
560 };
561 
566 enum class MGRCoarseGridMethod : HYPRE_Int
567 {
568  galerkin = 0,
569  nonGalerkin = 1,
571  cprLikeDiag = 2,
573  cprLikeBlockDiag = 3,
575  approximateInverse = 4
577 };
578 
583 enum class MGRFRelaxationType : HYPRE_Int
584 {
585  none = -1,
587  amgVCycle = 2,
591  jacobi = 7,
593  gsElim = 9,
594  l1forwardGaussSeidel = 13,
595  l1backwardGaussSeidel = 14,
596  l1jacobi = 18,
597  gsElimWPivoting = 99,
598  gsElimWInverse = 199
599 };
600 
605 enum class MGRGlobalSmootherType : HYPRE_Int
606 {
607  none = -1,
608  blockJacobi = 0,
609  blockGaussSeidel = 1,
610  jacobi = 2,
611  ilu0 = 16
612 };
613 
614 } // namespace hypre
615 
616 } // namespace geos
617 
618 #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(EXP, msg)
Conditionally raise a hard error and terminate the program.
Definition: Logger.hpp:142
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:382
MGRCoarseGridMethod
This enum class specifies the strategy for level coarse grid computation in MGR.
Definition: HypreUtils.hpp:567
@ 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:553
@ blockColLumped
Block 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:406
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:486
HYPRE_BigInt * toHypreBigInt(geos::globalIndex *const index)
Converts a non-const array from GEOSX globalIndex type to HYPRE_BigInt.
Definition: HypreUtils.hpp:189
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:121
HYPRE_Int getILUType(LinearSolverParameters::AMG::SmootherType const type)
Returns hypre's identifier of the AMG ILU smoother type.
Definition: HypreUtils.hpp:428
MGRFRelaxationType
This enum class specifies the F-relaxation type.
Definition: HypreUtils.hpp:584
@ 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:361
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:521
@ 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:80
HYPRE_Int getAMGCoarseType(LinearSolverParameters::AMG::CoarseType const &type)
Returns hypre's identifier of the AMG coarse solver type.
Definition: HypreUtils.hpp:443
constexpr LvArray::MemorySpace getLvArrayMemorySpace(HYPRE_MemoryLocation const location)
Definition: HypreUtils.hpp:99
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:346
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:467
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:154
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:538
@ classicalModifiedInterpolation
Classical modified interpolation.
@ approximateInverse
Approximate inverse.
constexpr LvArray::MemorySpace memorySpace
Memory space used by hypre matrix/vector objects.
Definition: HypreUtils.hpp:119
MGRGlobalSmootherType
This enum class specifies the global smoother type.
Definition: HypreUtils.hpp:606
@ none
no global smoothing is performed (default)
@ ilu0
incomplete LU factorization
parallelDevicePolicy<> execPolicy
Execution policy for operations on hypre data.
Definition: HypreUtils.hpp:117
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:54
HYPRE_Int(*)(HYPRE_Solver) DestroyFunc
Alias for destroy function type.
Definition: HypreUtils.hpp:62
HYPRE_Int(*)(HYPRE_Solver, HYPRE_ParCSRMatrix, HYPRE_ParVector, HYPRE_ParVector) SolveFunc
Alias for apply function type.
Definition: HypreUtils.hpp:59
HYPRE_Int(*)(HYPRE_Solver, HYPRE_ParCSRMatrix, HYPRE_ParVector, HYPRE_ParVector) SetupFunc
Alias for setup function type.
Definition: HypreUtils.hpp:56
HYPRE_Solver ptr
pointer to preconditioner
Definition: HypreUtils.hpp:64
SetupFunc setup
pointer to setup function
Definition: HypreUtils.hpp:65
DestroyFunc destroy
pointer to destroy function
Definition: HypreUtils.hpp:67
SolveFunc solve
pointer to apply function
Definition: HypreUtils.hpp:66
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)