OOPS
interpolatorunstrc_interface.F90
Go to the documentation of this file.
1 ! (C) Copyright 2020- UCAR
2 !
3 ! This software is licensed under the terms of the Apache Licence Version 2.0
4 ! which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
5 
7 
8 use atlas_module
9 use fckit_configuration_module, only: fckit_configuration
10 use fckit_log_module, only: fckit_log
11 use fckit_mpi_module, only: fckit_mpi_comm
12 use iso_c_binding
13 use kinds
14 use missing_values_mod
16 
17 private
18 ! ------------------------------------------------------------------------------
19 contains
20 !-------------------------------------------------------------------------------
21 !> Create Unstructured Interpolator from lonlat
22 !!
23 subroutine unstrc_create_c(c_key_unstrc, c_comm, c_lonlat1, &
24  c_lonlat2, c_config) bind(c, name='unstrc_create_f90')
25 implicit none
26 
27 ! Passed variables
28 integer(c_int), intent(inout) :: c_key_unstrc !< bump interpolator
29 type(c_ptr), value, intent(in) :: c_comm !< MPI Communicator
30 type(c_ptr), intent(in),value :: c_lonlat1 !< source grid (atlas field)
31 type(c_ptr), intent(in),value :: c_lonlat2 !< target grid (atlas field)
32 type(c_ptr), value, intent(in) :: c_config !< Configuration
33 
34 ! local variables
35 type(unstrc_interp), pointer :: unstrc_int
36 type(fckit_mpi_comm) :: f_comm
37 type(fckit_configuration) :: f_config
38 type(atlas_field) :: lonlat_field1, lonlat_field2
39 real(kind_real), pointer :: lonlat1(:,:), lonlat2(:,:)
40 real(kind_real), allocatable :: lon1(:), lat1(:)
41 real(kind_real), allocatable :: lon2(:), lat2(:)
42 integer :: ngrid1, ngrid2, nnearest
43 character(len=:), allocatable :: us_interp_type, infile
44 logical :: read_from_file
45 
46 f_comm = fckit_mpi_comm(c_comm)
47 f_config = fckit_configuration(c_config)
48 
49 ! check for specification of weight types and number of
50 ! neighbors in the configuration. If absent, implement defaults
51 
52 call unstrc_interp_registry%init()
53 call unstrc_interp_registry%add(c_key_unstrc)
54 call unstrc_interp_registry%get(c_key_unstrc, unstrc_int)
55 
56 ! This may need to be revised for multiple MPI tasks
57 lonlat_field1 = atlas_field(c_lonlat1)
58 call lonlat_field1%data(lonlat1)
59 ngrid1 = size(lonlat1(1,:))
60 
61 lonlat_field2 = atlas_field(c_lonlat2)
62 call lonlat_field2%data(lonlat2)
63 ngrid2 = size(lonlat2(1,:))
64 
65 if (f_config%has("read_from_file")) then
66  call f_config%get_or_die("read_from_file", read_from_file)
67 else
68  read_from_file = .false.
69 endif
70 
71 if (read_from_file) then
72  call fckit_log%info("Reading interpolator from file")
73  call f_config%get_or_die("infile", infile)
74  call unstrc_int%create(f_comm, infile)
75 
76  if ((unstrc_int%ngrid_in /= ngrid1) .or. (unstrc_int%ngrid_out /= ngrid2)) &
77  call abor1_ftn('unstructured interpolator input file is not consistent with input and output functionspaces')
78 
79 else
80 
81  if (f_config%has("us_interp_type")) then
82  call f_config%get_or_die("interpolation_type", us_interp_type)
83  else
84  us_interp_type = 'barycent'
85  endif
86 
87  if (f_config%has("nnearest")) then
88  call f_config%get_or_die("nnearest", nnearest)
89  else
90  nnearest = 4
91  endif
92 
93  ! For now copy the atlas fields into an unstructured grid
94  ! in the future we can make this more efficient
95  allocate(lon1(ngrid1), lat1(ngrid1))
96  lon1 = lonlat1(1,:)
97  lat1 = lonlat1(2,:)
98 
99  allocate(lon2(ngrid2), lat2(ngrid2))
100  lon2 = lonlat2(1,:)
101  lat2 = lonlat2(2,:)
102 
103  call unstrc_int%create(f_comm, nnearest, us_interp_type, ngrid1, lat1, lon1, &
104  ngrid2, lat2, lon2)
105 
106  deallocate(lon1, lat1, lon2, lat2)
107 endif
108 
109 ! release pointers
110 call lonlat_field1%final()
111 call lonlat_field2%final()
112 
113 end subroutine unstrc_create_c
114 
115 !-------------------------------------------------------------------------------
116 !> Write Unstructured Interpolator to a file
117 !!
118 subroutine unstrc_write_c(c_key_unstrc, c_config) bind(c, name='unstrc_write_f90')
119 implicit none
120 
121 ! Passed variables
122 integer(c_int), intent(inout) :: c_key_unstrc !< unstructured interpolator
123 type(c_ptr), value, intent(in) :: c_config !< Configuration
124 
125 ! local variables
126 type(unstrc_interp), pointer :: unstrc_int
127 type(fckit_configuration) :: f_config
128 character(len=:), allocatable :: outfile
129 
130 f_config = fckit_configuration(c_config)
131 
132 ! read name of output file from configuration
133 call f_config%get_or_die("outfile",outfile)
134 
135 call unstrc_interp_registry%get(c_key_unstrc, unstrc_int)
136 call unstrc_int%write(outfile)
137 
138 end subroutine unstrc_write_c
139 
140 !-------------------------------------------------------------------------------
141 !> Apply Unstructured Interpolator
142 !!
143 subroutine unstrc_apply_c(c_key_unstrc, c_infield, c_outfield) bind(c, name='unstrc_apply_f90')
144 
145 implicit none
146 
147 ! Passed variables
148 integer(c_int), intent(in) :: c_key_unstrc !< key to unstructured interpolator
149 type(c_ptr), intent(in), value :: c_infield !< input field
150 type(c_ptr), intent(in), value :: c_outfield !< output field
151 
152 ! Local variables
153 type(unstrc_interp), pointer :: unstrc_int
154 type(atlas_field) :: infield, outfield
155 real(kind_real), pointer :: fin(:,:), fout(:,:)
156 integer :: jlev
157 
158 call unstrc_interp_registry%get(c_key_unstrc, unstrc_int)
159 
160 infield = atlas_field(c_infield)
161 outfield = atlas_field(c_outfield)
162 
163 call infield%data(fin)
164 call outfield%data(fout)
165 
166 do jlev = 1, infield%levels()
167  call unstrc_int%apply(fin(jlev,:), fout(jlev,:))
168 enddo
169 
170 ! free up pointers and arrays
171 call infield%final()
172 call outfield%final()
173 
174 end subroutine unstrc_apply_c
175 
176 !-------------------------------------------------------------------------------
177 !> Apply Unstructured Interpolator Adjoint
178 !!
179 subroutine unstrc_apply_ad_c(c_key_unstrc, c_field2, c_field1) bind(c, name='unstrc_apply_ad_f90')
180 
181 implicit none
182 
183 ! Passed variables
184 integer(c_int), intent(in) :: c_key_unstrc !< key to unstructured interpolator
185 type(c_ptr), intent(in), value :: c_field2 !< input field
186 type(c_ptr), intent(in), value :: c_field1 !< output field
187 
188 ! Local variables
189 type(unstrc_interp), pointer :: unstrc_int
190 type(atlas_field) :: field_grid2, field_grid1
191 real(kind_real), pointer :: f2(:,:), f1(:,:)
192 integer :: jlev
193 
194 call unstrc_interp_registry%get(c_key_unstrc, unstrc_int)
195 
196 field_grid2 = atlas_field(c_field2)
197 field_grid1 = atlas_field(c_field1)
198 
199 call field_grid2%data(f2)
200 call field_grid1%data(f1)
201 
202 do jlev = 1, field_grid2%levels()
203  call unstrc_int%apply_ad(f1(jlev,:), f2(jlev,:))
204 enddo
205 
206 ! free up pointers
207 call field_grid2%final()
208 call field_grid1%final()
209 
210 end subroutine unstrc_apply_ad_c
211 
212 !------------------------------------------------------------------------------
213 !> Delete unstructured_interpolator
214 subroutine unstrc_delete_c(c_key_unstrc) bind(c, name='unstrc_delete_f90')
215 
216 implicit none
217 
218 ! Passed variables
219 integer(c_int), intent(inout) :: c_key_unstrc
220 
221 ! Local variables
222 type(unstrc_interp), pointer :: unstrc_int
223 
224 call unstrc_interp_registry%get(c_key_unstrc, unstrc_int)
225 
226 call unstrc_int%delete()
227 
228 call unstrc_interp_registry%remove(c_key_unstrc)
229 
230 end subroutine unstrc_delete_c
231 
232 ! ------------------------------------------------------------------------------
233 
subroutine unstrc_apply_ad_c(c_key_unstrc, c_field2, c_field1)
Apply Unstructured Interpolator Adjoint.
subroutine unstrc_create_c(c_key_unstrc, c_comm, c_lonlat1, c_lonlat2, c_config)
Create Unstructured Interpolator from lonlat.
subroutine unstrc_delete_c(c_key_unstrc)
Delete unstructured_interpolator.
subroutine unstrc_apply_c(c_key_unstrc, c_infield, c_outfield)
Apply Unstructured Interpolator.
subroutine unstrc_write_c(c_key_unstrc, c_config)
Write Unstructured Interpolator to a file.
type(registry_t), public unstrc_interp_registry
Registry for unstrc_interpo objects.