Skip to content

Commit aae3d3c

Browse files
committed
identify 1D and 2D coupler fields
1 parent 592216e commit aae3d3c

13 files changed

+108
-67
lines changed

docs/src/interfacer.md

+7-6
Original file line numberDiff line numberDiff line change
@@ -54,9 +54,10 @@ is used for checkpointing the simulation.
5454
- `add_coupler_fields!(coupler_field_names::Set, ::ComponentModelSimulation)`:
5555
A function that adds names of quantities the coupler must exchange
5656
to support this component model. These will be added for each model
57-
in addition to the existing defaults: `z0m_sfc`, `z0b_sfc`, `beta`,
58-
`F_turb_energy`, `F_turb_moisture`, `F_turb_ρτxz`, `F_turb_ρτyz`,
59-
`temp1`, and `temp2`.
57+
in addition to the existing defaults: `z0m_sfc` (scalar), `z0b_sfc` (scalar),
58+
`beta` (scalar), `F_turb_energy`, `F_turb_moisture`, `F_turb_ρτxz`, `F_turb_ρτyz`,
59+
`temp1`, and `temp2`. They are defined as 2D fields at the surface, unless
60+
specified otherwise. When extending this function, if a variable
6061

6162
### ComponentModelSimulation - optional functions
6263
- `update_sim!(::ComponentModelSimulation, csf, turbulent_fluxes)`: A
@@ -197,9 +198,9 @@ overwritten or used as-is. These currently include the following:
197198

198199
| Coupler name | Description | Units | Default value |
199200
|-------------------|-------------|-------|---------------|
200-
| `beta` | factor that scales evaporation based on its estimated level of saturation | | 1 |
201-
| `emissivity` | measure of how much energy a surface radiates | | 1 |
202-
| `height_disp` | displacement height relative to the surface | m | 0 |
201+
| `beta` | factor that scales evaporation based on its estimated level of saturation (scalar) | | 1 |
202+
| `emissivity` | measure of how much energy a surface radiates (scalar) | | 1 |
203+
| `height_disp` | displacement height relative to the surface (scalar) | m | 0 |
203204

204205
- `update_turbulent_fluxes!(::ComponentModelSimulation, fields::NamedTuple)`:
205206
This function updates the turbulent fluxes of the component model simulation

experiments/ClimaEarth/components/atmosphere/climaatmos.jl

+5-1
Original file line numberDiff line numberDiff line change
@@ -346,6 +346,8 @@ end
346346
Interfacer.reinit!(sim::ClimaAtmosSimulation) = Interfacer.reinit!(sim.integrator)
347347

348348
"""
349+
Interfacer.add_coupler_fields!(coupler_field_names, ::ClimaAtmosSimulation)
350+
349351
Extend Interfacer.add_coupler_fields! to add the fields required for ClimaAtmosSimulation.
350352
351353
The fields added are:
@@ -356,8 +358,10 @@ The fields added are:
356358
- `:q_sfc` (for moisture)
357359
"""
358360
function Interfacer.add_coupler_fields!(coupler_field_names, ::ClimaAtmosSimulation)
361+
# All fields are 2D; update the list of 2D coupler field names
359362
atmos_coupler_fields = [:surface_direct_albedo, :surface_diffuse_albedo, :ϵ_sfc, :T_sfc, :q_sfc]
360-
push!(coupler_field_names, atmos_coupler_fields...)
363+
push!(coupler_field_names[2], atmos_coupler_fields...)
364+
return nothing
361365
end
362366

363367
function FieldExchanger.update_sim!(sim::ClimaAtmosSimulation, csf, turbulent_fluxes)

experiments/ClimaEarth/components/land/climaland_bucket.jl

