34 int klo = geom.Domain().smallEnd(2);
35 int khi = geom.Domain().bigEnd(2);
38 #pragma omp parallel if (Gpu::notInLaunchRegion())
40 for (MFIter mfi(eddyViscosity,
TileNoZ()); mfi.isValid(); ++mfi) {
43 const Box& gbx = mfi.growntilebox(IntVect(1,1,0));
44 AMREX_ALWAYS_ASSERT( gbx.smallEnd(2) == klo &&
45 gbx.bigEnd(2) == khi );
61 const GeometryData gdata = geom.data();
62 const Box xybx = PerpendicularBox<ZDir>(gbx, IntVect{0, 0, 0});
63 FArrayBox pbl_height_predictor(xybx, 1, The_Async_Arena());
64 FArrayBox pbl_height_corrector(xybx, 1, The_Async_Arena());
65 IArrayBox pbl_index(xybx, 1, The_Async_Arena());
66 const auto& pblh_arr = pbl_height_predictor.array();
67 const auto& pblh_corr_arr = pbl_height_corrector.array();
68 const auto& pbli_arr = pbl_index.array();
71 const auto& cell_data = cons_in.const_array(mfi);
72 const auto& uvel =
xvel.const_array(mfi);
73 const auto& vvel =
yvel.const_array(mfi);
77 const auto& u_star_arr = SurfLayer->get_u_star(level)->const_array(mfi);
78 const auto& t_star_arr = SurfLayer->get_t_star(level)->const_array(mfi);
79 const auto& l_obuk_arr = SurfLayer->get_olen(level)->const_array(mfi);
80 const auto& t10av_arr = SurfLayer->get_mac_avg(level, 2)->const_array(mfi);
82 const Array4<Real const> z_nd_arr = z_phys_nd->array(mfi);
84 ParallelFor(xybx, [=] AMREX_GPU_DEVICE(
int i,
int j,
int) noexcept
88 const Real t_layer = t10av_arr(i, j, 0);
92 bool above_critical =
false;
93 while (!above_critical && ((kpbl + 1) <= khi)) {
97 zval = (use_terrain_fitted_coords)
99 : (kpbl + 0.5) * gdata.CellSize(2);
103 const Real ws2 = 0.25 * ( (uvel(i, j, kpbl) + uvel(i + 1, j, kpbl)) *
104 (uvel(i, j, kpbl) + uvel(i + 1, j, kpbl)) +
105 (vvel(i, j, kpbl) + vvel(i, j + 1, kpbl)) *
106 (vvel(i, j, kpbl) + vvel(i, j + 1, kpbl)) );
108 above_critical = (Rib >= Ribcr);
118 const Real pblh_emp = (use_terrain_fitted_coords)
120 : gdata.ProbLo(2) + 0.5 * gdata.CellSize(2);
124 pblh_arr(i, j, 0) = (above_critical) ? zval : pblh_emp;
125 pbli_arr(i, j, 0) = (above_critical) ? kpbl : klo+1;
136 ParallelFor(xybx, [=] AMREX_GPU_DEVICE(
int i,
int j,
int) noexcept
138 const Real t_layer = t10av_arr(i, j, 0);
139 const Real phiM = (l_obuk_arr(i, j, 0) > 0)
140 ? (1 + 5 * sf * pblh_arr(i, j, 0) / l_obuk_arr(i, j, 0))
142 (1 - 8 * sf * pblh_arr(i, j, 0) / l_obuk_arr(i, j, 0)),
144 const Real wstar = u_star_arr(i, j, 0) / phiM;
145 const Real t_excess = -const_b * u_star_arr(i, j, 0) * t_star_arr(i, j, 0) / wstar;
146 const Real t_surf = t_layer + std::max(std::min(t_excess, 3.0), 0.0);
149 Real zval0, zval, Rib0, Rib;
151 zval = (use_terrain_fitted_coords)
153 : gdata.ProbLo(2) + (kpbl + 0.5) * gdata.CellSize(2);
156 const Real ws2 = 0.25 * ( (uvel(i, j, kpbl) + uvel(i + 1, j, kpbl)) *
157 (uvel(i, j, kpbl) + uvel(i + 1, j, kpbl)) +
158 (vvel(i, j, kpbl) + vvel(i, j + 1, kpbl)) *
159 (vvel(i, j, kpbl) + vvel(i, j + 1, kpbl)) );
163 bool above_critical =
false;
164 while (!above_critical && ((kpbl + 1) <= khi)) {
169 zval = (use_terrain_fitted_coords)
171 : gdata.ProbLo(2) + (kpbl + 0.5) * gdata.CellSize(2);
174 const Real ws2 = 0.25 * ( (uvel(i, j, kpbl) + uvel(i + 1, j, kpbl)) *
175 (uvel(i, j, kpbl) + uvel(i + 1, j, kpbl)) +
176 (vvel(i, j, kpbl) + vvel(i, j + 1, kpbl)) *
177 (vvel(i, j, kpbl) + vvel(i, j + 1, kpbl)) );
179 above_critical = (Rib >= Ribcr);
182 if (above_critical) {
184 Real pblh_interp = zval0 + (zval - zval0) / (Rib - Rib0) * (Ribcr - Rib0);
185 pblh_corr_arr(i, j, 0) = pblh_interp;
186 pbli_arr(i, j, 0) = kpbl;
195 pblh_corr_arr(i, j, 0) = (use_terrain_fitted_coords)
197 : gdata.ProbLo(2) + 0.5 * gdata.CellSize(2);
198 pbli_arr(i, j, 0) = klo + 1;
212 const Array4<Real>& K_turb = eddyViscosity.array(mfi);
222 const auto& dxInv = geom.InvCellSizeArray();
223 const Real dz_inv = geom.InvCellSize(2);
224 const int izmin = geom.Domain().smallEnd(2);
225 const int izmax = geom.Domain().bigEnd(2);
227 ParallelFor(gbx, [=] AMREX_GPU_DEVICE(
int i,
int j,
int k) noexcept
229 const Real zval = (use_terrain_fitted_coords)
231 : gdata.ProbLo(2) + (k + 0.5) * gdata.CellSize(2);
233 const Real met_h_zeta = (use_terrain_fitted_coords)
235 const Real dz_terrain = met_h_zeta / dz_inv;
236 if (k < pbli_arr(i, j, 0)) {
237 const Real phiM = (l_obuk_arr(i, j, 0) > 0)
238 ? (1 + 5 * sf * pblh_arr(i, j, 0) / l_obuk_arr(i, j, 0))
240 (1 - 8 * sf * pblh_arr(i, j, 0) / l_obuk_arr(i, j, 0)),
242 const Real phit = (l_obuk_arr(i, j, 0) > 0)
243 ? (1 + 5 * sf * pblh_arr(i, j, 0) / l_obuk_arr(i, j, 0))
245 (1 - 16 * sf * pblh_arr(i, j, 0) / l_obuk_arr(i, j, 0)),
247 const Real Prt = phit / phiM + const_b *
KAPPA * sf;
248 const Real wstar = u_star_arr(i, j, 0) / phiM;
250 * (1 - zval / pblh_corr_arr(i, j, 0))
251 * (1 - zval / pblh_corr_arr(i, j, 0));
254 const Real lambda = 150.0;
255 const Real lscale = (
KAPPA * zval * lambda) / (
KAPPA * zval + lambda);
256 Real dthetadz, dudz, dvdz;
258 c_ext_dir_on_zlo, c_ext_dir_on_zhi, u_ext_dir_on_zlo,
259 u_ext_dir_on_zhi, v_ext_dir_on_zlo, v_ext_dir_on_zhi, dthetadz,
260 dudz, dvdz, moisture_indices);
261 const Real wind_shear = dudz * dudz + dvdz * dvdz + 1.0e-9;
264 grad_Ri = std::max(grad_Ri, -100.0);
277 Real Pr = 1.0 + 2.1 * grad_Ri;
278 const Real fm = (grad_Ri > 0)
279 ? 1.0 / ((1.0 + 5.0 * grad_Ri) * (1.0 + 5.0 * grad_Ri))
280 : 1 - 8 * grad_Ri / (1 + 1.746 * std::sqrt(-grad_Ri));
281 const Real ft = (grad_Ri > 0)
282 ? 1.0 / ((1.0 + 5.0 * grad_Ri) * (1.0 + 5.0 * grad_Ri))
283 : 1 - 8 * grad_Ri / (1 + 1.286 * std::sqrt(-grad_Ri));
284 const Real rl2wsp =
rho * lscale * lscale * std::sqrt(wind_shear);
286 Pr = std::max(0.25, std::min(Pr, 4.0));
295 constexpr
Real ckz = 0.001;
296 constexpr
Real Kmax = 1000.0;
297 const Real rhoKmin = ckz * dz_terrain *
rho;
298 const Real rhoKmax =
rho * Kmax;
301 constexpr
Real Kmin = 0.1;
302 constexpr
Real Kmax = 300.0;
303 const Real rhoKmin =
rho * Kmin;
304 const Real rhoKmax =
rho * Kmax;
315 ParallelFor(xybx, [=] AMREX_GPU_DEVICE(
int i,
int j,
int ) noexcept
constexpr amrex::Real KAPPA
Definition: ERF_Constants.H:20
constexpr amrex::Real CONST_GRAV
Definition: ERF_Constants.H:21
#define Rho_comp
Definition: ERF_IndexDefines.H:36
#define RhoTheta_comp
Definition: ERF_IndexDefines.H:37
AMREX_GPU_DEVICE AMREX_FORCE_INLINE void ComputeVerticalDerivativesPBL(int i, int j, int k, const amrex::Array4< const amrex::Real > &uvel, const amrex::Array4< const amrex::Real > &vvel, const amrex::Array4< const amrex::Real > &cell_data, const int izmin, const int izmax, const amrex::Real &dz_inv, const bool c_ext_dir_on_zlo, const bool c_ext_dir_on_zhi, const bool u_ext_dir_on_zlo, const bool u_ext_dir_on_zhi, const bool v_ext_dir_on_zlo, const bool v_ext_dir_on_zhi, amrex::Real &dthetadz, amrex::Real &dudz, amrex::Real &dvdz, const MoistureComponentIndices &moisture_indices)
Definition: ERF_PBLModels.H:181
TurbChoice turbChoice
Definition: ERF_SetupVertDiff.H:5
amrex::Real Real
Definition: ERF_ShocInterface.H:19
AMREX_FORCE_INLINE AMREX_GPU_DEVICE amrex::Real Compute_h_zeta_AtCellCenter(const int &i, const int &j, const int &k, const amrex::GpuArray< amrex::Real, AMREX_SPACEDIM > &cellSizeInv, const amrex::Array4< const amrex::Real > &z_nd)
Definition: ERF_TerrainMetrics.H:53
AMREX_GPU_DEVICE AMREX_FORCE_INLINE amrex::Real Compute_Zrel_AtCellCenter(const int &i, const int &j, const int &k, const amrex::Array4< const amrex::Real > &z_nd)
Definition: ERF_TerrainMetrics.H:387
AMREX_FORCE_INLINE amrex::IntVect TileNoZ()
Definition: ERF_TileNoZ.H:11
@ yvel_bc
Definition: ERF_IndexDefines.H:88
@ cons_bc
Definition: ERF_IndexDefines.H:76
@ xvel_bc
Definition: ERF_IndexDefines.H:87
@ ext_dir
Definition: ERF_IndexDefines.H:209
@ Theta_v
Definition: ERF_IndexDefines.H:176
@ Turb_lengthscale
Definition: ERF_IndexDefines.H:180
@ Q_v
Definition: ERF_IndexDefines.H:179
@ Mom_v
Definition: ERF_IndexDefines.H:175
@ theta
Definition: ERF_MM5.H:20
@ rho
Definition: ERF_Kessler.H:22
@ xvel
Definition: ERF_IndexDefines.H:141
@ yvel
Definition: ERF_IndexDefines.H:142
amrex::Real pbl_mrf_const_b
Definition: ERF_TurbStruct.H:428
amrex::Real pbl_mrf_Ribcr
Definition: ERF_TurbStruct.H:426
amrex::Real pbl_mrf_sf
Definition: ERF_TurbStruct.H:430