OOPS
interface/ObsErrorBase.h
Go to the documentation of this file.
1 /*
2  * (C) Crown copyright 2021, Met Office
3  *
4  * This software is licensed under the terms of the Apache Licence Version 2.0
5  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
6  */
7 
8 #ifndef OOPS_INTERFACE_OBSERRORBASE_H_
9 #define OOPS_INTERFACE_OBSERRORBASE_H_
10 
11 #include <memory>
12 #include <string>
13 
14 #include "eckit/system/ResourceUsage.h"
15 
16 #include "oops/base/ObsVector.h"
19 #include "oops/util/ObjectCounter.h"
20 #include "oops/util/Printable.h"
21 
22 namespace eckit {
23  class Configuration;
24 }
25 
26 namespace oops {
27 
28 namespace interface {
29 
30 // -----------------------------------------------------------------------------
31 /// \brief Base class for OBS-specific implementations of the ObsError interface.
32 ///
33 /// interface::ObsErrorBase overrides oops::ObsErrorBase methods to pass OBS-specific
34 /// implementations of ObsVector to OBS-specific implementations of ObsError.
35 ///
36 /// Note: each subclass should typedef `Parameters_` to the name of a subclass of
37 /// ObsErrorParametersBase holding its configuration settings and provide a constructor with the
38 /// following signature:
39 ///
40 /// ObsErrorBase(const Parameters_ &params, OBS::ObsSpace &obsspace,
41 /// const eckit::mpi::Comm &timeComm);
42 ///
43 /// This constructor should pass \c timeComm to the constructor of this class.
44 template <typename OBS>
45 class ObsErrorBase : public oops::ObsErrorBase<OBS> {
46  typedef typename OBS::ObsSpace ObsSpace_;
47  typedef typename OBS::ObsVector ObsVector_;
48 
49  public:
50  explicit ObsErrorBase(const eckit::mpi::Comm &timeComm)
51  : timeComm_(timeComm) {}
52 
53  // Overrides of oops::ObsErrorBase methods, converting between oops interface classes and
54  // OBS-specific classes operated upon by OBS-specific implementations of the ObsError interface.
55 
56  void multiply(oops::ObsVector<OBS> &dy) const final {
57  this->multiply(dy.obsvector());
58  }
59 
60  void inverseMultiply(oops::ObsVector<OBS> &dy) const final {
61  this->inverseMultiply(dy.obsvector());
62  }
63 
64  void randomize(oops::ObsVector<OBS> &dy) const final {
65  this->randomize(dy.obsvector());
66  }
67 
70  }
71 
72  void update(const oops::ObsVector<OBS> &dy) final {
73  this->update(dy.obsvector());
74  }
75 
78  }
79 
80  // The methods below need to be overridden in subclasses (along with save(), getRMSE() and
81  // print(), methods inherited from parent classes that neither take nor return instances of oops
82  // interface classes and therefore are still abstract).
83 
84  /// Multiply a Departure \p dy by \f$R\f$.
85  virtual void multiply(ObsVector_ &dy) const = 0;
86 
87  /// Multiply a Departure \p dy by \f$R^{-1}\f$.
88  virtual void inverseMultiply(ObsVector_ &dy) const = 0;
89 
90  /// Generate a random perturbation in \p dy.
91  virtual void randomize(ObsVector_ &dy) const = 0;
92 
93  /// Return a copy of obs error std. dev. If this ObsVector_ is modified (e.g. by obs filters),
94  /// it should be passed back to update() to ensure the covariance matrix stays consistent.
95  virtual std::unique_ptr<ObsVector_> getObsErrors() const = 0;
96 
97  /// Set the diagonal of the covariance matrix to \p stddev squared.
98  virtual void update(const ObsVector_ &stddev) = 0;
99 
100  /// Return the vector of inverse obs error variances.
101  virtual std::unique_ptr<ObsVector_> getInverseVariance() const = 0;
102 
103  private:
104  const eckit::mpi::Comm & timeComm_;
105 };
106 
107 // =============================================================================
108 
109 /// \brief A subclass of ObsErrorFactory able to create instances of T (a concrete subclass of
110 /// interface::ObsErrorBase<OBS>). Passes OBS::ObsSpace to the constructor of T.
111 template <class OBS, class T>
112 class ObsErrorMaker : public ObsErrorFactory<OBS> {
113  public:
117  typedef typename T::Parameters_ Parameters_;
118 
119  explicit ObsErrorMaker(const std::string & name) : ObsErrorFactory_(name) {}
120 
121  private:
122  std::unique_ptr<ObsErrorBase_> make(const ObsErrorParametersBase & parameters,
123  const ObsSpace_ & obsspace) override {
124  const auto &stronglyTypedParameters = dynamic_cast<const Parameters_&>(parameters);
125  return std::make_unique<T>(stronglyTypedParameters, obsspace.obsspace(), obsspace.timeComm());
126  }
127 
128  std::unique_ptr<ObsErrorParametersBase> makeParameters() const override {
129  return std::make_unique<Parameters_>();
130  }
131 };
132 
133 // -----------------------------------------------------------------------------
134 
135 } // namespace interface
136 
137 } // namespace oops
138 
139 #endif // OOPS_INTERFACE_OBSERRORBASE_H_
Base class for generic implementations of observation error covariance matrices.
A factory creating instances of concrete subclasses of ObsErrorBase.
Configuration parameters of an implementation of an observation error covariance matrix model.
ObsSpace_ & obsspace() const
Interfacing.
const eckit::mpi::Comm & timeComm() const
ObsVector class used in oops; subclass of interface class interface::ObsVector.
Base class for OBS-specific implementations of the ObsError interface.
void randomize(oops::ObsVector< OBS > &dy) const final
Generate random perturbation in dy.
virtual void multiply(ObsVector_ &dy) const =0
Multiply a Departure dy by .
virtual std::unique_ptr< ObsVector_ > getObsErrors() const =0
virtual void inverseMultiply(ObsVector_ &dy) const =0
Multiply a Departure dy by .
const eckit::mpi::Comm & timeComm_
void update(const oops::ObsVector< OBS > &dy) final
Set the diagonal of the covariance matrix to stddev squared.
ObsErrorBase(const eckit::mpi::Comm &timeComm)
void inverseMultiply(oops::ObsVector< OBS > &dy) const final
Multiply a Departure dy by .
virtual std::unique_ptr< ObsVector_ > getInverseVariance() const =0
Return the vector of inverse obs error variances.
virtual void randomize(ObsVector_ &dy) const =0
Generate a random perturbation in dy.
oops::ObsVector< OBS > inverseVariance() const final
Return the vector of inverse obs error variances.
virtual void update(const ObsVector_ &stddev)=0
Set the diagonal of the covariance matrix to stddev squared.
void multiply(oops::ObsVector< OBS > &dy) const final
Multiply a Departure dy by .
oops::ObsVector< OBS > obserrors() const final
A subclass of ObsErrorFactory able to create instances of T (a concrete subclass of interface::ObsErr...
ObsErrorMaker(const std::string &name)
oops::ObsErrorBase< OBS > ObsErrorBase_
std::unique_ptr< ObsErrorBase_ > make(const ObsErrorParametersBase &parameters, const ObsSpace_ &obsspace) override
std::unique_ptr< ObsErrorParametersBase > makeParameters() const override
oops::ObsErrorFactory< OBS > ObsErrorFactory_
Definition: FieldL95.h:22
The namespace for the main oops code.