diff --git a/.buildkite/longruns_gpu/pipeline.yml b/.buildkite/longruns_gpu/pipeline.yml index 0b70b63240..8b0cd40b72 100644 --- a/.buildkite/longruns_gpu/pipeline.yml +++ b/.buildkite/longruns_gpu/pipeline.yml @@ -37,6 +37,16 @@ steps: artifact_paths: "land_longrun_gpu/*png" agents: slurm_gpus: 1 - slurm_time: 12:00:00 + slurm_time: 01:00:00 + env: + CLIMACOMMS_DEVICE: "CUDA" + + - label: "Soil" + command: + - julia --color=yes --project=.buildkite experiments/long_runs/soil.jl + artifact_paths: "soil_longrun_gpu/*png" + agents: + slurm_gpus: 1 + slurm_time: 01:00:00 env: CLIMACOMMS_DEVICE: "CUDA" diff --git a/experiments/long_runs/soil.jl b/experiments/long_runs/soil.jl new file mode 100644 index 0000000000..c9f63722af --- /dev/null +++ b/experiments/long_runs/soil.jl @@ -0,0 +1,477 @@ +# # Global run of soil model + +# The code sets up and runs the soil model for on a spherical domain, +# using ERA5 data. In this simulation, we have +# turned lateral flow off because horizontal boundary conditions and the +# land/sea mask are not yet supported by ClimaCore. + +# Simulation Setup +# Number of spatial elements: 101 in horizontal, 15 in vertical +# Soil depth: 50 m +# Simulation duration: 365 d +# Timestep: 900 s +# Timestepper: ARS343 +# Fixed number of iterations: 1 +# Jacobian update: every new timestep +# Atmos forcing update: every 3 hours +import SciMLBase +import ClimaComms +@static pkgversion(ClimaComms) >= v"0.6" && ClimaComms.@import_required_backends +import ClimaTimeSteppers as CTS +using ClimaCore +using ClimaUtilities.ClimaArtifacts +import Interpolations +using Insolation + +using ClimaDiagnostics +using ClimaAnalysis +import ClimaAnalysis.Visualize as viz +using ClimaUtilities + +import ClimaUtilities.TimeVaryingInputs: TimeVaryingInput +import ClimaUtilities.SpaceVaryingInputs: SpaceVaryingInput +import ClimaUtilities.Regridders: InterpolationsRegridder +import ClimaUtilities.ClimaArtifacts: @clima_artifact +import ClimaParams as CP + +using ClimaLand +using ClimaLand.Soil +import ClimaLand +import ClimaLand.Parameters as LP + +using Statistics +using CairoMakie +using Dates +import NCDatasets + +const FT = Float64; + +regridder_type = :InterpolationsRegridder +context = ClimaComms.context() +device = ClimaComms.device() +device_suffix = device isa ClimaComms.CPUSingleThreaded ? "cpu" : "gpu" +root_path = "soil_longrun_$(device_suffix)" +diagnostics_outdir = joinpath(root_path, "global_diagnostics") +outdir = + ClimaUtilities.OutputPathGenerator.generate_output_path(diagnostics_outdir) + +function setup_prob(t0, tf, Δt; outdir = outdir, nelements = (101, 15)) + + earth_param_set = LP.LandParameters(FT) + radius = FT(6378.1e3) + depth = FT(50) + domain = ClimaLand.Domains.SphericalShell(; + radius = radius, + depth = depth, + nelements = nelements, + npolynomial = 1, + dz_tuple = FT.((10.0, 0.05)), + ) + surface_space = domain.space.surface + subsurface_space = domain.space.subsurface + + ref_time = DateTime(2021) + t_start = t0 + # Forcing data + era5_artifact_path = + ClimaLand.Artifacts.era5_land_forcing_data2021_folder_path(; context) # Precipitation: + precip = TimeVaryingInput( + joinpath(era5_artifact_path, "era5_2021_0.9x1.25_clima.nc"), + "rf", + surface_space; + reference_date = ref_time, + t_start, + regridder_type, + file_reader_kwargs = (; preprocess_func = (data) -> -data / 3600,), + ) + + snow_precip = TimeVaryingInput( + joinpath(era5_artifact_path, "era5_2021_0.9x1.25.nc"), + "sf", + surface_space; + reference_date = ref_time, + t_start, + regridder_type, + file_reader_kwargs = (; preprocess_func = (data) -> -data / 3600,), + ) + + u_atmos = TimeVaryingInput( + joinpath(era5_artifact_path, "era5_2021_0.9x1.25_clima.nc"), + "ws", + surface_space; + reference_date = ref_time, + t_start, + regridder_type, + ) + q_atmos = TimeVaryingInput( + joinpath(era5_artifact_path, "era5_2021_0.9x1.25_clima.nc"), + "q", + surface_space; + reference_date = ref_time, + t_start, + regridder_type, + ) + P_atmos = TimeVaryingInput( + joinpath(era5_artifact_path, "era5_2021_0.9x1.25.nc"), + "sp", + surface_space; + reference_date = ref_time, + t_start, + regridder_type, + ) + + T_atmos = TimeVaryingInput( + joinpath(era5_artifact_path, "era5_2021_0.9x1.25.nc"), + "t2m", + surface_space; + reference_date = ref_time, + t_start, + regridder_type, + ) + h_atmos = FT(10) + + atmos = PrescribedAtmosphere( + precip, + snow_precip, + T_atmos, + u_atmos, + q_atmos, + P_atmos, + ref_time, + h_atmos, + earth_param_set, + ) + + # Prescribed radiation + SW_d = TimeVaryingInput( + joinpath(era5_artifact_path, "era5_2021_0.9x1.25.nc"), + "ssrd", + surface_space; + reference_date = ref_time, + t_start, + regridder_type, + file_reader_kwargs = (; preprocess_func = (data) -> data / 3600,), + ) + LW_d = TimeVaryingInput( + joinpath(era5_artifact_path, "era5_2021_0.9x1.25.nc"), + "strd", + surface_space; + reference_date = ref_time, + t_start, + regridder_type, + file_reader_kwargs = (; preprocess_func = (data) -> data / 3600,), + ) + + function zenith_angle( + t, + ref_time; + latitude = ClimaCore.Fields.coordinate_field(surface_space).lat, + longitude = ClimaCore.Fields.coordinate_field(surface_space).long, + insol_params::Insolation.Parameters.InsolationParameters{FT} = earth_param_set.insol_params, + ) where {FT} + # This should be time in UTC + current_datetime = ref_time + Dates.Second(round(t)) + + # Orbital Data uses Float64, so we need to convert to our sim FT + d, δ, η_UTC = + FT.( + Insolation.helper_instantaneous_zenith_angle( + current_datetime, + ref_time, + insol_params, + ) + ) + + Insolation.instantaneous_zenith_angle.( + d, + δ, + η_UTC, + longitude, + latitude, + ).:1 + end + radiation = + PrescribedRadiativeFluxes(FT, SW_d, LW_d, ref_time; θs = zenith_angle) + + soil_params_artifact_path = + ClimaLand.Artifacts.soil_params_artifact_folder_path(; context) + extrapolation_bc = ( + Interpolations.Periodic(), + Interpolations.Flat(), + Interpolations.Flat(), + ) + soil_params_mask = SpaceVaryingInput( + joinpath( + soil_params_artifact_path, + "vGalpha_map_gupta_etal2020_1.0x1.0x4.nc", + ), + "α", + subsurface_space; + regridder_type, + regridder_kwargs = (; extrapolation_bc,), + file_reader_kwargs = (; preprocess_func = (data) -> data > 0,), + ) + oceans_to_value(field, mask, value) = + mask == 1.0 ? field : eltype(field)(value) + + vg_α = SpaceVaryingInput( + joinpath( + soil_params_artifact_path, + "vGalpha_map_gupta_etal2020_1.0x1.0x4.nc", + ), + "α", + subsurface_space; + regridder_type, + regridder_kwargs = (; extrapolation_bc,), + ) + vg_n = SpaceVaryingInput( + joinpath( + soil_params_artifact_path, + "vGn_map_gupta_etal2020_1.0x1.0x4.nc", + ), + "n", + subsurface_space; + regridder_type, + regridder_kwargs = (; extrapolation_bc,), + ) + x = parent(vg_α) + μ = mean(log10.(x[x .> 0])) + vg_α .= oceans_to_value.(vg_α, soil_params_mask, 10.0^μ) + + x = parent(vg_n) + μ = mean(x[x .> 0]) + vg_n .= oceans_to_value.(vg_n, soil_params_mask, μ) + + vg_fields_to_hcm_field(α::FT, n::FT) where {FT} = + ClimaLand.Soil.vanGenuchten{FT}(; @NamedTuple{α::FT, n::FT}((α, n))...) + hydrology_cm = vg_fields_to_hcm_field.(vg_α, vg_n) + + θ_r = SpaceVaryingInput( + joinpath( + soil_params_artifact_path, + "residual_map_gupta_etal2020_1.0x1.0x4.nc", + ), + "θ_r", + subsurface_space; + regridder_type, + regridder_kwargs = (; extrapolation_bc,), + ) + + ν = SpaceVaryingInput( + joinpath( + soil_params_artifact_path, + "porosity_map_gupta_etal2020_1.0x1.0x4.nc", + ), + "ν", + subsurface_space; + regridder_type, + regridder_kwargs = (; extrapolation_bc,), + ) + K_sat = SpaceVaryingInput( + joinpath( + soil_params_artifact_path, + "ksat_map_gupta_etal2020_1.0x1.0x4.nc", + ), + "Ksat", + subsurface_space; + regridder_type, + regridder_kwargs = (; extrapolation_bc,), + ) + + x = parent(K_sat) + μ = mean(log10.(x[x .> 0])) + K_sat .= oceans_to_value.(K_sat, soil_params_mask, 10.0^μ) + + ν .= oceans_to_value.(ν, soil_params_mask, 1) + + θ_r .= oceans_to_value.(θ_r, soil_params_mask, 0) + + + S_s = + oceans_to_value.( + ClimaCore.Fields.zeros(subsurface_space) .+ FT(1e-3), + soil_params_mask, + 1, + ) + ν_ss_om = + oceans_to_value.( + ClimaCore.Fields.zeros(subsurface_space) .+ FT(0.1), + soil_params_mask, + 0, + ) + ν_ss_quartz = + oceans_to_value.( + ClimaCore.Fields.zeros(subsurface_space) .+ FT(0.1), + soil_params_mask, + 0, + ) + ν_ss_gravel = + oceans_to_value.( + ClimaCore.Fields.zeros(subsurface_space) .+ FT(0.1), + soil_params_mask, + 0, + ) + PAR_albedo = ClimaCore.Fields.zeros(surface_space) .+ FT(0.2) + NIR_albedo = ClimaCore.Fields.zeros(surface_space) .+ FT(0.2) + soil_params = Soil.EnergyHydrologyParameters( + FT; + ν, + ν_ss_om, + ν_ss_quartz, + ν_ss_gravel, + hydrology_cm, + K_sat, + S_s, + θ_r, + PAR_albedo = PAR_albedo, + NIR_albedo = NIR_albedo, + ) + + soil_params_mask_sfc = + ClimaLand.Domains.top_center_to_surface(soil_params_mask) + + # Read in f_max data and land sea mask + infile_path = ClimaLand.Artifacts.topmodel_data_path() + f_max = + SpaceVaryingInput(infile_path, "fmax", surface_space; regridder_type) + mask = SpaceVaryingInput( + infile_path, + "landsea_mask", + surface_space; + regridder_type, + ) + # Unsure how to handle two masks + f_max = oceans_to_value.(f_max, mask, FT(0.0)) + f_max = oceans_to_value.(f_max, soil_params_mask_sfc, FT(0.0)) + f_over = FT(3.28) # 1/m + R_sb = FT(1.484e-4 / 1000) # m/s + runoff_model = ClimaLand.Soil.Runoff.TOPMODELRunoff{FT}(; + f_over = f_over, + f_max = f_max, + R_sb = R_sb, + ) + + soil_args = (domain = domain, parameters = soil_params) + soil_model_type = Soil.EnergyHydrology{FT} + sources = (Soil.PhaseChange{FT}(Δt),)# sublimation and subsurface runoff are added automatically + top_bc = ClimaLand.Soil.AtmosDrivenFluxBC(atmos, radiation, runoff_model) + zero_flux = Soil.HeatFluxBC((p, t) -> 0.0) + boundary_conditions = (; + top = top_bc, + bottom = Soil.WaterHeatBC(; + water = Soil.FreeDrainage(), + heat = zero_flux, + ), + ) + soil = soil_model_type(; + boundary_conditions = boundary_conditions, + sources = sources, + soil_args..., + ) + + + Y, p, cds = initialize(soil) + + init_soil(ν, θ_r) = θ_r + (ν - θ_r) / 2 + Y.soil.ϑ_l .= init_soil.(ν, θ_r) + Y.soil.θ_i .= FT(0.0) + T = FT(276.85) + ρc_s = + Soil.volumetric_heat_capacity.( + Y.soil.ϑ_l, + Y.soil.θ_i, + soil_params.ρc_ds, + soil_params.earth_param_set, + ) + Y.soil.ρe_int .= + Soil.volumetric_internal_energy.( + Y.soil.θ_i, + ρc_s, + T, + soil_params.earth_param_set, + ) + + set_initial_cache! = make_set_initial_cache(soil) + exp_tendency! = make_exp_tendency(soil) + imp_tendency! = ClimaLand.make_imp_tendency(soil) + jacobian! = ClimaLand.make_jacobian(soil) + set_initial_cache!(p, Y, t0) + + # set up jacobian info + jac_kwargs = + (; jac_prototype = ImplicitEquationJacobian(Y), Wfact = jacobian!) + + prob = SciMLBase.ODEProblem( + CTS.ClimaODEFunction( + T_exp! = exp_tendency!, + T_imp! = SciMLBase.ODEFunction(imp_tendency!; jac_kwargs...), + dss! = ClimaLand.dss!, + ), + Y, + (t0, tf), + p, + ) + + updateat = Array(t0:(3600 * 3):tf) + drivers = ClimaLand.get_drivers(soil) + updatefunc = ClimaLand.make_update_drivers(drivers) + + # ClimaDiagnostics + + nc_writer = ClimaDiagnostics.Writers.NetCDFWriter(subsurface_space, outdir) + + diags = + ClimaLand.CLD.default_diagnostics(soil, t0; output_writer = nc_writer) + + diagnostic_handler = + ClimaDiagnostics.DiagnosticsHandler(diags, Y, p, t0; dt = Δt) + + diag_cb = ClimaDiagnostics.DiagnosticsCallback(diagnostic_handler) + + driver_cb = ClimaLand.DriverUpdateCallback(updateat, updatefunc) + return prob, SciMLBase.CallbackSet(driver_cb, diag_cb) +end + +function setup_and_solve_problem(; greet = false) + + t0 = 0.0 + tf = 60 * 60.0 * 24 * 60 # keep short until it runs! * 365 + Δt = 900.0 + nelements = (101, 15) + if greet + @info "Run: Global Soil Model" + @info "Resolution: $nelements" + @info "Timestep: $Δt s" + @info "Duration: $(tf - t0) s" + end + + prob, cb = setup_prob(t0, tf, Δt; nelements) + + # Define timestepper and ODE algorithm + stepper = CTS.ARS111() + ode_algo = CTS.IMEXAlgorithm( + stepper, + CTS.NewtonsMethod( + max_iters = 3, + update_j = CTS.UpdateEvery(CTS.NewNewtonIteration), + ), + ) + SciMLBase.solve(prob, ode_algo; dt = Δt, callback = cb, adaptive = false) + return nothing +end + +setup_and_solve_problem(; greet = true); +# read in diagnostics and make some plots! +#### ClimaAnalysis #### +simdir = ClimaAnalysis.SimDir(outdir) +short_names_2D = ["slw", "si", "tsoil"] +times = 0.0:(60.0 * 60.0 * 24 * 20):(60.0 * 60.0 * 24 * 60) +for t in times + for short_name in short_names_2D + var = get(simdir; short_name) + fig = CairoMakie.Figure(size = (800, 600)) + kwargs = short_name in short_names_2D ? Dict() : Dict(:z => 1) + viz.plot!(fig, var, time = t; kwargs...) + CairoMakie.save(joinpath(root_path, "$short_name $t.png"), fig) + end +end diff --git a/src/diagnostics/Diagnostics.jl b/src/diagnostics/Diagnostics.jl index 52da2e8f91..81678eaee3 100644 --- a/src/diagnostics/Diagnostics.jl +++ b/src/diagnostics/Diagnostics.jl @@ -6,6 +6,8 @@ using ..Bucket: BucketModel import ..SoilCanopyModel +import ..Soil: EnergyHydrology + import ..Domains: top_center_to_surface import ClimaDiagnostics: diff --git a/src/diagnostics/default_diagnostics.jl b/src/diagnostics/default_diagnostics.jl index f5ef8cc44f..feadb6e2f3 100644 --- a/src/diagnostics/default_diagnostics.jl +++ b/src/diagnostics/default_diagnostics.jl @@ -122,3 +122,20 @@ function default_diagnostics( hourly_averages(soilcanopy_diagnostics...; output_writer, t_start) return [default_outputs...] end + + +# SoilModel +function default_diagnostics( + land_model::EnergyHydrology, + t_start; + output_writer, +) + + define_diagnostics!(land_model) + + soil_diagnostics = ["slw", "si", "tsoil"] + + default_outputs = + daily_averages(soil_diagnostics...; output_writer, t_start) + return [default_outputs...] +end diff --git a/src/diagnostics/soilcanopy_compute_methods.jl b/src/diagnostics/soilcanopy_compute_methods.jl index 7f99072c68..eef836b786 100644 --- a/src/diagnostics/soilcanopy_compute_methods.jl +++ b/src/diagnostics/soilcanopy_compute_methods.jl @@ -58,15 +58,27 @@ function compute_vapor_flux!(out, Y, p, t, land_model::SoilCanopyModel) end end -function compute_soil_temperature!(out, Y, p, t, land_model::SoilCanopyModel) +function compute_soil_temperature!( + out, + Y, + p, + t, + land_model::Union{SoilCanopyModel, EnergyHydrology}, +) if isnothing(out) - return copy(p.soil.T) # or is it p.soil.T? + return copy(top_center_to_surface(p.soil.T)) else - out .= p.soil.T + out .= top_center_to_surface(p.soil.T) end end -function compute_soil_water_liquid!(out, Y, p, t, land_model::SoilCanopyModel) +function compute_soil_water_liquid!( + out, + Y, + p, + t, + land_model::Union{SoilCanopyModel, EnergyHydrology}, +) if isnothing(out) return copy(top_center_to_surface(p.soil.θ_l)) # or is it in Y.soil? else @@ -74,7 +86,13 @@ function compute_soil_water_liquid!(out, Y, p, t, land_model::SoilCanopyModel) end end -function compute_infiltration(out, Y, p, t, land_model::SoilCanopyModel) +function compute_infiltration( + out, + Y, + p, + t, + land_model::Union{SoilCanopyModel, EnergyHydrology}, +) if isnothing(out) return copy(p.soil.infiltration) else @@ -570,7 +588,13 @@ function compute_canopy_temperature!(out, Y, p, t, land_model::SoilCanopyModel) end end -function compute_soil_ice!(out, Y, p, t, land_model::SoilCanopyModel) +function compute_soil_ice!( + out, + Y, + p, + t, + land_model::Union{SoilCanopyModel, EnergyHydrology}, +) if isnothing(out) return copy(top_center_to_surface(Y.soil.θ_i)) else diff --git a/src/integrated/soil_canopy_model.jl b/src/integrated/soil_canopy_model.jl index ee49351665..5258787fd3 100644 --- a/src/integrated/soil_canopy_model.jl +++ b/src/integrated/soil_canopy_model.jl @@ -88,9 +88,9 @@ function SoilCanopyModel{FT}(; MM <: Soil.Biogeochemistry.SoilCO2Model{FT}, } - (; atmos, radiation, soil_organic_carbon) = land_args + (; atmos, radiation, soil_organic_carbon, Δt) = land_args # These should always be set by the constructor. - sources = (RootExtraction{FT}(), Soil.PhaseChange{FT}()) + sources = (RootExtraction{FT}(), Soil.PhaseChange{FT}(Δt)) if :runoff ∈ propertynames(land_args) top_bc = ClimaLand.Soil.AtmosDrivenFluxBC( atmos, diff --git a/src/shared_utilities/Domains.jl b/src/shared_utilities/Domains.jl index 23b43428e3..7d1371a49b 100644 --- a/src/shared_utilities/Domains.jl +++ b/src/shared_utilities/Domains.jl @@ -753,20 +753,22 @@ bottom_center_to_surface(val) = val A function to return a tuple containing the distance between the top boundary and its closest center, and the bottom boundary and its closest center, -both as Fields. +both as Fields. It also returns the widths of each layer as a field. """ function get_Δz(z::ClimaCore.Fields.Field) # Extract the differences between levels of the face space fs = obtain_face_space(axes(z)) z_face = ClimaCore.Fields.coordinate_field(fs).z - Δz = ClimaCore.Fields.Δz_field(z_face) - + Δz_face = ClimaCore.Fields.Δz_field(z_face) Δz_top = ClimaCore.Fields.level( - Δz, + Δz_face, ClimaCore.Utilities.PlusHalf(ClimaCore.Spaces.nlevels(fs) - 1), ) - Δz_bottom = ClimaCore.Fields.level(Δz, ClimaCore.Utilities.PlusHalf(0)) - return Δz_top ./ 2, Δz_bottom ./ 2 + Δz_bottom = ClimaCore.Fields.level(Δz_face, ClimaCore.Utilities.PlusHalf(0)) + + #Layer widths: + Δz_center = ClimaCore.Fields.Δz_field(z) + return Δz_top ./ 2, Δz_bottom ./ 2, Δz_center end """ @@ -808,11 +810,17 @@ during the simulation. function get_additional_domain_fields(subsurface_space) surface_space = obtain_surface_space(subsurface_space) z = ClimaCore.Fields.coordinate_field(subsurface_space).z - Δz_top, Δz_bottom = get_Δz(z) + Δz_top, Δz_bottom, Δz = get_Δz(z) face_space = obtain_face_space(subsurface_space) z_face = ClimaCore.Fields.coordinate_field(face_space).z z_sfc = top_face_to_surface(z_face, surface_space) - fields = (; z = z, Δz_top = Δz_top, Δz_bottom = Δz_bottom, z_sfc = z_sfc) + fields = (; + z = z, + Δz_top = Δz_top, + Δz_bottom = Δz_bottom, + z_sfc = z_sfc, + Δz = Δz, + ) return fields end diff --git a/src/standalone/Soil/energy_hydrology.jl b/src/standalone/Soil/energy_hydrology.jl index b08699ca0c..493f45f7cb 100644 --- a/src/standalone/Soil/energy_hydrology.jl +++ b/src/standalone/Soil/energy_hydrology.jl @@ -331,7 +331,7 @@ function ClimaLand.make_compute_jacobian(model::EnergyHydrology{FT}) where {FT} ClimaLand.Soil.dψdϑ( hydrology_cm, Y.soil.ϑ_l, - ν, + ν - Y.soil.θ_i, #ν_eff θ_r, S_s, ), @@ -353,7 +353,7 @@ function ClimaLand.make_compute_jacobian(model::EnergyHydrology{FT}) where {FT} ClimaLand.Soil.dψdϑ( hydrology_cm, Y.soil.ϑ_l, - ν, + ν - Y.soil.θ_i, #ν_eff θ_r, S_s, ), @@ -572,9 +572,11 @@ end """ PhaseChange{FT} <: AbstractSoilSource{FT} -PhaseChange source type. +PhaseChange source type """ -struct PhaseChange{FT} <: AbstractSoilSource{FT} end +struct PhaseChange{FT} <: AbstractSoilSource{FT} + Δt::FT +end """ @@ -599,7 +601,7 @@ function ClimaLand.source!( (; ν, ρc_ds, θ_r, hydrology_cm, earth_param_set) = params _ρ_l = FT(LP.ρ_cloud_liq(earth_param_set)) _ρ_i = FT(LP.ρ_cloud_ice(earth_param_set)) - Δz_top = model.domain.fields.Δz_top # center face distance + Δz = model.domain.fields.Δz # center face distance @. dY.soil.ϑ_l += -phase_change_source( p.soil.θ_l, @@ -612,8 +614,9 @@ function ClimaLand.source!( ρc_ds, earth_param_set, ), - 2 * Δz_top, # the factor of 2 appears to get the face-face/layer thickness, Δz_top is center-face distance + Δz, p.soil.κ, + src.Δt, ), ν, θ_r, @@ -632,8 +635,9 @@ function ClimaLand.source!( ρc_ds, earth_param_set, ), - 2 * Δz_top, #the factor of 2 appears to get the face-face/layer thickness, Δz_top is center-face distance + Δz, p.soil.κ, + src.Δt, ), ν, θ_r, @@ -780,7 +784,7 @@ Returns the surface albedo field of the `EnergyHydrology` soil model. """ function ClimaLand.surface_albedo(model::EnergyHydrology{FT}, Y, p) where {FT} - return FT( + return @. FT( 0.5 * model.parameters.PAR_albedo + 0.5 * model.parameters.NIR_albedo, ) end diff --git a/src/standalone/Soil/soil_heat_parameterizations.jl b/src/standalone/Soil/soil_heat_parameterizations.jl index dc6fb35427..d90505d4fa 100644 --- a/src/standalone/Soil/soil_heat_parameterizations.jl +++ b/src/standalone/Soil/soil_heat_parameterizations.jl @@ -18,9 +18,12 @@ export volumetric_internal_energy, Returns the thermal timescale for temperature differences across a typical thickness Δz to equilibrate. + +Clip to 10x the timestep in order to respect CFL condition, +given that phase change is stepped explicitly. """ -function thermal_time(ρc::FT, Δz::FT, κ::FT) where {FT} - return ρc * Δz^2 / κ +function thermal_time(ρc::FT, Δz::FT, κ::FT, Δt::FT) where {FT} + return max(ρc * Δz^2 / κ, 10 * Δt) end """ @@ -209,7 +212,7 @@ Compute the expression for relative saturation. This is referred to as θ_sat in Balland and Arp's paper. """ function relative_saturation(θ_l::FT, θ_i::FT, ν::FT) where {FT} - return (θ_l + θ_i) / ν + return max((θ_l + θ_i) / ν, eps(FT)) end """ diff --git a/src/standalone/Soil/soil_hydrology_parameterizations.jl b/src/standalone/Soil/soil_hydrology_parameterizations.jl index b677c19bd5..6e405fe9cc 100644 --- a/src/standalone/Soil/soil_hydrology_parameterizations.jl +++ b/src/standalone/Soil/soil_hydrology_parameterizations.jl @@ -113,13 +113,14 @@ function pressure_head( end """ - dψdϑ(cm::vanGenuchten{FT}, ϑ, ν, θ_r, S_s) + dψdϑ(cm::vanGenuchten{FT}, ϑ_l, ν, θ_r, S_s) Computes and returns the derivative of the pressure head with respect to ϑ for the van Genuchten formulation. """ -function dψdϑ(cm::vanGenuchten{FT}, ϑ, ν, θ_r, S_s) where {FT} - S = effective_saturation(ν, ϑ, θ_r) +function dψdϑ(cm::vanGenuchten{FT}, ϑ_l, ν, θ_r, S_s) where {FT} + ϑ_l_safe = max(ϑ_l, θ_r + eps(FT)) + S = effective_saturation(ν, ϑ_l_safe, θ_r) (; α, m, n) = cm if S < 1.0 return FT( diff --git a/test/shared_utilities/domains.jl b/test/shared_utilities/domains.jl index b4a6ef7fdd..5805c3e8e1 100644 --- a/test/shared_utilities/domains.jl +++ b/test/shared_utilities/domains.jl @@ -42,7 +42,7 @@ FT = Float32 face_space = obtain_face_space(shell.space.subsurface) z_face = ClimaCore.Fields.coordinate_field(face_space).z @test shell.fields.z_sfc == top_face_to_surface(z_face, shell.space.surface) - Δz_top, Δz_bottom = get_Δz(shell.fields.z) + Δz_top, Δz_bottom, Δz = get_Δz(shell.fields.z) @test shell.fields.Δz_top == Δz_top @test shell.fields.Δz_bottom == Δz_bottom @test shell.radius == radius @@ -107,7 +107,7 @@ FT = Float32 face_space = obtain_face_space(box.space.subsurface) z_face = ClimaCore.Fields.coordinate_field(face_space).z @test box.fields.z_sfc == top_face_to_surface(z_face, box.space.surface) - Δz_top, Δz_bottom = get_Δz(box.fields.z) + Δz_top, Δz_bottom, Δz = get_Δz(box.fields.z) @test box.fields.Δz_top == Δz_top @test box.fields.Δz_bottom == Δz_bottom box_coords = coordinates(box).subsurface @@ -178,7 +178,9 @@ FT = Float32 z_face = ClimaCore.Fields.coordinate_field(face_space).z @test z_column.fields.z_sfc == top_face_to_surface(z_face, z_column.space.surface) - Δz_top, Δz_bottom = get_Δz(z_column.fields.z) + Δz_top, Δz_bottom, Δz = get_Δz(z_column.fields.z) + z = ClimaCore.Fields.coordinate_field(z_column.space.subsurface).z + @test z_column.fields.z == z @test z_column.fields.Δz_top == Δz_top @test z_column.fields.Δz_bottom == Δz_bottom column_coords = coordinates(z_column).subsurface diff --git a/test/standalone/Soil/soiltest.jl b/test/standalone/Soil/soiltest.jl index f84b38c1df..d07566ba81 100644 --- a/test/standalone/Soil/soiltest.jl +++ b/test/standalone/Soil/soiltest.jl @@ -663,8 +663,8 @@ for FT in (Float32, Float64) heat = zero_heat_flux_bc, ), ) - - sources = (PhaseChange{FT}(),) + Δt = FT(1000)# doesnt matter what this is here, we are checking dY.liquid water and dY.ice are in sync + sources = (PhaseChange{FT}(Δt),) ### hyd_off_en_on = Soil.EnergyHydrologyParameters(