From 98a6ac543d4dc70182b629f0626cfa22d10c4744 Mon Sep 17 00:00:00 2001 From: Milan Date: Wed, 20 Mar 2024 15:47:20 -0400 Subject: [PATCH 1/3] Convection with work arrays for thread safety --- src/physics/convection.jl | 64 +++++++++++++++++------------------- src/physics/define_column.jl | 16 ++++++--- test/convection.jl | 17 ++++++++++ 3 files changed, 58 insertions(+), 39 deletions(-) create mode 100644 test/convection.jl diff --git a/src/physics/convection.jl b/src/physics/convection.jl index c3b994ab3..73b20ebae 100644 --- a/src/physics/convection.jl +++ b/src/physics/convection.jl @@ -21,16 +21,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!( @@ -44,7 +38,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) @@ -70,11 +64,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] @@ -88,7 +85,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 @@ -111,7 +108,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 @@ -123,24 +120,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 @@ -157,9 +154,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 """ @@ -196,7 +193,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 @@ -205,7 +202,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 @@ -216,20 +213,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 @@ -256,13 +253,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!( @@ -291,9 +285,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, σ, @@ -314,13 +310,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 @@ -352,14 +348,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 diff --git a/src/physics/define_column.jl b/src/physics/define_column.jl index f814821e5..3b258939a 100644 --- a/src/physics/define_column.jl +++ b/src/physics/define_column.jl @@ -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] @@ -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 @@ -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 @@ -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 \ No newline at end of file diff --git a/test/convection.jl b/test/convection.jl new file mode 100644 index 000000000..71cf7c868 --- /dev/null +++ b/test/convection.jl @@ -0,0 +1,17 @@ +@testset "Convection" begin + + spectral_grid = SpectralGrid(trunc=31, nlev=8) + + for Convection in ( NoConvection, + SimplifiedBettsMiller, + DryBettsMiller) + for Model in ( PrimitiveDryModel, + PrimitiveWetModel) + + convection = Convection(spectral_grid) + model = Model(;spectral_grid, convection) + simulation = initialize!(model) + run!(simulation, period=Day(5)) + end + end +end \ No newline at end of file From e4c9c55ec5710426f4cbbc362ee2ea475755cef7 Mon Sep 17 00:00:00 2001 From: Milan Date: Wed, 20 Mar 2024 15:53:30 -0400 Subject: [PATCH 2/3] convection tests added --- src/physics/convection.jl | 1 + test/convection.jl | 12 ++++++++---- 2 files changed, 9 insertions(+), 4 deletions(-) diff --git a/src/physics/convection.jl b/src/physics/convection.jl index 73b20ebae..0111a72b0 100644 --- a/src/physics/convection.jl +++ b/src/physics/convection.jl @@ -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 diff --git a/test/convection.jl b/test/convection.jl index 71cf7c868..284145f6d 100644 --- a/test/convection.jl +++ b/test/convection.jl @@ -8,10 +8,14 @@ for Model in ( PrimitiveDryModel, PrimitiveWetModel) - convection = Convection(spectral_grid) - model = Model(;spectral_grid, convection) - simulation = initialize!(model) - run!(simulation, period=Day(5)) + # 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 \ No newline at end of file From 5b83db7d6b604f1efb62fb7131dbb5559ff27301 Mon Sep 17 00:00:00 2001 From: Milan Date: Wed, 20 Mar 2024 16:07:52 -0400 Subject: [PATCH 3/3] scaling information for diagn corrected --- src/dynamics/diagnostic_variables.jl | 10 +++++----- src/dynamics/scaling.jl | 4 ++++ src/dynamics/time_integration.jl | 4 ++-- 3 files changed, 11 insertions(+), 7 deletions(-) diff --git a/src/dynamics/diagnostic_variables.jl b/src/dynamics/diagnostic_variables.jl index 5337820da..ad2df9a92 100644 --- a/src/dynamics/diagnostic_variables.jl +++ b/src/dynamics/diagnostic_variables.jl @@ -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) @@ -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 @@ -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, diff --git a/src/dynamics/scaling.jl b/src/dynamics/scaling.jl index 716edc12d..801f05508 100644 --- a/src/dynamics/scaling.jl +++ b/src/dynamics/scaling.jl @@ -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 """ diff --git a/src/dynamics/time_integration.jl b/src/dynamics/time_integration.jl index 33ae7a4cf..0c836f0bc 100644 --- a/src/dynamics/time_integration.jl +++ b/src/dynamics/time_integration.jl @@ -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 @@ -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