-
Notifications
You must be signed in to change notification settings - Fork 19
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
b26b78b
commit 8d894d1
Showing
17 changed files
with
323 additions
and
175 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,225 @@ | ||
import StaticArrays as SA | ||
import ClimaCore.RecursiveApply: rzero, ⊞, ⊠ | ||
|
||
# TODO: write a test with scalars that are linear with z | ||
""" | ||
Diagnose horizontal covariances based on vertical gradients | ||
(i.e. taking turbulence production as the only term) | ||
""" | ||
function covariance_from_grad(coeff, mixing_length, ∇Φ, ∇Ψ) | ||
return 2 * coeff * mixing_length^2 * dot(∇Φ, ∇Ψ) | ||
end | ||
|
||
""" | ||
Compute the Smagorinsky length scale from | ||
- c_smag coefficient | ||
- N_eff - buoyancy frequency = sqrt(max(ᶜlinear_buoygrad, 0)) | ||
- dz - vertical grid scale | ||
- Pr - Prandtl number | ||
- ϵ_st - strain rate norm | ||
""" | ||
function compute_smagorinsky_length_scale(c_smag, N_eff, dz, Pr, ϵ_st) | ||
FT = eltype(c_smag) | ||
#return N_eff > FT(0) && N_eff < sqrt(2 * Pr * ϵ_st) ? | ||
return N_eff > FT(0) ? | ||
c_smag * dz * max(0, (1 - N_eff^2 / Pr / 2 / ϵ_st))^(1 / 4) : | ||
c_smag * dz | ||
end | ||
|
||
""" | ||
Compute the grid scale cloud fraction based on sub-grid scale properties | ||
""" | ||
function set_cloud_fraction!(Y, p, ::DryModel) | ||
@. p.precomputed.ᶜcloud_fraction = 0 | ||
end | ||
function set_cloud_fraction!(Y, p, ::Union{EquilMoistModel, NonEquilMoistModel}) | ||
(; SG_quad, params) = p | ||
|
||
FT = eltype(params) | ||
thermo_params = CAP.thermodynamics_params(params) | ||
ᶜdz = Fields.Δz_field(axes(Y.c)) | ||
ᶜlg = Fields.local_geometry_field(Y.c) | ||
(; ᶜts, ᶜp, ᶠu³, ᶜcloud_fraction) = p.precomputed | ||
(; obukhov_length) = p.precomputed.sfc_conditions | ||
|
||
ᶜlinear_buoygrad = p.scratch.ᶜtemp_scalar | ||
@. ᶜlinear_buoygrad = buoyancy_gradients( | ||
params, | ||
p.atmos.moisture_model, | ||
EnvBuoyGrad( | ||
BuoyGradMean(), | ||
TD.air_temperature(thermo_params, ᶜts), # t_sat | ||
TD.vapor_specific_humidity(thermo_params, ᶜts), # qv_sat | ||
TD.total_specific_humidity(thermo_params, ᶜts), # qt_sat | ||
TD.liquid_specific_humidity(thermo_params, ᶜts), # q_liq | ||
TD.ice_specific_humidity(thermo_params, ᶜts), # q_ice | ||
TD.dry_pottemp(thermo_params, ᶜts), # θ_sat | ||
TD.liquid_ice_pottemp(thermo_params, ᶜts), # θ_liq_ice_sat | ||
projected_vector_data( | ||
C3, | ||
ᶜgradᵥ(ᶠinterp(TD.virtual_pottemp(thermo_params, ᶜts))), | ||
ᶜlg, | ||
), # ∂θv∂z_unsat | ||
projected_vector_data( | ||
C3, | ||
ᶜgradᵥ(ᶠinterp(TD.total_specific_humidity(thermo_params, ᶜts))), | ||
ᶜlg, | ||
), # ∂qt∂z_sat | ||
projected_vector_data( | ||
C3, | ||
ᶜgradᵥ(ᶠinterp(TD.liquid_ice_pottemp(thermo_params, ᶜts))), | ||
ᶜlg, | ||
), # ∂θl∂z_sat | ||
ᶜp, # p | ||
ifelse(TD.has_condensate(thermo_params, ᶜts), 1, 0),# en_cld_frac | ||
Y.c.ρ, # ρ | ||
), | ||
) | ||
|
||
ᶠu = p.scratch.ᶠtemp_C123 | ||
@. ᶠu = C123(ᶠinterp(Y.c.uₕ)) + C123(ᶠu³) | ||
|
||
ᶜstrain_rate = p.scratch.ᶜtemp_UVWxUVW | ||
compute_strain_rate_center!(ᶜstrain_rate, ᶠu) | ||
|
||
ᶜprandtl_nvec = p.scratch.ᶜtemp_scalar | ||
@. ᶜprandtl_nvec = turbulent_prandtl_number( | ||
params, | ||
obukhov_length, | ||
ᶜlinear_buoygrad, | ||
norm_sqr(ᶜstrain_rate), | ||
) | ||
|
||
ᶜl_smag = p.scratch.ᶜtemp_scalar_2 | ||
@. ᶜl_smag = compute_smagorinsky_length_scale( | ||
CAP.c_smag(params), | ||
sqrt(max(ᶜlinear_buoygrad, 0)), #N_eff | ||
ᶜdz, | ||
ᶜprandtl_nvec, | ||
norm_sqr(ᶜstrain_rate), | ||
) | ||
|
||
coeff = FT(2.1) # TODO - move to parameters | ||
@. ᶜcloud_fraction = quad_loop( | ||
SG_quad, | ||
ᶜp, | ||
TD.total_specific_humidity(thermo_params, ᶜts), | ||
TD.liquid_ice_pottemp(thermo_params, ᶜts), | ||
Geometry.WVector( | ||
ᶜgradᵥ(ᶠinterp(TD.total_specific_humidity(thermo_params, ᶜts))), | ||
), | ||
Geometry.WVector( | ||
ᶜgradᵥ(ᶠinterp(TD.liquid_ice_pottemp(thermo_params, ᶜts))), | ||
), | ||
coeff, | ||
ᶜl_smag, # replace with mixing_length when using EDMF SGS | ||
thermo_params, | ||
) | ||
end | ||
|
||
""" | ||
function quad_loop(SG_quad, p_c, q_mean, θ_mean, ᶜ∇q, ᶜ∇θ, | ||
coeff, ᶜlength_scale, thermo_params) | ||
where: | ||
- SG_quad is a struct containing information about quadrature type and order | ||
- p_c is the atmospheric pressure | ||
- q_mean, θ_mean is the grid mean q_tot and liquid ice potential temperature | ||
- ᶜ∇q, ᶜ∇θ are the gradients of q_tot and liquid ice potential temperature | ||
- coeff - a free parameter (to be moved into params) | ||
- ᶜlength_scale - mixing length for simulations with EDMF and Smagorinsky | ||
length scale for simulations without EDMF | ||
- thermo params - thermodynamics parameters | ||
The function imposes additional limits on the quadrature points and | ||
returns cloud fraction computed as a sum over quadrature points. | ||
""" | ||
function quad_loop( | ||
SG_quad::SGSQuadrature, | ||
p_c, | ||
q_mean, | ||
θ_mean, | ||
ᶜ∇q, | ||
ᶜ∇θ, | ||
coeff, | ||
ᶜlength_scale, | ||
thermo_params, | ||
) | ||
# Returns the physical values based on quadrature sampling points | ||
# and limited covarainces | ||
function get_x_hat(χ1, χ2) | ||
|
||
@assert SG_quad.quadrature_type isa GaussianQuad | ||
FT = eltype(χ1) | ||
|
||
q′q′ = covariance_from_grad(coeff, ᶜlength_scale, ᶜ∇q, ᶜ∇q) | ||
θ′θ′ = covariance_from_grad(coeff, ᶜlength_scale, ᶜ∇θ, ᶜ∇θ) | ||
θ′q′ = covariance_from_grad(coeff, ᶜlength_scale, ᶜ∇θ, ᶜ∇q) | ||
|
||
# Epsilon defined per typical variable fluctuation | ||
eps_q = eps(FT) * max(eps(FT), q_mean) | ||
eps_θ = eps(FT) | ||
|
||
# limit σ_q to prevent negative q_tot_hat | ||
σ_q_lim = -q_mean / (sqrt(FT(2)) * SG_quad.a[1]) | ||
σ_q = min(sqrt(q′q′), σ_q_lim) | ||
# Do we also have to try to limit θ in the same way as q?? | ||
σ_θ = sqrt(θ′θ′) | ||
|
||
# Enforce Cauchy-Schwarz inequality, numerically stable compute | ||
_corr = (θ′q′ / max(σ_q, eps_q)) | ||
corr = max(min(_corr / max(σ_θ, eps_θ), FT(1)), FT(-1)) | ||
|
||
# Conditionals | ||
σ_c = sqrt(max(1 - corr * corr, 0)) * σ_θ | ||
|
||
μ_c = θ_mean + sqrt(FT(2)) * corr * σ_θ * χ1 | ||
θ_hat = μ_c + sqrt(FT(2)) * σ_c * χ2 | ||
q_hat = q_mean + sqrt(FT(2)) * σ_q * χ1 | ||
# The σ_q_lim limits q_tot_hat to be close to zero | ||
# for the negative sampling points. However due to numerical erros | ||
# we sometimes still get small negative numers here | ||
return (θ_hat, max(FT(0), q_hat)) | ||
end | ||
|
||
function f(x1_hat, x2_hat) | ||
FT = eltype(x1_hat) | ||
@assert(x1_hat >= FT(0)) | ||
@assert(x2_hat >= FT(0)) | ||
ts = thermo_state(thermo_params; p = p_c, θ = x1_hat, q_tot = x2_hat) | ||
hc = TD.has_condensate(thermo_params, ts) | ||
return (; | ||
cf = hc ? FT(1) : FT(0), # cloud fraction | ||
q_tot_sat = hc ? x2_hat : FT(0), # cloudy/dry for buoyancy in TKE | ||
) | ||
end | ||
|
||
return quad(f, get_x_hat, SG_quad).cf | ||
end | ||
|
||
""" | ||
Compute f(θ, q) as a sum over inner and outer quadrature points | ||
that approximate the sub-grid scale variability of θ and q. | ||
θ - liquid ice potential temperature | ||
q - total water specific humidity | ||
""" | ||
function quad(f::F, get_x_hat::F1, quad) where {F <: Function, F1 <: Function} | ||
χ = quad.a | ||
weights = quad.w | ||
quad_order = quadrature_order(quad) | ||
FT = eltype(χ) | ||
# zero outer quadrature points | ||
T = typeof(f(get_x_hat(χ[1], χ[1])...)) | ||
outer_env = rzero(T) | ||
@inbounds for m_q in 1:quad_order | ||
# zero inner quadrature points | ||
inner_env = rzero(T) | ||
for m_h in 1:quad_order | ||
x_hat = get_x_hat(χ[m_q], χ[m_h]) | ||
inner_env = inner_env ⊞ f(x_hat...) ⊠ weights[m_h] ⊠ FT(1 / sqrt(π)) | ||
end | ||
outer_env = outer_env ⊞ inner_env ⊠ weights[m_q] ⊠ FT(1 / sqrt(π)) | ||
end | ||
return outer_env | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.