Skip to content

Commit

Permalink
Add test for conservation of kinetic energy with topography
Browse files Browse the repository at this point in the history
  • Loading branch information
dennisYatunin committed Apr 26, 2024
1 parent e0dc193 commit 139879f
Show file tree
Hide file tree
Showing 9 changed files with 79 additions and 12 deletions.
6 changes: 6 additions & 0 deletions .buildkite/pipeline.yml
Original file line number Diff line number Diff line change
Expand Up @@ -181,6 +181,12 @@ steps:
--config_file $CONFIG_PATH/plane_density_current_test.yml
artifact_paths: "plane_density_current_test/output_active/*"

- label: ":computer: Kinetic energy conservation check for Schar mountain experiment (stretched grid)"
command: >
julia --color=yes --project=examples examples/hybrid/driver.jl
--config_file $CONFIG_PATH/plane_schar_mountain_kinetic_energy_check_stretched.yml
artifact_paths: "plane_schar_mountain_kinetic_energy_check_stretched/output_active/*"


- group: "Sphere Examples (Dycore)"
steps:
Expand Down
3 changes: 3 additions & 0 deletions config/default_configs/default_config.yml
Original file line number Diff line number Diff line change
Expand Up @@ -226,6 +226,9 @@ check_conservation:
check_precipitation:
help: "Sanity checks for 1-moment precipitation [`false` (default), `true`]"
value: false
check_kinetic_energy:
help: "Whether to disable the effects of pressure, gravity, and the Coriolis force on velocity, and also check that kinetic energy is conserved at the end of the simulation"
value: false
ls_adv:
help: "Large-scale advection [`nothing` (default), `Bomex`, `LifeCycleTan2018`, `Rico`, `ARM_SGP`, `GATE_III`]"
value: ~
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
check_kinetic_energy: true
dt_save_state_to_disk: "3600secs"
initial_condition: "ScharProfile"
x_max: 120000.0
z_elem: 45
dt: "0.5secs"
t_end: "120secs"
dz_top: 1000.0
x_elem: 100
dz_bottom: 200.0
use_reference_state: false
ode_algo: "SSP33ShuOsher"
config: "plane"
hyperdiff: "false"
z_max: 25000.0
topography: "Schar"
job_id: "plane_schar_mountain_kinetic_energy_check_stretched"
toml: [toml/plane_schar_mountain_test_stretched.toml]
diagnostics:
- short_name: wa
period: 4hours
12 changes: 12 additions & 0 deletions examples/hybrid/driver.jl
Original file line number Diff line number Diff line change
Expand Up @@ -163,6 +163,18 @@ if config.parsed_args["check_conservation"]
end
end

# Kinetic energy conservation check
if config.parsed_args["check_kinetic_energy"]
ᶜρ_init = sol.u[1].c.ρ
ᶜρ_final = sol.u[end].c.ρ

ᶜK_init = CA.compute_kinetic!(similar(ᶜρ_init), sol.u[1])
ᶜK_final = CA.compute_kinetic!(similar(ᶜρ_init), sol.u[end])

FT = eltype(ᶜρ_init)
@test sum(ᶜρ_init .* ᶜK_init) sum(ᶜρ_final .* ᶜK_final) rtol = 2 * eps(FT)
end

# Precipitation characteristic checks
if config.parsed_args["check_precipitation"]
# run some simple tests based on the output
Expand Down
5 changes: 5 additions & 0 deletions src/cache/cache.jl
Original file line number Diff line number Diff line change
Expand Up @@ -159,6 +159,11 @@ function build_cache(Y, atmos, params, surface_setup, sim_info, aerosol_names)
ᶠf¹² = nothing
end

if atmos.check_kinetic_energy
@. ᶜf³ *= 0
isnothing(ᶠf¹²) || @. ᶠf¹² *= 0
end

quadrature_style =
Spaces.quadrature_style(Spaces.horizontal_space(axes(Y.c)))
do_dss = quadrature_style isa Quadratures.GLL
Expand Down
24 changes: 17 additions & 7 deletions src/prognostic_equations/advection.jl
Original file line number Diff line number Diff line change
Expand Up @@ -39,8 +39,11 @@ NVTX.@annotate function horizontal_advection_tendency!(Yₜ, Y, p, t)
@. Yₜ.c.sgs⁰.ρatke -= wdivₕ(Y.c.sgs⁰.ρatke * ᶜu⁰)
end

@. Yₜ.c.uₕ -= C12(gradₕ(ᶜp - ᶜp_ref) / Y.c.ρ + gradₕ(ᶜK + ᶜΦ))
# Without the C12(), the right-hand side would be a C1 or C2 in 2D space.
if p.atmos.check_kinetic_energy
@. Yₜ.c.uₕ -= C12(gradₕ(ᶜK))
else
@. Yₜ.c.uₕ -= C12(gradₕ(ᶜp - ᶜp_ref) / Y.c.ρ + gradₕ(ᶜK + ᶜΦ))
end # Without the C12(), the right-hand side would be a C1 or C2 in 2D space.
return nothing
end

