OOPS
HtRinvHMatrix.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_ASSIMILATION_HTRINVHMATRIX_H_
12 #define OOPS_ASSIMILATION_HTRINVHMATRIX_H_
13 
14 #include <memory>
15 #include <utility>
16 
17 #include <boost/noncopyable.hpp>
18 
23 #include "oops/util/dot_product.h"
24 #include "oops/util/formats.h"
25 #include "oops/util/Logger.h"
26 #include "oops/util/PrintAdjTest.h"
27 
28 namespace oops {
29 
30 /// The \f$ H^T R^{-1} H \f$ matrix.
31 /*!
32  * The solvers represent matrices as objects that implement a "multiply"
33  * method. This class defines objects that apply a generalized
34  * \f$ H^T R^{-1} H \f$ matrix that also includes the equivalent
35  * operators for the other terms of the cost function.
36  */
37 
38 template<typename MODEL, typename OBS> class HtRinvHMatrix : private boost::noncopyable {
41 
42  public:
43  explicit HtRinvHMatrix(const CostFct_ & j, const bool test = false);
44 
45  void multiply(const CtrlInc_ & dx, CtrlInc_ & dz) const;
46 
47  private:
48  CostFct_ const & j_;
49  bool test_;
50  mutable int iter_;
51 };
52 
53 // -----------------------------------------------------------------------------
54 
55 template<typename MODEL, typename OBS>
57  : j_(j), test_(test), iter_(0)
58 {}
59 
60 // -----------------------------------------------------------------------------
61 
62 template<typename MODEL, typename OBS>
64 // Increment counter
65  iter_++;
66 
67 // Setup TL terms of cost function
69  for (unsigned jj = 0; jj < j_.nterms(); ++jj) {
70  j_.jterm(jj).setPostProcTL(dx, costtl);
71  }
72 
73 // Run TLM
74  CtrlInc_ mdx(dx);
75  j_.runTLM(mdx, costtl);
76 
77 // Get TLM outputs, multiply by covariance inverses, and setup ADJ forcing terms
78  j_.zeroAD(dz);
80 
83 
84  for (unsigned jj = 0; jj < j_.nterms(); ++jj) {
85  std::unique_ptr<GeneralizedDepartures> wtmp(j_.jterm(jj).newDualVector());
86  j_.jterm(jj).computeCostTL(dx, *wtmp);
87  zz.append(j_.jterm(jj).multiplyCoInv(*wtmp));
88  j_.jterm(jj).computeCostAD(zz.getv(jj), dz, costad);
89  if (test_) ww.append(std::move(wtmp));
90  }
91 
92 // Run ADJ
93  j_.runADJ(dz, costad);
94 
95  for (unsigned jj = 0; jj < j_.nterms(); ++jj) {
96  j_.jterm(jj).setPostProcAD();
97  }
98 
99  if (test_) {
100  // <G dx, dy>, where dy = Rinv H dx
101  double adj_tst_fwd = dot_product(ww, zz);
102  // <dx, Gt dy> , where dy = Rinv H dx
103  double adj_tst_bwd = dot_product(dx, dz);
104 
105  Log::info() << "Online adjoint test, iteration: " << iter_ << std::endl
106  << util::PrintAdjTest(adj_tst_fwd, adj_tst_bwd, "G") << std::endl;
107  }
108 }
109 
110 // -----------------------------------------------------------------------------
111 
112 } // namespace oops
113 
114 #endif // OOPS_ASSIMILATION_HTRINVHMATRIX_H_
Cost Function.
Definition: CostFunction.h:53
Container of dual space vectors for all terms of the cost function.
Definition: DualVector.h:35
void append(std::unique_ptr< GeneralizedDepartures > &&)
Definition: DualVector.h:110
std::shared_ptr< const GeneralizedDepartures > getv(const unsigned) const
Definition: DualVector.h:129
CostFunction< MODEL, OBS > CostFct_
Definition: HtRinvHMatrix.h:40
HtRinvHMatrix(const CostFct_ &j, const bool test=false)
Definition: HtRinvHMatrix.h:56
void multiply(const CtrlInc_ &dx, CtrlInc_ &dz) const
Definition: HtRinvHMatrix.h:63
CostFct_ const & j_
Definition: HtRinvHMatrix.h:48
ControlIncrement< MODEL, OBS > CtrlInc_
Definition: HtRinvHMatrix.h:39
Control model post processing.
The namespace for the main oops code.