ERF
Energy Research and Forecasting: An Atmospheric Modeling Code
ERF_ImplicitPre.H
Go to the documentation of this file.
1  // *****************************************************************************
2  // Do semi-implicit solve for diffusion: theta, u, and v
3  // *****************************************************************************
4  {
5  const Real l_vert_implicit_fac = solverChoice.vert_implicit_fac[nrk];
6 
8 
9  const bool l_do_implicit_theta = solverChoice.implicit_thermal_diffusion;
10  const bool l_do_implicit_mom = solverChoice.implicit_momentum_diffusion;
11 
12  // If we're doing an implicit solve for momenta (u and v only),
13  // then we use the explicit solution at this point to derive
14  // the surface velocity gradient -- to be consistent with the
15  // hoextrap used with the surface layer BC. We then subtract
16  // out the contribution from the gradient of tau_corr times
17  // (1 - implicit_fac) and add in the implicit solution scaled
18  // by implicit_fac.
19  MultiFab* Tau13corr = (l_do_implicit_mom) ? Tau_corr[level][0].get() : nullptr;
20  MultiFab* Tau23corr = (l_do_implicit_mom) ? Tau_corr[level][1].get() : nullptr;
21 #ifdef ERF_IMPLICIT_W
22  MultiFab* Tau33corr = (l_do_implicit_mom) ? Tau_corr[level][2].get() : nullptr;
23 #endif
24 
25  // BCs
26  const BCRec* bc_ptr_h = domain_bcs_type.data();
27  GpuArray<Real, AMREX_SPACEDIM*2> l_bc_neumann_vals_d;
28  for (int ori = 0; ori < 2*AMREX_SPACEDIM; ori++) {
29  l_bc_neumann_vals_d[ori] = m_bc_neumann_vals[RhoTheta_comp][ori];
30  }
31  const bool l_use_SurfLayer = (m_SurfaceLayer != nullptr);
32 
33  const bool l_use_turb = solverChoice.turbChoice[level].use_kturb;
34 
35  const bool l_use_stretched_dz = (solverChoice.mesh_type == MeshType::StretchedDz);
36 
37  for ( MFIter mfi(S_old[IntVars::cons],TileNoZ()); mfi.isValid(); ++mfi)
38  {
39  Box bx = mfi.tilebox();
40  Box tbx = mfi.nodaltilebox(0);
41  Box tby = mfi.nodaltilebox(1);
42 
43  const Array4< Real>& cell_data = scratch.array(mfi);
44 
45  const Array4< Real>& rho_u = (l_do_implicit_mom) ? scratch_xmom.array(mfi) : Array4<Real>{};
46  const Array4< Real>& rho_v = (l_do_implicit_mom) ? scratch_ymom.array(mfi) : Array4<Real>{};
47  const Array4<const Real>& tau13_corr = (l_do_implicit_mom) ? Tau13corr->array(mfi) : Array4<Real>{};
48  const Array4<const Real>& tau23_corr = (l_do_implicit_mom) ? Tau23corr->array(mfi) : Array4<Real>{};
49 
50 #ifdef ERF_IMPLICIT_W
51  Box tbz = mfi.nodaltilebox(2);
52  const Array4< Real>& rho_w = (l_do_implicit_mom) ? scratch_zmom.array(mfi) : Array4<Real>{};
53  const Array4<const Real>& tau33_corr = (l_do_implicit_mom) ? Tau33corr->array(mfi) : Array4<Real>{};
54 #endif
55 
56  const Array4<const Real>& z_nd_arr = z_phys_nd[level]->const_array(mfi);
57  const Array4<const Real>& detJ_arr = detJ_cc[level]->const_array(mfi);
58 
59  const Array4<const Real>& mu_turb = l_use_turb ? eddyDiffs->const_array(mfi) : Array4<const Real>{};
60 
61  const Array4<const Real> tau13 = Tau[level][TauType::tau13]->array(mfi);
62  const Array4<const Real> tau23 = Tau[level][TauType::tau23]->array(mfi);
63  [[maybe_unused]] const Array4<const Real> tau33 = Tau[level][TauType::tau33]->array(mfi);
64  const Array4<const Real>& hfx_z = Hfx3->const_array(mfi);
65 
66  if (l_use_stretched_dz) {
67  if (l_do_implicit_theta) {
68  ImplicitDiffForStateLU_S(bx, fine_geom.Domain(), level, RhoTheta_comp,
69  slow_dt, l_bc_neumann_vals_d, cell_data,
70  stretched_dz_d[level], hfx_z,
71  mu_turb, solverChoice,
72  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
73  }
74  if (l_do_implicit_mom) {
75  ImplicitDiffForMomLU_S<0>(tbx, fine_geom.Domain(), level, slow_dt,
76  cell_data, rho_u, tau13, tau13_corr,
77  stretched_dz_d[level],
78  mu_turb, solverChoice,
79  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
80 
81  ImplicitDiffForMomLU_S<1>(tby, fine_geom.Domain(), level, slow_dt,
82  cell_data, rho_v, tau23, tau23_corr,
83  stretched_dz_d[level],
84  mu_turb, solverChoice,
85  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
86 #ifdef ERF_IMPLICIT_W
87  ImplicitDiffForMomLU_S<2>(tbz, fine_geom.Domain(), level, slow_dt,
88  cell_data, rho_w, tau33, tau33_corr,
89  stretched_dz_d[level],
90  mu_turb, solverChoice,
91  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
92 #endif
93  }
94  } else if (l_use_terrain_fitted_coords) {
95  if (l_do_implicit_theta) {
96  ImplicitDiffForStateLU_T(bx, fine_geom.Domain(), level, RhoTheta_comp,
97  slow_dt, l_bc_neumann_vals_d, cell_data,
98  z_nd_arr, detJ_arr, dxInv, hfx_z,
99  mu_turb, solverChoice,
100  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
101  }
102  if (l_do_implicit_mom) {
103  ImplicitDiffForMomLU_T<0>(tbx, fine_geom.Domain(), level, slow_dt,
104  cell_data, rho_u, tau13, tau13_corr,
105  z_nd_arr, detJ_arr, dxInv,
106  mu_turb, solverChoice,
107  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
108 
109  ImplicitDiffForMomLU_T<1>(tby, fine_geom.Domain(), level, slow_dt,
110  cell_data, rho_v, tau23, tau23_corr,
111  z_nd_arr, detJ_arr, dxInv,
112  mu_turb, solverChoice,
113  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
114 #ifdef ERF_IMPLICIT_W
115  ImplicitDiffForMomLU_T<2>(tbz, fine_geom.Domain(), level, slow_dt,
116  cell_data, rho_w, tau33, tau33_corr,
117  z_nd_arr, detJ_arr, dxInv,
118  mu_turb, solverChoice,
119  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
120 #endif
121  }
122  } else { // no terrain
123  if (l_do_implicit_theta) {
124  ImplicitDiffForStateLU_N(bx, fine_geom.Domain(), level, RhoTheta_comp,
125  slow_dt, l_bc_neumann_vals_d, cell_data,
126  dxInv, hfx_z, mu_turb, solverChoice,
127  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
128  }
129  if (l_do_implicit_mom) {
130  ImplicitDiffForMomLU_N<0>(tbx, fine_geom.Domain(), level, slow_dt,
131  cell_data, rho_u, tau13, tau13_corr,
132  dxInv,
133  mu_turb, solverChoice,
134  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
135 
136  ImplicitDiffForMomLU_N<1>(tby, fine_geom.Domain(), level, slow_dt,
137  cell_data, rho_v, tau23, tau23_corr,
138  dxInv,
139  mu_turb, solverChoice,
140  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
141 #ifdef ERF_IMPLICIT_W
142  ImplicitDiffForMomLU_N<2>(tbz, fine_geom.Domain(), level, slow_dt,
143  cell_data, rho_w, tau33, tau33_corr,
144  dxInv,
145  mu_turb, solverChoice,
146  bc_ptr_h, l_use_SurfLayer, l_vert_implicit_fac);
147 #endif
148  }
149  }
150  } // mfi
151 
152  } // if do implicit solve for diffusive contribution to (rho theta) update
153  } // wrapper
constexpr amrex::Real zero
Definition: ERF_Constants.H:6
@ tau23
Definition: ERF_DataStruct.H:32
@ tau33
Definition: ERF_DataStruct.H:32
@ tau13
Definition: ERF_DataStruct.H:32
void ImplicitDiffForStateLU_T(const amrex::Box &bx, const amrex::Box &domain, const int level, const int n, const amrex::Real dt, const amrex::GpuArray< amrex::Real, AMREX_SPACEDIM *2 > &bc_neumann_vals, const amrex::Array4< amrex::Real > &cell_data, const amrex::Array4< const amrex::Real > &z_nd, const amrex::Array4< const amrex::Real > &detJ, const amrex::GpuArray< amrex::Real, AMREX_SPACEDIM > &cellSizeInv, const amrex::Array4< const amrex::Real > &hfx_z, const amrex::Array4< const amrex::Real > &mu_turb, const SolverChoice &solverChoice, const amrex::BCRec *bc_ptr, const bool use_SurfLayer, const amrex::Real implicit_fac)
void ImplicitDiffForStateLU_N(const amrex::Box &bx, const amrex::Box &domain, const int level, const int n, const amrex::Real dt, const amrex::GpuArray< amrex::Real, AMREX_SPACEDIM *2 > &bc_neumann_vals, const amrex::Array4< amrex::Real > &cell_data, const amrex::GpuArray< amrex::Real, AMREX_SPACEDIM > &cellSizeInv, const amrex::Array4< const amrex::Real > &hfx_z, const amrex::Array4< const amrex::Real > &mu_turb, const SolverChoice &solverChoice, const amrex::BCRec *bc_ptr, const bool use_SurfLayer, const amrex::Real implicit_fac)
void ImplicitDiffForStateLU_S(const amrex::Box &bx, const amrex::Box &domain, const int level, const int n, const amrex::Real dt, const amrex::GpuArray< amrex::Real, AMREX_SPACEDIM *2 > &bc_neumann_vals, const amrex::Array4< amrex::Real > &cell_data, const amrex::Gpu::DeviceVector< amrex::Real > &stretched_dz_d, const amrex::Array4< const amrex::Real > &hfx_z, const amrex::Array4< const amrex::Real > &mu_turb, const SolverChoice &solverChoice, const amrex::BCRec *bc_ptr, const bool use_SurfLayer, const amrex::Real implicit_fac)
const Real l_vert_implicit_fac
Definition: ERF_ImplicitPost.H:5
#define RhoTheta_comp
Definition: ERF_IndexDefines.H:37
amrex::GpuArray< Real, AMREX_SPACEDIM > dxInv
Definition: ERF_InitCustomPertVels_ParticleTests.H:17
pp get("wavelength", wavelength)
amrex::Real Real
Definition: ERF_ShocInterface.H:19
AMREX_FORCE_INLINE amrex::IntVect TileNoZ()
Definition: ERF_TileNoZ.H:11
@ cons
Definition: ERF_IndexDefines.H:176