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 
212 HYPRE_Vector parVectorToVectorAll( HYPRE_ParVector const vec );
213 
220 HYPRE_Int dummySetup( HYPRE_Solver,
221  HYPRE_ParCSRMatrix,
222  HYPRE_ParVector,
223  HYPRE_ParVector );
224 
233 HYPRE_Int SuperLUDistSolve( HYPRE_Solver solver,
234  HYPRE_ParCSRMatrix A,
235  HYPRE_ParVector b,
236  HYPRE_ParVector x );
237 
243 HYPRE_Int SuperLUDistDestroy( HYPRE_Solver solver );
244 
251 HYPRE_Int relaxationCreate( HYPRE_Solver & solver,
252  HYPRE_Int const type );
253 
262 HYPRE_Int relaxationSetup( HYPRE_Solver solver,
263  HYPRE_ParCSRMatrix A,
264  HYPRE_ParVector b,
265  HYPRE_ParVector x );
266 
275 HYPRE_Int relaxationSolve( HYPRE_Solver solver,
276  HYPRE_ParCSRMatrix A,
277  HYPRE_ParVector b,
278  HYPRE_ParVector x );
279 
285 HYPRE_Int relaxationDestroy( HYPRE_Solver solver );
286 
293 {
295  {
298  };
299  return findOption( typeMap, type, "multigrid cycle", "HyprePreconditioner" );
300 }
301 
308 {
310  {
319  };
320  return findOption( typeMap, type, "multigrid relaxation", "HyprePreconditioner" );
321 }
322 
329 {
331  {
343  };
344  return findOption( typeMap, type, "multigrid interpolation", "HyprePreconditioner" );
345 }
346 
353 {
355  {
365  };
366  return findOption( typeMap, type, "multigrid aggressive interpolation", "HyprePreconditioner" );
367 }
368 
375 {
377  {
380  };
381  return findOption( typeMap, type, "ILU", "HyprePreconditioner" );
382 }
383 
390 {
392  {
404  };
405  return findOption( typeMap, type, "multigrid coarse solver", "HyprePreconditioner" );
406 }
407 
414 {
416  {
423  };
424  return findOption( typeMap, type, "multigrid coarsening", "HyprePreconditioner" );
425 }
426 
433 {
435  {
443  };
444  return findOption( typeMap, type, "relaxation", "HyprePreconditioner" );
445 }
446 
453 {
455  {
458  };
459  return findOption( typeMap, type, "ILU", "HyprePreconditioner" );
460 }
461 
466 enum class AMGCoarseningType : HYPRE_Int
467 {
468  CLJP = 0,
469  Ruge_Stueben = 3,
470  Falgout = 6,
471  CLJPDebug = 7,
472  PMIS = 8,
473  PMISDebug = 9,
474  HMIS = 10,
475  CGC = 21,
476  CGC_E = 22
477 };
478 
483 enum class MGRInterpolationType : HYPRE_Int
484 {
485  injection = 0,
486  l1jacobi = 1,
487  jacobi = 2,
489  approximateInverse = 4,
490  blockJacobi = 12
491 };
492 
498 enum class MGRRestrictionType : HYPRE_Int
499 {
500  injection = 0,
501  jacobi = 2,
502  approximateInverse = 3,
503  blockJacobi = 12,
504  cprLike = 13,
505  blockColLumped = 14
506 };
507 
512 enum class MGRCoarseGridMethod : HYPRE_Int
513 {
514  galerkin = 0,
515  nonGalerkin = 1,
517  cprLikeDiag = 2,
519  cprLikeBlockDiag = 3,
521  approximateInverse = 4
523 };
524 
529 enum class MGRFRelaxationType : HYPRE_Int
530 {
531  none = -1,
533  amgVCycle = 2,
537  jacobi = 7,
539  gsElim = 9,
540  l1forwardGaussSeidel = 13,
541  l1backwardGaussSeidel = 14,
542  l1jacobi = 18,
543  gsElimWPivoting = 99,
544  gsElimWInverse = 199
545 };
546 
551 enum class MGRGlobalSmootherType : HYPRE_Int
552 {
553  none = -1,
554  blockJacobi = 0,
555  blockGaussSeidel = 1,
556  jacobi = 2,
557  ilu0 = 16
558 };
559 
560 } // namespace hypre
561 
562 } // namespace geos
563 
564 #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.
Definition: DataTypes.hpp:329
HYPRE_Int getAMGInterpolationType(LinearSolverParameters::AMG::InterpType const &type)
Returns hypre's identifier of the AMG interpolation type.
Definition: HypreUtils.hpp:328
MGRCoarseGridMethod
This enum class specifies the strategy for level coarse grid computation in MGR.
Definition: HypreUtils.hpp:513
@ 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:499
@ blockColLumped
Block column-lumped approximation.
@ approximateInverse
Approximate inverse.
@ cprLike
CPR-like restriction.
HYPRE_Int getAMGAggressiveInterpolationType(LinearSolverParameters::AMG::AggInterpType const &type)
Returns hypre's identifier of the AMG aggressive interpolation type.
Definition: HypreUtils.hpp:352
HYPRE_Vector parVectorToVectorAll(HYPRE_ParVector const vec)
Gather a parallel vector on a every rank.
HYPRE_Int getRelaxationType(LinearSolverParameters::PreconditionerType const type)
Returns hypre's identifier of the relaxation preconditioner type.
Definition: HypreUtils.hpp:432
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:374
MGRFRelaxationType
This enum class specifies the F-relaxation type.
Definition: HypreUtils.hpp:530
@ 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:307
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:467
@ 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:389
constexpr LvArray::MemorySpace getLvArrayMemorySpace(HYPRE_MemoryLocation const location)
Definition: HypreUtils.hpp:99
HYPRE_Int getAMGCycleType(LinearSolverParameters::AMG::CycleType const &type)
Returns hypre's identifier of the AMG cycle type.
Definition: HypreUtils.hpp:292
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:413
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
MGRInterpolationType
This enum class specifies the strategy for computing the level intepolation operator in MGR.
Definition: HypreUtils.hpp:484
@ 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:552
@ 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:88
GEOS_LOCALINDEX_TYPE localIndex
Local index type (for indexing objects within an MPI partition).
Definition: DataTypes.hpp:85
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.
@ 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)