OOPS
HybridCovariance.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_HYBRIDCOVARIANCE_H_
12 #define OOPS_BASE_HYBRIDCOVARIANCE_H_
13 
14 #include <memory>
15 #include <string>
16 #include <utility>
17 #include <vector>
18 
19 #include "eckit/config/LocalConfiguration.h"
20 #include "eckit/exception/Exceptions.h"
22 #include "oops/base/Geometry.h"
24 #include "oops/base/Increment.h"
26 #include "oops/base/State.h"
27 #include "oops/base/Variables.h"
28 #include "oops/util/DateTime.h"
29 #include "oops/util/Logger.h"
30 
31 namespace oops {
32 
33 /// Generic hybrid static-ensemble model space error covariance.
34 
35 // -----------------------------------------------------------------------------
36 template <typename MODEL>
41 
42  public:
43  HybridCovariance(const Geometry_ &, const Variables &,
44  const eckit::Configuration &, const State_ &, const State_ &);
46 
47  private:
48  void doRandomize(Increment_ &) const override;
49  void doMultiply(const Increment_ &, Increment_ &) const override;
50  void doInverseMultiply(const Increment_ &, Increment_ &) const override;
51 
52  std::vector< std::unique_ptr< ModelSpaceCovarianceBase<MODEL> > > Bcomponents_;
53  std::vector< std::string > weightTypes_;
54  std::vector< double > valueWeights_;
55  std::vector< Increment_ > incrementWeights_;
56 };
57 
58 // =============================================================================
59 
60 /// Constructor, destructor
61 // -----------------------------------------------------------------------------
62 template<typename MODEL>
64  const eckit::Configuration & config,
65  const State_ & xb, const State_ & fg)
66  : ModelSpaceCovarianceBase<MODEL>(xb, fg, resol, config)
67 {
68  std::vector<eckit::LocalConfiguration> confs;
69  config.get("components", confs);
70  for (const auto & conf : confs) {
71  // B component
72  const eckit::LocalConfiguration covarConf(conf, "covariance");
73  std::unique_ptr< ModelSpaceCovarianceBase<MODEL> > B(
74  CovarianceFactory<MODEL>::create(covarConf, resol, vars, xb, fg));
75  Bcomponents_.push_back(std::move(B));
76 
77  // Weight
78  const eckit::LocalConfiguration weightConf(conf, "weight");
79  if (weightConf.has("value")) {
80  // Scalar weight provided in the configuration
81  weightTypes_.push_back("value");
82  valueWeights_.push_back(weightConf.getDouble("value"));
83  } else {
84  // 3D weight read from a file
85  weightTypes_.push_back("increment");
86  Increment_ weight(resol, vars, xb.validTime());
87  weight.read(weightConf);
88  incrementWeights_.push_back(weight);
89  }
90  }
91  Log::trace() << "HybridCovariance created." << std::endl;
92 }
93 // -----------------------------------------------------------------------------
94 template<typename MODEL>
96  Log::trace() << "HybridCovariance destructed" << std::endl;
97 }
98 // -----------------------------------------------------------------------------
99 template<typename MODEL>
101  dxo.zero();
102  Increment_ tmp(dxo);
103  int valueIndex = 0;
104  int incrementIndex = 0;
105  for (size_t jcomp = 0; jcomp < Bcomponents_.size(); ++jcomp) {
106  Bcomponents_[jcomp]->multiply(dxi, tmp);
107  if (weightTypes_[jcomp] == "value") {
108  tmp *= valueWeights_[valueIndex];
109  valueIndex += 1;
110  }
111  if (weightTypes_[jcomp] == "increment") {
112  tmp.schur_product_with(incrementWeights_[incrementIndex]);
113  incrementIndex += 1;
114  }
115  dxo += tmp;
116  }
117 }
118 // -----------------------------------------------------------------------------
119 template<typename MODEL>
122  dxo.zero();
123  GMRESR(dxo, dxi, *this, Id, 10, 1.0e-3);
124 }
125 // -----------------------------------------------------------------------------
126 template<typename MODEL>
128  dx.zero();
129  Increment_ tmp(dx);
130  int valueIndex = 0;
131  for (size_t jcomp = 0; jcomp < Bcomponents_.size(); ++jcomp) {
132  Bcomponents_[jcomp]->randomize(tmp);
133  if (weightTypes_[jcomp] == "value") {
134  tmp *= std::sqrt(valueWeights_[valueIndex]);
135  valueIndex += 1;
136  }
137  if (weightTypes_[jcomp] == "increment") {
138  throw eckit::NotImplemented("HybridCovariance::doRandomize: no square-root", Here());
139  }
140  dx += tmp;
141  }
142 }
143 // -----------------------------------------------------------------------------
144 } // namespace oops
145 
146 #endif // OOPS_BASE_HYBRIDCOVARIANCE_H_
GMRESR solver for Ax=b.
Geometry class used in oops; subclass of interface class interface::Geometry.
Generic hybrid static-ensemble model space error covariance.
std::vector< std::unique_ptr< ModelSpaceCovarianceBase< MODEL > > > Bcomponents_
std::vector< double > valueWeights_
HybridCovariance(const Geometry_ &, const Variables &, const eckit::Configuration &, const State_ &, const State_ &)
Constructor, destructor.
Increment< MODEL > Increment_
std::vector< Increment_ > incrementWeights_
Geometry< MODEL > Geometry_
std::vector< std::string > weightTypes_
void doMultiply(const Increment_ &, Increment_ &) const override
void doInverseMultiply(const Increment_ &, Increment_ &) const override
void doRandomize(Increment_ &) const override
Identity matrix.
Increment class used in oops.
State class used in oops; subclass of interface class interface::State.
void schur_product_with(const Increment &other)
Compute Schur product of this Increment with other, assign to this Increment.
void read(const eckit::Configuration &)
Read this Increment from file.
void zero()
Zero out this Increment.
const util::DateTime validTime() const
Accessor to the time of this State.
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