GEOS
ResidualNormKernel.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_PHYSICSSOLVERS_FLUIDFLOW_SINGLEPHASE_RESIDUALNORMKERNEL_HPP
21 #define GEOS_PHYSICSSOLVERS_FLUIDFLOW_SINGLEPHASE_RESIDUALNORMKERNEL_HPP
22 
26 
27 namespace geos
28 {
29 
30 namespace singlePhaseBaseKernels
31 {
32 
33 /******************************** ResidualNormKernel ********************************/
34 
39 {
40 public:
41 
44  using Base::m_rankOffset;
46  using Base::m_dofNumber;
47 
48  IsothermalResidualNormKernel( globalIndex const rankOffset,
49  arrayView1d< real64 const > const & localResidual,
50  arrayView1d< globalIndex const > const & dofNumber,
52  ElementSubRegionBase const & subRegion,
53  real64 const minNormalizer )
54  : Base( rankOffset,
55  localResidual,
56  dofNumber,
57  ghostRank,
58  minNormalizer ),
59  m_mass_n( subRegion.template getField< fields::flow::mass_n >() )
60  {}
61 
63  virtual void computeLinf( localIndex const ei,
64  LinfStackVariables & stack ) const override
65  {
66  real64 const massNormalizer = LvArray::math::max( m_minNormalizer, m_mass_n[ei] );
67  real64 const valMass = LvArray::math::abs( m_localResidual[stack.localRow] ) / massNormalizer;
68  if( valMass > stack.localValue[0] )
69  {
70  stack.localValue[0] = valMass;
71  }
72  }
73 
75  virtual void computeL2( localIndex const ei,
76  L2StackVariables & stack ) const override
77  {
78  real64 const massNormalizer = LvArray::math::max( m_minNormalizer, m_mass_n[ei] );
79  stack.localValue[0] += m_localResidual[stack.localRow] * m_localResidual[stack.localRow];
80  stack.localNormalizer[0] += massNormalizer;
81  }
82 
83 
84 protected:
85 
88 
89 };
90 
95 {
96 public:
97 
100  using Base::m_rankOffset;
101  using Base::m_localResidual;
102  using Base::m_dofNumber;
103 
104  ThermalResidualNormKernel( globalIndex const rankOffset,
105  arrayView1d< real64 const > const & localResidual,
106  arrayView1d< globalIndex const > const & dofNumber,
108  ElementSubRegionBase const & subRegion,
109  real64 const minNormalizer )
110  : Base( rankOffset,
111  localResidual,
112  dofNumber,
113  ghostRank,
114  minNormalizer ),
115  m_mass_n( subRegion.template getField< fields::flow::mass_n >() ),
116  m_energy_n( subRegion.template getField< fields::flow::energy_n >() )
117  {}
118 
120  void computeMassEnergyNormalizers( localIndex const ei,
121  real64 & massNormalizer,
122  real64 & energyNormalizer ) const
123  {
124  massNormalizer = LvArray::math::max( m_minNormalizer, m_mass_n[ei] );
125  energyNormalizer = LvArray::math::max( m_minNormalizer, LvArray::math::abs( m_energy_n[ei] ) ); // energy can be negative
126  }
127 
129  virtual void computeLinf( localIndex const ei,
130  LinfStackVariables & stack ) const override
131  {
132  real64 massNormalizer = 0.0, energyNormalizer = 0.0;
133  computeMassEnergyNormalizers( ei, massNormalizer, energyNormalizer );
134 
135  // step 1: mass residual
136 
137  real64 const valMass = LvArray::math::abs( m_localResidual[stack.localRow] ) / massNormalizer;
138  if( valMass > stack.localValue[0] )
139  {
140  stack.localValue[0] = valMass;
141  }
142 
143  // step 2: energy residual
144  real64 const valEnergy = LvArray::math::abs( m_localResidual[stack.localRow + 1] ) / energyNormalizer;
145  if( valEnergy > stack.localValue[1] )
146  {
147  stack.localValue[1] = valEnergy;
148  }
149  }
150 
152  virtual void computeL2( localIndex const ei,
153  L2StackVariables & stack ) const override
154  {
155  real64 massNormalizer = 0.0, energyNormalizer = 0.0;
156  computeMassEnergyNormalizers( ei, massNormalizer, energyNormalizer );
157 
158  // step 1: mass residual
159 
160  stack.localValue[0] += m_localResidual[stack.localRow] * m_localResidual[stack.localRow];
161  stack.localNormalizer[0] += massNormalizer;
162 
163  // step 2: energy residual
164 
165  stack.localValue[1] += m_localResidual[stack.localRow + 1] * m_localResidual[stack.localRow + 1];
166  stack.localNormalizer[1] += energyNormalizer;
167  }
168 
169 
170 protected:
171 
174 
177 
178 };
179 
184 {
185 public:
186 
200  template< typename POLICY >
201  static void
203  globalIndex const rankOffset,
204  string const dofKey,
205  arrayView1d< real64 const > const & localResidual,
206  ElementSubRegionBase const & subRegion,
207  real64 const minNormalizer,
208  real64 (& residualNorm)[1],
209  real64 (& residualNormalizer)[1] )
210  {
211  arrayView1d< globalIndex const > const dofNumber = subRegion.getReference< array1d< globalIndex > >( dofKey );
212  arrayView1d< integer const > const ghostRank = subRegion.ghostRank();
213 
214  IsothermalResidualNormKernel kernel( rankOffset, localResidual, dofNumber, ghostRank, subRegion, minNormalizer );
215  if( normType == physicsSolverBaseKernels::NormType::Linf )
216  {
217  IsothermalResidualNormKernel::launchLinf< POLICY >( subRegion.size(), kernel, residualNorm );
218  }
219  else // L2 norm
220  {
221  IsothermalResidualNormKernel::launchL2< POLICY >( subRegion.size(), kernel, residualNorm, residualNormalizer );
222  }
223  }
224 
239  template< typename POLICY >
240  static void
242  globalIndex const rankOffset,
243  string const & dofKey,
244  arrayView1d< real64 const > const & localResidual,
245  ElementSubRegionBase const & subRegion,
246  real64 const minNormalizer,
247  real64 (& residualNorm)[2],
248  real64 (& residualNormalizer)[2] )
249  {
250  arrayView1d< globalIndex const > const dofNumber = subRegion.getReference< array1d< globalIndex > >( dofKey );
251  arrayView1d< integer const > const ghostRank = subRegion.ghostRank();
252 
253  ThermalResidualNormKernel kernel( rankOffset, localResidual, dofNumber, ghostRank, subRegion, minNormalizer );
254  if( normType == physicsSolverBaseKernels::NormType::Linf )
255  {
256  ThermalResidualNormKernel::launchLinf< POLICY >( subRegion.size(), kernel, residualNorm );
257  }
258  else // L2 norm
259  {
260  ThermalResidualNormKernel::launchL2< POLICY >( subRegion.size(), kernel, residualNorm, residualNormalizer );
261  }
262  }
263 
264 };
265 
266 } // namespace singlePhaseBaseKernels
267 
268 } // namespace geos
269 
270 #endif //GEOS_PHYSICSSOLVERS_FLUIDFLOW_SINGLEPHASE_RESIDUALNORMKERNEL_HPP
#define GEOS_HOST_DEVICE
Marks a host-device function.
Definition: GeosxMacros.hpp:49
NormType
Type of norm used to check convergence TODO: find a way to put this inside the class.
array1d< integer > const & ghostRank()
Get the ghost information of each object.
GEOS_DECLTYPE_AUTO_RETURN getReference(LOOKUP_TYPE const &lookup) const
Look up a wrapper and get reference to wrapped object.
Definition: Group.hpp:1275
localIndex size() const
Get the "size" of the group, which determines the number of elements in resizable wrappers.
Definition: Group.hpp:1317
Define the base interface for the residual calculations.
real64 const m_minNormalizer
Value used to make sure that normalizers are never zero.
arrayView1d< globalIndex const > const m_dofNumber
View on the dof numbers.
GEOS_HOST_DEVICE integer ghostRank(localIndex const i) const
Getter for the ghost rank.
arrayView1d< real64 const > const m_localResidual
View on the local residual.
virtual GEOS_HOST_DEVICE void computeLinf(localIndex const ei, LinfStackVariables &stack) const override
Compute the local values for the Linf norm.
arrayView1d< real64 const > const m_mass_n
View on mass at the previous converged time step.
virtual GEOS_HOST_DEVICE void computeL2(localIndex const ei, L2StackVariables &stack) const override
Compute the local values and normalizer for the L2 norm.
static void createAndLaunch(physicsSolverBaseKernels::NormType const normType, globalIndex const rankOffset, string const dofKey, arrayView1d< real64 const > const &localResidual, ElementSubRegionBase const &subRegion, real64 const minNormalizer, real64(&residualNorm)[1], real64(&residualNormalizer)[1])
Create a new kernel and launch (isothermal version)
static void createAndLaunch(physicsSolverBaseKernels::NormType const normType, globalIndex const rankOffset, string const &dofKey, arrayView1d< real64 const > const &localResidual, ElementSubRegionBase const &subRegion, real64 const minNormalizer, real64(&residualNorm)[2], real64(&residualNormalizer)[2])
Create a new kernel and launch (thermal version)
virtual GEOS_HOST_DEVICE void computeL2(localIndex const ei, L2StackVariables &stack) const override
Compute the local values and normalizer for the L2 norm.
arrayView1d< real64 const > const m_mass_n
View on mass at the previous converged time step.
virtual GEOS_HOST_DEVICE void computeLinf(localIndex const ei, LinfStackVariables &stack) const override
Compute the local values for the Linf norm.
arrayView1d< real64 const > const m_energy_n
View on energy at the previous converged time step.
ArrayView< T, 1 > arrayView1d
Alias for 1D array view.
Definition: DataTypes.hpp:180
GEOS_GLOBALINDEX_TYPE globalIndex
Global index type (for indexing objects across MPI partitions).
Definition: DataTypes.hpp:88
double real64
64-bit floating point type.
Definition: DataTypes.hpp:99
GEOS_LOCALINDEX_TYPE localIndex
Local index type (for indexing objects within an MPI partition).
Definition: DataTypes.hpp:85
Array< T, 1 > array1d
Alias for 1D array.
Definition: DataTypes.hpp:176