OOPS
ModelSpaceCovarianceBase.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
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  * In applying this licence, ECMWF does not waive the privileges and immunities
7  * granted to it by virtue of its status as an intergovernmental organisation nor
8  * does it submit to any jurisdiction.
9  */
10 
11 #ifndef OOPS_BASE_MODELSPACECOVARIANCEBASE_H_
12 #define OOPS_BASE_MODELSPACECOVARIANCEBASE_H_
13 
14 #include <map>
15 #include <memory>
16 #include <string>
17 #include <vector>
18 
19 #include <boost/make_unique.hpp>
20 #include <boost/noncopyable.hpp>
21 #include <boost/ptr_container/ptr_vector.hpp>
22 #include <boost/range/adaptor/reversed.hpp>
23 
24 #include "eckit/config/LocalConfiguration.h"
25 #include "eckit/exception/Exceptions.h"
27 #include "oops/base/Geometry.h"
29 #include "oops/base/Increment.h"
32 #include "oops/base/State.h"
33 #include "oops/base/Variables.h"
34 #include "oops/util/AssociativeContainers.h"
35 #include "oops/util/Logger.h"
36 #include "oops/util/parameters/ConfigurationParameter.h"
37 #include "oops/util/parameters/HasParameters_.h"
38 #include "oops/util/parameters/OptionalParameter.h"
39 #include "oops/util/parameters/Parameters.h"
40 #include "oops/util/parameters/RequiredPolymorphicParameter.h"
41 #include "oops/util/Random.h"
42 
43 namespace util {
44  class DateTime;
45 }
46 
47 namespace oops {
48 
49 // -----------------------------------------------------------------------------
50 
51 // Should this be one with the ErrorCovariance class in the interface directory? YT
52 
53 /// Abstract base class for model space error covariances.
54 ///
55 /// Note: subclasses can opt to extract their settings either from a Configuration object or from a
56 /// subclass of Parameters.
57 ///
58 /// In the former case, they should provide a constructor taking a const reference to an
59 /// eckit::Configuration object. In the latter case, the implementer should first define a subclass
60 /// of Parameters holding the settings of the model-space covariance change in question. The latter
61 /// should then typedef `Parameters_` to the name of that subclass and provide a constructor taking
62 /// a const reference to an instance of that subclass.
63 template <typename MODEL>
69  typedef typename boost::ptr_vector<LinearVariableChangeBase_> ChvarVec_;
70  typedef typename ChvarVec_::iterator iter_;
71  typedef typename ChvarVec_::const_iterator icst_;
72  typedef typename ChvarVec_::const_reverse_iterator ircst_;
73 
74  public:
75  ModelSpaceCovarianceBase(const State_ &, const State_ &,
77  ModelSpaceCovarianceBase(const State_ &, const State_ &,
78  const Geometry_ &, const eckit::Configuration &);
80 
81  void randomize(Increment_ &) const;
82  void multiply(const Increment_ &, Increment_ &) const;
83  void inverseMultiply(const Increment_ &, Increment_ &) const;
84  void getVariance(Increment_ &) const;
85 
86  private:
87  virtual void doRandomize(Increment_ &) const = 0;
88  virtual void doMultiply(const Increment_ &, Increment_ &) const = 0;
89  virtual void doInverseMultiply(const Increment_ &, Increment_ &) const = 0;
90 
96 };
97 
98 // =============================================================================
99 
100 template <typename MODEL>
101 class CovarianceFactory;
102 
103 // -----------------------------------------------------------------------------
104 
105 /// \brief A subclass of ModelSpaceCovarianceParametersBase storing the values of all options in a
106 /// single Configuration object.
107 ///
108 /// This object can be accessed by calling the value() method of the \p config member variable.
109 ///
110 /// The ConfigurationParameter class does not perform any parameter validation; models using
111 /// GenericModelSpaceCovarianceParameters should therefore ideally be refactored, replacing this
112 /// class with a dedicated subclass of ModelSpaceCovarianceParametersBase storing each parameter in
113 /// a separate (Optional/Required)Parameter object.
114 ///
115 template <typename MODEL>
117  OOPS_CONCRETE_PARAMETERS(GenericModelSpaceCovarianceParameters,
119  public:
120  ConfigurationParameter config{this};
121 };
122 
123 // -----------------------------------------------------------------------------
124 
125 /// \brief Contains a polymorphic parameter holding an instance of a subclass of
126 /// ModelSpaceCovarianceParametersBase.
127 template <typename MODEL>
128 class ModelSpaceCovarianceParametersWrapper : public Parameters {
129  OOPS_CONCRETE_PARAMETERS(ModelSpaceCovarianceParametersWrapper, Parameters)
130  public:
131  /// After deserialization, holds an instance of a subclass of ModelSpaceCovarianceParametersBase
132  /// controlling the behavior of a covariance model. The type of the subclass is determined by
133  /// the value of the "name" key in the Configuration object from which this object is
134  /// deserialized.
135  RequiredPolymorphicParameter<ModelSpaceCovarianceParametersBase<MODEL>, CovarianceFactory<MODEL>>
136  covarianceParameters{"covariance model", this};
137 };
138 
139 // =============================================================================
140 
141 /// Covariance Factory
142 template <typename MODEL>
146 
147  public:
148  /// \brief Create and return a new covariance model.
149  ///
150  /// The covariance model is determined by the \c covarianceModel attribute of \p parameters.
151  /// \p parameters must be an instance of the subclass of ModelSpaceCovarianceParametersBase
152  /// associated with that covariance model, otherwise an exception will be thrown.
154  const Geometry_ &, const Variables &,
155  const State_ &, const State_ &);
156 
157  /// \brief Create and return a new covariance model.
158  ///
159  /// Deprecated overload taking a Configuration instead of a ModelSpaceCovarianceParametersBase.
160  static ModelSpaceCovarianceBase<MODEL> * create(const eckit::Configuration &,
161  const Geometry_ &, const Variables &,
162  const State_ &, const State_ &);
163 
164  /// \brief Create and return an instance of the subclass of ModelSpaceCovarianceParametersBase
165  /// storing parameters of the specified covariance model.
166  static std::unique_ptr<ModelSpaceCovarianceParametersBase<MODEL>> createParameters(
167  const std::string &covarianceModel);
168 
169  /// \brief Return the names of all covariance models that can be created by one of the
170  /// registered makers.
171  static std::vector<std::string> getMakerNames() {
172  return keys(getMakers());
173  }
174 
175  virtual ~CovarianceFactory() = default;
176 
177  protected:
178  /// \brief Register a maker able to create covariance models of type \p name.
179  explicit CovarianceFactory(const std::string &name);
180 
181  private:
183  const Geometry_ &, const Variables &,
184  const State_ &, const State_ &) = 0;
185 
186  virtual std::unique_ptr<ModelSpaceCovarianceParametersBase<MODEL>> makeParameters() const = 0;
187 
188  static std::map < std::string, CovarianceFactory<MODEL> * > & getMakers() {
189  static std::map < std::string, CovarianceFactory<MODEL> * > makers_;
190  return makers_;
191  }
192 };
193 
194 // -----------------------------------------------------------------------------
195 
196 template<class MODEL, class COVAR>
197 class CovarMaker : public CovarianceFactory<MODEL> {
198  /// Defined as T::Parameters_ if COVAR defines a Parameters_ type; otherwise as
199  /// GenericModelSpaceCovarianceParameters<MODEL>.
200  typedef TParameters_IfAvailableElseFallbackType_t<
202 
205 
208  const Geometry_ & resol, const Variables & vars,
209  const State_ & xb, const State_ & fg) override {
210  const auto &stronglyTypedParams = dynamic_cast<const Parameters_&>(params);
211  return new COVAR(resol, vars,
212  parametersOrConfiguration<HasParameters_<COVAR>::value>(stronglyTypedParams),
213  xb, fg);
214  }
215 
216  std::unique_ptr<ModelSpaceCovarianceParametersBase<MODEL>> makeParameters() const override {
217  return boost::make_unique<Parameters_>();
218  }
219 
220  public:
221  explicit CovarMaker(const std::string & name) : CovarianceFactory<MODEL>(name) {}
222 };
223 
224 // =============================================================================
225 
226 template <typename MODEL>
228  if (getMakers().find(name) != getMakers().end()) {
229  throw std::runtime_error(name + " already registered in covariance factory.");
230  }
231  getMakers()[name] = this;
232 }
233 
234 // -----------------------------------------------------------------------------
235 
236 template <typename MODEL>
239  const Geometry_ & resol,
240  const Variables & vars,
241  const State_ & xb, const State_ & fg) {
242  const std::string id = parameters.covarianceModel.value().value();
243  Log::trace() << "ModelSpaceCovarianceBase type = " << id << std::endl;
244  typename std::map<std::string, CovarianceFactory<MODEL>*>::iterator jcov = getMakers().find(id);
245  if (jcov == getMakers().end()) {
246  Log::error() << id << " does not exist in CovarianceFactory." << std::endl;
247  Log::error() << "CovarianceFactory has " << getMakers().size() << " elements:" << std::endl;
248  for (typename std::map<std::string, CovarianceFactory<MODEL>*>::const_iterator
249  jj = getMakers().begin(); jj != getMakers().end(); ++jj) {
250  Log::error() << "A " << jj->first << " B" << std::endl;
251  }
252  throw std::runtime_error(id + " does not exist in covariance factory.");
253  }
254  Variables vars_in(vars);
255  Variables vars_out;
256  for (const LinearVariableChangeParametersWrapper<MODEL> &variableChange :
257  boost::adaptors::reverse(parameters.variableChanges.value())) {
258  const LinearVariableChangeParametersBase &variableChangeParameters =
259  variableChange.variableChangeParameters;
260  if (variableChangeParameters.inputVariables.value() != boost::none &&
261  variableChangeParameters.outputVariables.value() != boost::none) {
262  vars_out = *variableChangeParameters.outputVariables.value();
263  if (!(vars_in == vars_out)) {
264  Log::error() << "Input variables: " << vars_in << std::endl;
265  Log::error() << "Output variables: " << vars_out << std::endl;
266  throw eckit::BadParameter("Sequence of variable changes is not consistent");
267  }
268  vars_in = *variableChangeParameters.inputVariables.value();
269  }
270  }
271  return (*jcov).second->make(parameters, resol, vars_in, xb, fg);
272 }
273 
274 // -----------------------------------------------------------------------------
275 
276 template <typename MODEL>
278  const eckit::Configuration & conf,
279  const Geometry_ & resol,
280  const Variables & vars,
281  const State_ & xb, const State_ & fg) {
283  parameters.validateAndDeserialize(conf);
284  return create(parameters.covarianceParameters, resol, vars, xb, fg);
285 }
286 
287 // -----------------------------------------------------------------------------
288 
289 template <typename MODEL>
290 std::unique_ptr<ModelSpaceCovarianceParametersBase<MODEL>>
292  const std::string &name) {
293  typename std::map<std::string, CovarianceFactory<MODEL>*>::iterator it =
294  getMakers().find(name);
295  if (it == getMakers().end()) {
296  throw std::runtime_error(name + " does not exist in CovarianceFactory");
297  }
298  return it->second->makeParameters();
299 }
300 
301 // =============================================================================
302 
303 template <typename MODEL>
305  const State_ & bg, const State_ & fg,
306  const Geometry_ & resol,
307  const ModelSpaceCovarianceParametersBase<MODEL> & parameters) {
308  for (const LinearVariableChangeParametersWrapper<MODEL> &variableChange :
309  parameters.variableChanges.value()) {
311  bg, fg, resol, variableChange.variableChangeParameters));
312  }
313  randomizationSize_ = parameters.randomizationSize;
314  fullInverse_ = parameters.fullInverse;
315  fullInverseIterations_ = parameters.fullInverseIterations;
316  fullInverseAccuracy_ = parameters.fullInverseAccuracy;
317 }
318 
319 // -----------------------------------------------------------------------------
320 
321 template <typename MODEL>
323  const State_ & bg, const State_ & fg,
324  const Geometry_ & resol,
325  const eckit::Configuration & conf)
327  bg, fg, resol, validateAndDeserialize<GenericModelSpaceCovarianceParameters<MODEL>>(conf))
328 {}
329 
330 // -----------------------------------------------------------------------------
331 
332 template <typename MODEL>
334  // TODO(notguillaume): Generalize to non-square change of variable
335  if (chvars_.size()) {
336  this->doRandomize(dx); // dx = C^1/2 dx
337  // K_N K_N-1 ... K_1
338  for (icst_ it = chvars_.begin(); it != chvars_.end(); ++it) {
339  dx = it->multiply(dx); // dx = K_i dx
340  }
341  } else {
342  this->doRandomize(dx);
343  }
344 }
345 
346 // -----------------------------------------------------------------------------
347 
348 template <typename MODEL>
350  Increment_ & dxo) const {
351  if (chvars_.size()) {
352  // K_1^T K_2^T .. K_N^T
353  std::unique_ptr<Increment_> dxchvarin(new Increment_(dxi));
354  for (ircst_ it = chvars_.rbegin(); it != chvars_.rend(); ++it) {
355  Increment_ dxchvarout = it->multiplyAD(*dxchvarin);
356  dxchvarin.reset(new Increment_(dxchvarout));
357  }
358  Increment_ dxchvarout(*dxchvarin, false);
359 
360  this->doMultiply(*dxchvarin, dxchvarout);
361 
362  // K_N K_N-1 ... K_1
363  dxchvarin.reset(new Increment_(dxchvarout));
364  for (icst_ it = chvars_.begin(); it != chvars_.end(); ++it) {
365  Increment_ dxchvarout = it->multiply(*dxchvarin);
366  dxchvarin.reset(new Increment_(dxchvarout));
367  }
368  dxo = *dxchvarin;
369  } else {
370  this->doMultiply(dxi, dxo);
371  }
372 }
373 
374 // -----------------------------------------------------------------------------
375 
376 template <typename MODEL>
378  Increment_ & dxo) const {
379  if (fullInverse_) {
380  // Approximate full inverse using GMRESR
382  dxo.zero();
383  GMRESR(dxo, dxi, *this, Id, fullInverseIterations_, fullInverseAccuracy_);
384  } else {
385  if (chvars_.size()) {
386  // K_1^{-1} K_2^{-1} .. K_N^{-1}
387  std::unique_ptr<Increment_> dxchvarin(new Increment_(dxi));
388  for (ircst_ it = chvars_.rbegin(); it != chvars_.rend(); ++it) {
389  Increment_ dxchvarout = it->multiplyInverse(*dxchvarin);
390  dxchvarin.reset(new Increment_(dxchvarout));
391  }
392  Increment_ dxchvarout(*dxchvarin, false);
393 
394  this->doInverseMultiply(*dxchvarin, dxchvarout);
395 
396  // K_N^T^{-1} K_N-1^T^{-1} ... K_1^T^{-1}
397  dxchvarin.reset(new Increment_(dxchvarout));
398  for (icst_ it = chvars_.begin(); it != chvars_.end(); ++it) {
399  Increment_ dxchvarout = it->multiplyInverseAD(*dxchvarin);
400  dxchvarin.reset(new Increment_(dxchvarout));
401  }
402  dxo = *dxchvarin;
403  } else {
404  this->doInverseMultiply(dxi, dxo);
405  }
406  }
407 }
408 
409 // -----------------------------------------------------------------------------
410 
411 template <typename MODEL>
413  Increment_ dx(variance);
414  Increment_ dxsq(variance);
415  Increment_ mean(variance);
416  mean.zero();
417  variance.zero();
418  for (size_t ie = 0; ie < randomizationSize_; ++ie) {
419  this->randomize(dx);
420  dx -= mean;
421  dxsq = dx;
422  dxsq.schur_product_with(dx);
423  double rk_var = static_cast<double>(ie)/static_cast<double>(ie+1);
424  double rk_mean = 1.0/static_cast<double>(ie+1);
425  variance.axpy(rk_var, dxsq, false);
426  mean.axpy(rk_mean, dx, false);
427  }
428  double rk_norm = 1.0/static_cast<double>(randomizationSize_-1);
429  variance *= rk_norm;
430 }
431 
432 // -----------------------------------------------------------------------------
433 
434 } // namespace oops
435 
436 #endif // OOPS_BASE_MODELSPACECOVARIANCEBASE_H_
GMRESR solver for Ax=b.
Geometry< MODEL > Geometry_
CovarMaker(const std::string &name)
ModelSpaceCovarianceBase< MODEL > * make(const ModelSpaceCovarianceParametersBase< MODEL > &params, const Geometry_ &resol, const Variables &vars, const State_ &xb, const State_ &fg) override
TParameters_IfAvailableElseFallbackType_t< COVAR, GenericModelSpaceCovarianceParameters< MODEL > > Parameters_
std::unique_ptr< ModelSpaceCovarianceParametersBase< MODEL > > makeParameters() const override
virtual std::unique_ptr< ModelSpaceCovarianceParametersBase< MODEL > > makeParameters() const =0
static std::vector< std::string > getMakerNames()
Return the names of all covariance models that can be created by one of the registered makers.
CovarianceFactory(const std::string &name)
Register a maker able to create covariance models of type name.
virtual ModelSpaceCovarianceBase< MODEL > * make(const ModelSpaceCovarianceParametersBase< MODEL > &, const Geometry_ &, const Variables &, const State_ &, const State_ &)=0
virtual ~CovarianceFactory()=default
static std::unique_ptr< ModelSpaceCovarianceParametersBase< MODEL > > createParameters(const std::string &covarianceModel)
Create and return an instance of the subclass of ModelSpaceCovarianceParametersBase storing parameter...
static ModelSpaceCovarianceBase< MODEL > * create(const ModelSpaceCovarianceParametersBase< MODEL > &, const Geometry_ &, const Variables &, const State_ &, const State_ &)
Create and return a new covariance model.
static std::map< std::string, CovarianceFactory< MODEL > * > & getMakers()
A subclass of ModelSpaceCovarianceParametersBase storing the values of all options in a single Config...
Geometry class used in oops; subclass of interface class interface::Geometry.
Identity matrix.
Increment class used in oops.
Base class of classes storing parameters controlling specific linear variable changes.
Contains a polymorphic parameter holding an instance of a subclass of LinearVariableChangeParametersB...
ChvarVec_::const_reverse_iterator ircst_
LinearVariableChangeBase< MODEL > LinearVariableChangeBase_
virtual void doMultiply(const Increment_ &, Increment_ &) const =0
boost::ptr_vector< LinearVariableChangeBase_ > ChvarVec_
ModelSpaceCovarianceBase(const State_ &, const State_ &, const Geometry_ &, const ModelSpaceCovarianceParametersBase< MODEL > &)
virtual void doInverseMultiply(const Increment_ &, Increment_ &) const =0
void inverseMultiply(const Increment_ &, Increment_ &) const
virtual void doRandomize(Increment_ &) const =0
void multiply(const Increment_ &, Increment_ &) const
Base class for classes storing parameters of a particular model-space error covariance implementation...
Parameter< std::vector< LinearVariableChangeParametersWrapper< MODEL > > > variableChanges
OptionalParameter< std::string > covarianceModel
Covariance model name.
Contains a polymorphic parameter holding an instance of a subclass of ModelSpaceCovarianceParametersB...
RequiredPolymorphicParameter< ModelSpaceCovarianceParametersBase< MODEL >, CovarianceFactory< MODEL > > covarianceParameters
State class used in oops; subclass of interface class interface::State.
void axpy(const double &w, const Increment &dx, const bool check=true)
void schur_product_with(const Increment &other)
Compute Schur product of this Increment with other, assign to this Increment.
void zero()
Zero out this Increment.
int error
Definition: compare.py:168
The namespace for the main oops code.
double GMRESR(VECTOR &xx, const VECTOR &bb, const AMATRIX &A, const PMATRIX &precond, const int maxiter, const double tolerance)
Definition: GMRESR.h:64
Definition: TLML95.h:34