Expand Down Expand Up @@ -230,11 +233,18 @@ function edmfx_sgs_vertical_advection_tendency!(
ᶜleft_bias(ᶠKᵥʲs.:($$j)[colidx]),
ᶜright_bias(ᶠKᵥʲs.:($$j)[colidx]),
)
# For the updraft u_3 equation, we assume the grid-mean to be hydrostatic
# and calcuate the buoyancy term relative to the grid-mean density.
@. Yₜ.f.sgsʲs.:($$j).u₃[colidx] -=
(ᶠinterp(ᶜρʲs.:($$j)[colidx] - Y.c.ρ[colidx]) * ᶠgradᵥ_ᶜΦ[colidx]) /
ᶠinterp(ᶜρʲs.:($$j)[colidx]) + ᶠgradᵥ(ᶜKᵥʲ[colidx])

if p.atmos.check_kinetic_energy
@. Yₜ.f.sgsʲs.:($$j).u₃[colidx] -= ᶠgradᵥ(ᶜKᵥʲ[colidx])
else
# Assume the grid-mean to be hydrostatic and calcuate the buoyancy
# term relative to the grid-mean density.
@. Yₜ.f.sgsʲs.:($$j).u₃[colidx] -=
(
ᶠinterp(ᶜρʲs.:($$j)[colidx] - Y.c.ρ[colidx]) *
ᶠgradᵥ_ᶜΦ[colidx]
) / ᶠinterp(ᶜρʲs.:($$j)[colidx]) + ᶠgradᵥ(ᶜKᵥʲ[colidx])
end

# buoyancy term in mse equation
@. Yₜ.c.sgsʲs.:($$j).mse[colidx] +=
Expand Down
13 changes: 8 additions & 5 deletions src/prognostic_equations/implicit/implicit_tendency.jl
Original file line number Diff line number Diff line change
Expand Up @@ -160,6 +160,7 @@ vertical_advection!(ᶜρχₜ, ᶠu³, ᶜχ, ::Val{:third_order}) =

function implicit_vertical_advection_tendency!(Yₜ, Y, p, t, colidx)
(; moisture_model, turbconv_model, rayleigh_sponge, precip_model) = p.atmos
(; check_kinetic_energy) = p.atmos
(; dt) = p
n = n_mass_flux_subdomains(turbconv_model)
ᶜJ = Fields.local_geometry_field(Y.c).J
Expand Down Expand Up @@ -213,11 +214,13 @@ function implicit_vertical_advection_tendency!(Yₜ, Y, p, t, colidx)
)
end

@. Yₜ.f.u₃[colidx] +=
-(
ᶠgradᵥ(ᶜp[colidx] - ᶜp_ref[colidx]) +
ᶠinterp(Y.c.ρ[colidx] - ᶜρ_ref[colidx]) * ᶠgradᵥ_ᶜΦ[colidx]
) / ᶠinterp(Y.c.ρ[colidx])
if !check_kinetic_energy
@. Yₜ.f.u₃[colidx] +=
-(
ᶠgradᵥ(ᶜp[colidx] - ᶜp_ref[colidx]) +
ᶠinterp(Y.c.ρ[colidx] - ᶜρ_ref[colidx]) * ᶠgradᵥ_ᶜΦ[colidx]
) / ᶠinterp(Y.c.ρ[colidx])
end

if rayleigh_sponge isa RayleighSponge
(; ᶠβ_rayleigh_w) = p.rayleigh_sponge
Expand Down
6 changes: 6 additions & 0 deletions src/solver/type_getters.jl
Original file line number Diff line number Diff line change
Expand Up @@ -96,9 +96,15 @@ function get_atmos(config::AtmosConfig, params)
surface_model = get_surface_model(parsed_args),
surface_albedo = get_surface_albedo_model(parsed_args, params, FT),
numerics = get_numerics(parsed_args),
check_kinetic_energy = parsed_args["check_kinetic_energy"],
)
@assert !@any_reltype(atmos, (UnionAll, DataType))

if atmos.check_kinetic_energy
@assert isnothing(atmos.hyperdiff)
@assert isnothing(atmos.rayleigh_sponge)
end

@info "AtmosModel: \n$(summary(atmos))"
return atmos
end
Expand Down
1 change: 1 addition & 0 deletions src/solver/types.jl
Original file line number Diff line number Diff line change
Expand Up @@ -393,6 +393,7 @@ Base.@kwdef struct AtmosModel{
surface_model::SM = nothing
surface_albedo::SA = nothing
numerics::NUM = nothing
check_kinetic_energy::Bool
end

Base.broadcastable(x::AtmosModel) = tuple(x)
Expand Down

0 comments on commit 139879f

Please sign in to comment.