Loading [MathJax]/extensions/tex2jax.js
OOPS
All Classes Namespaces Files Functions Variables Typedefs Macros Pages
CostJbTotal.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_COSTJBTOTAL_H_
12 #define OOPS_ASSIMILATION_COSTJBTOTAL_H_
13 
14 #include <limits>
15 #include <memory>
16 
17 #include "eckit/config/LocalConfiguration.h"
21 #include "oops/base/Geometry.h"
23 #include "oops/base/ObsSpaces.h"
25 #include "oops/base/State.h"
27 #include "oops/util/Logger.h"
28 
29 namespace oops {
30  template<typename MODEL> class JqTermTLAD;
31 
32 // -----------------------------------------------------------------------------
33 
34 /// Total Jb cost function for all components of the control variable.
35 
36 template<typename MODEL, typename OBS> class CostJbTotal {
47 
48  public:
49 /// Construct \f$ J_b\f$.
50  CostJbTotal(const CtrlVar_ &, JbState_ *, const eckit::Configuration &,
51  const Geometry_ &, const ObsSpaces_ & odb);
52 
53 /// Destructor
55 
56 /// Initialize before nonlinear model integration.
57  void initialize(const CtrlVar_ &) const;
58  void initializeTraj(const CtrlVar_ &, const Geometry_ &,
59  const eckit::Configuration &, PostProcTLAD_ &);
60 
61 /// Finalize computation after nonlinear model integration.
62  double finalize(const CtrlVar_ &) const;
63  void finalizeTraj();
64 
65 /// Initialize before starting the TL run.
66  void initializeTL(PostProcTLAD_ &) const;
67  void finalizeTL(const CtrlInc_ &, CtrlInc_ &) const;
68 
69 /// Initialize before starting the AD run.
70  void initializeAD(CtrlInc_ &, const CtrlInc_ &, PostProcTLAD_ &) const;
71  void finalizeAD() const;
72 
73 /// Multiply by covariance matrix and its inverse.
74  void multiplyB(const CtrlInc_ &, CtrlInc_ &) const;
75  void multiplyBinv(const CtrlInc_ &, CtrlInc_ &) const;
76 
77 /// Randomize
78  void randomize(CtrlInc_ &) const;
79 
80 /// Add Jb gradient at first guess.
81  void addGradientFG(CtrlInc_ &) const;
82  void addGradientFG(CtrlInc_ &, CtrlInc_ &) const;
83 
84 /// Return background.
85  const CtrlVar_ & getBackground() const {return xb_;}
86 
87 /// Return first guess \f$ x_0-x_b\f$.
88  const CtrlInc_ & getFirstGuess() const {return *dxFG_;}
89 
90 /// Jb terms for ControlIncrement constructor.
91  const Geometry_ & resolution() const {return *resol_;}
92  const JbState_ & jbState() const {return *jb_;}
93  const ModelAuxCovar_ & jbModBias() const {return jbModBias_;}
94  const ObsAuxCovars_ & jbObsBias() const {return jbObsBias_;}
95  const util::DateTime & windowBegin() const {return windowBegin_;}
96  const util::DateTime & windowEnd() const {return windowEnd_;}
97 
98  private:
99  double evaluate(const CtrlInc_ &) const;
100 
101  const CtrlVar_ & xb_;
102  std::unique_ptr<JbState_> jb_;
105 
106  const CtrlVar_ mutable * fg_;
107 
108 /// First guess increment \f$x_0-x_b\f$ or more generally \f$ x_i-M(x_{i-1})\f$.
109  std::unique_ptr<CtrlInc_> dxFG_;
110 
111 /// Inner loop resolution
112  std::unique_ptr<Geometry_> resol_;
113  const util::DateTime windowBegin_;
114  const util::DateTime windowEnd_;
115 
117  std::shared_ptr<JqTermTLAD_> jqtraj_;
118  mutable std::shared_ptr<JqTermTLAD_> jqtl_;
119  mutable std::shared_ptr<JqTermTLAD_> jqad_;
120 };
121 
122 // =============================================================================
123 
124 template<typename MODEL, typename OBS>
126  const eckit::Configuration & conf,
127  const Geometry_ & resol, const ObsSpaces_ & odb)
128  : xb_(xb), jb_(jb),
129  jbModBias_(conf.getSubConfiguration("model aux error"), resol),
130  jbObsBias_(odb, conf.getSubConfiguration("observations")), dxFG_(), resol_(),
131  windowBegin_(conf.getString("window begin")),
132  windowEnd_(windowBegin_ + util::Duration(conf.getString("window length"))),
133  jqtraj_()
134 {
135  jbEvaluation_ = conf.getBool("jb evaluation", true);
136  Log::trace() << "CostJbTotal contructed." << std::endl;
137 }
138 
139 // -----------------------------------------------------------------------------
140 
141 template<typename MODEL, typename OBS>
143  Log::trace() << "CostJbTotal::initialize start" << std::endl;
144  fg_ = &fg;
145  Log::trace() << "CostJbTotal::initialize done" << std::endl;
146 }
147 
148 // -----------------------------------------------------------------------------
149 
150 template<typename MODEL, typename OBS>
151 double CostJbTotal<MODEL, OBS>::finalize(const CtrlVar_ & mx) const {
152  Log::trace() << "CostJbTotal::finalize start" << std::endl;
153  ASSERT(fg_);
154  CtrlInc_ dx(*this);
155 
156 // Compute x_0 - x_b for Jb (and Jq is present)
157  jb_->computeIncrement(xb_.state(), fg_->state(), mx.state(), dx.state());
158 
159 // Model and Obs biases
160  dx.modVar().diff(fg_->modVar(), xb_.modVar());
161  dx.obsVar().diff(fg_->obsVar(), xb_.obsVar());
162 
163 // Print increment
164  Log::info() << "CostJb: FG-BG" << dx << std::endl;
165 
166 // Compute Jb value
167  double zjb = 0.0;
168  if (jbEvaluation_) zjb = this->evaluate(dx);
169  Log::trace() << "CostJbTotal::finalize done" << std::endl;
170  return zjb;
171 }
172 
173 // -----------------------------------------------------------------------------
174 
175 template<typename MODEL, typename OBS>
177  const eckit::Configuration & inner,
178  PostProcTLAD_ & pptraj) {
179  Log::trace() << "CostJbTotal::initializeTraj start" << std::endl;
180  fg_ = &fg;
181  resol_.reset(new Geometry_(resol));
182 // Linearize terms
183  jb_->linearize(fg.state(), *resol_);
184  jbModBias_.linearize(fg.modVar(), *resol_);
185  jbObsBias_.linearize(fg.obsVar(), inner);
186 
187  jqtraj_.reset(jb_->initializeJqTLAD());
188  pptraj.enrollProcessor(jqtraj_);
189 
190  Log::trace() << "CostJbTotal::initializeTraj done" << std::endl;
191 }
192 
193 // -----------------------------------------------------------------------------
194 
195 template<typename MODEL, typename OBS>
197  Log::trace() << "CostJbTotal::finalizeTraj start" << std::endl;
198  ASSERT(fg_);
199 // Compute and save first guess increment.
200  dxFG_.reset(new CtrlInc_(*this));
201 
202 // Compute x_0 - x_b for Jb (and Jq is present)
203  const State_ * mx = &fg_->state();
204  if (jqtraj_) mx = &jqtraj_->getMxi();
205  jb_->computeIncrement(xb_.state(), fg_->state(), *mx, dxFG_->state());
206 
207 // Model and Obs biases
208  dxFG_->modVar().diff(fg_->modVar(), xb_.modVar());
209  dxFG_->obsVar().diff(fg_->obsVar(), xb_.obsVar());
210 
211 // Print increment
212  Log::info() << "CostJb: FG-BG" << *dxFG_ << std::endl;
213  jqtraj_.reset();
214  Log::trace() << "CostJbTotal::finalizeTraj done" << std::endl;
215 }
216 
217 // -----------------------------------------------------------------------------
218 
219 template<typename MODEL, typename OBS>
220 double CostJbTotal<MODEL, OBS>::evaluate(const CtrlInc_ & dx) const {
221  Log::trace() << "CostJbTotal::evaluate start" << std::endl;
222  CtrlInc_ gg(*this);
223  this->multiplyBinv(dx, gg);
224 
225  double zjb = 0.0;
226  double zz = 0.5 * dot_product(dx.state(), gg.state());
227  Log::info() << "CostJb : Nonlinear Jb State = " << zz << std::endl;
228  zjb += zz;
229  zz = 0.5 * dot_product(dx.modVar(), gg.modVar());
230  Log::info() << "CostJb : Nonlinear Jb Model Aux = " << zz << std::endl;
231  zjb += zz;
232  zz = 0.5 * dot_product(dx.obsVar(), gg.obsVar());
233  Log::info() << "CostJb : Nonlinear Jb Obs Aux = " << zz << std::endl;
234  zjb += zz;
235 
236  Log::info() << "CostJb : Nonlinear Jb = " << zjb << std::endl;
237 
238 // Get rid of very small values for test
239  double ztest = zjb;
240  if (zjb >= 0.0 && zjb <= std::numeric_limits<double>::epsilon()) ztest = 0.0;
241  Log::test() << "CostJb : Nonlinear Jb = " << ztest << std::endl;
242 
243  Log::trace() << "CostJbTotal::evaluate done" << std::endl;
244  return zjb;
245 }
246 
247 // -----------------------------------------------------------------------------
248 
249 template<typename MODEL, typename OBS>
251  Log::trace() << "CostJbTotal::addGradientFG 1 start" << std::endl;
252  CtrlInc_ gg(grad, false);
253  this->multiplyBinv(*dxFG_, gg);
254  grad += gg;
255  Log::trace() << "CostJbTotal::addGradientFG 1 done" << std::endl;
256 }
257 
258 // -----------------------------------------------------------------------------
259 
260 template<typename MODEL, typename OBS>
262  Log::trace() << "CostJbTotal::addGradientFG 2 start" << std::endl;
263  jb_->addGradient(dxFG_->state(), grad.state(), gradJb.state());
264  grad.modVar() += gradJb.modVar();
265  grad.obsVar() += gradJb.obsVar();
266  Log::trace() << "CostJbTotal::addGradientFG 2 done" << std::endl;
267 }
268 
269 // -----------------------------------------------------------------------------
270 
271 template<typename MODEL, typename OBS>
273  Log::trace() << "CostJbTotal::initializeTL start" << std::endl;
274  jqtl_.reset(jb_->initializeJqTL());
275  pptl.enrollProcessor(jqtl_);
276  Log::trace() << "CostJbTotal::initializeTL done" << std::endl;
277 }
278 
279 // -----------------------------------------------------------------------------
280 
281 template<typename MODEL, typename OBS>
282 void CostJbTotal<MODEL, OBS>::finalizeTL(const CtrlInc_ & bgns, CtrlInc_ & dx) const {
283  Log::trace() << "CostJbTotal::finalizeTL start" << std::endl;
284  dx = bgns;
285  if (jqtl_) jqtl_->computeModelErrorTL(dx.state());
286  jqtl_.reset();
287  Log::trace() << "CostJbTotal::finalizeTL done" << std::endl;
288 }
289 
290 // -----------------------------------------------------------------------------
291 
292 template<typename MODEL, typename OBS>
294  PostProcTLAD_ & ppad) const {
295  Log::trace() << "CostJbTotal::initializeAD start" << std::endl;
296  jqad_.reset(jb_->initializeJqAD(dx.state()));
297  bgns += dx;
298  ppad.enrollProcessor(jqad_);
299  Log::trace() << "CostJbTotal::initializeAD done" << std::endl;
300 }
301 
302 // -----------------------------------------------------------------------------
303 
304 template<typename MODEL, typename OBS>
306  Log::trace() << "CostJbTotal::finalizeAD start" << std::endl;
307  if (jqad_) jqad_->clear();
308  jqad_.reset();
309  Log::trace() << "CostJbTotal::finalizeAD done" << std::endl;
310 }
311 
312 // -----------------------------------------------------------------------------
313 
314 template<typename MODEL, typename OBS>
315 void CostJbTotal<MODEL, OBS>::multiplyB(const CtrlInc_ & dxin, CtrlInc_ & dxout) const {
316  Log::trace() << "CostJbTotal::multiplyB start" << std::endl;
317  jb_->Bmult(dxin.state(), dxout.state());
318  jbModBias_.multiply(dxin.modVar(), dxout.modVar());
319  jbObsBias_.multiply(dxin.obsVar(), dxout.obsVar());
320  Log::trace() << "CostJbTotal::multiplyB done" << std::endl;
321 }
322 
323 // -----------------------------------------------------------------------------
324 
325 template<typename MODEL, typename OBS>
326 void CostJbTotal<MODEL, OBS>::multiplyBinv(const CtrlInc_ & dxin, CtrlInc_ & dxout) const {
327  Log::trace() << "CostJbTotal::multiplyBinv start" << std::endl;
328  jb_->Bminv(dxin.state(), dxout.state());
329  jbModBias_.inverseMultiply(dxin.modVar(), dxout.modVar());
330  jbObsBias_.inverseMultiply(dxin.obsVar(), dxout.obsVar());
331  Log::trace() << "CostJbTotal::multiplyBinv done" << std::endl;
332 }
333 
334 // -----------------------------------------------------------------------------
335 
336 template<typename MODEL, typename OBS>
338  Log::trace() << "CostJbTotal::randomize start" << std::endl;
339  jb_->randomize(dx.state());
340  jbModBias_.randomize(dx.modVar());
341  jbObsBias_.randomize(dx.obsVar());
342  Log::trace() << "CostJbTotal::randomize done" << std::endl;
343 }
344 
345 // -----------------------------------------------------------------------------
346 
347 } // namespace oops
348 
349 #endif // OOPS_ASSIMILATION_COSTJBTOTAL_H_
ObsAuxIncrs_ & obsVar()
Get augmented observation control variable.
Increment_ & state()
Get state control variable.
ModelAuxIncr_ & modVar()
Get augmented model control variable.
Control variable.
ObsAuxCtrls_ & obsVar()
Get augmented observation control variable.
ModelAux_ & modVar()
Get augmented model control variable.
State_ & state()
Get state control variable.
Jb Cost Function Base Class.
Definition: CostJbState.h:37
Control variable increment.
Definition: CostJbTotal.h:36
PostProcessorTLAD< MODEL > PostProcTLAD_
Definition: CostJbTotal.h:46
double evaluate(const CtrlInc_ &) const
Definition: CostJbTotal.h:220
std::shared_ptr< JqTermTLAD_ > jqad_
Definition: CostJbTotal.h:119
void initialize(const CtrlVar_ &) const
Initialize before nonlinear model integration.
Definition: CostJbTotal.h:142
const Geometry_ & resolution() const
Jb terms for ControlIncrement constructor.
Definition: CostJbTotal.h:91
const util::DateTime & windowBegin() const
Definition: CostJbTotal.h:95
void addGradientFG(CtrlInc_ &) const
Add Jb gradient at first guess.
Definition: CostJbTotal.h:250
ControlIncrement< MODEL, OBS > CtrlInc_
Definition: CostJbTotal.h:37
void finalizeTL(const CtrlInc_ &, CtrlInc_ &) const
Definition: CostJbTotal.h:282
const JbState_ & jbState() const
Definition: CostJbTotal.h:92
JqTermTLAD< MODEL > JqTermTLAD_
Definition: CostJbTotal.h:41
std::unique_ptr< CtrlInc_ > dxFG_
First guess increment or more generally .
Definition: CostJbTotal.h:109
const CtrlInc_ & getFirstGuess() const
Return first guess .
Definition: CostJbTotal.h:88
const CtrlVar_ & getBackground() const
Return background.
Definition: CostJbTotal.h:85
ControlVariable< MODEL, OBS > CtrlVar_
Definition: CostJbTotal.h:38
Geometry< MODEL > Geometry_
Definition: CostJbTotal.h:42
void initializeAD(CtrlInc_ &, const CtrlInc_ &, PostProcTLAD_ &) const
Initialize before starting the AD run.
Definition: CostJbTotal.h:293
ObsAuxCovariances< OBS > ObsAuxCovars_
Definition: CostJbTotal.h:44
ModelAuxCovariance< MODEL > ModelAuxCovar_
Definition: CostJbTotal.h:43
std::unique_ptr< Geometry_ > resol_
Inner loop resolution.
Definition: CostJbTotal.h:112
ModelAuxCovar_ jbModBias_
Definition: CostJbTotal.h:103
double finalize(const CtrlVar_ &) const
Finalize computation after nonlinear model integration.
Definition: CostJbTotal.h:151
CostJbState< MODEL > JbState_
Definition: CostJbTotal.h:40
const ObsAuxCovars_ & jbObsBias() const
Definition: CostJbTotal.h:94
const util::DateTime windowBegin_
Definition: CostJbTotal.h:113
const CtrlVar_ * fg_
Definition: CostJbTotal.h:106
ObsSpaces< OBS > ObsSpaces_
Definition: CostJbTotal.h:45
std::shared_ptr< JqTermTLAD_ > jqtraj_
Definition: CostJbTotal.h:117
~CostJbTotal()
Destructor.
Definition: CostJbTotal.h:54
const CtrlVar_ & xb_
Definition: CostJbTotal.h:101
std::unique_ptr< JbState_ > jb_
Definition: CostJbTotal.h:102
const util::DateTime windowEnd_
Definition: CostJbTotal.h:114
void multiplyBinv(const CtrlInc_ &, CtrlInc_ &) const
Definition: CostJbTotal.h:326
void initializeTraj(const CtrlVar_ &, const Geometry_ &, const eckit::Configuration &, PostProcTLAD_ &)
Definition: CostJbTotal.h:176
void multiplyB(const CtrlInc_ &, CtrlInc_ &) const
Multiply by covariance matrix and its inverse.
Definition: CostJbTotal.h:315
std::shared_ptr< JqTermTLAD_ > jqtl_
Definition: CostJbTotal.h:118
void initializeTL(PostProcTLAD_ &) const
Initialize before starting the TL run.
Definition: CostJbTotal.h:272
void randomize(CtrlInc_ &) const
Randomize.
Definition: CostJbTotal.h:337
CostJbTotal(const CtrlVar_ &, JbState_ *, const eckit::Configuration &, const Geometry_ &, const ObsSpaces_ &odb)
Construct .
Definition: CostJbTotal.h:125
const ModelAuxCovar_ & jbModBias() const
Definition: CostJbTotal.h:93
const util::DateTime & windowEnd() const
Definition: CostJbTotal.h:96
void finalizeAD() const
Definition: CostJbTotal.h:305
State< MODEL > State_
Definition: CostJbTotal.h:39
ObsAuxCovars_ jbObsBias_
Definition: CostJbTotal.h:104
Geometry class used in oops; subclass of interface class interface::Geometry.
Auxiliary Error Covariance related to model, not used at the moment.
void diff(const ModelAuxControl_ &, const ModelAuxControl_ &)
Sets this ModelAuxIncrement to the difference between two ModelAuxControl objects.
Holds a vector of ObsAuxCovariance.
void diff(const ObsAuxControls_ &, const ObsAuxControls_ &)
Linear algebra operators.
Control model post processing.
void enrollProcessor(PostBaseTLAD_ *pp)
State class used in oops; subclass of interface class interface::State.
State_ & state()
Accessor.
The namespace for the main oops code.
Definition: TLML95.h:34