Skip to content

Commit

Permalink
Merge pull request #498 from SpeedyWeather/mk/convection
Browse files Browse the repository at this point in the history
Convection with work arrays for thread safety
  • Loading branch information
milankl authored Mar 20, 2024
2 parents 3b6adfe + 5b83db7 commit 4baf70f
Show file tree
Hide file tree
Showing 6 changed files with 74 additions and 46 deletions.
10 changes: 5 additions & 5 deletions src/dynamics/diagnostic_variables.jl
Original file line number Diff line number Diff line change
Expand Up @@ -57,7 +57,7 @@ Base.@kwdef struct DynamicsVariables{NF<:AbstractFloat, Grid<:AbstractGrid{NF}}
trunc::Int

# MULTI-PURPOSE VECTOR (a, b), work array to be reused in various places, examples:
# uω_coslat⁻¹, vω_coslat⁻¹ = a, b (all models)
# uω_coslat⁻¹, vω_coslat⁻¹ = a, b (all models)
# uω_coslat⁻¹_grid, vω_coslat⁻¹_grid = a_grid, b_grid (all models)
# uh_coslat⁻¹, vh_coslat⁻¹ = a, b (ShallowWaterModel)
# uh_coslat⁻¹_grid, vh_coslat⁻¹_grid = a_grid, b_grid (ShallowWaterModel)
Expand Down Expand Up @@ -138,8 +138,8 @@ Base.@kwdef struct SurfaceVariables{NF<:AbstractFloat, Grid<:AbstractGrid{NF}}
div_mean::LTM{Complex{NF}} = zeros(LTM{Complex{NF}}, trunc+2, trunc+1) # divergence (in spectral though)

precip_large_scale::Grid = zeros(Grid, nlat_half) # large scale precipitation (for output)
precip_convection::Grid = zeros(Grid, nlat_half) # convective precipitation (for output)
cloud_top::Grid = zeros(Grid, nlat_half) # cloud top [hPa]
precip_convection::Grid = zeros(Grid, nlat_half) # convective precipitation (for output)
cloud_top::Grid = zeros(Grid, nlat_half) # cloud top [hPa]
soil_moisture_availability::Grid = zeros(Grid, nlat_half)
end

