ERF
Energy Research and Forecasting: An Atmospheric Modeling Code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
ERF_ComputeDiffusivityYSU.cpp File Reference
#include "ERF_SurfaceLayer.H"
#include "ERF_DirectionSelector.H"
#include "ERF_Diffusion.H"
#include "ERF_Constants.H"
#include "ERF_TurbStruct.H"
#include "ERF_PBLModels.H"
Include dependency graph for ERF_ComputeDiffusivityYSU.cpp:

Functions

void ComputeDiffusivityYSU (const MultiFab &xvel, const MultiFab &yvel, const MultiFab &cons_in, MultiFab &eddyViscosity, const Geometry &geom, const TurbChoice &turbChoice, std::unique_ptr< SurfaceLayer > &SurfLayer, bool use_terrain_fitted_coords, bool, int level, const BCRec *bc_ptr, bool, const std::unique_ptr< MultiFab > &z_phys_nd)
 

Function Documentation

◆ ComputeDiffusivityYSU()

void ComputeDiffusivityYSU ( const MultiFab &  xvel,
const MultiFab &  yvel,
const MultiFab &  cons_in,
MultiFab &  eddyViscosity,
const Geometry &  geom,
const TurbChoice turbChoice,
std::unique_ptr< SurfaceLayer > &  SurfLayer,
bool  use_terrain_fitted_coords,
bool  ,
int  level,
const BCRec *  bc_ptr,
bool  ,
const std::unique_ptr< MultiFab > &  z_phys_nd 
)
24 {
25  {
26  /*
27  YSU PBL initially introduced by S.-Y. Hong, Y. Noh, and J. Dudhia, MWR, 2006 [HND06]
28 
29  Further Modifications from S.-Y. Hong, Q. J. R. Meteorol. Soc., 2010 [H10]
30 
31  Implementation follows WRF as of early 2024 with some simplifications
32  */
33 
34 #ifdef _OPENMP
35 #pragma omp parallel if (Gpu::notInLaunchRegion())
36 #endif
37  for ( MFIter mfi(eddyViscosity,TilingIfNotGPU()); mfi.isValid(); ++mfi) {
38 
39  // Pull out the box we're working on, make sure it covers full domain in z-direction
40  const Box &bx = mfi.growntilebox(1);
41  const Box &dbx = geom.Domain();
42  Box sbx(bx.smallEnd(), bx.bigEnd());
43  sbx.grow(2,-1);
44  AMREX_ALWAYS_ASSERT(sbx.smallEnd(2) == dbx.smallEnd(2) && sbx.bigEnd(2) == dbx.bigEnd(2));
45 
46  // Get some data in arrays
47  const auto& cell_data = cons_in.const_array(mfi);
48  const auto& uvel = xvel.const_array(mfi);
49  const auto& vvel = yvel.const_array(mfi);
50 
51  const auto& z0_arr = SurfLayer->get_z0(level)->const_array();
52  const auto& ws10av_arr = SurfLayer->get_mac_avg(level,5)->const_array(mfi);
53  const auto& t10av_arr = SurfLayer->get_mac_avg(level,2)->const_array(mfi);
54  const auto& t_surf_arr = SurfLayer->get_t_surf(level)->const_array(mfi);
55  const auto& over_land_arr = (SurfLayer->get_lmask(level)) ? SurfLayer->get_lmask(level)->const_array(mfi) :
56  Array4<int> {};
57  const Array4<Real const> z_nd_arr = z_phys_nd->array(mfi);
58  const Real most_zref = SurfLayer->get_zref();
59 
60  // Require that MOST zref is 10 m so we get the wind speed at 10 m from most
61  bool invalid_zref = false;
62  if (use_terrain_fitted_coords) {
63  invalid_zref = most_zref != 10.0;
64  } else {
65  // zref gets reset to nearest cell center, so assert that zref is in the same cell as the 10m point
66  Real dz = geom.CellSize(2);
67  invalid_zref = int((most_zref - 0.5*dz)/dz) != int((10.0 - 0.5*dz)/dz);
68  }
69  if (invalid_zref) {
70  Print() << "most_zref = " << most_zref << std::endl;
71  Abort("MOST Zref must be 10m for YSU PBL scheme");
72  }
73 
74  // create flattened boxes to store PBL height
75  const GeometryData gdata = geom.data();
76  const Box xybx = PerpendicularBox<ZDir>(bx, IntVect{0,0,0});
77  FArrayBox pbl_height(xybx,1);
78  IArrayBox pbl_index(xybx,1);
79  const auto& pblh_arr = pbl_height.array();
80  const auto& pbli_arr = pbl_index.array();
81 
82  // -- Diagnose PBL height - starting out assuming non-moist --
83  // loop is only over i,j in order to find height at each x,y
84  const Real f0 = turbChoice.pbl_ysu_coriolis_freq;
85  const bool force_over_water = turbChoice.pbl_ysu_force_over_water;
86  const Real land_Ribcr = turbChoice.pbl_ysu_land_Ribcr;
87  const Real unst_Ribcr = turbChoice.pbl_ysu_unst_Ribcr;
88  ParallelFor(xybx, [=] AMREX_GPU_DEVICE (int i, int j, int) noexcept
89  {
90  // Reconstruct a surface bulk Richardson number from the surface layer model
91  // In WRF, this value is supplied to YSU by the MM5 surface layer model
92  const Real t_surf = t_surf_arr(i,j,0);
93  const Real t_layer = t10av_arr(i,j,0);
94  const Real ws_layer = ws10av_arr(i,j,0);
95  const Real Rib_layer = CONST_GRAV * most_zref / (ws_layer*ws_layer) * (t_layer - t_surf)/(t_layer);
96 
97  // For now, we only support stable boundary layers
98  if (Rib_layer < unst_Ribcr) {
99  Abort("For now, YSU PBL only supports stable conditions");
100  }
101 
102  // TODO: unstable BLs
103 
104  // PBL Height: Stable Conditions
105  Real Rib_cr;
106  bool over_land = (over_land_arr) ? over_land_arr(i,j,0) : 1;
107  if (over_land && !force_over_water) {
108  Rib_cr = land_Ribcr;
109  } else { // over water
110  // Velocity at z=10 m comes from MOST -> currently the average using whatever averaging MOST uses.
111  // TODO: Revisit this calculation with local ws10?
112  const Real z0 = z0_arr(i,j,0);
113  const Real Rossby = ws_layer/(f0*z0);
114  Rib_cr = min(0.16*std::pow(1.0e-7*Rossby,-0.18),0.3); // Note: upper bound in WRF code, but not H10 paper
115  }
116 
117  bool above_critical = false;
118  int kpbl = 0;
119  Real Rib_up = Rib_layer, Rib_dn;
120  const Real base_theta = cell_data(i,j,0,RhoTheta_comp) / cell_data(i,j,0,Rho_comp);
121  while (!above_critical and bx.contains(i,j,kpbl+1)) {
122  kpbl += 1;
123  const Real zval = use_terrain_fitted_coords ?
124  Compute_Zrel_AtCellCenter(i,j,kpbl,z_nd_arr) : gdata.ProbLo(2) + (kpbl + 0.5)*gdata.CellSize(2);
125  const Real ws2_level = 0.25*( (uvel(i,j,kpbl)+uvel(i+1,j ,kpbl))*(uvel(i,j,kpbl)+uvel(i+1,j ,kpbl))
126  + (vvel(i,j,kpbl)+vvel(i ,j+1,kpbl))*(vvel(i,j,kpbl)+vvel(i ,j+1,kpbl)) );
127  const Real theta = cell_data(i,j,kpbl,RhoTheta_comp) / cell_data(i,j,kpbl,Rho_comp);
128  Rib_dn = Rib_up;
129  Rib_up = (theta-base_theta)/base_theta * CONST_GRAV * zval / ws2_level;
130  above_critical = Rib_up >= Rib_cr;
131  }
132 
133  Real interp_fact;
134  if (Rib_dn >= Rib_cr) {
135  interp_fact = 0.0;
136  } else if (Rib_up <= Rib_cr)
137  interp_fact = 1.0;
138  else {
139  interp_fact = (Rib_cr - Rib_dn) / (Rib_up - Rib_dn);
140  }
141 
142  const Real zval_up = use_terrain_fitted_coords ?
143  Compute_Zrel_AtCellCenter(i,j,kpbl,z_nd_arr) : gdata.ProbLo(2) + (kpbl + 0.5)*gdata.CellSize(2);
144  const Real zval_dn = use_terrain_fitted_coords ?
145  Compute_Zrel_AtCellCenter(i,j,kpbl-1,z_nd_arr) : gdata.ProbLo(2) + (kpbl-1 + 0.5)*gdata.CellSize(2);
146  pblh_arr(i,j,0) = zval_dn + interp_fact*(zval_up-zval_dn);
147 
148  const Real zval_0 = use_terrain_fitted_coords ?
149  Compute_Zrel_AtCellCenter(i,j,0,z_nd_arr) : gdata.ProbLo(2) + (0.5)*gdata.CellSize(2);
150  const Real zval_1 = use_terrain_fitted_coords ?
151  Compute_Zrel_AtCellCenter(i,j,1,z_nd_arr) : gdata.ProbLo(2) + (1.5)*gdata.CellSize(2);
152  if (pblh_arr(i,j,0) < 0.5*(zval_0+zval_1) ) {
153  kpbl = 0;
154  }
155  pbli_arr(i,j,0) = kpbl;
156  });
157 
158  // -- Compute nonlocal/countergradient mixing parameters --
159  // Not included for stable so nothing to do until unstable treatment is added
160 
161  // -- Compute entrainment parameters --
162  // 0 for stable so nothing to do?
163 
164  // -- Compute diffusion coefficients --
165 
166  const auto& u_star_arr = SurfLayer->get_u_star(level)->const_array(mfi);
167  const auto& l_obuk_arr = SurfLayer->get_olen(level)->const_array(mfi);
168  const Array4<Real > &K_turb = eddyViscosity.array(mfi);
169 
170  // Dirichlet flags to switch derivative stencil
171  bool c_ext_dir_on_zlo = ( (bc_ptr[BCVars::cons_bc].lo(2) == ERFBCType::ext_dir) );
172  bool c_ext_dir_on_zhi = ( (bc_ptr[BCVars::cons_bc].lo(5) == ERFBCType::ext_dir) );
173  bool u_ext_dir_on_zlo = ( (bc_ptr[BCVars::xvel_bc].lo(2) == ERFBCType::ext_dir) );
174  bool u_ext_dir_on_zhi = ( (bc_ptr[BCVars::xvel_bc].lo(5) == ERFBCType::ext_dir) );
175  bool v_ext_dir_on_zlo = ( (bc_ptr[BCVars::yvel_bc].lo(2) == ERFBCType::ext_dir) );
176  bool v_ext_dir_on_zhi = ( (bc_ptr[BCVars::yvel_bc].lo(5) == ERFBCType::ext_dir) );
177 
178  const auto& dxInv = geom.InvCellSizeArray();
179  const Real dz_inv = geom.InvCellSize(2);
180  const int izmin = geom.Domain().smallEnd(2);
181  const int izmax = geom.Domain().bigEnd(2);
182 
183  ParallelFor(bx, [=] AMREX_GPU_DEVICE (int i, int j, int k) noexcept
184  {
185  const Real zval = use_terrain_fitted_coords ?
186  Compute_Zrel_AtCellCenter(i,j,k,z_nd_arr) : gdata.ProbLo(2) + (k + 0.5)*gdata.CellSize(2);
187  const Real rho = cell_data(i,j,k,Rho_comp);
188  const Real met_h_zeta = use_terrain_fitted_coords ? Compute_h_zeta_AtCellCenter(i,j,k,dxInv,z_nd_arr) : 1.0;
189  const Real dz_terrain = met_h_zeta/dz_inv;
190  if (k < pbli_arr(i,j,0)) {
191  // -- Compute diffusion coefficients within PBL
192  constexpr Real zfacmin = 1e-8; // value from WRF
193  constexpr Real phifac = 8.0; // value from H10 and WRF
194  constexpr Real wstar3 = 0.0; // only nonzero for unstable
195  constexpr Real pfac = 2.0; // profile exponent
196  const Real zfac = std::min(std::max(1 - zval / pblh_arr(i,j,0), zfacmin ), 1.0);
197  // Not including YSU top down PBL term (not in H10, added to WRF later)
198  const Real ust3 = u_star_arr(i,j,0) * u_star_arr(i,j,0) * u_star_arr(i,j,0);
199  Real wscalek = ust3 + phifac * KAPPA * wstar3 * (1.0 - zfac);
200  wscalek = std::pow(wscalek, 1.0/3.0);
201  // stable only
202  const Real phi_term = 1 + 5 * zval / l_obuk_arr(i,j,0); // phi_term appears in WRF but not papers
203  wscalek = std::max(u_star_arr(i,j,0) / phi_term, 0.001); // 0.001 limit appears in WRF but not papers
204  K_turb(i,j,k,EddyDiff::Mom_v) = rho * wscalek * KAPPA * zval * std::pow(zfac, pfac);
205  K_turb(i,j,k,EddyDiff::Theta_v) = K_turb(i,j,k,EddyDiff::Mom_v);
206  } else {
207  // -- Compute coefficients in free stream above PBL
208  constexpr Real lam0 = 30.0;
209  constexpr Real min_richardson = -100.0;
210  constexpr Real prandtl_max = 4.0;
211  Real dthetadz, dudz, dvdz;
212  const int RhoQv_comp = -1;
213  const int RhoQc_comp = -1;
214  const int RhoQr_comp = -1;
216  uvel, vvel, cell_data, izmin, izmax, 1.0/dz_terrain,
217  c_ext_dir_on_zlo, c_ext_dir_on_zhi,
218  u_ext_dir_on_zlo, u_ext_dir_on_zhi,
219  v_ext_dir_on_zlo, v_ext_dir_on_zhi,
220  dthetadz, dudz, dvdz, RhoQv_comp, RhoQc_comp, RhoQr_comp);
221  const Real shear_squared = dudz*dudz + dvdz*dvdz + 1.0e-9; // 1.0e-9 from WRF to avoid divide by zero
222  const Real theta = cell_data(i,j,k,RhoTheta_comp) / cell_data(i,j,k,Rho_comp);
223  Real richardson = CONST_GRAV / theta * dthetadz / shear_squared;
224  const Real lambdadz = std::min(std::max(0.1*dz_terrain , lam0), 300.0); // in WRF, H10 paper just says use lam0
225  const Real lengthscale = lambdadz * KAPPA * zval / (lambdadz + KAPPA * zval);
226  const Real turbfact = lengthscale * lengthscale * std::sqrt(shear_squared);
227 
228  if (richardson < 0) {
229  richardson = max(richardson, min_richardson);
230  Real sqrt_richardson = std::sqrt(-richardson);
231  K_turb(i,j,k,EddyDiff::Mom_v) = rho * turbfact * (1.0 - 8.0 * richardson / (1.0 + 1.746 * sqrt_richardson));
232  K_turb(i,j,k,EddyDiff::Theta_v) = rho * turbfact * (1.0 - 8.0 * richardson / (1.0 + 1.286 * sqrt_richardson));
233  } else {
234  const Real oneplus5ri = 1.0 + 5.0 * richardson;
235  K_turb(i,j,k,EddyDiff::Theta_v) = rho * turbfact / (oneplus5ri * oneplus5ri);
236  const Real prandtl = std::min(1.0+2.1*richardson, prandtl_max); // limit from WRF
237  K_turb(i,j,k,EddyDiff::Mom_v) = K_turb(i,j,k,EddyDiff::Theta_v) * prandtl;
238  }
239  }
240 
241  // limit both diffusion coefficients - from WRF, not documented in papers
242  constexpr Real ckz = 0.001;
243  constexpr Real Kmax = 1000.0;
244  const Real rhoKmin = ckz * dz_terrain * rho;
245  const Real rhoKmax = rho * Kmax;
246  K_turb(i,j,k,EddyDiff::Mom_v) = std::max(std::min(K_turb(i,j,k,EddyDiff::Mom_v) ,rhoKmax), rhoKmin);
247  K_turb(i,j,k,EddyDiff::Theta_v) = std::max(std::min(K_turb(i,j,k,EddyDiff::Theta_v) ,rhoKmax), rhoKmin);
248  K_turb(i,j,k,EddyDiff::Turb_lengthscale) = pblh_arr(i,j,0);
249  });
250 
251  // HACK set bottom ghost cell to 1st cell
252  ParallelFor(bx, [=] AMREX_GPU_DEVICE (int i, int j, int k) noexcept
253  {
254  if (k==-1) {
255  K_turb(i,j,k,EddyDiff::Mom_v) = K_turb(i,j,0,EddyDiff::Mom_v);
256  K_turb(i,j,k,EddyDiff::Theta_v) = K_turb(i,j,0,EddyDiff::Theta_v);
257  }
258  });
259  }
260  }
261 }
constexpr amrex::Real KAPPA
Definition: ERF_Constants.H:20
constexpr amrex::Real CONST_GRAV
Definition: ERF_Constants.H:21
TurbChoice turbChoice
Definition: ERF_DiffSetup.H:2
#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 int RhoQv_comp, const int RhoQc_comp, const int RhoQr_comp)
Definition: ERF_PBLModels.H:119
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:47
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:390
@ 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
@ 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
real(c_double), parameter prandtl
Definition: ERF_module_model_constants.F90:88
amrex::Real pbl_ysu_land_Ribcr
Definition: ERF_TurbStruct.H:292
amrex::Real pbl_ysu_coriolis_freq
Definition: ERF_TurbStruct.H:289
bool pbl_ysu_force_over_water
Definition: ERF_TurbStruct.H:291
amrex::Real pbl_ysu_unst_Ribcr
Definition: ERF_TurbStruct.H:293

Referenced by ComputeTurbulentViscosity().

Here is the call graph for this function:
Here is the caller graph for this function: