GEOS
C1PPUPhaseFlux.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_COMPOSITIONAL_C1PPUPHASEFLUX_HPP
21 #define GEOS_PHYSICSSOLVERS_FLUIDFLOW_COMPOSITIONAL_C1PPUPHASEFLUX_HPP
22 
23 #include "common/DataLayouts.hpp"
24 #include "common/DataTypes.hpp"
25 #include "constitutive/fluid/multifluid/Layouts.hpp"
26 #include "constitutive/capillaryPressure/layouts.hpp"
29 
30 
31 namespace geos
32 {
33 
34 namespace isothermalCompositionalMultiphaseFVMKernelUtilities
35 {
36 
37 // TODO make input parameter
38 static constexpr real64 epsC1PPU = 5000;
39 
40 template< typename VIEWTYPE >
41 using ElementViewConst = ElementRegionManager::ElementViewConst< VIEWTYPE >;
42 
43 using Deriv = constitutive::multifluid::DerivativeOffset;
44 
46 {
74  template< integer numComp, integer numFluxSupportPoints >
76  static void
77  compute( integer const numPhase,
78  integer const ip,
79  integer const hasCapPressure,
80  integer const checkPhasePresenceInGravity,
81  localIndex const ( &seri )[numFluxSupportPoints],
82  localIndex const ( &sesri )[numFluxSupportPoints],
83  localIndex const ( &sei )[numFluxSupportPoints],
84  real64 const ( &trans )[2],
85  real64 const ( &dTrans_dPres )[2],
86  ElementViewConst< arrayView1d< real64 const > > const & pres,
87  ElementViewConst< arrayView1d< real64 const > > const & gravCoef,
88  ElementViewConst< arrayView2d< real64 const, compflow::USD_PHASE > > const & phaseMob,
89  ElementViewConst< arrayView3d< real64 const, compflow::USD_PHASE_DC > > const & dPhaseMob,
90  ElementViewConst< arrayView2d< real64 const, compflow::USD_PHASE > > const & phaseVolFrac,
91  ElementViewConst< arrayView3d< real64 const, compflow::USD_PHASE_DC > > const & dPhaseVolFrac,
92  ElementViewConst< arrayView3d< real64 const, compflow::USD_COMP_DC > > const & dCompFrac_dCompDens,
93  ElementViewConst< arrayView3d< real64 const, constitutive::multifluid::USD_PHASE > > const & phaseMassDens,
94  ElementViewConst< arrayView4d< real64 const, constitutive::multifluid::USD_PHASE_DC > > const & dPhaseMassDens,
95  ElementViewConst< arrayView3d< real64 const, constitutive::cappres::USD_CAPPRES > > const & phaseCapPressure,
96  ElementViewConst< arrayView4d< real64 const, constitutive::cappres::USD_CAPPRES_DS > > const & dPhaseCapPressure_dPhaseVolFrac,
97  real64 & potGrad,
98  real64 ( &phaseFlux ),
99  real64 ( & dPhaseFlux_dP )[numFluxSupportPoints],
100  real64 ( & dPhaseFlux_dC )[numFluxSupportPoints][numComp] )
101  {
102  real64 dPotGrad_dTrans = 0.0;
103  real64 dPresGrad_dP[numFluxSupportPoints]{};
104  real64 dPresGrad_dC[numFluxSupportPoints][numComp]{};
105  real64 dGravHead_dP[numFluxSupportPoints]{};
106  real64 dGravHead_dC[numFluxSupportPoints][numComp]{};
107  PotGrad::compute< numComp, numFluxSupportPoints >( numPhase, ip, hasCapPressure, checkPhasePresenceInGravity,
108  seri, sesri, sei, trans, dTrans_dPres, pres, gravCoef,
109  phaseVolFrac, dPhaseVolFrac, dCompFrac_dCompDens, phaseMassDens, dPhaseMassDens,
110  phaseCapPressure, dPhaseCapPressure_dPhaseVolFrac,
111  potGrad, dPotGrad_dTrans, dPresGrad_dP,
112  dPresGrad_dC, dGravHead_dP, dGravHead_dC );
113 
114  // gravity head
115  real64 gravHead = 0.0;
116  for( integer i = 0; i < numFluxSupportPoints; i++ )
117  {
118  localIndex const er = seri[i];
119  localIndex const esr = sesri[i];
120  localIndex const ei = sei[i];
121 
122  real64 const gravD = trans[i] * gravCoef[er][esr][ei];
123 
124  gravHead += gravD;
125  }
126 
127  // *** upwinding ***
128 
129  // phase flux and derivatives
130 
131  // assuming TPFA in the code below
132 
133  real64 Ttrans = fabs( trans[0] );
134  potGrad = potGrad / Ttrans;
135 
136  real64 const mobility_i = phaseMob[seri[0]][sesri[0]][sei[0]][ip];
137  real64 const mobility_j = phaseMob[seri[1]][sesri[1]][sei[1]][ip];
138 
139  // compute phase flux, see Eqs. (66) and (69) from the reference above
140  real64 smoEps = epsC1PPU;
141  if( fabs( gravHead ) <= 1e-20 )
142  smoEps = 1000;
143  real64 const tmpSqrt = sqrt( potGrad * potGrad + smoEps * smoEps );
144  real64 const smoMax = 0.5 * (-potGrad + tmpSqrt);
145 
146  phaseFlux = Ttrans * ( potGrad * mobility_i - smoMax * (mobility_j - mobility_i) );
147 
148  // derivativess
149 
150  // first part, mobility derivative
151 
152  // dP
153  {
154  real64 const dMob_dP = dPhaseMob[seri[0]][sesri[0]][sei[0]][ip][Deriv::dP];
155  dPhaseFlux_dP[0] += Ttrans * potGrad * dMob_dP;
156  }
157 
158  // dC
159  {
160  arraySlice1d< real64 const, compflow::USD_PHASE_DC - 2 >
161  dPhaseMobSub = dPhaseMob[seri[0]][sesri[0]][sei[0]][ip];
162  for( integer jc = 0; jc < numComp; ++jc )
163  {
164  dPhaseFlux_dC[0][jc] += Ttrans * potGrad * dPhaseMobSub[Deriv::dC + jc];
165  }
166  }
167 
168  real64 const tmpInv = 1.0 / tmpSqrt;
169  real64 const dSmoMax_x = 0.5 * (1.0 - potGrad * tmpInv);
170 
171  // pressure gradient and mobility difference depend on all points in the stencil
172  real64 const dMobDiff_sign[numFluxSupportPoints] = {-1.0, 1.0};
173  for( integer ke = 0; ke < numFluxSupportPoints; ++ke )
174  {
175  // dP
176 
177  real64 const dPotGrad_dP = dPresGrad_dP[ke] - dGravHead_dP[ke];
178 
179  // first part
180  dPhaseFlux_dP[ke] += dPotGrad_dP * mobility_i;
181 
182  // second part
183  real64 const dSmoMax_dP = -dPotGrad_dP * dSmoMax_x;
184  dPhaseFlux_dP[ke] += -dSmoMax_dP * (mobility_j - mobility_i);
185 
186  real64 const dMob_dP = dPhaseMob[seri[ke]][sesri[ke]][sei[ke]][ip][Deriv::dP];
187  dPhaseFlux_dP[ke] += -Ttrans * smoMax * dMobDiff_sign[ke] * dMob_dP;
188 
189  // dC
190 
191  arraySlice1d< real64 const, compflow::USD_PHASE_DC - 2 >
192  dPhaseMobSub = dPhaseMob[seri[ke]][sesri[ke]][sei[ke]][ip];
193 
194  for( integer jc = 0; jc < numComp; ++jc )
195  {
196  real64 const dPotGrad_dC = dPresGrad_dC[ke][jc] - dGravHead_dC[ke][jc];
197 
198  // first part
199  dPhaseFlux_dC[ke][jc] += dPotGrad_dC * mobility_i;
200 
201  // second part
202  real64 const dSmoMax_dC = -dPotGrad_dC * dSmoMax_x;
203  dPhaseFlux_dC[ke][jc] += -dSmoMax_dC * (mobility_j - mobility_i);
204  dPhaseFlux_dC[ke][jc] += -Ttrans * smoMax * dMobDiff_sign[ke] * dPhaseMobSub[Deriv::dC + jc];
205  }
206  }
207 
208  potGrad = potGrad * Ttrans;
209  }
210 };
211 
212 } // namespace isothermalCompositionalMultiPhaseFVMKernelUtilities
213 
214 } // namespace geos
215 
216 
217 #endif // GEOS_PHYSICSSOLVERS_FLUIDFLOW_COMPOSITIONAL_C1PPUPHASEFLUX_HPP
#define GEOS_HOST_DEVICE
Marks a host-device function.
Definition: GeosxMacros.hpp:49
ArrayView< T, 1 > arrayView1d
Alias for 1D array view.
Definition: DataTypes.hpp:180
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
ArraySlice< T, 1, USD > arraySlice1d
Alias for 1D array slice.
Definition: DataTypes.hpp:184
std::int32_t integer
Signed integer type.
Definition: DataTypes.hpp:82
ArrayView< T, 4, USD > arrayView4d
Alias for 4D array view.
Definition: DataTypes.hpp:228
ArrayView< T, 2, USD > arrayView2d
Alias for 2D array view.
Definition: DataTypes.hpp:196
ArrayView< T, 3, USD > arrayView3d
Alias for 3D array view.
Definition: DataTypes.hpp:212
static GEOS_HOST_DEVICE void compute(integer const numPhase, integer const ip, integer const hasCapPressure, integer const checkPhasePresenceInGravity, localIndex const (&seri)[numFluxSupportPoints], localIndex const (&sesri)[numFluxSupportPoints], localIndex const (&sei)[numFluxSupportPoints], real64 const (&trans)[2], real64 const (&dTrans_dPres)[2], ElementViewConst< arrayView1d< real64 const > > const &pres, ElementViewConst< arrayView1d< real64 const > > const &gravCoef, ElementViewConst< arrayView2d< real64 const, compflow::USD_PHASE > > const &phaseMob, ElementViewConst< arrayView3d< real64 const, compflow::USD_PHASE_DC > > const &dPhaseMob, ElementViewConst< arrayView2d< real64 const, compflow::USD_PHASE > > const &phaseVolFrac, ElementViewConst< arrayView3d< real64 const, compflow::USD_PHASE_DC > > const &dPhaseVolFrac, ElementViewConst< arrayView3d< real64 const, compflow::USD_COMP_DC > > const &dCompFrac_dCompDens, ElementViewConst< arrayView3d< real64 const, constitutive::multifluid::USD_PHASE > > const &phaseMassDens, ElementViewConst< arrayView4d< real64 const, constitutive::multifluid::USD_PHASE_DC > > const &dPhaseMassDens, ElementViewConst< arrayView3d< real64 const, constitutive::cappres::USD_CAPPRES > > const &phaseCapPressure, ElementViewConst< arrayView4d< real64 const, constitutive::cappres::USD_CAPPRES_DS > > const &dPhaseCapPressure_dPhaseVolFrac, real64 &potGrad, real64(&phaseFlux), real64(&dPhaseFlux_dP)[numFluxSupportPoints], real64(&dPhaseFlux_dC)[numFluxSupportPoints][numComp])
Form the PhasePotentialUpwind from pressure gradient and gravitational head.