Expand Down Expand Up @@ -217,10 +217,10 @@ function Base.zeros(
nthreads = Threads.nthreads()
columns = [ColumnVariables{NF}(; nlev) for _ in 1:nthreads]

# particle work arrays
# particle work arrays
particles = zeros(ParticleVariables, SG)

scale = Ref(convert(SG.NF, SG.radius))
scale = Ref(one(NF))

return DiagnosticVariables{NF, Grid{NF}, Model}(
layers, surface, columns, particles,
Expand Down
4 changes: 4 additions & 0 deletions src/dynamics/scaling.jl
Original file line number Diff line number Diff line change
Expand Up @@ -41,11 +41,15 @@ $(TYPEDSIGNATURES)
Scales the prognostic variables vorticity and divergence with
the Earth's radius which is used in the dynamical core."""
function scale!(progn::PrognosticVariables,
diagn::DiagnosticVariables,
scale::Real)
new_scale = scale/progn.scale[] # undo previous scale and new scale in one go
scale!(progn, :vor, new_scale)
scale!(progn, :div, new_scale)
progn.scale[] = scale # store scaling information
diagn.scale[] = scale # store scaling information
# no need to actually scale the diagnostic variables as they will be
# overwritten by the transform of the prognostic variables anyway
end

"""
Expand Down
4 changes: 2 additions & 2 deletions src/dynamics/time_integration.jl
Original file line number Diff line number Diff line change
Expand Up @@ -383,7 +383,7 @@ function time_stepping!(
(; time_stepping) = model

# SCALING: we use vorticity*radius, divergence*radius in the dynamical core
scale!(progn, model.spectral_grid.radius)
scale!(progn, diagn, model.spectral_grid.radius)

# OUTPUT INITIALISATION AND STORING INITIAL CONDITIONS + FEEDBACK
# propagate spectral state to grid variables for initial condition output
Expand All @@ -410,7 +410,7 @@ function time_stepping!(
callback!(model.callbacks, progn, diagn, model)
end

# UNSCALE, CLOSE, FINISH
# UNSCALE, CLOSE, FINISH
finish!(feedback) # finish the progress meter, do first for benchmark accuracy
unscale!(progn) # undo radius-scaling for vor, div from the dynamical core
unscale!(diagn) # undo radius-scaling for vor, div from the dynamical core
Expand Down
65 changes: 31 additions & 34 deletions src/physics/convection.jl
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ abstract type AbstractConvection <: AbstractParameterization end

export NoConvection
struct NoConvection <: AbstractConvection end
NoConvection(::SpectralGrid) = NoConvection()
initialize!(::NoConvection, ::PrimitiveEquation) = nothing
convection!(::ColumnVariables, ::NoConvection, ::PrimitiveEquation) = nothing

Expand All @@ -21,16 +22,10 @@ Base.@kwdef struct SimplifiedBettsMiller{NF} <: AbstractConvection

"[OPTION] Relative humidity for reference profile"
relative_humidity::NF = 0.7

"temperature [K] reference profile to adjust to"
temp_ref_profile::Vector{NF} = zeros(NF, nlev)

"specific humidity [kg/kg] profile to adjust to"
humid_ref_profile::Vector{NF} = zeros(NF, nlev)
end

SimplifiedBettsMiller(SG::SpectralGrid; kwargs...) = SimplifiedBettsMiller{SG.NF}(nlev=SG.nlev; kwargs...)
initialize!(::SimplifiedBettsMiller, ::PrimitiveWet) = nothing
initialize!(::SimplifiedBettsMiller, ::PrimitiveEquation) = nothing

# function barrier for all AbstractConvection
function convection!(
Expand All @@ -44,7 +39,7 @@ end
function convection!(
column::ColumnVariables,
scheme::SimplifiedBettsMiller,
model::PrimitiveEquation,
model::PrimitiveWet,
)
convection!(column, scheme, model.clausius_clapeyron,
model.geometry, model.planet, model.atmosphere, model.time_stepping)
Expand All @@ -70,11 +65,14 @@ function convection!(
σ = geometry.σ_levels_full
σ_half = geometry.σ_levels_half
Δσ = geometry.σ_levels_thick
(; temp_ref_profile, humid_ref_profile) = SBM
(; geopot, nlev, temp, temp_virt, humid, temp_tend, humid_tend) = column
pₛ = column.pres[end]
(; Lᵥ, cₚ) = clausius_clapeyron


# use work arrays for temp_ref_profile, humid_ref_profile
temp_ref_profile = column.a # temperature [K] reference profile to adjust to
humid_ref_profile = column.b # specific humidity [kg/kg] profile to adjust to

# CONVECTIVE CRITERIA AND FIRST GUESS RELAXATION
temp_parcel = temp[nlev]
humid_parcel = humid[nlev]
Expand All @@ -88,7 +86,7 @@ function convection!(
humid_ref_profile[k] = qsat*SBM.relative_humidity
end

local Pq::NF = 0 # precipitation due to drying
local Pq::NF = 0 # precipitation due to drying
local PT::NF = 0 # precipitation due to cooling
local ΔT::NF = 0 # vertically uniform temperature profile adjustment
local Qref::NF = 0 # = ∫_pₛ^p_LZB -humid_ref_profile dp
Expand All @@ -111,7 +109,7 @@ function convection!(
deep_convection = Pq > 0 && PT > 0
shallow_convection = Pq <= 0 && PT > 0

# escape immediately for no convection
# escape immediately for no convection
no_convection = !(deep_convection || shallow_convection)
no_convection && return nothing

Expand All @@ -123,24 +121,24 @@ function convection!(
ΔT = (PT - Pq*Lᵥ/cₚ)/Δσ_lzb # eq (5) but reusing PT, Pq, and /cₚ already included

for k in level_zero_buoyancy:nlev
temp_ref_profile[k] -= ΔT # equation (6)
temp_ref_profile[k] -= ΔT # equation (6)
end

elseif shallow_convection

# FRIERSON'S QREF SCHEME
# "changing the reference profiles for both temperature and humidity so the
# precipitation is zero.
# precipitation is zero.

for k in level_zero_buoyancy:nlev
Qref -= humid_ref_profile[k]*Δσ[k] # eq (11) but in σ coordinates
Qref -= humid_ref_profile[k]*Δσ[k] # eq (11) but in σ coordinates
end
fq = 1 - Pq/Qref # = 1 - Δq/Qref in eq (12) but we reuse Pq
fq = 1 - Pq/Qref # = 1 - Δq/Qref in eq (12) but we reuse Pq

ΔT = PT/Δσ_lzb # equation (14), reuse PT and in σ coordinates
for k in level_zero_buoyancy:nlev
humid_ref_profile[k] *= fq # update humidity profile, eq (13)
temp_ref_profile[k] -= ΔT # update temperature profile, eq (15)
humid_ref_profile[k] *= fq # update humidity profile, eq (13)
temp_ref_profile[k] -= ΔT # update temperature profile, eq (15)
end
end

Expand All @@ -157,9 +155,9 @@ function convection!(
(; gravity) = planet
(; water_density) = atmosphere
(; Δt_sec) = time_stepping
pₛΔt_gρ = (pₛ * Δt_sec / gravity / water_density) * deep_convection # enfore no precip for shallow conv
pₛΔt_gρ = (pₛ * Δt_sec / gravity / water_density) * deep_convection # enfore no precip for shallow conv
column.precip_convection *= pₛΔt_gρ # convert to [m] of rain during Δt
column.cloud_top = min(column.cloud_top, level_zero_buoyancy) # clouds reach to top of convection
column.cloud_top = min(column.cloud_top, level_zero_buoyancy) # clouds reach to top of convection
end

"""
Expand Down Expand Up @@ -196,7 +194,7 @@ function pseudo_adiabat!(
local k::Int = nlev # layer index top to surface
local temp_virt_parcel::NF = temp_parcel * (1 + μ*humid_parcel)

while buoyant && k > 1 # calculate moist adiabat while buoyant till top
while buoyant && k > 1 # calculate moist adiabat while buoyant till top
k -= 1 # one level up

if !saturated # if not saturated yet follow dry adiabat
Expand All @@ -205,7 +203,7 @@ function pseudo_adiabat!(
sat_humid = saturation_humidity(temp_parcel_dry, σ[k]*pres, clausius_clapeyron)

# set to saturated when the dry adiabatic ascent would reach saturation
# then follow moist adiabat instead (see below)
# then follow moist adiabat instead (see below)
saturated = humid_parcel >= sat_humid
end

Expand All @@ -216,20 +214,20 @@ function pseudo_adiabat!(
B = q*Lᵥ^2 / ((1-q)^2 * cₚ * R_vapour)
Γ = (1 + A/Tᵥ) / (1 + B/T^2)

ΔΦ = geopot[k] - geopot[k+1] # vertical gradient in geopotential
ΔΦ = geopot[k] - geopot[k+1] # vertical gradient in geopotential
temp_parcel = temp_parcel - ΔΦ/cₚ*Γ # new temperature of parcel at k

# at new (lower) temperature condensation occurs immediately
# new humidity equals to that saturation humidity
humid_parcel = saturation_humidity(temp_parcel, σ[k]*pres, clausius_clapeyron)
else
temp_parcel = temp_parcel_dry # else parcel temperature following dry adiabat
temp_parcel = temp_parcel_dry # else parcel temperature following dry adiabat
end

# use dry/moist adiabatic ascent for reference profile
temp_ref_profile[k] = temp_parcel

# check whether parcel is still buoyant wrt to environment
# check whether parcel is still buoyant wrt to environment
# use virtual temperature as it's equivalent to density
temp_virt_parcel = temp_parcel*(1 + μ*humid_parcel) # virtual temperature of parcel
buoyant = temp_virt_parcel > temp_virt_environment[k] ? true : false
Expand All @@ -256,13 +254,10 @@ Base.@kwdef struct DryBettsMiller{NF} <: AbstractConvection

"[OPTION] Relaxation time for profile adjustment"
time_scale::Second = Hour(4)

"temperature [K] reference profile to adjust to"
temp_ref_profile::Vector{NF} = zeros(NF, nlev)
end

DryBettsMiller(SG::SpectralGrid; kwargs...) = DryBettsMiller{SG.NF}(nlev=SG.nlev; kwargs...)
initialize!(::DryBettsMiller, ::PrimitiveWet) = nothing
initialize!(::DryBettsMiller, ::PrimitiveEquation) = nothing

# function barrier to unpack model
function convection!(
Expand Down Expand Up @@ -291,9 +286,11 @@ function convection!(
σ = geometry.σ_levels_full
σ_half = geometry.σ_levels_half
Δσ = geometry.σ_levels_thick
(; temp_ref_profile) = DBM
(; nlev, temp, temp_tend) = column

# use work arrays for temp_ref_profile
temp_ref_profile = column.a # temperature [K] reference profile to adjust to

# CONVECTIVE CRITERIA AND FIRST GUESS RELAXATION
level_zero_buoyancy = dry_adiabat!(temp_ref_profile,
temp, σ,
Expand All @@ -314,13 +311,13 @@ function convection!(

# ADJUST PROFILES FOLLOWING FRIERSON 2007
convection = PT > 0
convection || return nothing # escape immediately for no convection
convection || return nothing # escape immediately for no convection

# height of zero buoyancy level in σ coordinates
Δσ_lzb = σ_half[nlev+1] - σ_half[level_zero_buoyancy]
ΔT = PT/Δσ_lzb # eq (5) or (14) but reusing PT
for k in level_zero_buoyancy:nlev
temp_ref_profile[k] -= ΔT # equation (6) or equation (15)
temp_ref_profile[k] -= ΔT # equation (6) or equation (15)
temp_tend[k] -= (temp[k] - temp_ref_profile[k]) / DBM.time_scale.value
end
end
Expand Down Expand Up @@ -352,14 +349,14 @@ function dry_adiabat!(
local buoyant::Bool = true # is the parcel still buoyant?
local k::Int = nlev # layer index top to surface

while buoyant && k > 1 # calculate moist adiabat while buoyant till top
while buoyant && k > 1 # calculate moist adiabat while buoyant till top
k -= 1 # one level up

# dry adiabatic ascent
temp_parcel = temp_parcel*(σ[k]/σ[k+1])^R_cₚ
temp_ref_profile[k] = temp_parcel

# check whether parcel is still buoyant wrt to environment
# check whether parcel is still buoyant wrt to environment
buoyant = temp_parcel > temp_environment[k] ? true : false
end

Expand Down
16 changes: 11 additions & 5 deletions src/physics/define_column.jl
Original file line number Diff line number Diff line change
Expand Up @@ -13,12 +13,12 @@ Base.@kwdef mutable struct ColumnVariables{NF<:AbstractFloat} <: AbstractColumnV
const nlev::Int = 0 # number of vertical levels

# COORDINATES
ij::Int = 0 # grid-point index
ij::Int = 0 # grid-point index
jring::Int = 0 # latitude ring the column is on
lond::NF = 0 # longitude
latd::NF = 0 # latitude, needed for shortwave radiation
land_fraction::NF = 0 # fraction of the column that is over land
orography::NF = 0 # orography height [m]
orography::NF = 0 # orography height [m]

# PROGNOSTIC VARIABLES
const u::Vector{NF} = zeros(NF, nlev) # zonal velocity [m/s]
Expand All @@ -31,9 +31,9 @@ Base.@kwdef mutable struct ColumnVariables{NF<:AbstractFloat} <: AbstractColumnV
const pres::Vector{NF} = zeros(NF, nlev+1) # pressure [Pa]

# TENDENCIES to accumulate the parametrizations into
const u_tend::Vector{NF} = zeros(NF, nlev) # zonal velocity [m/s²]
const u_tend::Vector{NF} = zeros(NF, nlev) # zonal velocity [m/s²]
const v_tend::Vector{NF} = zeros(NF, nlev) # meridional velocity [m/s²]
const temp_tend::Vector{NF} = zeros(NF, nlev) # absolute temperature [K/s]
const temp_tend::Vector{NF} = zeros(NF, nlev) # absolute temperature [K/s]
const humid_tend::Vector{NF} = zeros(NF, nlev) # specific humidity [kg/kg/s]

# FLUXES, arrays to be used for various parameterizations, on half levels incl top and bottom
Expand All @@ -49,7 +49,7 @@ Base.@kwdef mutable struct ColumnVariables{NF<:AbstractFloat} <: AbstractColumnV
const flux_humid_upward::Vector{NF} = zeros(NF, nlev+1)
const flux_humid_downward::Vector{NF} = zeros(NF, nlev+1)

# boundary layer
# boundary layer
boundary_layer_depth::Int = 0
boundary_layer_drag::NF = 0
surface_geopotential::NF = 0
Expand All @@ -74,4 +74,10 @@ Base.@kwdef mutable struct ColumnVariables{NF<:AbstractFloat} <: AbstractColumnV
cloud_top::Int = nlev + 1 # layer index k of top-most layer with clouds
precip_convection::NF = 0 # Precipitation due to convection [m]
precip_large_scale::NF = 0 # precipitation due to large-scale condensation [m]

# WORK ARRAYS
const a::Vector{NF} = zeros(NF, nlev)
const b::Vector{NF} = zeros(NF, nlev)
const c::Vector{NF} = zeros(NF, nlev)
const d::Vector{NF} = zeros(NF, nlev)
end
21 changes: 21 additions & 0 deletions test/convection.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
@testset "Convection" begin

spectral_grid = SpectralGrid(trunc=31, nlev=8)

for Convection in ( NoConvection,
SimplifiedBettsMiller,
DryBettsMiller)
for Model in ( PrimitiveDryModel,
PrimitiveWetModel)

# that combination is not defined
if ~(Convection == SimplifiedBettsMiller && Model == PrimitiveDryModel)
convection = Convection(spectral_grid)
model = Model(;spectral_grid, convection)
model.feedback.verbose = false
simulation = initialize!(model)
run!(simulation, period=Day(1))
end
end
end
end

0 comments on commit 4baf70f

Please sign in to comment.