OOPS
EnsembleCovariance.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_ENSEMBLECOVARIANCE_H_
12 #define OOPS_BASE_ENSEMBLECOVARIANCE_H_
13 
14 #include <memory>
15 #include <string>
16 #include <vector>
17 
18 #include "eckit/config/LocalConfiguration.h"
19 #include "eckit/exception/Exceptions.h"
20 #include "eckit/system/ResourceUsage.h"
21 
23 #include "oops/base/Geometry.h"
25 #include "oops/base/Increment.h"
27 #include "oops/base/Localization.h"
29 #include "oops/base/State.h"
30 #include "oops/base/Variables.h"
31 #include "oops/util/Logger.h"
32 #include "oops/util/ObjectCounter.h"
33 
34 namespace oops {
35 
36 /// Generic ensemble based model space error covariance.
37 
38 // -----------------------------------------------------------------------------
39 template <typename MODEL>
41  private util::ObjectCounter<EnsembleCovariance<MODEL>> {
47  typedef std::shared_ptr<IncrementEnsemble<MODEL>> EnsemblePtr_;
48 
49  public:
50  static const std::string classname() {return "oops::EnsembleCovariance";}
51 
52  EnsembleCovariance(const Geometry_ &, const Variables &,
53  const eckit::Configuration &, const State_ &, const State_ &);
55 
56  private:
57  void doRandomize(Increment_ &) const override;
58  void doMultiply(const Increment_ &, Increment_ &) const override;
59  void doInverseMultiply(const Increment_ &, Increment_ &) const override;
60 
62  std::unique_ptr<Localization_> loc_;
63  int seed_ = 7; // For reproducibility
64 };
65 
66 // =============================================================================
67 
68 /// Constructor, destructor
69 // -----------------------------------------------------------------------------
70 template<typename MODEL>
72  const eckit::Configuration & conf,
73  const State_ & xb, const State_ & fg)
74  : ModelSpaceCovarianceBase<MODEL>(xb, fg, resol, conf), ens_(), loc_()
75 {
76  Log::trace() << "EnsembleCovariance::EnsembleCovariance start" << std::endl;
77  size_t init = eckit::system::ResourceUsage().maxResidentSetSize();
78  ens_.reset(new Ensemble_(conf, xb, fg, resol, vars));
79  if (conf.has("localization")) {
80  const eckit::LocalConfiguration confloc(conf, "localization");
81  loc_.reset(new Localization_(resol, confloc));
82  }
83  size_t current = eckit::system::ResourceUsage().maxResidentSetSize();
84  this->setObjectSize(current - init);
85  Log::trace() << "EnsembleCovariance::EnsembleCovariance done" << std::endl;
86 }
87 // -----------------------------------------------------------------------------
88 template<typename MODEL>
90  Log::trace() << "EnsembleCovariance destructed." << std::endl;
91 }
92 // -----------------------------------------------------------------------------
93 template<typename MODEL>
95  dx.zero();
96  if (loc_) {
97  // Localized covariance matrix
98  for (unsigned int ie = 0; ie < ens_->size(); ++ie) {
99  Increment_ tmp(dx);
100  loc_->randomize(tmp);
101  tmp.schur_product_with((*ens_)[ie]);
102  dx.axpy(1.0, tmp, false);
103  }
104  } else {
105  // Raw covariance matrix
106  util::NormalDistribution<double> normalDist(ens_->size(), 0.0, 1.0, seed_);
107  for (unsigned int ie = 0; ie < ens_->size(); ++ie) {
108  dx.axpy(normalDist[ie], (*ens_)[ie]);
109  }
110  }
111  dx *= 1.0/sqrt(static_cast<double>(ens_->size()-1));
112 }
113 // -----------------------------------------------------------------------------
114 template<typename MODEL>
116  dxo.zero();
117  for (unsigned int ie = 0; ie < ens_->size(); ++ie) {
118  if (loc_) {
119  // Localized covariance matrix
120  Increment_ dx(dxi);
121  dx.schur_product_with((*ens_)[ie]);
122  loc_->multiply(dx);
123  dx.schur_product_with((*ens_)[ie]);
124  dxo.axpy(1.0, dx, false);
125  } else {
126  // Raw covariance matrix
127  double wgt = dxi.dot_product_with((*ens_)[ie]);
128  dxo.axpy(wgt, (*ens_)[ie], false);
129  }
130  }
131  const double rk = 1.0/static_cast<double>(ens_->size()-1);
132  dxo *= rk;
133 }
134 // -----------------------------------------------------------------------------
135 template<typename MODEL>
138  dxo.zero();
139  GMRESR(dxo, dxi, *this, Id, 10, 1.0e-3);
140 }
141 // -----------------------------------------------------------------------------
142 } // namespace oops
143 
144 #endif // OOPS_BASE_ENSEMBLECOVARIANCE_H_
GMRESR solver for Ax=b.
Generic ensemble based model space error covariance.
EnsembleCovariance(const Geometry_ &, const Variables &, const eckit::Configuration &, const State_ &, const State_ &)
Constructor, destructor.
Increment< MODEL > Increment_
Geometry< MODEL > Geometry_
static const std::string classname()
void doMultiply(const Increment_ &, Increment_ &) const override
std::shared_ptr< IncrementEnsemble< MODEL > > EnsemblePtr_
std::unique_ptr< Localization_ > loc_
void doRandomize(Increment_ &) const override
void doInverseMultiply(const Increment_ &, Increment_ &) const override
Localization< MODEL > Localization_
IncrementEnsemble< MODEL > Ensemble_
Geometry class used in oops; subclass of interface class interface::Geometry.
Identity matrix.
Ensemble of inrements.
Increment class used in oops.
double dot_product_with(const Increment &other) const
dot product with the other increment
Abstract model-space localization class used by high level algorithms and applications.
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.
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