+5-1
Original file line numberDiff line numberDiff line change
@@ -312,15 +312,19 @@ Interfacer.step!(sim::BucketSimulation, t) = Interfacer.step!(sim.integrator, t
312312
Interfacer.reinit!(sim::BucketSimulation) = Interfacer.reinit!(sim.integrator)
313313

314314
"""
315+
Interfacer.add_coupler_fields!(coupler_field_names, ::BucketSimulation)
316+
315317
Extend Interfacer.add_coupler_fields! to add the fields required for BucketSimulation.
316318
317319
The fields added are:
318320
- `:ρ_sfc`
319321
- `:F_radiative` (for radiation input)
320322
"""
321323
function Interfacer.add_coupler_fields!(coupler_field_names, ::BucketSimulation)
324+
# All fields are 2D; update the list of 2D coupler field names
322325
bucket_coupler_fields = [:ρ_sfc, :F_radiative]
323-
push!(coupler_field_names, bucket_coupler_fields...)
326+
push!(coupler_field_names[2], bucket_coupler_fields...)
327+
return nothing
324328
end
325329

326330
# extensions required by FluxCalculator (partitioned fluxes)

experiments/ClimaEarth/components/land/climaland_integrated.jl

+13-8
Original file line numberDiff line numberDiff line change
@@ -416,8 +416,8 @@ end
416416
function Interfacer.update_field!(sim::ClimaLandSimulation, ::Val{:air_humidity}, field)
417417
parent(sim.integrator.p.drivers.q) .= parent(field)
418418
end
419-
function Interfacer.update_field!(sim::ClimaLandSimulation, ::Val{:c_co2}, field)
420-
sim.integrator.p.drivers.c_co2 .= field
419+
function Interfacer.update_field!(sim::ClimaLandSimulation, ::Val{:c_co2}, scalar)
420+
sim.integrator.p.drivers.c_co2 .= scalar
421421
end
422422
function Interfacer.update_field!(sim::ClimaLandSimulation, ::Val{:liquid_precipitation}, field)
423423
# Arbitrarily take parameters from the soil (they are the same for all land sub-components)
@@ -471,29 +471,34 @@ function FieldExchanger.import_atmos_fields!(csf, sim::ClimaLandSimulation, atmo
471471
FieldExchanger.dummmy_remap!(csf.q_air, Interfacer.get_field(atmos_sim, Val(:specific_humidity)))
472472
FieldExchanger.dummmy_remap!(csf.P_liq, Interfacer.get_field(atmos_sim, Val(:liquid_precipitation)))
473473
FieldExchanger.dummmy_remap!(csf.P_snow, Interfacer.get_field(atmos_sim, Val(:snow_precipitation)))
474-
# CO2 is a scalar so it doesn't need remapping TODO store CO2 as a scalar in the coupler fields
475-
csf.c_co2 .= Interfacer.get_field(atmos_sim, Val(:co2))
474+
# CO2 is a scalar so it doesn't need remapping, but we do need to access the Ref
475+
csf.c_co2[] = Interfacer.get_field(atmos_sim, Val(:co2))
476476
end
477477

478478
"""
479+
Interfacer.add_coupler_fields!(coupler_field_names, ::ClimaLandSimulation)
480+
479481
Extend Interfacer.add_coupler_fields! to add the fields required for ClimaLandSimulation.
480482
481483
The fields added are:
484+
- `:c_co2` (for photosynthesis, biogeochemistry) - scalar
482485
- `:SW_d` (for radiative transfer)
483486
- `:LW_d` (for radiative transfer)
484487
- `:cos_zenith_angle` (for radiative transfer)
485488
- `:diffuse_fraction` (for radiative transfer)
486-
- `:c_co2` (for photosynthesis, biogeochemistry)
487489
- `:P_air` (for canopy conductance)
488490
- `:T_air` (for canopy conductance)
489491
- `:q_air` (for canopy conductance)
490492
- `P_liq` (for moisture fluxes)
491493
- `P_snow` (for moisture fluxes)
492494
"""
493495
function Interfacer.add_coupler_fields!(coupler_field_names, ::ClimaLandSimulation)
494-
land_coupler_fields =
495-
[:SW_d, :LW_d, :cos_zenith_angle, :diffuse_fraction, :c_co2, :P_air, :T_air, :q_air, :P_liq, :P_snow]
496-
push!(coupler_field_names, land_coupler_fields...)
496+
land_coupler_fields_1d = [:c_co2]
497+
land_coupler_fields_2d =
498+
[:SW_d, :LW_d, :cos_zenith_angle, :diffuse_fraction, :P_air, :T_air, :q_air, :P_liq, :P_snow]
499+
push!(coupler_field_names[1], land_coupler_fields_1d...)
500+
push!(coupler_field_names[2], land_coupler_fields_2d...)
501+
return nothing
497502
end
498503

499504
function Checkpointer.get_model_prog_state(sim::ClimaLandSimulation)

experiments/ClimaEarth/components/ocean/eisenman_seaice.jl

+5-1
Original file line numberDiff line numberDiff line change
@@ -158,15 +158,19 @@ Interfacer.step!(sim::EisenmanIceSimulation, t) = Interfacer.step!(sim.integrato
158158
Interfacer.reinit!(sim::EisenmanIceSimulation) = Interfacer.reinit!(sim.integrator)
159159

160160
"""
161+
Interfacer.add_coupler_fields!(coupler_field_names, ::EisenmanIceSimulation)
162+
161163
Extend Interfacer.add_coupler_fields! to add the fields required for EisenmanIceSimulation.
162164
163165
The fields added are:
164166
- `:ρ_sfc` (for humidity calculation)
165167
- `:F_radiative` (for radiation input)
166168
"""
167169
function Interfacer.add_coupler_fields!(coupler_field_names, ::EisenmanIceSimulation)
170+
# All fields are 2D; update the list of 2D coupler field names
168171
eisenman_coupler_fields = [:ρ_sfc, :F_radiative]
169-
push!(coupler_field_names, eisenman_coupler_fields...)
172+
push!(coupler_field_names[2], eisenman_coupler_fields...)
173+
return nothing
170174
end
171175

172176
# extensions required by FluxCalculator (partitioned fluxes)

experiments/ClimaEarth/components/ocean/prescr_seaice.jl

+5-1
Original file line numberDiff line numberDiff line change
@@ -190,15 +190,19 @@ Interfacer.step!(sim::PrescribedIceSimulation, t) = Interfacer.step!(sim.integra
190190
Interfacer.reinit!(sim::PrescribedIceSimulation) = Interfacer.reinit!(sim.integrator)
191191

192192
"""
193+
Interfacer.add_coupler_fields!(coupler_field_names, ::PrescribedIceSimulation)
194+
193195
Extend Interfacer.add_coupler_fields! to add the fields required for PrescribedIceSimulation.
194196
195197
The fields added are:
196198
- `:ρ_sfc` (for humidity calculation)
197199
- `:F_radiative` (for radiation input)
198200
"""
199201
function Interfacer.add_coupler_fields!(coupler_field_names, ::PrescribedIceSimulation)
202+
# All fields are 2D; update the list of 2D coupler field names
200203
ice_coupler_fields = [:ρ_sfc, :F_radiative]
201-
push!(coupler_field_names, ice_coupler_fields...)
204+
push!(coupler_field_names[2], ice_coupler_fields...)
205+
return nothing
202206
end
203207

204208
# extensions required by FluxCalculator (partitioned fluxes)

experiments/ClimaEarth/components/ocean/slab_ocean.jl

+5-1
Original file line numberDiff line numberDiff line change
@@ -148,15 +148,19 @@ Interfacer.step!(sim::SlabOceanSimulation, t) = Interfacer.step!(sim.integrator,
148148
Interfacer.reinit!(sim::SlabOceanSimulation) = Interfacer.reinit!(sim.integrator)
149149

150150
"""
151+
Interfacer.add_coupler_fields!(coupler_field_names, ::SlabOceanSimulation)
152+
151153
Extend Interfacer.add_coupler_fields! to add the fields required for SlabOceanSimulation.
152154
153155
The fields added are:
154156
- `:ρ_sfc` (for humidity calculation)
155157
- `:F_radiative` (for radiation input)
156158
"""
157159
function Interfacer.add_coupler_fields!(coupler_field_names, ::SlabOceanSimulation)
160+
# All fields are 2D; update the list of 2D coupler field names
158161
ocean_coupler_fields = [:ρ_sfc, :F_radiative]
159-
push!(coupler_field_names, ocean_coupler_fields...)
162+
push!(coupler_field_names[2], ocean_coupler_fields...)
163+
return nothing
160164
end
161165

162166
# extensions required by FluxCalculator (partitioned fluxes)

experiments/ClimaEarth/setup_run.jl

+2-2
Original file line numberDiff line numberDiff line change
@@ -490,8 +490,8 @@ function setup_and_run(config_dict::AbstractDict)
490490
for sim in model_sims
491491
Interfacer.add_coupler_fields!(coupler_field_names, sim)
492492
end
493-
# add coupler fields required to track conservation, if specified
494-
energy_check && push!(coupler_field_names, :radiative_energy_flux_toa, :P_net)
493+
# add 2D coupler fields required to track conservation, if specified
494+
energy_check && push!(coupler_field_names[2], :radiative_energy_flux_toa, :P_net)
495495

496496
# allocate space for the coupler fields
497497
coupler_fields = Interfacer.init_coupler_fields(FT, coupler_field_names, boundary_space)

experiments/ClimaEarth/test/component_model_tests/climaland_tests.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,7 @@ end
7777

7878
# Initialize the coupler fields so we can perform exchange
7979
coupler_field_names = Interfacer.default_coupler_fields()
80-
map(sim -> Interfacer.add_coupler_fields!(coupler_field_names, sim), values(model_sims))
80+
map(sim -> Interfacer.add_coupler_fields!(coupler_field_names, sim), model_sims)
8181
coupler_fields = Interfacer.init_coupler_fields(FT, coupler_field_names, boundary_space)
8282
flux_type = FluxCalculator.PartitionedStateFluxes()
8383

src/FluxCalculator.jl

+2-2
Original file line numberDiff line numberDiff line change
@@ -105,7 +105,7 @@ function calculate_surface_air_density(atmos_sim::Interfacer.AtmosModelSimulatio
105105
end
106106

107107
"""
108-
partitioned_turbulent_fluxes!(model_sims::NamedTuple, fields::CC.Fields.Field, boundary_space::CC.Spaces.AbstractSpace, surface_scheme, thermo_params::TD.Parameters.ThermodynamicsParameters)
108+
partitioned_turbulent_fluxes!(model_sims::NamedTuple, fields::NamedTuple, boundary_space::CC.Spaces.AbstractSpace, surface_scheme, thermo_params::TD.Parameters.ThermodynamicsParameters)
109109
110110
The current setup calculates the aerodynamic fluxes in the coupler (assuming no regridding is needed)
111111
using adapter function `get_surface_fluxes!`, which calls `SurfaceFluxes.jl`. The coupler saves
@@ -128,7 +128,7 @@ TODO:
128128
"""
129129
function partitioned_turbulent_fluxes!(
130130
model_sims::NamedTuple,
131-
csf::CC.Fields.Field,
131+
csf::NamedTuple,
132132
boundary_space::CC.Spaces.AbstractSpace,
133133
surface_scheme,
134134
thermo_params::TD.Parameters.ThermodynamicsParameters,

src/Interfacer.jl

+37-28
Original file line numberDiff line numberDiff line change
@@ -99,42 +99,51 @@ float_type(::CoupledSimulation{FT}) where {FT} = FT
9999
"""
100100
default_coupler_fields()
101101
102-
Return a list of default coupler fields needed to run a simulation.
103-
"""
104-
default_coupler_fields() = [
105-
# fields needed for flux calculations
106-
:z0m_sfc,
107-
:z0b_sfc,
108-
:beta,
109-
:emissivity,
110-
# fields used for flux exchange
111-
:F_turb_energy,
112-
:F_turb_moisture,
113-
:F_turb_ρτxz,
114-
:F_turb_ρτyz,
115-
# fields used to track water conservation, and for water fluxes
116-
:P_liq,
117-
:P_snow,
118-
# fields used for temporary storage during calculations
119-
:temp1,
120-
:temp2,
121-
]
102+
Return 2 vectors of default coupler fields needed to run a simulation,
103+
corresponding to 1D fields (scalars) and 2D fields.
104+
"""
105+
function default_coupler_fields()
106+
keys_1d = [
107+
# scalars needed for flux calculations
108+
:z0m_sfc,
109+
:z0b_sfc,
110+
:beta,
111+
:emissivity,
112+
]
113+
keys_2d = [
114+
# fields used for flux exchange
115+
:F_turb_energy,
116+
:F_turb_moisture,
117+
:F_turb_ρτxz,
118+
:F_turb_ρτyz,
119+
# fields used to track water conservation, and for water fluxes
120+
:P_liq,
121+
:P_snow,
122+
# fields used for temporary storage during calculations
123+
:temp1,
124+
:temp2,
125+
]
126+
return (keys_1d, keys_2d)
127+
end
122128

123129
"""
124130
init_coupler_fields(FT, coupler_field_names, boundary_space)
125131
126-
Allocate a Field of NamedTuples on the provided boundary space to store
127-
the provided coupler fields.
132+
Allocate a NamedTuple of Floats for scalars, and Fields on the provided
133+
boundary space to store the provided coupler fields.
128134
"""
129135
function init_coupler_fields(FT, coupler_field_names, boundary_space)
136+
keys_1d, keys_2d = coupler_field_names
130137
# First remove any duplicate field names
131-
unique!(coupler_field_names)
138+
unique!(keys_1d)
139+
unique!(keys_2d)
140+
141+
# Allocate space for the 1D and 2D variables
142+
vals_1d = ((Ref(zero(FT)) for _ in 1:length(keys_1d))...,)
143+
vals_2d = ((zeros(boundary_space) for _ in 1:length(keys_2d))...,)
132144

133-
key_types = (coupler_field_names...,)
134-
val_types = Tuple{(FT for _ in 1:length(coupler_field_names))...}
135-
nt_type = NamedTuple{key_types, val_types}
136-
coupler_fields = zeros(nt_type, boundary_space)
137-
return coupler_fields
145+
# Dict(zip((keys_1d..., keys_2d...), (vals_1d..., vals_2d...)))
146+
return NamedTuple{(keys_1d..., keys_2d...)}((vals_1d..., vals_2d...))
138147
end
139148

140149
"""

src/surface_stub.jl

+4-1
Original file line numberDiff line numberDiff line change
@@ -97,14 +97,17 @@ The stub surface simulation is not updated by this function. Extends `SciMLBase.
9797
reinit!(::AbstractSurfaceStub) = nothing
9898

9999
"""
100+
Interfacer.add_coupler_fields!(coupler_field_names, ::AbstractSurfaceStub)
101+
100102
Extend Interfacer.add_coupler_fields! to add the fields required for AbstractSurfaceStub.
101103
102104
The fields added are:
103105
- `:ρ_sfc` (for humidity calculation)
104106
"""
105107
function Interfacer.add_coupler_fields!(coupler_field_names, ::AbstractSurfaceStub)
108+
# Add to the 2D coupler fields
106109
surface_coupler_fields = [:ρ_sfc]
107-
push!(coupler_field_names, surface_coupler_fields...)
110+
push!(coupler_field_names[2], surface_coupler_fields...)
108111
end
109112

110113
"""

test/flux_calculator_tests.jl

+17-14
Original file line numberDiff line numberDiff line change
@@ -199,20 +199,23 @@ for FT in (Float32, Float64)
199199

200200
model_sims = (; atmos_sim, ocean_sim, ocean_sim2)
201201

202-
coupler_cache_names = [
203-
:T_sfc,
204-
:surface_direct_albedo,
205-
:surface_diffuse_albedo,
206-
:F_R_sfc,
207-
:F_R_toa,
208-
:P_liq,
209-
:P_snow,
210-
:P_net,
211-
:F_turb_energy,
212-
:F_turb_ρτxz,
213-
:F_turb_ρτyz,
214-
:F_turb_moisture,
215-
]
202+
coupler_cache_names = (
203+
[],
204+
[
205+
:T_sfc,
206+
:surface_direct_albedo,
207+
:surface_diffuse_albedo,
208+
:F_R_sfc,
209+
:F_R_toa,
210+
:P_liq,
211+
:P_snow,
212+
:P_net,
213+
:F_turb_energy,
214+
:F_turb_ρτxz,
215+
:F_turb_ρτyz,
216+
:F_turb_moisture,
217+
],
218+
)
216219
fields = Interfacer.init_coupler_fields(FT, coupler_cache_names, boundary_space)
217220

218221
# calculate turbulent fluxes

0 commit comments

Comments
 (0)