14 use missing_values_mod
32 type(c_ptr),
intent(in) :: odb
33 integer(c_int),
intent(in) :: nn
34 real(c_double),
intent(inout) :: zz
41 real(kind_real),
allocatable :: values(:,:)
42 real(kind_real) :: missing
45 #define LISTED_TYPE qg_obsvec
48 #include "oops/util/linkedList_i.f"
58 #include "oops/util/linkedList_c.f"
67 integer,
intent(in) :: nlev
68 integer,
intent(in) :: nobs
75 if (
allocated(self%values))
deallocate(self%values)
78 allocate(self%values(self%nlev,self%nobs))
81 self%values = 0.0_kind_real
82 self%missing = missing_value(self%missing)
96 self%nlev = other%nlev
97 self%nobs = other%nobs
98 self%missing = other%missing
101 allocate(self%values(self%nlev,self%nobs))
114 deallocate(self%values)
127 if ((other%nlev/=self%nlev).or.(other%nobs/=self%nobs))
then
129 deallocate(self%values)
132 self%nlev = other%nlev
133 self%nobs = other%nobs
134 self%missing = other%missing
137 allocate(self%values(self%nlev,self%nobs))
141 self%values = other%values
165 integer,
intent(in) :: i
168 self%values(:,i) = self%missing
193 if ((self%nobs/=mask%nobs).or.(self%nlev/=mask%nlev))
call abor1_ftn(
'qg_obsvec_mask: inconsistent sizes')
195 where(mask%values == 1) self%values = self%missing
207 if ((self%nobs/=mask%nobs).or.(self%nlev/=mask%nlev))
call abor1_ftn(
'qg_obsvec_mask: inconsistent sizes')
209 where(mask%values == mask%missing) self%values = self%missing
220 real(kind_real),
intent(in) :: zz
223 where(self%values /= self%missing) self%values = zz*self%values
236 if ((self%nobs/=other%nobs).or.(self%nlev/=other%nlev))
call abor1_ftn(
'qg_obsvec_add: inconsistent sizes')
239 where(self%values /= self%missing .and. other%values /= other%missing)
240 self%values = self%values+other%values
242 self%values = self%missing
256 if ((self%nobs/=other%nobs).or.(self%nlev/=other%nlev))
call abor1_ftn(
'qg_obsvec_sub: inconsistent sizes')
259 where(self%values /= self%missing .and. other%values /= other%missing)
260 self%values = self%values-other%values
262 self%values = self%missing
276 if ((self%nobs/=other%nobs).or.(self%nlev/=other%nlev))
call abor1_ftn(
'qg_obsvec_mul: inconsistent sizes')
279 where(self%values /= self%missing .and. other%values /= other%missing)
280 self%values = self%values*other%values
282 self%values = self%missing
296 if ((self%nobs/=other%nobs).or.(self%nlev/=other%nlev))
call abor1_ftn(
'qg_obsvec_div: inconsistent sizes')
299 where(self%values /= self%missing .and. other%values /= other%missing)
300 self%values = self%values/other%values
302 self%values = self%missing
314 real(kind_real),
intent(in) :: zz
317 if ((self%nobs/=other%nobs).or.(self%nlev/=other%nlev))
call abor1_ftn(
'qg_obsvec_axpy: inconsistent sizes')
320 where(self%values /= self%missing .and. other%values /= other%missing)
321 self%values = self%values+zz*other%values
323 self%values = self%missing
337 where(self%values /= self%missing) self%values = 1.0/self%values
347 type(c_ptr),
intent(in) :: c_odb
354 nval = self%nobs*self%nlev
369 real(kind_real),
intent(inout) :: zz
375 if ((obsvec1%nobs/=obsvec2%nobs).or.(obsvec1%nlev/=obsvec2%nlev))
call abor1_ftn(
'qg_obsvec_dotprod: inconsistent sizes')
381 do jobs=1,obsvec1%nobs
382 do jlev=1,obsvec1%nlev
383 if (obsvec1%values(jlev, jobs) /= obsvec1%missing .and. &
384 obsvec2%values(jlev, jobs) /= obsvec2%missing) &
385 zz = zz+obsvec1%values(jlev,jobs)*obsvec2%values(jlev,jobs)
398 real(kind_real),
intent(inout) :: zmin
399 real(kind_real),
intent(inout) :: zmax
400 real(kind_real),
intent(inout) :: zavg
402 if (self%nobs*self%nlev>0)
then
404 if (.not.
allocated(self%values))
call abor1_ftn(
'qg_obsvec_stats: obs vector not allocated')
405 zmin = minval(self%values, mask = (self%values /= self%missing))
406 zmax = maxval(self%values, mask = (self%values /= self%missing))
407 zavg = sum(self%values, mask = (self%values /= self%missing)) / &
408 count(mask = (self%values /= self%missing))
423 integer,
intent(inout) :: kobs
424 kobs =
size(self%values) + 2
435 integer,
intent(inout) :: kobs
438 kobs = count(mask = (self%values /= self%missing))
451 integer,
intent(inout) :: kobs
454 kobs = count(mask = (self%values /= self%missing) .and. &
455 (obsmask%values /= obsmask%missing))
468 integer,
intent(in) :: nvals
469 real(kind_real),
dimension(nvals),
intent(out) :: vals
471 integer :: jobs, jlev, jval
477 if ((self%values(jlev, jobs) /= self%missing) .and. &
478 (obsmask%values(jlev, jobs) /= obsmask%missing))
then
479 if (jval > nvals)
call abor1_ftn(
'qg_obsvec_get: inconsistent vector size')
480 vals(jval) = self%values(jlev, jobs)
subroutine, public qg_obsvec_dotprod(obsvec1, obsvec2, zz)
Compute dot product between observation vectors.
subroutine, public qg_obsvec_random(c_odb, self)
Generate random observation vector.
subroutine, public qg_obsvec_ones(self)
Set observation vector to ones.
subroutine, public qg_obsvec_mask(self, mask)
Mask observation vector (set values to missing values where mask == 1)
subroutine, public qg_obsvec_mul(self, other)
Multiply observation vector.
subroutine, public qg_obsvec_zero(self)
Set observation vector to zero.
subroutine, public qg_obsvec_settomissing_ith(self, i)
Set i-th value of observation vector to missing value.
subroutine, public qg_obsvec_delete(self)
Delete observation vector.
subroutine, public qg_obsvec_mul_scal(self, zz)
Multiply observation vector with a scalar.
type(registry_t), public qg_obsvec_registry
Linked list interface - defines registry_t type.
subroutine, public qg_obsvec_size(self, kobs)
Get observation vector size.
subroutine, public qg_obsvec_mask_with_missing(self, mask)
Mask observation vector (set values to missing values where mask == missing value)
subroutine, public qg_obsvec_clone(self, other)
Clone observation vector.
subroutine, public qg_obsvec_get_withmask(self, obsmask, vals, nvals)
Get non-missing values from observation vector into vals array.
subroutine, public qg_obsvec_div(self, other)
Divide observation vector.
subroutine, public qg_obsvec_copy(self, other)
Copy observation vector.
subroutine, public qg_obsvec_setup(self, nlev, nobs)
Linked list implementation.
subroutine, public qg_obsvec_invert(self)
Invert observation vector.
subroutine, public qg_obsvec_nobs_withmask(self, obsmask, kobs)
Get observation vector size (only non-masked observations)
subroutine, public qg_obsvec_add(self, other)
Add observation vector.
subroutine, public qg_obsvec_stats(self, zmin, zmax, zavg)
Compute observation vector statistics.
subroutine, public qg_obsvec_axpy(self, zz, other)
Apply axpy on observation vector.
subroutine, public qg_obsvec_sub(self, other)
Subtract observation vector.
subroutine, public qg_obsvec_nobs(self, kobs)
Get observation vector size.