OOPS
CostJbJq.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_COSTJBJQ_H_
12 #define OOPS_ASSIMILATION_COSTJBJQ_H_
13 
14 #include <memory>
15 #include <vector>
16 
17 #include "eckit/config/LocalConfiguration.h"
20 #include "oops/base/Geometry.h"
21 #include "oops/base/Increment.h"
23 #include "oops/base/State.h"
24 #include "oops/base/Variables.h"
25 #include "oops/util/dot_product.h"
26 #include "oops/util/Duration.h"
27 #include "oops/util/Logger.h"
28 
29 namespace oops {
30 
31 // -----------------------------------------------------------------------------
32 
33 /// Jb + Jq Cost Function
34 /*!
35  * CostJbJq encapsulates the generalized Jb term of the cost weak
36  * constraint 4D-Var function (ie Jb+Jq).
37  */
38 
39 template<typename MODEL> class CostJbJq : public CostJbState<MODEL> {
43 
44  public:
45 /// Construct \f$ J_b\f$.
46  CostJbJq(const eckit::Configuration &, const eckit::mpi::Comm &,
47  const Geometry_ &, const Variables &, const State_ &);
48 
49 /// Destructor
50  virtual ~CostJbJq() {}
51 
52 /// Get increment from state (usually first guess).
53  void computeIncrement(const State_ &, const State_ &, const State_ &,
54  Increment_ &) const override;
55 
56 /// Linearize before the linear computations.
57  void linearize(const State_ &, const Geometry_ &) override;
58 
59 /// Add Jb gradient.
60  void addGradient(const Increment_ &, Increment_ &, Increment_ &) const override;
61 
62 /// Finalize \f$ J_q\f$ after the model run.
63  JqTermTLAD<MODEL> * initializeJqTLAD() const override;
64 
65 /// Finalize \f$ J_q\f$ after the TL run.
66  JqTermTLAD<MODEL> * initializeJqTL() const override;
67 
68 /// Initialize \f$ J_q\f$ forcing before the AD run.
69  JqTermTLAD<MODEL> * initializeJqAD(const Increment_ &) const override;
70 
71 /// Multiply by \f$ B\f$ and \f$ B^{-1}\f$.
72  void Bmult(const Increment_ &, Increment_ &) const override;
73  void Bminv(const Increment_ &, Increment_ &) const override;
74 
75 /// Randomize
76  void randomize(Increment_ &) const override;
77 
78 /// Create new increment (set to 0).
79  Increment_ * newStateIncrement() const override;
80 
81  private:
82  std::unique_ptr<ModelSpaceCovarianceBase<MODEL> > B_;
83  const State_ & xb_;
85  std::unique_ptr<const Geometry_> resol_;
86  const eckit::LocalConfiguration conf_;
87  const eckit::mpi::Comm & commTime_;
88  const bool first_;
89 };
90 
91 // =============================================================================
92 
93 // Generalized Jb Term of Cost Function
94 // -----------------------------------------------------------------------------
95 
96 template<typename MODEL>
97 CostJbJq<MODEL>::CostJbJq(const eckit::Configuration & config, const eckit::mpi::Comm & comm,
98  const Geometry_ & resolouter, const Variables & ctlvars,
99  const State_ & xb)
100  : B_(), xb_(xb), ctlvars_(ctlvars), resol_(), conf_(config), commTime_(comm),
101  first_(comm.rank() == 0)
102 {
103  Log::trace() << "CostJbJq contructed." << std::endl;
104 }
105 
106 // -----------------------------------------------------------------------------
107 
108 template<typename MODEL>
109 void CostJbJq<MODEL>::linearize(const State_ & fg, const Geometry_ & lowres) {
110  Log::trace() << "CostJbJq::linearize start" << std::endl;
111  resol_.reset(new Geometry_(lowres));
112  const eckit::LocalConfiguration covConf(conf_, "background error");
113 
114  std::vector<eckit::LocalConfiguration> confs;
115  covConf.get("covariances", confs);
116  ASSERT(confs.size() == lowres.timeComm().size());
117  eckit::LocalConfiguration myconf = confs[lowres.timeComm().rank()];
118 
119  B_.reset(CovarianceFactory<MODEL>::create(myconf, lowres, ctlvars_, xb_, fg));
120  Log::trace() << "CostJbJq::linearize done" << std::endl;
121 }
122 
123 // -----------------------------------------------------------------------------
124 
125 template<typename MODEL>
126 void CostJbJq<MODEL>::computeIncrement(const State_ & xb, const State_ & fg, const State_ & mx,
127  Increment_ & dx) const {
128  Log::trace() << "CostJbJq::computeIncrement start" << std::endl;
129  static int tag = 13579;
130  size_t mytime = commTime_.rank();
131  State_ mxim1(fg);
132 
133 // Send values of M(x_i) at end of my subwindow to next subwindow
134  if (mytime + 1 < commTime_.size()) {
135  oops::mpi::send(commTime_, mx, mytime+1, tag);
136  }
137 
138 // Receive values at beginning of my subwindow from previous subwindow
139  if (mytime > 0) {
140  oops::mpi::receive(commTime_, mxim1, mytime-1, tag);
141  } else {
142  mxim1 = xb;
143  }
144 
145 // Compute x_i - M(x_{i-1})
146  dx.diff(fg, mxim1);
147 
148  ++tag;
149  Log::info() << "CostJbJq: x_i - M(x_{i-1})" << dx << std::endl;
150  Log::trace() << "CostJbJq::computeIncrement done" << std::endl;
151 }
152 
153 // -----------------------------------------------------------------------------
154 
155 template<typename MODEL>
157  Increment_ & gradJb) const {
158  Log::trace() << "CostJbJq::addGradient start" << std::endl;
159  if (first_) {
160 // Jb from pre-computed gradient
161  grad += gradJb;
162 
163  } else {
164 // Compute and add Jq gradient Qi^{-1} ( x_i - M(x_{i-1}) )
165  Increment_ gg(grad, false);
166  B_->inverseMultiply(dxFG, gg);
167  grad += gg;
168  }
169  Log::trace() << "CostJbJq::addGradient done" << std::endl;
170 }
171 
172 // -----------------------------------------------------------------------------
173 
174 template<typename MODEL>
176  Log::trace() << "CostJbJq::initializeJqTLAD" << std::endl;
177  return new JqTermTLAD<MODEL>(commTime_);
178 }
179 
180 // -----------------------------------------------------------------------------
181 
182 template<typename MODEL>
184  Log::trace() << "CostJbJq::initializeJqTL start" << std::endl;
185  JqTermTLAD<MODEL> * jqtl = new JqTermTLAD<MODEL>(commTime_);
186  Log::trace() << "CostJbJq::initializeJqTL done" << std::endl;
187  return jqtl;
188 }
189 
190 // -----------------------------------------------------------------------------
191 
192 template<typename MODEL>
194  Log::trace() << "CostJbJq::initializeJqAD start" << std::endl;
195  JqTermTLAD<MODEL> * jqad = new JqTermTLAD<MODEL>(commTime_);
196  jqad->setupAD(dx);
197  Log::trace() << "CostJbJq::initializeJqAD done" << std::endl;
198  return jqad;
199 }
200 
201 // -----------------------------------------------------------------------------
202 
203 template<typename MODEL>
204 void CostJbJq<MODEL>::Bmult(const Increment_ & dxin, Increment_ & dxout) const {
205  Log::trace() << "CostJbJq::Bmult start" << std::endl;
206  B_->multiply(dxin, dxout);
207  Log::trace() << "CostJbJq::Bmult done" << std::endl;
208 }
209 
210 // -----------------------------------------------------------------------------
211 
212 template<typename MODEL>
213 void CostJbJq<MODEL>::Bminv(const Increment_ & dxin, Increment_ & dxout) const {
214  Log::trace() << "CostJbJq::Bminv start" << std::endl;
215  B_->inverseMultiply(dxin, dxout);
216  Log::trace() << "CostJbJq::Bminv done" << std::endl;
217 }
218 
219 // -----------------------------------------------------------------------------
220 
221 template<typename MODEL>
223  Log::trace() << "CostJbJq::randomize start" << std::endl;
224  B_->randomize(dx);
225  Log::trace() << "CostJbJq::randomize done" << std::endl;
226 }
227 
228 // -----------------------------------------------------------------------------
229 
230 template<typename MODEL>
232  Log::trace() << "CostJbJq::newStateIncrement start" << std::endl;
233  Increment_ * incr = new Increment_(*resol_, ctlvars_, xb_.validTime());
234  Log::trace() << "CostJbJq::newStateIncrement done" << std::endl;
235  return incr;
236 }
237 
238 // -----------------------------------------------------------------------------
239 
240 } // namespace oops
241 
242 #endif // OOPS_ASSIMILATION_COSTJBJQ_H_
Jb + Jq Cost Function.
Definition: CostJbJq.h:39
void Bmult(const Increment_ &, Increment_ &) const override
Multiply by and .
Definition: CostJbJq.h:204
const State_ & xb_
Definition: CostJbJq.h:83
std::unique_ptr< const Geometry_ > resol_
Definition: CostJbJq.h:85
void randomize(Increment_ &) const override
Randomize.
Definition: CostJbJq.h:222
const eckit::mpi::Comm & commTime_
Definition: CostJbJq.h:87
Increment_ * newStateIncrement() const override
Create new increment (set to 0).
Definition: CostJbJq.h:231
CostJbJq(const eckit::Configuration &, const eckit::mpi::Comm &, const Geometry_ &, const Variables &, const State_ &)
Construct .
Definition: CostJbJq.h:97
const bool first_
Definition: CostJbJq.h:88
void computeIncrement(const State_ &, const State_ &, const State_ &, Increment_ &) const override
Get increment from state (usually first guess).
Definition: CostJbJq.h:126
const Variables ctlvars_
Definition: CostJbJq.h:84
JqTermTLAD< MODEL > * initializeJqTL() const override
Finalize after the TL run.
Definition: CostJbJq.h:183
State< MODEL > State_
Definition: CostJbJq.h:42
void linearize(const State_ &, const Geometry_ &) override
Linearize before the linear computations.
Definition: CostJbJq.h:109
Increment< MODEL > Increment_
Definition: CostJbJq.h:41
virtual ~CostJbJq()
Destructor.
Definition: CostJbJq.h:50
std::unique_ptr< ModelSpaceCovarianceBase< MODEL > > B_
Definition: CostJbJq.h:82
JqTermTLAD< MODEL > * initializeJqAD(const Increment_ &) const override
Initialize forcing before the AD run.
Definition: CostJbJq.h:193
void addGradient(const Increment_ &, Increment_ &, Increment_ &) const override
Add Jb gradient.
Definition: CostJbJq.h:156
void Bminv(const Increment_ &, Increment_ &) const override
Definition: CostJbJq.h:213
JqTermTLAD< MODEL > * initializeJqTLAD() const override
Finalize after the model run.
Definition: CostJbJq.h:175
Geometry< MODEL > Geometry_
Definition: CostJbJq.h:40
const eckit::LocalConfiguration conf_
Definition: CostJbJq.h:86
Jb Cost Function Base Class.
Definition: CostJbState.h:37
Geometry class used in oops; subclass of interface class interface::Geometry.
const eckit::mpi::Comm & timeComm() const
Accessor to the MPI communicator for distribution in time.
Increment class used in oops.
void setupAD(const Increment_ &dx)
Definition: JqTermTLAD.h:160
State class used in oops; subclass of interface class interface::State.
void diff(const State_ &state1, const State_ &state2)
Set this Increment to be difference between state1 and state2.
void send(const eckit::mpi::Comm &comm, const SERIALIZABLE &sendobj, const int dest, const int tag)
Extend eckit Comm for Serializable oops objects.
Definition: oops/mpi/mpi.h:44
void receive(const eckit::mpi::Comm &comm, SERIALIZABLE &recvobj, const int source, const int tag)
Definition: oops/mpi/mpi.h:55
The namespace for the main oops code.