forked from NOAA-GFDL/SIS2
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSIS_ctrl_types.F90
383 lines (323 loc) · 21.4 KB
/
SIS_ctrl_types.F90
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
!~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~!
! SIS_types contains a number of common SIS types, along with subroutines to !
! perform various tasks on these types, including allocation, deallocation, !
! registration for restarts, and checksums. !
!~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~!
module SIS_ctrl_types
! use mpp_mod, only: mpp_sum, stdout, input_nml_file, PE_here => mpp_pe
! use mpp_domains_mod, only: domain2D, mpp_get_compute_domain, CORNER, EAST, NORTH
use mpp_domains_mod, only: domain2D, CORNER, EAST, NORTH
! use mpp_parameter_mod, only: CGRID_NE, BGRID_NE, AGRID
! use fms_mod, only: open_namelist_file, check_nml_error, close_file
! use fms_io_mod, only: save_restart, restore_state, query_initialized
! use fms_io_mod, only: register_restart_field, restart_file_type
use time_manager_mod, only: time_type, time_type_to_real
use coupler_types_mod,only: coupler_2d_bc_type, coupler_3d_bc_type
use SIS_dyn_trans, only : dyn_trans_CS
use SIS_fast_thermo, only : fast_thermo_CS
use SIS_slow_thermo, only : slow_thermo_CS
use SIS_hor_grid, only : SIS_hor_grid_type
use ice_grid, only : ice_grid_type
use SIS_types, only : ice_state_type, ice_ocean_flux_type, ocean_sfc_state_type
use SIS_types, only : fast_ice_avg_type, ice_rad_type, simple_OSS_type
! use SIS2_ice_thm, only : ice_thermo_type !, SIS2_ice_thm_CS, enth_from_TS, energy_melt_EnthS
! use SIS2_ice_thm, only : get_SIS2_thermo_coefs, temp_from_En_S
use SIS_optics, only : SIS_optics_CS
use MOM_coms, only : PE_here
use MOM_error_handler, only : SIS_error=>MOM_error, FATAL, WARNING, SIS_mesg=>MOM_mesg, is_root_pe
use MOM_file_parser, only : param_file_type
use MOM_hor_index, only : hor_index_type
use SIS_diag_mediator, only : SIS_diag_ctrl, post_data=>post_SIS_data
use SIS_diag_mediator, only : register_SIS_diag_field, register_static_field
! use MOM_checksums, only : chksum, Bchksum, hchksum, uchksum, vchksum
! use SIS_error_checking, only : check_redundant_B, check_redundant_C
use SIS_sum_output_type, only : SIS_sum_out_CS
! use SIS_tracer_registry, only : SIS_tracer_registry_type
use SIS_tracer_flow_control, only : SIS_tracer_flow_control_CS
implicit none ; private
public :: SIS_fast_CS, SIS_slow_CS
public :: ice_diagnostics_init, ice_diags_fast_init
!~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~!
!> The SIS_fast_CS type is the control structure for the fast portion of the
!! SIS2 solver. Typically, this control structure and everything under it is
!! found on the atmospheric processors.
type SIS_fast_CS
type(time_type) :: Time
type(time_type) :: Time_step_fast
type(time_type) :: Time_step_slow
! logical :: slab_ice ! If true, do the old style GFDL slab ice.
! logical :: Cgrid_dyn ! If true use a C-grid discretization of the
! ! sea-ice dynamics.
logical :: bounds_check ! If true, check for sensible values of thicknesses
! temperatures, fluxes, etc.
logical :: debug ! If true, write verbose checksums for debugging purposes.
logical :: Eulerian_tsurf ! If true, use previous calculations of the ice-top
! surface skin temperature for tsurf at the start of
! atmospheric time stepping, including interpolating between
! tsurf values from other categories in the same location.
! type(SIS_tracer_registry_type), pointer :: TrReg => NULL()
type(ice_state_type), pointer :: IST => NULL()
type(fast_thermo_CS), pointer :: fast_thermo_CSp => NULL()
type(SIS_optics_CS), pointer :: optics_CSp => NULL()
type(SIS_diag_ctrl), pointer :: diag ! A structure that regulates diagnostics.
! diag here might point to its own structure, or it might point
! to the same structure as is used by SIS_slow_CS.
type(ice_rad_type), pointer :: Rad => NULL() ! A structure with fields related to
! the absorption, reflection and transmission of
! shortwave radiation.
type(SIS_hor_grid_type), pointer :: G => NULL() ! A structure containing metrics and grid info.
type(ice_grid_type), pointer :: IG => NULL() ! A structure containing sea-ice specific grid info.
! type(ice_state_type), pointer :: Ice_state => NULL() ! A structure containing the internal
! ! representation of the ice state.
type(simple_OSS_type), pointer :: sOSS => NULL() ! A structure containing the arrays
! that describe the ocean's surface state, as it is revealed
! to the atmosphere or the fast ice thermodynamics modules.
type(fast_ice_avg_type), pointer :: FIA => NULL() ! A structure of the fluxes and other
! fields that are calculated during the fast ice step but
! stored for later use by the slow ice step or the ocean.
end type SIS_fast_CS
!~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~!
!> The SIS_slow_CS type is the control structure for the slow portion of the
!! SIS2 solver. This control structure and everything under it may be found on
!! the atmospheric processors with the traditional FMS approach to concurrent
!! coupling, or they may be on the ocean processors with the new embedded-ice
!! approach.
type SIS_slow_CS
type(time_type) :: Time
type(time_type) :: Time_step_slow
! logical :: slab_ice ! If true, do the old style GFDL slab ice.
logical :: Cgrid_dyn ! If true use a C-grid discretization of the
! sea-ice dynamics.
logical :: specified_ice ! If true, the sea ice is specified and there is
! no need for ice dynamics.
logical :: do_icebergs ! If true, use the Lagrangian iceberg code, which
! modifies the calving field among other things.
logical :: pass_iceberg_area_to_ocean ! If true, iceberg area is passed through coupler
! (must have ICEBERGS_APPLY_RIGID_BOUNDARY=True in MOM_input)
logical :: berg_windstress_bug = .false. ! If true, use older code that applied
! an old ice-ocean stress to the icebergs in place of
! the current air-ice stress. This option exists for
! backward compatibility, but should be avoided.
logical :: bounds_check ! If true, check for sensible values of thicknesses
! temperatures, fluxes, etc.
logical :: debug ! If true, write verbose checksums for debugging purposes.
! type(SIS_tracer_registry_type), pointer :: TrReg => NULL()
type(ice_state_type), pointer :: IST => NULL()
type(slow_thermo_CS), pointer :: slow_thermo_CSp => NULL()
type(dyn_trans_CS), pointer :: dyn_trans_CSp => NULL()
type(SIS_tracer_flow_control_CS), pointer :: SIS_tracer_flow_CSp => NULL()
type(ice_ocean_flux_type), pointer :: IOF => NULL() ! A structure containing fluxes from
! the ice to the ocean that are calculated by the ice model.
type(SIS_diag_ctrl) :: diag ! A structure that regulates diagnostics.
type(SIS_hor_grid_type), pointer :: G => NULL() ! A structure containing metrics and grid info.
type(ice_grid_type), pointer :: IG => NULL() ! A structure containing sea-ice specific grid info.
! type(ice_state_type), pointer :: Ice_state => NULL() ! A structure containing the internal
! ! representation of the ice state.
type(ocean_sfc_state_type), pointer :: OSS => NULL() ! A structure containing the arrays
! that describe the ocean's surface state, as it is revealed
! to the ice model.
type(simple_OSS_type), pointer :: sOSS => NULL() ! A structure containing the arrays
! that describe the ocean's surface state, as it is revealed
! to the atmosphere or the fast ice thermodynamics modules.
type(fast_ice_avg_type), pointer :: FIA => NULL() ! A structure of the fluxes and other
! fields that are calculated during the fast ice step but
! stored for later use by the slow ice step or the ocean.
end type SIS_slow_CS
contains
!=======================================================================
!> ice_diagnostics_init does the registration for a variety of sea-ice model
!! diagnostics and saves several static diagnotic fields.
subroutine ice_diagnostics_init(IOF, OSS, FIA, G, IG, diag, Time, Cgrid)
type(ice_ocean_flux_type), intent(inout) :: IOF
type(ocean_sfc_state_type), intent(inout) :: OSS
type(fast_ice_avg_type), intent(inout) :: FIA
type(SIS_hor_grid_type), intent(inout) :: G
type(ice_grid_type), intent(in) :: IG
type(SIS_diag_ctrl), intent(in) :: diag
type(time_type), intent(inout) :: Time
logical, optional, intent(in) :: Cgrid
real, dimension(G%isc:G%iec,G%jsc:G%jec) :: tmp_diag ! A temporary diagnostic array
real :: I_area_Earth ! The inverse of the area of the sphere, in m-2.
real, parameter :: missing = -1e34 ! The fill value for missing data.
integer :: id_geo_lon, id_geo_lat, id_sin_rot, id_cos_rot, id_cell_area
logical :: Cgrid_dyn
logical :: sent
integer :: i, j, k, isc, iec, jsc, jec, n, nLay
character(len=8) :: nstr
isc = G%isc ; iec = G%iec ; jsc = G%jsc ; jec = G%jec
nLay = IG%NkIce
Cgrid_dyn = .true. ; if (present(Cgrid)) Cgrid_dyn = Cgrid
id_sin_rot = register_static_field('ice_model', 'SINROT', diag%axesT1, &
'-SINROT,COSROT points north', 'none')
id_cos_rot = register_static_field('ice_model', 'COSROT', diag%axesT1, &
'COSROT,SINROT points east','none')
id_geo_lon = register_static_field('ice_model', 'GEOLON', diag%axesT1, 'longitude', &
'degrees')
id_geo_lat = register_static_field('ice_model', 'GEOLAT', diag%axesT1, 'latitude', &
'degrees')
id_cell_area = register_static_field('ice_model', 'CELL_AREA', diag%axesT1, &
'cell area', 'sphere')
FIA%id_sh = register_SIS_diag_field('ice_model','SH' ,diag%axesT1, Time, &
'sensible heat flux', 'W/m^2', missing_value=missing)
FIA%id_lh = register_SIS_diag_field('ice_model','LH' ,diag%axesT1, Time, &
'latent heat flux', 'W/m^2', missing_value=missing)
FIA%id_sw = register_SIS_diag_field('ice_model','SW' ,diag%axesT1, Time, &
'shortwave heat flux', 'W/m^2', missing_value=missing)
FIA%id_lw = register_SIS_diag_field('ice_model','LW' ,diag%axesT1, Time, &
'longwave heat flux over ice', 'W/m^2', missing_value=missing)
FIA%id_snofl = register_SIS_diag_field('ice_model','SNOWFL' ,diag%axesT1, Time, &
'rate of snow fall', 'kg/(m^2*s)', missing_value=missing)
FIA%id_rain = register_SIS_diag_field('ice_model','RAIN' ,diag%axesT1, Time, &
'rate of rain fall', 'kg/(m^2*s)', missing_value=missing)
FIA%id_runoff = register_SIS_diag_field('ice_model','RUNOFF' ,diag%axesT1, Time, &
'liquid runoff', 'kg/(m^2*s)', missing_value=missing)
FIA%id_calving = register_SIS_diag_field('ice_model','CALVING',diag%axesT1, Time, &
'frozen runoff', 'kg/(m^2*s)', missing_value=missing)
FIA%id_runoff_hflx = register_SIS_diag_field('ice_model','RUNOFF_HFLX' ,diag%axesT1, Time, &
'liquid runoff sensible heat flux', 'W/m^2', missing_value=missing)
FIA%id_calving_hflx = register_SIS_diag_field('ice_model','CALVING_HFLX',diag%axesT1, Time, &
'frozen runoff sensible heat flux', 'W/m^2', missing_value=missing)
FIA%id_evap = register_SIS_diag_field('ice_model','EVAP',diag%axesT1, Time, &
'evaporation', 'kg/(m^2*s)', missing_value=missing)
IOF%id_saltf = register_SIS_diag_field('ice_model','SALTF' ,diag%axesT1, Time, &
'ice to ocean salt flux', 'kg/(m^2*s)', missing_value=missing)
FIA%id_tmelt = register_SIS_diag_field('ice_model','TMELT' ,diag%axesT1, Time, &
'upper surface melting energy flux', 'W/m^2', missing_value=missing)
FIA%id_bmelt = register_SIS_diag_field('ice_model','BMELT' ,diag%axesT1, Time, &
'bottom surface melting energy flux', 'W/m^2', missing_value=missing)
FIA%id_bheat = register_SIS_diag_field('ice_model','BHEAT' ,diag%axesT1, Time, &
'ocean to ice heat flux', 'W/m^2', missing_value=missing)
!### THIS DIAGNOSTIC IS MISSING.
! XYZ%id_strna = register_SIS_diag_field('ice_model','STRAIN_ANGLE', diag%axesT1,Time, &
! 'strain angle', 'none', missing_value=missing)
FIA%id_sw_dn = register_SIS_diag_field('ice_model','SWDN' ,diag%axesT1, Time, &
'Downward shortwave heat flux at the bottom of the atmosphere', 'W/m^2', missing_value=missing)
FIA%id_albedo = register_SIS_diag_field('ice_model','ALB' ,diag%axesT1, Time, &
'Shortwave flux weighted surface albedo, or 1 if no SW', '0-1', missing_value=missing)
FIA%id_sw_vis = register_SIS_diag_field('ice_model','SW_VIS' ,diag%axesT1, Time, &
'visible shortwave heat flux', 'W/m^2', missing_value=missing)
FIA%id_sw_dir = register_SIS_diag_field('ice_model','SW_DIR' ,diag%axesT1, Time, &
'direct shortwave heat flux', 'W/m^2', missing_value=missing)
FIA%id_sw_dif = register_SIS_diag_field('ice_model','SW_DIF' ,diag%axesT1, Time, &
'diffuse shortwave heat flux', 'W/m^2', missing_value=missing)
FIA%id_sw_vis_dir = register_SIS_diag_field('ice_model','SW_VIS_DIR' ,diag%axesT1, Time, &
'visible direct shortwave heat flux', 'W/m^2', missing_value=missing)
FIA%id_sw_vis_dif = register_SIS_diag_field('ice_model','SW_VIS_DIF' ,diag%axesT1, Time, &
'visible diffuse shortwave heat flux', 'W/m^2', missing_value=missing)
FIA%id_sw_nir_dir = register_SIS_diag_field('ice_model','SW_NIR_DIR' ,diag%axesT1, Time, &
'near IR direct shortwave heat flux', 'W/m^2', missing_value=missing)
FIA%id_sw_nir_dif = register_SIS_diag_field('ice_model','SW_NIR_DIF' ,diag%axesT1, Time, &
'near IR diffuse shortwave heat flux', 'W/m^2', missing_value=missing)
! diagnostics for quantities produced outside the ice model
FIA%id_slp = register_SIS_diag_field('ice_model', 'SLP', diag%axesT1, Time, &
'sea level pressure', 'Pa', missing_value=missing)
! diagnostics for quantities produced outside the ice model
OSS%id_sst = register_SIS_diag_field('ice_model', 'SST', diag%axesT1, Time, &
'sea surface temperature', 'deg-C', missing_value=missing)
OSS%id_sss = register_SIS_diag_field('ice_model', 'SSS', diag%axesT1, Time, &
'sea surface salinity', 'psu', missing_value=missing)
OSS%id_ssh = register_SIS_diag_field('ice_model', 'SSH', diag%axesT1, Time, &
'sea surface height', 'm', missing_value=missing)
if (Cgrid_dyn) then
OSS%id_uo = register_SIS_diag_field('ice_model', 'UO', diag%axesCu1, Time, &
'surface current - x component', 'm/s', missing_value=missing)
OSS%id_vo = register_SIS_diag_field('ice_model', 'VO', diag%axesCv1, Time, &
'surface current - y component', 'm/s', missing_value=missing)
else
OSS%id_uo = register_SIS_diag_field('ice_model', 'UO', diag%axesB1, Time, &
'surface current - x component', 'm/s', missing_value=missing)
OSS%id_vo = register_SIS_diag_field('ice_model', 'VO', diag%axesB1, Time, &
'surface current - y component', 'm/s', missing_value=missing)
endif
OSS%id_frazil = register_SIS_diag_field('ice_model','FRAZIL' ,diag%axesT1, Time, &
'energy flux of frazil formation', 'W/m^2', missing_value=missing)
!### THIS DIAGNOSTIC IS MISSING.
! XYZ%id_obi = register_SIS_diag_field('ice_model', 'OBI', diag%axesT1, Time, &
! 'ice observed', '0 or 1', missing_value=missing)
! Use whether the appropriate arrays are allocated to determine whether the
! following iceberg diagnostics should be offered.
if (associated(IOF%ustar_berg)) &
IOF%id_ustar_berg = register_SIS_diag_field('ice_model', 'USTAR_BERG', diag%axesT1, Time, &
'iceberg ustar', 'm/s', missing_value=missing)
if (associated(IOF%area_berg)) &
IOF%id_area_berg = register_SIS_diag_field('ice_model', 'AREA_BERG', diag%axesT1, Time, &
'icebergs area', 'm2/m2', missing_value=missing)
if (associated(IOF%mass_berg)) &
IOF%id_mass_berg = register_SIS_diag_field('ice_model', 'MASS_BERG', diag%axesT1, Time, &
'icebergs mass', 'kg/m2', missing_value=missing)
! Write out static fields.
if (id_sin_rot>0) call post_data(id_sin_rot, G%sin_rot, diag, is_static=.true.)
if (id_cos_rot>0) call post_data(id_cos_rot, G%cos_rot, diag, is_static=.true.)
if (id_geo_lon>0) call post_data(id_geo_lon, G%geoLonT, diag, is_static=.true.)
if (id_geo_lat>0) call post_data(id_geo_lat, G%geoLatT, diag, is_static=.true.)
if (id_cell_area>0) then
I_area_Earth = 1.0 / (16.0*atan(1.0)*G%Rad_Earth**2)
!$OMP parallel do default(none) shared(isc,iec,jsc,jec,G,I_area_Earth,tmp_diag)
do j=jsc,jec ; do i=isc,iec
tmp_diag(i,j) = (G%areaT(i,j) * G%mask2dT(i,j)) * I_area_Earth
enddo ; enddo
call post_data(id_cell_area, tmp_diag, diag, is_static=.true.)
endif
end subroutine ice_diagnostics_init
!> ice_diags_fast_init does the registration for a variety of sea-ice model
!! diagnostics associated with the rapid physics updates.
subroutine ice_diags_fast_init(Rad, G, IG, diag, Time, component)
type(ice_rad_type), intent(inout) :: Rad
type(SIS_hor_grid_type), intent(inout) :: G
type(ice_grid_type), intent(in) :: IG
type(SIS_diag_ctrl), intent(in) :: diag
type(time_type), intent(inout) :: Time
character(len=*), optional, intent(in) :: component
real, parameter :: missing = -1e34 ! The fill value for missing data.
integer :: i, j, k, isc, iec, jsc, jec, n, nLay
character(len=8) :: nstr
character(len=40) :: comp_name ! The name for this component in the diag tables.
comp_name = "ice_model" ; if (present(component)) comp_name = component
isc = G%isc ; iec = G%iec ; jsc = G%jsc ; jec = G%jec
nLay = IG%NkIce
Rad%id_swdn = register_SIS_diag_field(trim(comp_name),'SWDN' ,diag%axesT1, Time, &
'downward shortwave flux', 'W/m^2', missing_value=missing)
Rad%id_lwdn = register_SIS_diag_field(trim(comp_name),'LWDN' ,diag%axesT1, Time, &
'downward longwave flux', 'W/m^2', missing_value=missing)
Rad%id_alb = register_SIS_diag_field(trim(comp_name),'ALB',diag%axesT1, Time, &
'surface albedo','0-1', missing_value=missing )
Rad%id_coszen = register_SIS_diag_field(trim(comp_name),'coszen',diag%axesT1, Time, &
'cosine of the solar zenith angle for the next radiation step','-1:1', missing_value=missing )
Rad%id_sw_abs_sfc= register_SIS_diag_field(trim(comp_name),'sw_abs_sfc',diag%axesT1, Time, &
'SW frac. abs. at the ice surface','0-1', missing_value=missing )
Rad%id_sw_abs_snow= register_SIS_diag_field(trim(comp_name),'sw_abs_snow',diag%axesT1, Time, &
'SW frac. abs. in snow','0-1', missing_value=missing )
call safe_alloc_ids_1d(Rad%id_sw_abs_ice, nLay)
do n=1,nLay
write(nstr, '(I4)') n ; nstr = adjustl(nstr)
Rad%id_sw_abs_ice(n) = register_SIS_diag_field(trim(comp_name),'sw_abs_ice'//trim(nstr), &
diag%axesT1, Time, 'SW frac. abs. in ice layer '//trim(nstr), &
'0:1', missing_value=missing )
enddo
Rad%id_sw_pen= register_SIS_diag_field(trim(comp_name),'sw_pen',diag%axesT1, Time, &
'SW frac. pen. surf.','0:1', missing_value=missing )
Rad%id_sw_abs_ocn= register_SIS_diag_field(trim(comp_name),'sw_abs_ocn',diag%axesT1, Time, &
'SW frac. sent to the ocean','0:1', missing_value=missing )
Rad%id_alb_vis_dir = register_SIS_diag_field(trim(comp_name),'alb_vis_dir',diag%axesT1, Time, &
'ice surface albedo vis_dir','0-1', missing_value=missing )
Rad%id_alb_vis_dif = register_SIS_diag_field(trim(comp_name),'alb_vis_dif',diag%axesT1, Time, &
'ice surface albedo vis_dif','0-1', missing_value=missing )
Rad%id_alb_nir_dir = register_SIS_diag_field(trim(comp_name),'alb_nir_dir',diag%axesT1, Time, &
'ice surface albedo nir_dir','0-1', missing_value=missing )
Rad%id_alb_nir_dif = register_SIS_diag_field(trim(comp_name),'alb_nir_dif',diag%axesT1, Time, &
'ice surface albedo nir_dif','0-1', missing_value=missing )
Rad%id_tskin = register_SIS_diag_field(trim(comp_name),'Tskin', diag%axesTc, Time, &
'Skin temperature','Kelvin', missing_value=missing )
Rad%id_cn = register_SIS_diag_field(trim(comp_name),'CN_fast', diag%axesTc, Time, &
'Category concentration','0-1', missing_value=missing )
Rad%id_mi = register_SIS_diag_field(trim(comp_name),'MI_fast', diag%axesTc, Time, &
'Category concentration','0-1', missing_value=missing )
end subroutine ice_diags_fast_init
subroutine safe_alloc_ids_1d(ids, nids)
integer, allocatable :: ids(:)
integer, intent(in) :: nids
if (.not.ALLOCATED(ids)) then
allocate(ids(nids)) ; ids(:) = -1
endif
end subroutine safe_alloc_ids_1d
end module SIS_ctrl_types