OOPS
CostFunction.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_COSTFUNCTION_H_
12 #define OOPS_ASSIMILATION_COSTFUNCTION_H_
13 
14 #include <map>
15 #include <memory>
16 #include <string>
17 #include <vector>
18 #include <boost/noncopyable.hpp>
19 #include <boost/ptr_container/ptr_vector.hpp>
20 
21 #include "eckit/config/LocalConfiguration.h"
22 #include "eckit/mpi/Comm.h"
31 #include "oops/base/Geometry.h"
32 #include "oops/base/Increment.h"
35 #include "oops/base/State.h"
36 #include "oops/base/Variables.h"
37 #include "oops/mpi/mpi.h"
38 #include "oops/util/DateTime.h"
39 #include "oops/util/dot_product.h"
40 #include "oops/util/Duration.h"
41 #include "oops/util/Logger.h"
42 
43 namespace oops {
44 
45 // -----------------------------------------------------------------------------
46 
47 /// Cost Function
48 /*!
49  * The CostFunction defines and manages the computation of all the terms
50  * of the variational data assimilation cost function.
51  */
52 
53 template<typename MODEL, typename OBS> class CostFunction : private boost::noncopyable {
62 
63  public:
64  explicit CostFunction(const eckit::Configuration &);
65  virtual ~CostFunction() {}
66 
67  double evaluate(const CtrlVar_ &,
68  const eckit::Configuration & config = eckit::LocalConfiguration(),
70  double linearize(const CtrlVar_ &, const eckit::Configuration &,
72 
75  const bool idModel = false) const = 0;
78  const bool idModel = false) const = 0;
79  virtual void zeroAD(CtrlInc_ &) const = 0;
80 
81  virtual void runNL(CtrlVar_ &, PostProcessor<State_>&) const = 0;
82 
83  void addIncrement(CtrlVar_ &, const CtrlInc_ &,
85  void resetLinearization();
86 
87 /// Compute cost function gradient at first guess (without Jb).
88  void computeGradientFG(CtrlInc_ &) const;
89 
90 /// Access \f$ J_b\f$
91  const JbTotal_ & jb() const {return *jb_;}
92 /// Access terms of the cost function other than \f$ J_b\f$
93  const CostBase_ & jterm(const size_t ii) const {return jterms_[ii];}
94  size_t nterms() const {return jterms_.size();}
95  double getCostJb() const {return costJb_;}
96  double getCostJoJc() const {return costJoJc_;}
97 
98  protected:
99  void setupTerms(const eckit::Configuration &);
100  const CtrlVar_ & background() const {return *xb_;}
101 
102  private:
103  virtual void addIncr(CtrlVar_ &, const CtrlInc_ &, PostProcessor<Increment_>&) const = 0;
104 
105  virtual CostJbState<MODEL> * newJb(const eckit::Configuration &, const Geometry_ &,
106  const CtrlVar_ &) const = 0;
107  virtual CostJo<MODEL, OBS> * newJo(const eckit::Configuration &) const = 0;
108  virtual CostTermBase<MODEL, OBS> * newJc(const eckit::Configuration &,
109  const Geometry_ &) const = 0;
110  virtual void doLinearize(const Geometry_ &, const eckit::Configuration &,
111  const CtrlVar_ &, const CtrlVar_ &,
113  virtual const Geometry_ & geometry() const = 0;
114 
115 // Data members
116  std::unique_ptr<const CtrlVar_> xb_;
117  std::unique_ptr<JbTotal_> jb_;
118  boost::ptr_vector<CostBase_> jterms_;
119 
120  mutable double costJb_;
121  mutable double costJoJc_;
122 };
123 
124 // -----------------------------------------------------------------------------
125 
126 /// Cost Function Factory
127 template <typename MODEL, typename OBS>
128 class CostFactory {
129  public:
130  static CostFunction<MODEL, OBS> * create(const eckit::Configuration &, const eckit::mpi::Comm &);
131  virtual ~CostFactory() = default;
132 
133  protected:
134  explicit CostFactory(const std::string &);
135  private:
136  virtual CostFunction<MODEL, OBS> * make(const eckit::Configuration &,
137  const eckit::mpi::Comm &) = 0;
138  static std::map < std::string, CostFactory<MODEL, OBS> * > & getMakers() {
139  static std::map < std::string, CostFactory<MODEL, OBS> * > makers_;
140  return makers_;
141  }
142 };
143 
144 template<class MODEL, class OBS, class FCT>
145 class CostMaker : public CostFactory<MODEL, OBS> {
146  private:
147  CostFunction<MODEL, OBS> * make(const eckit::Configuration & config,
148  const eckit::mpi::Comm & comm) override
149  {return new FCT(config, comm);}
150  public:
151  explicit CostMaker(const std::string & name) : CostFactory<MODEL, OBS>(name) {}
152 };
153 
154 // =============================================================================
155 
156 // Factory
157 // -----------------------------------------------------------------------------
158 
159 template <typename MODEL, typename OBS>
160 CostFactory<MODEL, OBS>::CostFactory(const std::string & name) {
161  if (getMakers().find(name) != getMakers().end()) {
162  throw std::runtime_error(name + " already registered in cost function factory.");
163  }
164  getMakers()[name] = this;
165 }
166 
167 // -----------------------------------------------------------------------------
168 
169 template <typename MODEL, typename OBS>
170 CostFunction<MODEL, OBS>* CostFactory<MODEL, OBS>::create(const eckit::Configuration & config,
171  const eckit::mpi::Comm & comm) {
172  std::string id = config.getString("cost type");
173  Log::trace() << "Variational Assimilation Type=" << id << std::endl;
174  typename std::map<std::string, CostFactory<MODEL, OBS>*>::iterator j = getMakers().find(id);
175  if (j == getMakers().end()) {
176  throw std::runtime_error(id + " does not exist in cost function factory.");
177  }
178  Log::trace() << "CostFactory::create found cost function type" << std::endl;
179  return (*j).second->make(config, comm);
180 }
181 
182 // -----------------------------------------------------------------------------
183 // Cost Function
184 // -----------------------------------------------------------------------------
185 
186 template<typename MODEL, typename OBS>
187 CostFunction<MODEL, OBS>::CostFunction(const eckit::Configuration & config)
188  : jb_(), jterms_()
189 {}
190 
191 // -----------------------------------------------------------------------------
192 
193 template<typename MODEL, typename OBS>
194 void CostFunction<MODEL, OBS>::setupTerms(const eckit::Configuration & config) {
195  Log::trace() << "CostFunction::setupTerms start" << std::endl;
196 
197 // Jo
198  eckit::LocalConfiguration obsconf(config, "observations");
199  CostJo<MODEL, OBS> * jo = this->newJo(obsconf);
200  jterms_.push_back(jo);
201  Log::trace() << "CostFunction::setupTerms Jo added" << std::endl;
202 
203 // Jb
204  xb_.reset(new CtrlVar_(config, this->geometry(), jo->obspaces()));
205  jb_.reset(new JbTotal_(*xb_, this->newJb(config, this->geometry(), *xb_),
206  config, this->geometry(), jo->obspaces()));
207  Log::trace() << "CostFunction::setupTerms Jb added" << std::endl;
208 
209 // Other constraints
210  std::vector<eckit::LocalConfiguration> jcs;
211  config.get("constraints", jcs);
212  for (size_t jj = 0; jj < jcs.size(); ++jj) {
213  CostTermBase<MODEL, OBS> * jc = this->newJc(jcs[jj], this->geometry());
214  jterms_.push_back(jc);
215  }
216  Log::trace() << "CostFunction::setupTerms done" << std::endl;
217 }
218 
219 // -----------------------------------------------------------------------------
220 
221 template<typename MODEL, typename OBS>
223  const eckit::Configuration & config,
224  PostProcessor<State_> post) {
225  Log::trace() << "CostFunction::evaluate start" << std::endl;
226 // Setup terms of cost function
227  jb_->initialize(fguess);
228  PostProcessor<State_> pp(post);
229  for (size_t jj = 0; jj < jterms_.size(); ++jj) {
230  jterms_[jj].setPostProc(fguess, config, pp);
231  }
232 
233 // Run NL model
234  CtrlVar_ mfguess(fguess);
235  this->runNL(mfguess, pp);
236 
237 // Cost function value
238  double zzz = 0.0;
239  costJb_ = jb_->finalize(mfguess);
240  zzz += costJb_;
241  costJoJc_ = 0.0;
242  for (size_t jj = 0; jj < jterms_.size(); ++jj) {
243  costJoJc_ += jterms_[jj].computeCost();
244  }
245  zzz += costJoJc_;
246  Log::test() << "CostFunction: Nonlinear J = " << zzz << std::endl;
247  Log::trace() << "CostFunction::evaluate done" << std::endl;
248  return zzz;
249 }
250 
251 // -----------------------------------------------------------------------------
252 
253 template<typename MODEL, typename OBS>
255  const eckit::Configuration & innerConf,
256  PostProcessor<State_> post) {
257  Log::trace() << "CostFunction::linearize start" << std::endl;
258 // Inner loop resolution
259  const eckit::LocalConfiguration resConf(innerConf, "geometry");
260  const Geometry_ lowres(resConf, this->geometry().getComm(), this->geometry().timeComm());
261 
262 // Setup trajectory for terms of cost function
264  jb_->initializeTraj(fguess, lowres, innerConf, pptraj);
265  for (size_t jj = 0; jj < jterms_.size(); ++jj) {
266  jterms_[jj].setPostProcTraj(fguess, innerConf, lowres, pptraj);
267  }
268 
269 // Specific linearization if needed (including TLM)
270  this->doLinearize(lowres, innerConf, *xb_, fguess, post, pptraj);
271 
272 // Run NL model
273  double zzz = this->evaluate(fguess, innerConf, post);
274 
275 // Finalize trajectory setup
276  jb_->finalizeTraj();
277  for (size_t jj = 0; jj < jterms_.size(); ++jj) {
278  jterms_[jj].computeCostTraj();
279  }
280 
281  Log::trace() << "CostFunction::linearize done" << std::endl;
282  return zzz;
283 }
284 
285 // -----------------------------------------------------------------------------
286 
287 template<typename MODEL, typename OBS>
289  Log::trace() << "CostFunction::computeGradientFG start" << std::endl;
292  this->zeroAD(grad);
293 
294  for (size_t jj = 0; jj < jterms_.size(); ++jj) {
295  std::shared_ptr<const GeneralizedDepartures> tmp(jterms_[jj].newGradientFG());
296  jterms_[jj].computeCostAD(tmp, grad, costad);
297  }
298 
299  this->runADJ(grad, costad, pp);
300 
301  for (size_t jj = 0; jj < jterms_.size(); ++jj) {
302  jterms_[jj].setPostProcAD();
303  }
304 
305  Log::info() << "CostFunction::computeGradientFG: gradient:" << grad << std::endl;
306  Log::trace() << "CostFunction::computeGradientFG done" << std::endl;
307 }
308 
309 // -----------------------------------------------------------------------------
310 
311 template<typename MODEL, typename OBS>
313  PostProcessor<Increment_> post) const {
314  Log::trace() << "CostFunction::addIncrement start" << std::endl;
315  Log::info() << "CostFunction::addIncrement: First guess:" << xx << std::endl;
316  Log::info() << "CostFunction::addIncrement: Increment:" << dx << std::endl;
317 
318  xx.obsVar() += dx.obsVar();
319  xx.modVar() += dx.modVar();
320  this->addIncr(xx, dx, post);
321 
322  Log::info() << "CostFunction::addIncrement: Analysis: " << xx << std::endl;
323  Log::test() << "CostFunction::addIncrement: Analysis: " << xx << std::endl;
324  Log::trace() << "CostFunction::addIncrement done" << std::endl;
325 }
326 
327 // -----------------------------------------------------------------------------
328 
329 template<typename MODEL, typename OBS>
331  Log::trace() << "CostFunction::resetLinearization start" << std::endl;
332  for (size_t jj = 0; jj < jterms_.size(); ++jj) {
333  jterms_[jj].resetLinearization();
334  }
335  Log::trace() << "CostFunction::resetLinearization done" << std::endl;
336 }
337 
338 // -----------------------------------------------------------------------------
339 
340 } // namespace oops
341 
342 #endif // OOPS_ASSIMILATION_COSTFUNCTION_H_
ObsAuxIncrs_ & obsVar()
Get augmented observation 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.
Cost Function Factory.
Definition: CostFunction.h:128
virtual CostFunction< MODEL, OBS > * make(const eckit::Configuration &, const eckit::mpi::Comm &)=0
virtual ~CostFactory()=default
CostFactory(const std::string &)
Definition: CostFunction.h:160
static CostFunction< MODEL, OBS > * create(const eckit::Configuration &, const eckit::mpi::Comm &)
Definition: CostFunction.h:170
static std::map< std::string, CostFactory< MODEL, OBS > * > & getMakers()
Definition: CostFunction.h:138
Cost Function.
Definition: CostFunction.h:53
CostFunction(const eckit::Configuration &)
Definition: CostFunction.h:187
double evaluate(const CtrlVar_ &, const eckit::Configuration &config=eckit::LocalConfiguration(), PostProcessor< State_ > post=PostProcessor< State_ >())
Definition: CostFunction.h:222
const JbTotal_ & jb() const
Access .
Definition: CostFunction.h:91
virtual void runTLM(CtrlInc_ &, PostProcessorTLAD< MODEL > &, PostProcessor< Increment_ > post=PostProcessor< Increment_ >(), const bool idModel=false) const =0
virtual void runNL(CtrlVar_ &, PostProcessor< State_ > &) const =0
ControlIncrement< MODEL, OBS > CtrlInc_
Definition: CostFunction.h:54
JqTermTLAD< MODEL > JqTermTLAD_
Definition: CostFunction.h:58
virtual void runADJ(CtrlInc_ &, PostProcessorTLAD< MODEL > &, PostProcessor< Increment_ > post=PostProcessor< Increment_ >(), const bool idModel=false) const =0
CostJbTotal< MODEL, OBS > JbTotal_
Definition: CostFunction.h:56
size_t nterms() const
Definition: CostFunction.h:94
const CostBase_ & jterm(const size_t ii) const
Access terms of the cost function other than .
Definition: CostFunction.h:93
Geometry< MODEL > Geometry_
Definition: CostFunction.h:59
ControlVariable< MODEL, OBS > CtrlVar_
Definition: CostFunction.h:55
double getCostJoJc() const
Definition: CostFunction.h:96
double getCostJb() const
Definition: CostFunction.h:95
void setupTerms(const eckit::Configuration &)
Definition: CostFunction.h:194
boost::ptr_vector< CostBase_ > jterms_
Definition: CostFunction.h:118
virtual CostJo< MODEL, OBS > * newJo(const eckit::Configuration &) const =0
virtual const Geometry_ & geometry() const =0
virtual CostJbState< MODEL > * newJb(const eckit::Configuration &, const Geometry_ &, const CtrlVar_ &) const =0
void addIncrement(CtrlVar_ &, const CtrlInc_ &, PostProcessor< Increment_ > post=PostProcessor< Increment_ >()) const
Definition: CostFunction.h:312
CostTermBase< MODEL, OBS > CostBase_
Definition: CostFunction.h:57
double linearize(const CtrlVar_ &, const eckit::Configuration &, PostProcessor< State_ > post=PostProcessor< State_ >())
Definition: CostFunction.h:254
std::unique_ptr< const CtrlVar_ > xb_
Definition: CostFunction.h:116
virtual CostTermBase< MODEL, OBS > * newJc(const eckit::Configuration &, const Geometry_ &) const =0
virtual void zeroAD(CtrlInc_ &) const =0
virtual void doLinearize(const Geometry_ &, const eckit::Configuration &, const CtrlVar_ &, const CtrlVar_ &, PostProcessor< State_ > &, PostProcessorTLAD< MODEL > &)=0
void computeGradientFG(CtrlInc_ &) const
Compute cost function gradient at first guess (without Jb).
Definition: CostFunction.h:288
virtual void addIncr(CtrlVar_ &, const CtrlInc_ &, PostProcessor< Increment_ > &) const =0
const CtrlVar_ & background() const
Definition: CostFunction.h:100
std::unique_ptr< JbTotal_ > jb_
Definition: CostFunction.h:117
Increment< MODEL > Increment_
Definition: CostFunction.h:61
virtual ~CostFunction()
Definition: CostFunction.h:65
State< MODEL > State_
Definition: CostFunction.h:60
Jb Cost Function Base Class.
Definition: CostJbState.h:37
Control variable increment.
Definition: CostJbTotal.h:36
Jo Cost Function.
Definition: CostJo.h:52
const ObsSpaces_ & obspaces() const
Accessor...
Definition: CostJo.h:111
CostMaker(const std::string &name)
Definition: CostFunction.h:151
CostFunction< MODEL, OBS > * make(const eckit::Configuration &config, const eckit::mpi::Comm &comm) override
Definition: CostFunction.h:147
Base Class for Cost Function Terms.
Definition: CostTermBase.h:36
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
Control model post processing.
Definition: PostProcessor.h:30
Control model post processing.
void initializeTraj(const State_ &xx, const util::DateTime &end, const util::Duration &step)
Set linearization state.
State class used in oops; subclass of interface class interface::State.
The namespace for the main oops code.