OOPS
CostJo.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_COSTJO_H_
12 #define OOPS_ASSIMILATION_COSTJO_H_
13 
14 #include <memory>
15 #include <string>
16 #include <utility>
17 #include <vector>
18 
19 #include <boost/noncopyable.hpp>
20 
21 #include "eckit/config/LocalConfiguration.h"
25 #include "oops/base/Departures.h"
26 #include "oops/base/Geometry.h"
28 #include "oops/base/ObsErrors.h"
29 #include "oops/base/Observations.h"
30 #include "oops/base/Observers.h"
32 #include "oops/base/ObsSpaces.h"
35 #include "oops/base/State.h"
36 #include "oops/mpi/mpi.h"
37 #include "oops/util/DateTime.h"
38 #include "oops/util/Logger.h"
39 
40 namespace oops {
41 
42 // -----------------------------------------------------------------------------
43 
44 /// Jo Cost Function
45 /*!
46  * The CostJo class encapsulates the Jo term of the cost function.
47  * The Observers to be called during the model integration is managed
48  * inside the CostJo class.
49  */
50 
51 template<typename MODEL, typename OBS> class CostJo : public CostTermBase<MODEL, OBS>,
52  private boost::noncopyable {
66 
67  public:
68  /// Construct \f$ J_o\f$ from \f$ R\f$ and \f$ y_{obs}\f$.
69  CostJo(const eckit::Configuration &, const eckit::mpi::Comm &,
70  const util::DateTime &, const util::DateTime &,
71  const eckit::mpi::Comm & ctime = oops::mpi::myself());
72 
73  /// Destructor
74  virtual ~CostJo();
75 
76  /// Initialize \f$ J_o\f$ before starting the integration of the model.
77  void setPostProc(const CtrlVar_ &, const eckit::Configuration &, PostProc_ &) override;
78  /// Finalize \f$ J_o\f$ after the integration of the model.
79  double computeCost() override;
80 
81  /// Initialize \f$ J_o\f$ for the trajectory run
82  void setPostProcTraj(const CtrlVar_ &, const eckit::Configuration &,
83  const Geometry_ &, PostProcTLAD_ &) override;
84  void computeCostTraj() override;
85 
86  /// Initialize \f$ J_o\f$ before starting the TL run.
87  void setPostProcTL(const CtrlInc_ &, PostProcTLAD_ &) const override;
88  void computeCostTL(const CtrlInc_ &, GeneralizedDepartures &) const override;
89 
90  /// Initialize \f$ J_o\f$ before starting the AD run.
91  void computeCostAD(std::shared_ptr<const GeneralizedDepartures>,
92  CtrlInc_ &, PostProcTLAD_ &) const override;
93  void setPostProcAD() const override;
94 
95  /// Multiply by \f$ R\f$ and \f$ R^{-1}\f$.
96  std::unique_ptr<GeneralizedDepartures>
97  multiplyCovar(const GeneralizedDepartures &) const override;
98  std::unique_ptr<GeneralizedDepartures>
99  multiplyCoInv(const GeneralizedDepartures &) const override;
100 
101  /// Provide new departure.
102  std::unique_ptr<GeneralizedDepartures> newDualVector() const override;
103 
104  /// Return gradient at first guess ie \f$ R^{-1} {\cal H}(x^t ) - y\f$.
105  std::unique_ptr<GeneralizedDepartures> newGradientFG() const override;
106 
107  /// Reset obs operator trajectory.
108  void resetLinearization() override;
109 
110  /// Accessor...
111  const ObsSpaces_ & obspaces() const {return obspaces_;}
112 
113  private:
114  const eckit::LocalConfiguration obsconf_;
116  std::unique_ptr<Observations_> yobs_;
119 
120  /// Jo Gradient at first guess : \f$ R^{-1} (H(x_{fg})-y_{obs}) \f$.
121  std::unique_ptr<Departures_> gradFG_;
122 
123  /// Linearized observation operators.
124  std::shared_ptr<ObserversTLAD_> obstlad_;
125 
126  /// Configuration for current initialize/finalize pair
127  std::unique_ptr<eckit::LocalConfiguration> currentConf_;
128 };
129 
130 // =============================================================================
131 
132 template<typename MODEL, typename OBS>
133 CostJo<MODEL, OBS>::CostJo(const eckit::Configuration & joConf, const eckit::mpi::Comm & comm,
134  const util::DateTime & winbgn, const util::DateTime & winend,
135  const eckit::mpi::Comm & ctime)
136  : obsconf_(joConf), obspaces_(obsconf_, comm, winbgn, winend, ctime),
137  Rmat_(obsconf_, obspaces_), observers_(obspaces_, joConf),
138  gradFG_(), obstlad_(), currentConf_()
139 {
140  Log::trace() << "CostJo::CostJo" << std::endl;
141 }
142 
143 // -----------------------------------------------------------------------------
144 
145 template<typename MODEL, typename OBS>
147  obspaces_.save();
148  Log::trace() << "CostJo::~CostJo" << std::endl;
149 }
150 
151 // -----------------------------------------------------------------------------
152 
153 template<typename MODEL, typename OBS>
154 void CostJo<MODEL, OBS>::setPostProc(const CtrlVar_ & xx, const eckit::Configuration & conf,
155  PostProc_ & pp) {
156  Log::trace() << "CostJo::setPostProc start" << std::endl;
157  gradFG_.reset();
158 
159  currentConf_.reset(new eckit::LocalConfiguration(conf));
160 
161  observers_.initialize(xx.state().geometry(), xx.obsVar(), Rmat_, pp, conf);
162 
163  Log::trace() << "CostJo::setPostProc done" << std::endl;
164 }
165 
166 // -----------------------------------------------------------------------------
167 
168 template<typename MODEL, typename OBS>
170  Log::trace() << "CostJo::computeCost start" << std::endl;
171 
172  // Obs, simulated obs and departures (held here for nice prints and diagnostics)
173  if (!yobs_)
174  yobs_.reset(new Observations_(obspaces_, "ObsValue"));
175  Observations_ yeqv(obspaces_);
176  observers_.finalize(yeqv);
177 
178  // Perturb observations according to obs error statistics
179  bool obspert = currentConf_->getBool("obs perturbations", false);
180  if (obspert) {
181  yobs_->perturb(Rmat_);
182  Log::info() << "Perturbed observations: " << *yobs_ << std::endl;
183  }
184 
185  // Compute observations departures and save to output file
186  Departures_ ydep(yeqv - *yobs_);
187  if (currentConf_->has("diagnostics.departures")) {
188  const std::string depname = currentConf_->getString("diagnostics.departures");
189  ydep.save(depname);
190  }
191 
192  // Gradient at first guess (to define inner loop rhs)
193  gradFG_.reset(new Departures_(ydep));
194  Rmat_.inverseMultiply(*gradFG_);
195 
196  // Print diagnostics
197  Log::info() << "Jo Observations:" << std::endl << *yobs_
198  << "End Jo Observations" << std::endl;
199 
200  Log::info() << "Jo Observations Equivalent:" << std::endl << yeqv
201  << "End Jo Observations Equivalent" << std::endl;
202 
203  Log::info() << "Jo Bias Corrected Departures:" << std::endl << ydep
204  << "End Jo Bias Corrected Departures" << std::endl;
205 
206  Log::info() << "Jo Observations Errors:" << std::endl << Rmat_
207  << "End Jo Observations Errors" << std::endl;
208 
209  // Print Jo table
210  double zjo = 0.0;
211  std::vector<eckit::LocalConfiguration> typeconfs = obsconf_.getSubConfigurations();
212  for (size_t jj = 0; jj < obspaces_.size(); ++jj) {
213  double zz = 0.0;
214  const unsigned nobs = (*gradFG_)[jj].nobs();
215  Log::test() << "CostJo : Nonlinear Jo(" << obspaces_[jj].obsname() << ") = ";
216 
217  if (nobs > 0) {
218  zz = 0.5 * dot_product(ydep[jj], (*gradFG_)[jj]);
219  const double err = Rmat_[jj].getRMSE();
220  Log::test() << zz << ", nobs = " << nobs << ", Jo/n = " << zz/nobs << ", err = " << err;
221  } else {
222  Log::test() << zz << " --- No Observations";
223  }
224 
225  if (typeconfs[jj].getBool("monitoring only", false)) {
226  Log::test() << " (Monitoring only)";
227  } else {
228  zjo += zz;
229  }
230  Log::test() << std::endl;
231  }
232 
233  Log::trace() << "CostJo::computeCost done" << std::endl;
234  return zjo;
235 }
236 
237 // -----------------------------------------------------------------------------
238 
239 template<typename MODEL, typename OBS>
240 void CostJo<MODEL, OBS>::setPostProcTraj(const CtrlVar_ & xx, const eckit::Configuration & conf,
241  const Geometry_ & lowres, PostProcTLAD_ & pptraj) {
242  Log::trace() << "CostJo::setPostProcTraj start" << std::endl;
243  obstlad_.reset(new ObserversTLAD_(obspaces_, obsconf_));
244  obstlad_->initializeTraj(lowres, xx.obsVar(), pptraj);
245  Log::trace() << "CostJo::setPostProcTraj done" << std::endl;
246 }
247 
248 // -----------------------------------------------------------------------------
249 
250 template<typename MODEL, typename OBS>
252  obstlad_->finalizeTraj();
253  Log::trace() << "CostJo::computeCostTraj done" << std::endl;
254 }
255 
256 // -----------------------------------------------------------------------------
257 
258 template<typename MODEL, typename OBS>
260  Log::trace() << "CostJo::setPostProcTL start" << std::endl;
261  ASSERT(obstlad_);
262  obstlad_->initializeTL(pptl);
263  Log::trace() << "CostJo::setPostProcTL done" << std::endl;
264 }
265 
266 // -----------------------------------------------------------------------------
267 
268 template<typename MODEL, typename OBS>
270  Log::trace() << "CostJo::computeCostTL start" << std::endl;
271  ASSERT(obstlad_);
272  Departures_ & ydep = dynamic_cast<Departures_ &>(gdep);
273  obstlad_->finalizeTL(dx.obsVar(), ydep);
274  Log::trace() << "CostJo::computeCostTL done" << std::endl;
275 }
276 
277 // -----------------------------------------------------------------------------
278 
279 template<typename MODEL, typename OBS>
280 void CostJo<MODEL, OBS>::computeCostAD(std::shared_ptr<const GeneralizedDepartures> pv,
281  CtrlInc_ & dx, PostProcTLAD_ & ppad) const {
282  Log::trace() << "CostJo::computeCostAD start" << std::endl;
283  ASSERT(obstlad_);
284  std::shared_ptr<const Departures_> dy = std::dynamic_pointer_cast<const Departures_>(pv);
285  obstlad_->initializeAD(*dy, dx.obsVar(), ppad);
286  Log::trace() << "CostJo::computeCostAD done" << std::endl;
287 }
288 
289 // -----------------------------------------------------------------------------
290 
291 template<typename MODEL, typename OBS>
293  Log::trace() << "CostJo::setPostProcAD start" << std::endl;
294  ASSERT(obstlad_);
295  obstlad_->finalizeAD();
296  Log::trace() << "CostJo::setPostProcAD done" << std::endl;
297 }
298 
299 // -----------------------------------------------------------------------------
300 
301 template<typename MODEL, typename OBS>
302 std::unique_ptr<GeneralizedDepartures>
304  Log::trace() << "CostJo::multiplyCovar start" << std::endl;
305  std::unique_ptr<Departures_> y1(new Departures_(dynamic_cast<const Departures_ &>(v1)));
306  Rmat_.multiply(*y1);
307  return std::move(y1);
308 }
309 
310 // -----------------------------------------------------------------------------
311 
312 template<typename MODEL, typename OBS>
313 std::unique_ptr<GeneralizedDepartures>
315  Log::trace() << "CostJo::multiplyCoInv start" << std::endl;
316  std::unique_ptr<Departures_> y1(new Departures_(dynamic_cast<const Departures_ &>(v1)));
317  Rmat_.inverseMultiply(*y1);
318  return std::move(y1);
319 }
320 
321 // -----------------------------------------------------------------------------
322 
323 template<typename MODEL, typename OBS>
324 std::unique_ptr<GeneralizedDepartures> CostJo<MODEL, OBS>::newDualVector() const {
325  Log::trace() << "CostJo::newDualVector start" << std::endl;
326  std::unique_ptr<Departures_> ydep(new Departures_(obspaces_));
327  ydep->zero();
328  Log::trace() << "CostJo::newDualVector done" << std::endl;
329  return std::move(ydep);
330 }
331 
332 // -----------------------------------------------------------------------------
333 
334 template<typename MODEL, typename OBS>
335 std::unique_ptr<GeneralizedDepartures> CostJo<MODEL, OBS>::newGradientFG() const {
336  return std::unique_ptr<Departures_>(new Departures_(*gradFG_));
337 }
338 
339 // -----------------------------------------------------------------------------
340 
341 template<typename MODEL, typename OBS>
343  Log::trace() << "CostJo::resetLinearization start" << std::endl;
344  obstlad_.reset();
345  Log::trace() << "CostJo::resetLinearization done" << std::endl;
346 }
347 
348 // -----------------------------------------------------------------------------
349 
350 } // namespace oops
351 
352 #endif // OOPS_ASSIMILATION_COSTJO_H_
ObsAuxIncrs_ & obsVar()
Get augmented observation control variable.
Control variable.
ObsAuxCtrls_ & obsVar()
Get augmented observation control variable.
State_ & state()
Get state control variable.
Jo Cost Function.
Definition: CostJo.h:52
std::shared_ptr< ObserversTLAD_ > obstlad_
Linearized observation operators.
Definition: CostJo.h:124
CostJo(const eckit::Configuration &, const eckit::mpi::Comm &, const util::DateTime &, const util::DateTime &, const eckit::mpi::Comm &ctime=oops::mpi::myself())
Construct from and .
Definition: CostJo.h:133
void computeCostTL(const CtrlInc_ &, GeneralizedDepartures &) const override
Finish cost computation after TL model integration.
Definition: CostJo.h:269
std::unique_ptr< GeneralizedDepartures > newDualVector() const override
Provide new departure.
Definition: CostJo.h:324
void setPostProcTL(const CtrlInc_ &, PostProcTLAD_ &) const override
Initialize before starting the TL run.
Definition: CostJo.h:259
PostProcessor< State_ > PostProc_
Definition: CostJo.h:64
ObsSpaces_ obspaces_
Definition: CostJo.h:115
double computeCost() override
Finalize after the integration of the model.
Definition: CostJo.h:169
std::unique_ptr< GeneralizedDepartures > newGradientFG() const override
Return gradient at first guess ie .
Definition: CostJo.h:335
std::unique_ptr< GeneralizedDepartures > multiplyCoInv(const GeneralizedDepartures &) const override
Definition: CostJo.h:314
GetValuePosts< MODEL, OBS > GetValuePosts_
Definition: CostJo.h:58
void resetLinearization() override
Reset obs operator trajectory.
Definition: CostJo.h:342
Departures< OBS > Departures_
Definition: CostJo.h:55
virtual ~CostJo()
Destructor.
Definition: CostJo.h:146
ObsSpaces< OBS > ObsSpaces_
Definition: CostJo.h:61
void setPostProcTraj(const CtrlVar_ &, const eckit::Configuration &, const Geometry_ &, PostProcTLAD_ &) override
Initialize for the trajectory run.
Definition: CostJo.h:240
std::unique_ptr< GeneralizedDepartures > multiplyCovar(const GeneralizedDepartures &) const override
Multiply by and .
Definition: CostJo.h:303
ObsErrors_ Rmat_
Definition: CostJo.h:117
void setPostProc(const CtrlVar_ &, const eckit::Configuration &, PostProc_ &) override
Initialize before starting the integration of the model.
Definition: CostJo.h:154
void computeCostTraj() override
Finish cost computation and trajectory handling after nonlinear model integration.
Definition: CostJo.h:251
void setPostProcAD() const override
Adjoint ot setPostProcTL (clean-up)
Definition: CostJo.h:292
PostProcessorTLAD< MODEL > PostProcTLAD_
Definition: CostJo.h:65
ObsErrors< OBS > ObsErrors_
Definition: CostJo.h:60
Observations< OBS > Observations_
Definition: CostJo.h:56
ObserversTLAD< MODEL, OBS > ObserversTLAD_
Definition: CostJo.h:63
ControlVariable< MODEL, OBS > CtrlVar_
Definition: CostJo.h:53
std::unique_ptr< eckit::LocalConfiguration > currentConf_
Configuration for current initialize/finalize pair.
Definition: CostJo.h:127
void computeCostAD(std::shared_ptr< const GeneralizedDepartures >, CtrlInc_ &, PostProcTLAD_ &) const override
Initialize before starting the AD run.
Definition: CostJo.h:280
const ObsSpaces_ & obspaces() const
Accessor...
Definition: CostJo.h:111
ControlIncrement< MODEL, OBS > CtrlInc_
Definition: CostJo.h:54
std::unique_ptr< Departures_ > gradFG_
Jo Gradient at first guess : .
Definition: CostJo.h:121
Observers_ observers_
Definition: CostJo.h:118
const eckit::LocalConfiguration obsconf_
Definition: CostJo.h:114
Geometry< MODEL > Geometry_
Definition: CostJo.h:57
Observers< MODEL, OBS > Observers_
Definition: CostJo.h:62
std::unique_ptr< Observations_ > yobs_
Definition: CostJo.h:116
State< MODEL > State_
Definition: CostJo.h:59
Base Class for Cost Function Terms.
Definition: CostTermBase.h:36
Difference between two observation vectors.
Definition: Departures.h:44
void save(const std::string &) const
Save departures values.
Definition: Departures.h:249
size_t nobs() const
Return number of departures (excluding departures that are masked out)
Definition: Departures.h:200
Abstract base class for quantities.
Geometry class used in oops; subclass of interface class interface::Geometry.
Fills GeoVaLs with requested variables at requested locations during model run.
Definition: GetValuePosts.h:29
Container for ObsErrors for all observation types that are used in DA.
Definition: ObsErrors.h:34
Observations Class.
Definition: Observations.h:35
Computes observation operator (from GeoVaLs), applies bias correction and runs QC filters.
Definition: Observers.h:36
Computes observation equivalent TL and AD to/from increments.
Definition: ObserversTLAD.h:37
Control model post processing.
Definition: PostProcessor.h:30
Control model post processing.
State class used in oops; subclass of interface class interface::State.
Geometry_ geometry() const
Accessor to geometry associated with this State.
const eckit::mpi::Comm & myself()
Default communicator with each MPI task by itself.
Definition: oops/mpi/mpi.cc:90
The namespace for the main oops code.