GEOS
RateAndStateKernelsBase.hpp
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) 2018-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 
16 #ifndef GEOS_PHYSICSSOLVERS_INDUCEDSEISMICITY_KERNELS_RATEANDSTATEKERNELSBASE_HPP_
17 #define GEOS_PHYSICSSOLVERS_INDUCEDSEISMICITY_KERNELS_RATEANDSTATEKERNELSBASE_HPP_
18 
19 #include "common/DataTypes.hpp"
20 #include "common/GEOS_RAJA_Interface.hpp"
21 #include "constitutive/contact/RateAndStateFriction.hpp"
23 #include "constitutive/ConstitutivePassThru.hpp"
24 
25 namespace geos
26 {
27 
28 namespace rateAndStateKernels
29 {
30 
31 // TBD: Pass the kernel and add getters for relevant fields to make this function general purpose and avoid
32 // wrappers?
34 static void projectSlipRateBase( localIndex const k,
35  real64 const frictionCoefficient,
36  real64 const shearImpedance,
37  arrayView1d< real64 const > const normalTraction,
38  arrayView2d< real64 const > const shearTraction,
39  arrayView1d< real64 const > const slipRate,
40  arrayView2d< real64 > const slipVelocity )
41 {
42  // Project slip rate onto shear traction to get slip velocity components
43  real64 const frictionForce = normalTraction[k] * frictionCoefficient;
44  real64 const projectionScaling = 1.0 / ( shearImpedance + frictionForce / slipRate[k] );
45  slipVelocity[k][0] = projectionScaling * shearTraction[k][0];
46  slipVelocity[k][1] = projectionScaling * shearTraction[k][1];
47 }
48 
49 template< typename POLICY, typename KERNEL_TYPE >
50 static bool newtonSolve( SurfaceElementSubRegion & subRegion,
51  KERNEL_TYPE & kernel,
52  real64 const dt,
53  integer const maxNewtonIter,
54  real64 const newtonTol )
55 {
56  bool allConverged = false;
57  for( integer iter = 0; iter < maxNewtonIter; iter++ )
58  {
59  RAJA::ReduceMin< parallelDeviceReduce, int > converged( 1 );
60  RAJA::ReduceMax< parallelDeviceReduce, real64 > residualNorm( 0.0 );
61  forAll< POLICY >( subRegion.size(), [=] GEOS_HOST_DEVICE ( localIndex const k )
62  {
63  typename KERNEL_TYPE::StackVariables stack;
64  kernel.setup( k, dt, stack );
65  kernel.solve( k, stack );
66  auto const [elementConverged, elementResidualNorm] = kernel.checkConvergence( stack, newtonTol );
67  converged.min( elementConverged );
68  residualNorm.max( elementResidualNorm );
69  } );
70 
71  real64 const maxResidualNorm = MpiWrapper::max( residualNorm.get() );
72  GEOS_LOG_RANK_0( GEOS_FMT( " Newton iter {} : residual = {:.10e} ", iter, maxResidualNorm ) );
73 
74  if( converged.get() )
75  {
76  allConverged = true;
77  break;
78  }
79  }
80  return allConverged;
81 }
82 template< typename FRICTION_TYPE >
83 void enforceRateAndVelocityConsistency( FRICTION_TYPE const & frictionLawKernelWrapper,
84  SurfaceElementSubRegion & subRegion,
85  real64 const & shearImpedance )
86 {
87  arrayView2d< real64 > const slipVelocity = subRegion.getField< fields::rateAndState::slipVelocity >();
88  arrayView1d< real64 > const slipRate = subRegion.getField< fields::rateAndState::slipRate >();
89  arrayView1d< real64 const > const stateVariable = subRegion.getField< fields::rateAndState::stateVariable >();
90 
91  arrayView2d< real64 > const backgroundShearStress = subRegion.getField< fields::rateAndState::backgroundShearStress >();
92  arrayView1d< real64 > const backgroundNormalStress = subRegion.getField< fields::rateAndState::backgroundNormalStress >();
93 
94  RAJA::ReduceMax< parallelDeviceReduce, int > negativeSlipRate( 0 );
95  RAJA::ReduceMax< parallelDeviceReduce, int > bothNonZero( 0 );
96 
97  forAll< parallelDevicePolicy<> >( subRegion.size(), [=] GEOS_HOST_DEVICE ( localIndex const k )
98  {
99  if( slipRate[k] < 0.0 )
100  {
101  negativeSlipRate.max( 1 );
102  }
103  else if( LvArray::tensorOps::l2Norm< 2 >( slipVelocity[k] ) > 0.0 && slipRate[k] > 0.0 )
104  {
105  bothNonZero.max( 1 );
106  }
107  else if( LvArray::tensorOps::l2Norm< 2 >( slipVelocity[k] ) > 0.0 )
108  {
109  slipRate[k] = LvArray::tensorOps::l2Norm< 2 >( slipVelocity[k] );
110  }
111  else if( slipRate[k] > 0.0 )
112  {
113  real64 const frictionCoefficient = frictionLawKernelWrapper.frictionCoefficient( k, slipRate[k], stateVariable[k] );
114  projectSlipRateBase( k,
115  frictionCoefficient,
116  shearImpedance,
117  backgroundNormalStress,
118  backgroundShearStress,
119  slipRate,
120  slipVelocity );
121  }
122  } );
123 
124  GEOS_ERROR_IF( negativeSlipRate.get() > 0, "SlipRate cannot be negative." );
125  GEOS_ERROR_IF( bothNonZero.get() > 0, "Only one between slipRate and slipVelocity can be specified as i.c." );
126 }
127 
132 template< template< typename FRICTION_TYPE > class KERNEL_TYPE,
133  typename POLICY,
134  typename FRICTION_TYPE >
135 static void
136 createAndLaunch( SurfaceElementSubRegion & subRegion,
137  FRICTION_TYPE & frictionLaw,
138  real64 const shearImpedance,
139  integer const maxNewtonIter,
140  real64 const newtonTol,
141  real64 const time_n,
142  real64 const totalDt )
143 {
145 
146  GEOS_UNUSED_VAR( time_n );
147 
148  KERNEL_TYPE kernel( subRegion, frictionLaw, shearImpedance );
149 
150  real64 dtRemaining = totalDt;
151  real64 dt = totalDt;
152  for( integer subStep = 0; subStep < 5 && dtRemaining > 0.0; ++subStep )
153  {
154  real64 dtAccepted = KERNEL_TYPE< FRICTION_TYPE >::template solveRateAndStateEquation< POLICY >( subRegion, kernel, dt, maxNewtonIter, newtonTol );
155  dtRemaining -= dtAccepted;
156 
157  if( dtRemaining > 0.0 )
158  {
159  dt = dtAccepted;
160  }
161  GEOS_LOG_RANK_0( GEOS_FMT( " sub-step = {} completed, dt = {}, remaining dt = {}", subStep, dt, dtRemaining ) );
162  }
163 }
164 
165 } /* namespace rateAndStateKernels */
166 
167 } /* namespace geos */
168 
169 #endif /* GEOS_PHYSICSSOLVERS_INDUCEDSEISMICITY_KERNELS_RATEANDSTATEKERNELSBASE_HPP_ */
#define GEOS_HOST_DEVICE
Marks a host-device function.
Definition: GeosxMacros.hpp:49
#define GEOS_UNUSED_VAR(...)
Mark an unused variable and silence compiler warnings.
Definition: GeosxMacros.hpp:84
#define GEOS_LOG_RANK_0(msg)
Log a message on screen on rank 0.
Definition: Logger.hpp:101
#define GEOS_ERROR_IF(EXP, msg)
Conditionally raise a hard error and terminate the program.
Definition: Logger.hpp:142
#define GEOS_MARK_FUNCTION
Mark function with both Caliper and NVTX if enabled.
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
std::int32_t integer
Signed integer type.
Definition: DataTypes.hpp:82
static T max(T const &value, MPI_Comm comm=MPI_COMM_GEOS)
Convenience function for a MPI_Allreduce using a MPI_MAX operation.