OOPS
HBHtMatrix.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_HBHTMATRIX_H_
12 #define OOPS_ASSIMILATION_HBHTMATRIX_H_
13 
14 #include <memory>
15 #include <utility>
16 
17 #include <boost/noncopyable.hpp>
18 
23 #include "oops/util/PrintAdjTest.h"
24 
25 namespace oops {
26 
27 /// The \f$ H B H^T \f$ matrix.
28 /*!
29  * The solvers represent matrices as objects that implement a "multiply"
30  * method. This class defines objects that apply a generalized
31  * \f$ H B H ^T\f$ matrix which includes \f$ H \f$ and the equivalent
32  * operators for the other terms of the cost function.
33  */
34 
35 template<typename MODEL, typename OBS> class HBHtMatrix : private boost::noncopyable {
39 
40  public:
41  explicit HBHtMatrix(const CostFct_ & j, const bool test = false);
42 
43  void multiply(const Dual_ & dy, Dual_ & dz) const;
44 
45  private:
46  CostFct_ const & j_;
47  bool test_;
48  mutable int iter_;
49 };
50 
51 // -----------------------------------------------------------------------------
52 
53 template<typename MODEL, typename OBS>
55  : j_(j), test_(test), iter_(0)
56 {}
57 
58 // -----------------------------------------------------------------------------
59 
60 template<typename MODEL, typename OBS>
61 void HBHtMatrix<MODEL, OBS>::multiply(const Dual_ & dy, Dual_ & dz) const {
62 // Increment counter
63  iter_++;
64 
65 // Run ADJ
66  CtrlInc_ ww(j_.jb());
67  j_.zeroAD(ww);
69  for (unsigned jj = 0; jj < j_.nterms(); ++jj) {
70  j_.jterm(jj).computeCostAD(dy.getv(jj), ww, costad);
71  }
72  j_.runADJ(ww, costad);
73  for (unsigned jj = 0; jj < j_.nterms(); ++jj) {
74  j_.jterm(jj).setPostProcAD();
75  }
76 
77 // Multiply by B
78  CtrlInc_ zz(j_.jb());
79  j_.jb().multiplyB(ww, zz);
80 
81 // Run TLM
83  for (unsigned jj = 0; jj < j_.nterms(); ++jj) {
84  j_.jterm(jj).setPostProcTL(zz, costtl);
85  }
86 
87  CtrlInc_ mzz(zz);
88  j_.runTLM(mzz, costtl);
89 
90 // Get TLM outputs
91  dz.clear();
92  for (unsigned jj = 0; jj < j_.nterms(); ++jj) {
93  std::unique_ptr<GeneralizedDepartures> ztmp = j_.jterm(jj).newDualVector();
94  j_.jterm(jj).computeCostTL(zz, *ztmp);
95  dz.append(std::move(ztmp));
96  }
97 
98 // Tests
99  if (test_) {
100  // <G dx, dy >, where dx = B Gt dy
101  double adj_tst_fwd = dot_product(dz, dy);
102  // < dx, Gt dy>, where dx = B Gt dy
103  double adj_tst_bwd = dot_product(zz, ww);
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_HBHTMATRIX_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
The matrix.
Definition: HBHtMatrix.h:35
void multiply(const Dual_ &dy, Dual_ &dz) const
Definition: HBHtMatrix.h:61
CostFct_ const & j_
Definition: HBHtMatrix.h:46
DualVector< MODEL, OBS > Dual_
Definition: HBHtMatrix.h:38
ControlIncrement< MODEL, OBS > CtrlInc_
Definition: HBHtMatrix.h:36
CostFunction< MODEL, OBS > CostFct_
Definition: HBHtMatrix.h:37
HBHtMatrix(const CostFct_ &j, const bool test=false)
Definition: HBHtMatrix.h:54
Control model post processing.
The namespace for the main oops code.