MONC
kidreader.F90
Go to the documentation of this file.
1 
6  use state_mod, only : model_state_type
8  use collections_mod, only : map_type
14  use netcdf, only : nf90_noerr, nf90_global, nf90_nowrite, nf90_inquire_attribute, nf90_open, nf90_strerror, &
15  nf90_inq_dimid, nf90_inquire_dimension, nf90_inq_varid, nf90_get_var, nf90_inquire, nf90_close, nf90_get_att
16  implicit none
17 
18 #ifndef TEST_MODE
19  private
20 #endif
21 
22  integer, parameter :: number_q_coords = 100
23  character(len=*), parameter :: time_key = "time",& !< Corresponding NetCDF data time key
24  z_key = "z",&
25  z_half_key = "z_half",&
26  x_key = "x",&
27  x_half_key = "x_half",&
28  u_key="u",&
29  w_key="w"
30 
31  logical :: flood_q = .false., float_q= .false., clone_to_3d = .false., rotate_xy=.false.
32  integer :: domain_multiplication=1
33 
34 
35  integer, dimension(NUMBER_Q_COORDS), save :: q_coordinates_x, q_coordinates_y, q_coordinates_z, q_coordinates_value
36  !real :: rhobous, thref0, surface_pressure !< Boussinesq density, reference potential temperature and surface pressure
37  character(len=STRING_LENGTH) :: configuration_file
38 
40 
41 contains
42 
46  kidreader_get_descriptor%name="kidreader"
47  kidreader_get_descriptor%version=0.1
49  end function kidreader_get_descriptor
50 
54  subroutine initialise_callback(current_state)
55  type(model_state_type), target, intent(inout) :: current_state
56 
57  integer :: ncid, time_dim, z_dim, z_half_dim, x_dim, x_half_dim
58  real, dimension(:), allocatable :: time, x, z, x_half, z_half
59  real, dimension(:,:,:), allocatable :: u, w, v
60 
61  if (options_has_key(current_state%options_database, "restart")) then
62  call log_master_log(log_debug, "Ignoring KiD reader as restart from checkpoint file selected")
63  return
64  end if
65 
66  configuration_file=options_get_string(current_state%options_database, "kid_configuration_file")
67  current_state%rhobous=options_get_real(current_state%options_database, "rhobous")
68  current_state%thref0=options_get_real(current_state%options_database, "thref0")
69  current_state%surface_pressure=options_get_real(current_state%options_database, "surface_pressure")
70  flood_q=options_get_logical(current_state%options_database, "flood_q")
71  float_q=options_get_logical(current_state%options_database, "float_q")
72  clone_to_3d=options_get_logical(current_state%options_database, "clone_to_3d")
73  rotate_xy=options_get_logical(current_state%options_database, "rotate_xy")
74  domain_multiplication=options_get_integer(current_state%options_database, "domain_multiplication")
75 
76  call options_get_integer_array(current_state%options_database, "q_coordinates_x", q_coordinates_x)
77  call options_get_integer_array(current_state%options_database, "q_coordinates_y", q_coordinates_y)
78  call options_get_integer_array(current_state%options_database, "q_coordinates_z", q_coordinates_z)
79  call options_get_integer_array(current_state%options_database, "q_coordinates_value", q_coordinates_value)
80 
81  call check_status(nf90_open(path = trim(configuration_file), mode = nf90_nowrite, ncid = ncid))
82  call check_kinematics_file(ncid)
83  if (log_get_logging_level() .ge. log_debug) call read_global_attributes(ncid, current_state%parallel%my_rank)
84  call read_dimensions(ncid, time_dim, z_dim, z_half_dim, x_dim, x_half_dim)
85  call read_variables(ncid, time_dim, z_dim, z_half_dim, x_dim, x_half_dim, time, x, z, x_half, z_half, u, w, v)
86  call check_status(nf90_close(ncid))
87 
88  call create_grid(current_state%global_grid, z_half, x_half, z_half_dim, x_half_dim, current_state%parallel%my_rank)
89  call define_vertical_levels(current_state, z_half, z_half_dim)
90 
91  call decompose_grid(current_state)
92 
93  if (rotate_xy) then
94  call initialise_velocity_field(current_state%local_grid, current_state%v, dual_grid, primal_grid, dual_grid, v)
95  else
96  call initialise_velocity_field(current_state%local_grid, current_state%u, dual_grid, dual_grid, primal_grid, u)
97  end if
98  call initialise_velocity_field(current_state%local_grid, current_state%w, primal_grid, dual_grid, dual_grid, w)
99  if (clone_to_3d) call initialise_velocity_field(current_state%local_grid, current_state%v, dual_grid, primal_grid, dual_grid, v)
100  call initalise_source_and_z_fields(current_state)
101  call set_up_q_fields(current_state)
102  current_state%initialised=.true.
103  call log_master_log(log_info, "Initialised configuration from KiD model file `"//trim(configuration_file)//"`")
104  end subroutine initialise_callback
105 
106  subroutine set_up_q_fields(current_state)
107  type(model_state_type), intent(inout) :: current_state
108 
109  integer :: x_size, y_size, z_size, i
110 
111  z_size = current_state%local_grid%size(z_index) + current_state%local_grid%halo_size(z_index) * 2
112  y_size = current_state%local_grid%size(y_index) + current_state%local_grid%halo_size(y_index) * 2
113  x_size = current_state%local_grid%size(x_index) + current_state%local_grid%halo_size(x_index) * 2
114 
115  if (flood_q) current_state%number_q_fields=current_state%number_q_fields+1
116  if (float_q) current_state%number_q_fields=current_state%number_q_fields+1
117  allocate(current_state%q(current_state%number_q_fields), current_state%zq(current_state%number_q_fields), &
118  current_state%sq(current_state%number_q_fields))
119  do i=1,current_state%number_q_fields
120  call initialise_single_q_field(current_state, i, z_size, y_size, x_size)
121  if (flood_q .and. i == 1) current_state%q(i)%data(:,:,:) = 1.
122  if (float_q .and. (.not. flood_q .or. i==2)) then
123  call populate_q_tracer(current_state, current_state%q(i))
124  end if
125  end do
126  end subroutine set_up_q_fields
127 
131  subroutine populate_q_tracer(current_state, q_field)
132  type(model_state_type), intent(inout) :: current_state
133  type(prognostic_field_type), intent(inout) :: q_field
134 
135  integer :: i, x_local, y_local
136 
137  do i=1,number_q_coords
138  if (q_coordinates_x(i) .ne. -1 .and. q_coordinates_y(i) .ne. -1) then
139  if (q_coordinates_x(i) .ne. 0 .and. .not. (q_coordinates_x(i) .ge. current_state%local_grid%start(x_index) .and. &
140  q_coordinates_x(i) .le. current_state%local_grid%end(x_index))) cycle
141  if (q_coordinates_y(i) .ne. 0 .and. .not. (q_coordinates_y(i) .ge. current_state%local_grid%start(y_index) .and. &
142  q_coordinates_y(i) .le. current_state%local_grid%end(y_index))) cycle
143  x_local=(q_coordinates_x(i) - (current_state%local_grid%start(x_index)-1)) + current_state%local_grid%halo_size(x_index)
144  y_local=(q_coordinates_y(i) - (current_state%local_grid%start(y_index)-1)) + current_state%local_grid%halo_size(y_index)
145  if (q_coordinates_z(i) == 0 .and. q_coordinates_y(i) == 0 .and. q_coordinates_x(i) == 0) then
146  q_field%data(:,current_state%local_grid%local_domain_start_index(y_index):&
147  current_state%local_grid%local_domain_end_index(y_index),current_state%local_grid%local_domain_start_index(x_index):&
148  current_state%local_grid%local_domain_end_index(x_index)) = q_coordinates_value(i)
149  else if (q_coordinates_z(i) == 0 .and. q_coordinates_y(i) == 0 .and. q_coordinates_x(i) .ne. 0) then
150  q_field%data(:,current_state%local_grid%local_domain_start_index(y_index):&
151  current_state%local_grid%local_domain_end_index(y_index),x_local) = q_coordinates_value(i)
152  else if (q_coordinates_z(i) == 0 .and. q_coordinates_y(i) .ne. 0 .and. q_coordinates_x(i) == 0) then
153  q_field%data(:,y_local,current_state%local_grid%local_domain_start_index(x_index):&
154  current_state%local_grid%local_domain_end_index(x_index)) = q_coordinates_value(i)
155  else if (q_coordinates_z(i) == 0 .and. q_coordinates_y(i) .ne. 0 .and. q_coordinates_x(i) .ne. 0) then
156  q_field%data(:,y_local,x_local) = q_coordinates_value(i)
157  else if (q_coordinates_z(i) .ne. 0 .and. q_coordinates_y(i) == 0 .and. q_coordinates_x(i) == 0) then
158  q_field%data(q_coordinates_z(i),current_state%local_grid%local_domain_start_index(y_index):&
159  current_state%local_grid%local_domain_end_index(y_index),current_state%local_grid%local_domain_start_index(x_index):&
160  current_state%local_grid%local_domain_end_index(x_index)) = q_coordinates_value(i)
161  else if (q_coordinates_z(i) .ne. 0 .and. q_coordinates_y(i) == 0 .and. q_coordinates_x(i) .ne. 0) then
162  q_field%data(q_coordinates_z(i),current_state%local_grid%local_domain_start_index(y_index):&
163  current_state%local_grid%local_domain_end_index(y_index),x_local) = q_coordinates_value(i)
164  else if (q_coordinates_z(i) .ne. 0 .and. q_coordinates_y(i) .ne. 0 .and. q_coordinates_x(i) == 0) then
165  q_field%data(q_coordinates_z(i),y_local,current_state%local_grid%local_domain_start_index(x_index):&
166  current_state%local_grid%local_domain_end_index(x_index)) = q_coordinates_value(i)
167  else if (q_coordinates_z(i) .ne. 0 .and. q_coordinates_y(i) .ne. 0 .and. q_coordinates_x(i) .ne. 0) then
168  q_field%data(q_coordinates_z(i),y_local,x_local) = q_coordinates_value(i)
169  end if
170  end if
171  end do
172  end subroutine populate_q_tracer
173 
174  subroutine initialise_single_q_field(current_state, q_id, z_size, y_size, x_size)
175  type(model_state_type), intent(inout) :: current_state
176  integer, intent(in) :: q_id, x_size, y_size, z_size
177 
178  allocate(current_state%q(q_id)%data(z_size, y_size, x_size), current_state%zq(q_id)%data(z_size, y_size, x_size), &
179  current_state%sq(q_id)%data(z_size, y_size, x_size))
180  current_state%q(q_id)%data(:,:,:) = 0.
181  current_state%zq(q_id)%data(:,:,:) = 0.
182  current_state%sq(q_id)%data(:,:,:) = 0.
183  current_state%q(q_id)%active=.true.
184  current_state%zq(q_id)%active=.true.
185  current_state%sq(q_id)%active=.true.
186  end subroutine initialise_single_q_field
187 
191  subroutine decompose_grid(current_state)
192  type(model_state_type), intent(inout) :: current_state
193 
194  if (associated(current_state%parallel%decomposition_procedure)) then
195  call current_state%parallel%decomposition_procedure(current_state)
196  else
197  call log_master_log(log_error, "No decomposition specified")
198  end if
199  end subroutine decompose_grid
200 
203  subroutine initalise_source_and_z_fields(current_state)
204  type(model_state_type), intent(inout) :: current_state
205 
206  integer :: x_size, y_size, z_size
207 
208  z_size = current_state%local_grid%size(z_index) + current_state%local_grid%halo_size(z_index) * 2
209  y_size = current_state%local_grid%size(y_index) + current_state%local_grid%halo_size(y_index) * 2
210  x_size = current_state%local_grid%size(x_index) + current_state%local_grid%halo_size(x_index) * 2
211 
212 #ifdef U_ACTIVE
213  allocate(current_state%zu%data(z_size, y_size, x_size))
214  allocate(current_state%su%data(z_size, y_size, x_size))
215  allocate(current_state%savu%data(z_size, y_size, x_size))
216  current_state%zu%data(:,:,:)= 0.0
217 #endif
218 #ifdef V_ACTIVE
219  allocate(current_state%zv%data(z_size, y_size, x_size))
220  allocate(current_state%sv%data(z_size, y_size, x_size))
221  allocate(current_state%savv%data(z_size, y_size, x_size))
222  current_state%zv%data(:,:,:)= 0.0
223 #endif
224 #ifdef W_ACTIVE
225  allocate(current_state%zw%data(z_size, y_size, x_size))
226  allocate(current_state%sw%data(z_size, y_size, x_size))
227  allocate(current_state%savw%data(z_size, y_size, x_size))
228  current_state%zw%data(:,:,:)= 0.0
229 #endif
230  if (current_state%th%active) then
231  allocate(current_state%zth%data(z_size, y_size, x_size))
232  allocate(current_state%sth%data(z_size, y_size, x_size))
233  current_state%zth%data(:,:,:)= 0.0
234  end if
235  end subroutine initalise_source_and_z_fields
236 
244  subroutine initialise_velocity_field(local_grid, field, z_grid, y_grid, x_grid, data)
245  type(local_grid_type), intent(inout) :: local_grid
246  type(prognostic_field_type), intent(inout) :: field
247  integer, intent(in) :: z_grid, y_grid, x_grid
248  real, dimension(:,:,:), allocatable, intent(in) :: data
249 
250  integer :: i, j, k, preMulSizeY, preMulSizeX
251 
252  field%grid(z_index) = z_grid
253  field%grid(y_index) = y_grid
254  field%grid(x_index) = x_grid
255  field%active = .true.
256 
257  allocate(field%data(local_grid%size(z_index) + local_grid%halo_size(z_index) * 2, local_grid%size(y_index) + &
258  local_grid%halo_size(y_index) * 2, local_grid%size(x_index) + local_grid%halo_size(x_index) * 2))
259  field%data=0.0
260 
261  ! Divisions here are for the multiplication - we just fill up the original size and then duplicate this across dimensions
262  do i=ceiling(local_grid%start(x_index)/real(domain_multiplication)),&
263  ceiling(local_grid%end(x_index)/real(domain_multiplication))
264  do j=ceiling(local_grid%start(y_index)/real(domain_multiplication)),&
265  ceiling(local_grid%end(y_index)/real(domain_multiplication))
266  do k=local_grid%start(z_index),local_grid%end(z_index)
267  field%data(local_grid%halo_size(z_index)+(k-((local_grid%start(z_index)-1))), local_grid%halo_size(y_index)+&
268  (j-((ceiling(local_grid%start(y_index)/ real(domain_multiplication))-1))), local_grid%halo_size(x_index)+&
269  (i- ((ceiling(local_grid%start(x_index)/real(domain_multiplication))-1)))) = &
270  real(data(1, k, merge(j, i, rotate_xy)), kind=default_precision)
271  end do
272  end do
273  end do
274 
275  if (domain_multiplication .ge. 2) then
276  ! If there is a domain multiplication then duplicate data across dimensions
277  premulsizey = ceiling(local_grid%size(y_index) / real(domain_multiplication))
278  premulsizex = ceiling(local_grid%size(x_index) / real(domain_multiplication))
279  if (local_grid%active(y_index)) then
280  do i=1,domain_multiplication-1
281  field%data(:, local_grid%halo_size(y_index) + i*premulsizey +1 : local_grid%halo_size(y_index) + (i+1)*premulsizey, &
282  local_grid%halo_size(x_index)+1: local_grid%halo_size(x_index)+premulsizex) = &
283  field%data(:, local_grid%halo_size(y_index)+1 : local_grid%halo_size(y_index) + premulsizey, &
284  local_grid%halo_size(x_index)+1 : local_grid%halo_size(x_index)+premulsizex)
285  end do
286  end if
287  if (local_grid%active(x_index)) then
288  do i=1,domain_multiplication-1
289  field%data(:, local_grid%local_domain_start_index(y_index) : local_grid%local_domain_end_index(y_index), &
290  local_grid%halo_size(x_index) + i*premulsizex + 1 : local_grid%halo_size(x_index) + (i+1)*premulsizex) = &
291  field%data(:, local_grid%local_domain_start_index(y_index) : local_grid%local_domain_end_index(y_index), &
292  local_grid%halo_size(x_index)+1 : local_grid%halo_size(x_index)+premulsizex)
293  end do
294  end if
295  end if
296  end subroutine initialise_velocity_field
297 
304  subroutine create_grid(specific_grid, z, x, z_dim, x_dim, my_rank)
305  type(global_grid_type), intent(inout) :: specific_grid
306  integer, intent(in) :: z_dim, x_dim, my_rank
307  real, dimension(:), intent(in) :: z, x
308 
309  specific_grid%bottom(z_index) = int(z(1))
310  specific_grid%bottom(merge(y_index, x_index, rotate_xy)) = int(x(1))
311  if (clone_to_3d) specific_grid%bottom(y_index) = int(x(1))
312 
313  specific_grid%top(z_index) = int(z(z_dim))
314  specific_grid%top(merge(y_index, x_index, rotate_xy)) = int(x(x_dim)) * domain_multiplication
315  if (clone_to_3d) specific_grid%top(y_index) = int(x(x_dim)) * domain_multiplication
316 
317  specific_grid%resolution(z_index) = int(z(2) - z(1))
318  specific_grid%resolution(merge(y_index, x_index, rotate_xy)) = int(x(2) - x(1))
319  if (clone_to_3d) specific_grid%resolution(y_index) = specific_grid%resolution(x_index)
320 
321  specific_grid%size(z_index) = z_dim
322  specific_grid%size(merge(y_index, x_index, rotate_xy)) = x_dim * domain_multiplication
323  if (clone_to_3d) specific_grid%size(y_index) = x_dim * domain_multiplication
324 
325  specific_grid%active(z_index) = .true.
326  specific_grid%active(merge(y_index, x_index, rotate_xy)) = .true.
327  if (clone_to_3d) specific_grid%active(y_index) = .true.
328 
329 #ifdef U_ACTIVE
330  if (.not. specific_grid%active(x_index)) call log_master_log(log_error, &
331  "Model compiled with X active but inactive in configuration")
332 #else
333  if (specific_grid%active(x_index)) call log_master_log(log_error, &
334  "Model compiled with X inactive but active in configuration")
335 #endif
336 #ifdef V_ACTIVE
337  if (.not. specific_grid%active(y_index)) call log_master_log(log_error, &
338  "Model compiled with Y active but inactive in configuration")
339 #else
340  if (specific_grid%active(y_index)) call log_master_log(log_error, &
341  "Model compiled with Y inactive but active in configuration")
342 #endif
343 #ifdef W_ACTIVE
344  if (.not. specific_grid%active(z_index)) call log_master_log(log_error, &
345  "Model compiled with Z active but inactive in configuration")
346 #else
347  if (specific_grid%active(z_index)) call log_master_log(log_error, &
348  "Model compiled with Z inactive but active in configuration")
349 #endif
350  specific_grid%dimensions = merge(3, 2, clone_to_3d)
351  end subroutine create_grid
352 
358  subroutine define_vertical_levels(current_state, z, z_size)
359  type(model_state_type), intent(inout) :: current_state
360  integer, intent(in) :: z_size
361  real, dimension(:), intent(in) :: z
362 
363  integer :: i
364 
365  allocate(current_state%global_grid%configuration%vertical%kgd(z_size), &
366  current_state%global_grid%configuration%vertical%hgd(z_size))
367 
368  do i=1,z_size
369  current_state%global_grid%configuration%vertical%kgd(i) = i
370  current_state%global_grid%configuration%vertical%hgd(i) = real(z(i))
371  end do
372  end subroutine define_vertical_levels
373 
378  subroutine check_kinematics_file(ncid)
379  integer, intent(in) :: ncid
380 
381  integer :: ndims_in, nvars_in, ngatts_in, unlimdimid_in
382 
383  call check_status(nf90_inquire(ncid, ndims_in, nvars_in, ngatts_in, unlimdimid_in))
384  if (ndims_in /= 5) call log_log(log_error, "NetCDF KiD number of model dimensions must equal 5")
385  if (nvars_in /= 7) call log_log(log_error, "NetCDF KiD number of model variables must equal 5")
386  if (ngatts_in .le. 0) call log_log(log_error, "NetCDF KiD global attributes must be specified")
387  if (unlimdimid_in .gt. 0) call log_log(log_error, "NetCDF KiD model number of unlimited dimensions must be 0")
388  end subroutine check_kinematics_file
389 
404  subroutine read_variables(ncid, time_dim, z_dim, z_half_dim, x_dim, x_half_dim, time, x, z, x_half, z_half, u, w, v)
405  integer, intent(in) :: ncid, time_dim, z_dim, z_half_dim, x_dim, x_half_dim
406  real, dimension(:), allocatable, intent(inout) :: time, x, z, x_half, z_half
407  real, dimension(:,:,:), allocatable, intent(inout) :: u, w, v
408 
409  allocate(time(time_dim))
410  allocate(z(z_dim))
411  allocate(x(x_dim))
412  allocate(z_half(z_half_dim))
413  allocate(x_half(x_half_dim))
414  ! Due to NetCDF being in C, need to reverse the data order for F2003
415  if (rotate_xy) then
416  allocate(v(time_dim, z_dim, x_half_dim))
417  else
418  allocate(u(time_dim, z_dim, x_half_dim))
419  end if
420  allocate(w(time_dim, z_half_dim, x_half_dim))
421 
422  call read_single_variable(ncid, time_key, data1d=time)
423  call read_single_variable(ncid, z_key, data1d=z)
424  call read_single_variable(ncid, z_half_key, data1d=z_half)
425  call read_single_variable(ncid, x_key, data1d=x)
426  call read_single_variable(ncid, x_half_key, data1d=x_half)
427  if (rotate_xy) then
428  call read_single_variable(ncid, u_key, data3d=v)
429  else
430  call read_single_variable(ncid, u_key, data3d=u)
431  end if
432  call read_single_variable(ncid, w_key, data3d=w)
433 
434  if (clone_to_3d) then
435  if (.not. allocated(v)) then
436  allocate(v(time_dim, z_dim, x_half_dim))
437  v=u
438  end if
439  if (.not. allocated(u)) then
440  allocate(u(time_dim, z_dim, x_half_dim))
441  u=v
442  end if
443  end if
444  end subroutine read_variables
445 
451  subroutine read_single_variable(ncid, key, data1d, data3d)
452  integer, intent(in) :: ncid
453  character(len=*), intent(in) :: key
454  real, dimension(:), intent(inout), optional :: data1d
455  real, dimension(:,:,:), intent(inout), optional :: data3d
456 
457  integer :: variable_id
458  real, dimension(:,:,:), allocatable :: sdata
459 
460  call check_status(nf90_inq_varid(ncid, key, variable_id))
461 
462  if (.not. present(data1d) .and. .not. present(data3d)) return
463 
464  if (present(data1d)) then
465  call check_status(nf90_get_var(ncid, variable_id, data1d))
466  else
467  ! 3D will reshape the data to take account of the column-row major C-F transposition
468  allocate(sdata(size(data3d,1),size(data3d,3), size(data3d,2)))
469  call check_status(nf90_get_var(ncid, variable_id, sdata))
470  data3d(:,:,:)=reshape(sdata(:,:,:),(/size(data3d,1),size(data3d,2),size(data3d,3)/))
471  deallocate(sdata)
472  end if
473  end subroutine read_single_variable
474 
482  subroutine read_dimensions(ncid, time_dim, z_dim, z_half_dim, x_dim, x_half_dim)
483  integer, intent(in) :: ncid
484  integer, intent(out) :: time_dim, z_dim, z_half_dim, x_dim, x_half_dim
485 
486  integer :: time_dimid, z_dimid, z_half_dimid, x_dimid, x_half_dimid
487 
488  call check_status(nf90_inq_dimid(ncid, time_key, time_dimid))
489  call check_status(nf90_inq_dimid(ncid, z_key, z_dimid))
490  call check_status(nf90_inq_dimid(ncid, z_half_key, z_half_dimid))
491  call check_status(nf90_inq_dimid(ncid, x_key, x_dimid))
492  call check_status(nf90_inq_dimid(ncid, x_half_key, x_half_dimid))
493 
494  call check_status(nf90_inquire_dimension(ncid, time_dimid, len=time_dim))
495  call check_status(nf90_inquire_dimension(ncid, z_dimid, len=z_dim))
496  call check_status(nf90_inquire_dimension(ncid, z_half_dimid, len=z_half_dim))
497  call check_status(nf90_inquire_dimension(ncid, x_dimid, len=x_dim))
498  call check_status(nf90_inquire_dimension(ncid, x_half_dimid, len=x_half_dim))
499  end subroutine read_dimensions
500 
503  subroutine read_global_attributes(ncid, pid)
504  integer, intent(in) :: ncid, pid
505 
506  character(len=:), allocatable :: attributeValue
507 
508  attributevalue=read_specific_global_attribute(ncid, "title")
509  call log_master_log(log_debug, "KiD file title: "//attributevalue)
510  deallocate(attributevalue)
511 
512  attributevalue=read_specific_global_attribute(ncid, "creation date")
513  call log_master_log(log_debug, "KiD file created: "//attributevalue)
514  deallocate(attributevalue)
515  end subroutine read_global_attributes
516 
521  function read_specific_global_attribute(ncid, key)
522  integer, intent(in) :: ncid
523  character(len=*), intent(in) :: key
524 
525  integer :: length
526  character(len=:),allocatable,target :: read_specific_global_attribute
527 
528  call check_status(nf90_inquire_attribute(ncid, nf90_global, key, len = length))
529  allocate(character(length) :: read_specific_global_attribute)
530  call check_status(nf90_get_att(ncid, nf90_global, key, read_specific_global_attribute))
531  end function read_specific_global_attribute
532 
535  subroutine check_status(status)
536  integer, intent(in) :: status
537 
538  if (status /= nf90_noerr) then
539  call log_log(log_error, "NetCDF returned error code of "//trim(nf90_strerror(status)))
540  end if
541  end subroutine check_status
542 end module kidreader_mod
integer, dimension(number_q_coords), save q_coordinates_z
Definition: kidreader.F90:35
subroutine read_variables(ncid, time_dim, z_dim, z_half_dim, x_dim, x_half_dim, time, x, z, x_half, z_half, u, w, v)
Reads the variables from the NetCDF KiD model file.
Definition: kidreader.F90:405
subroutine set_up_q_fields(current_state)
Definition: kidreader.F90:107
integer, parameter number_q_coords
Number of Q field value coords that can be specified.
Definition: kidreader.F90:22
character(len= *), parameter x_key
Corresponding NetCDF data x (primal grid) key.
Definition: kidreader.F90:23
character(len= *), parameter w_key
Corresponding NetCDF data w flow field key.
Definition: kidreader.F90:23
integer, parameter, public dual_grid
Definition: grids.F90:16
integer, dimension(number_q_coords), save q_coordinates_x
Definition: kidreader.F90:35
subroutine populate_q_tracer(current_state, q_field)
Populates the Q tracer field based upon the configuration that has been read in from the simulation f...
Definition: kidreader.F90:132
integer, parameter, public log_error
Only log ERROR messages.
Definition: logging.F90:11
character(len=string_length) function, public options_get_string(options_database, key, index)
Retrieves a string value from the database that matches the provided key.
Contains prognostic field definitions and functions.
Definition: prognostics.F90:2
subroutine read_dimensions(ncid, time_dim, z_dim, z_half_dim, x_dim, x_half_dim)
Reads the dimensions from the NetCDF file.
Definition: kidreader.F90:483
A prognostic field which is assumed to be 3D.
Definition: prognostics.F90:13
Logging utility.
Definition: logging.F90:2
integer, parameter, public default_precision
MPI communication type which we use for the prognostic and calculation data.
Definition: datadefn.F90:17
integer, parameter, public z_index
Grid index parameters.
Definition: grids.F90:14
type(component_descriptor_type) function, public kidreader_get_descriptor()
Provides the descriptor back to the caller and is used in component registration. ...
Definition: kidreader.F90:46
character(len= *), parameter time_key
Corresponding NetCDF data time key.
Definition: kidreader.F90:23
Contains common definitions for the data and datatypes used by MONC.
Definition: datadefn.F90:2
The ModelState which represents the current state of a run.
Definition: state.F90:39
character(len= *), parameter x_half_key
Corresponding NetCDF data x half (primal grid) key.
Definition: kidreader.F90:23
subroutine, public log_master_log(level, message)
Will log just from the master process.
Definition: logging.F90:47
integer, parameter, public log_debug
Log DEBUG, INFO, WARNING and ERROR messages.
Definition: logging.F90:14
subroutine, public options_get_integer_array(options_database, key, array_data, from, to)
Retrieves an entire (or subset) integer array.
integer domain_multiplication
The Q field coordinates configured by the user.
Definition: kidreader.F90:32
Component to set up the model based upon a KiD model configuration.
Definition: kidreader.F90:4
character(len=:) function, allocatable, target read_specific_global_attribute(ncid, key)
Will read a specific global NetCDF attribute.
Definition: kidreader.F90:522
subroutine, public log_log(level, message, str)
Logs a message at the specified level. If the level is above the current level then the message is ig...
Definition: logging.F90:75
Map data structure that holds string (length 20 maximum) key value pairs.
Definition: collections.F90:86
subroutine check_status(status)
Will check a NetCDF status and write to log_log error any decoded statuses.
Definition: kidreader.F90:536
Defines the global grid.
Definition: grids.F90:100
Interfaces and types that MONC components must specify.
character(len= *), parameter u_key
Corresponding NetCDF data u flow field key.
Definition: kidreader.F90:23
subroutine define_vertical_levels(current_state, z, z_size)
Defines the vertical levels of the grid. This is both the grid points and corresponding height for ea...
Definition: kidreader.F90:359
subroutine initalise_source_and_z_fields(current_state)
Based upon the local grid this will initialise the Source, Z and SAV fields for each prognostic...
Definition: kidreader.F90:204
Defined the local grid, i.e. the grid held on this process after decomposition.
Definition: grids.F90:111
character(len= *), parameter z_half_key
Corresponding NetCDF data z half (dual grid) key.
Definition: kidreader.F90:23
Collection data structures.
Definition: collections.F90:7
integer, parameter, public log_warn
Log WARNING and ERROR messages.
Definition: logging.F90:12
integer, parameter, public string_length
Default length of strings.
Definition: datadefn.F90:10
subroutine read_single_variable(ncid, key, data1d, data3d)
Reads a single variable out of a NetCDF file.
Definition: kidreader.F90:452
subroutine check_kinematics_file(ncid)
Checks that the kinematics file that has been loaded is consistent with what we expect.
Definition: kidreader.F90:379
integer, dimension(number_q_coords), save q_coordinates_value
Definition: kidreader.F90:35
logical rotate_xy
Definition: kidreader.F90:31
integer, parameter, public primal_grid
Grid type parameters (usually applied to each dimension of a prognostic)
Definition: grids.F90:16
real(kind=default_precision) function, public options_get_real(options_database, key, index)
Retrieves a real value from the database that matches the provided key.
integer, dimension(number_q_coords), save q_coordinates_y
Definition: kidreader.F90:35
Functionality to support the different types of grid and abstraction between global grids and local o...
Definition: grids.F90:5
integer function, public options_get_integer(options_database, key, index)
Retrieves an integer value from the database that matches the provided key.
Manages the options database. Contains administration functions and deduce runtime options from the c...
integer, parameter, public log_info
Log INFO, WARNING and ERROR messages.
Definition: logging.F90:13
logical function, public options_get_logical(options_database, key, index)
Retrieves a logical value from the database that matches the provided key.
subroutine initialise_callback(current_state)
Initialisation hook which will parse the configuration NetCDF file and set up the model based upon th...
Definition: kidreader.F90:55
logical clone_to_3d
Definition: kidreader.F90:31
subroutine initialise_velocity_field(local_grid, field, z_grid, y_grid, x_grid, data)
Will initialise a velocity field with the loaded data.
Definition: kidreader.F90:245
subroutine create_grid(specific_grid, z, x, z_dim, x_dim, my_rank)
Creates a specific grid based upon the data read from the KiD model NetCDF file.
Definition: kidreader.F90:305
logical flood_q
Definition: kidreader.F90:31
logical function, public options_has_key(options_database, key)
Determines whether a specific key is in the database.
The model state which represents the current state of a run.
Definition: state.F90:2
integer function, public log_get_logging_level()
Retrieves the current logging level.
Definition: logging.F90:122
integer, parameter, public y_index
Definition: grids.F90:14
subroutine initialise_single_q_field(current_state, q_id, z_size, y_size, x_size)
Definition: kidreader.F90:175
logical float_q
Definition: kidreader.F90:31
character(len=string_length) configuration_file
NetCDF model file to load.
Definition: kidreader.F90:37
integer, parameter, public x_index
Definition: grids.F90:14
subroutine read_global_attributes(ncid, pid)
Will read the global attributes of the NetCDF KiD model dump and log_log them to debug.
Definition: kidreader.F90:504
subroutine decompose_grid(current_state)
Calls the decomposition procedure to decompose the grid and determine neighbouring processes If no de...
Definition: kidreader.F90:192
character(len= *), parameter z_key
Corresponding NetCDF data z (primal grid) key.
Definition: kidreader.F90:23