OOPS
oops/base/LinearModel.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
3  * (C) Copyright 2018-2021 UCAR.
4  *
5  * This software is licensed under the terms of the Apache Licence Version 2.0
6  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7  */
8 
9 #ifndef OOPS_BASE_LINEARMODEL_H_
10 #define OOPS_BASE_LINEARMODEL_H_
11 
12 #include <memory>
13 #include <string>
14 #include <utility>
15 
16 #include <boost/noncopyable.hpp>
17 
18 #include "oops/base/Geometry.h"
19 #include "oops/base/State.h"
24 #include "oops/util/Duration.h"
25 #include "oops/util/Logger.h"
26 #include "oops/util/Printable.h"
27 #include "oops/util/Timer.h"
28 
29 namespace eckit {
30  class Configuration;
31 }
32 
33 namespace oops {
34 
35 /// \brief Abstract linear forecast model used by high level algorithms and applications.
36 ///
37 /// Note: to see methods that need to be implemented in a generic linear forecast model
38 /// implementation, see LinearModelBase class in generic/LinearModelBase.h. To see methods that need
39 /// to be implemented in a MODEL-specific linear forecast model implementation, see
40 /// interface::LinearModelBase class in interface/LinearModelBase.h.
41 ///
42 // -----------------------------------------------------------------------------
43 
44 template <typename MODEL>
45 class LinearModel : public util::Printable,
46  private boost::noncopyable,
47  private util::ObjectCounter<LinearModel<MODEL> > {
54 
55  public:
56  static const std::string classname() {return "oops::LinearModel";}
57 
59  LinearModel(const Geometry_ &, const eckit::Configuration &);
60  virtual ~LinearModel();
61 
62  /// \brief Run the linear forecast from increment \p dx for \p len time, with \p post
63  /// postprocessors. Does not need to be implemented in the subclasses of LinearModelBase
64  void forecastTL(Increment_ & dx, const ModelAuxInc_ &,
65  const util::Duration & len,
68  const bool idmodel = false) const;
69 
70  /// \brief Run the adjoint linear forecast from increment \p dx for \p len -time, with \p post
71  /// postprocessors. Note that the clock ticks backwards. Does not need to be implemented in the
72  /// subclasses of LinearModelBase
73  void forecastAD(Increment_ & dx, ModelAuxInc_ &,
74  const util::Duration & len,
77  const bool idmodel = false) const;
78 
79  /// \brief Set the trajectory for the linear model
80  void setTrajectory(const State_ &, State_ &, const ModelAuxCtl_ &);
81 
82  /// \brief Time step for running LinearModel's forecast in oops (frequency with which the
83  /// State will be updated)
84  const util::Duration & timeResolution() const {return linearmodel_->timeResolution();}
85  /// \brief LinearModel variables (only used in 4DVar)
86  const oops::Variables & variables() const {return linearmodel_->variables();}
87 
88  private:
89  /// \brief Tangent linear forecast initialization, called before every run
90  void initializeTL(Increment_ &) const;
91  /// \brief Tangent linear forecast "step", called during run; updates increment to the next time
92  void stepTL(Increment_ &, const ModelAuxInc_ &) const;
93  /// \brief Tangent linear forecast finalization; called after each run
94  void finalizeTL(Increment_ &) const;
95  /// \brief Adjoint forecast initialization, called before every run
96  void initializeAD(Increment_ &) const;
97  /// \brief Adjoint forecast "step", called during run; updates increment to the next time
98  void stepAD(Increment_ &, ModelAuxInc_ &) const;
99  /// \brief Adjoint forecast finalization; called after each run
100  void finalizeAD(Increment_ &) const;
101 
102  /// \brief Print, used in logging
103  void print(std::ostream &) const;
104 
105  /// \brief Pointer to the LinearModel implementation
106  std::unique_ptr<LinearModelBase_> linearmodel_;
107 };
108 
109 // =============================================================================
110 
111 template<typename MODEL>
113  : linearmodel_()
114 {
115  Log::trace() << "LinearModel<MODEL>::LinearModel starting" << std::endl;
116  util::Timer timer(classname(), "LinearModel");
117  Log::info() << "LinearModel configuration is:" << params << std::endl;
118  linearmodel_.reset(LinearModelFactory<MODEL>::create(resol, params));
119  Log::trace() << "LinearModel<MODEL>::LinearModel done" << std::endl;
120 }
121 
122 // -----------------------------------------------------------------------------
123 
124 template<typename MODEL>
125 LinearModel<MODEL>::LinearModel(const Geometry_ & resol, const eckit::Configuration & conf)
126  : LinearModel(resol,
127  validateAndDeserialize<LinearModelParametersWrapper<MODEL>>(conf).linearModelParameters)
128 {}
129 
130 // -----------------------------------------------------------------------------
131 
132 template<typename MODEL>
134  Log::trace() << "LinearModel<MODEL>::~LinearModel starting" << std::endl;
135  util::Timer timer(classname(), "~LinearModel");
136  linearmodel_.reset();
137  Log::trace() << "LinearModel<MODEL>::~LinearModel done" << std::endl;
138 }
139 
140 // -----------------------------------------------------------------------------
141 
142 template<typename MODEL>
144  const util::Duration & len,
147  const bool idmodel) const {
148  Log::trace() << "LinearModel<MODEL>::forecastTL starting" << std::endl;
149 
150  const util::DateTime end(dx.validTime() + len);
151  const util::Duration tstep(linearmodel_->timeResolution());
152  Log::info() << "LinearModel<MODEL>::forecastTL: Starting " << dx << std::endl;
153  this->initializeTL(dx);
154  cost.initializeTL(dx, end, tstep);
155  post.initialize(dx, end, tstep);
156  cost.processTL(dx);
157  post.process(dx);
158  if (idmodel) {
159  while (dx.validTime() < end) {
160  dx.updateTime(tstep);
161  cost.processTL(dx);
162  post.process(dx);
163  }
164  } else {
165  while (dx.validTime() < end) {
166  this->stepTL(dx, maux);
167  cost.processTL(dx);
168  post.process(dx);
169  }
170  }
171  cost.finalizeTL(dx);
172  post.finalize(dx);
173  this->finalizeTL(dx);
174  Log::info() << "LinearModel<MODEL>::forecastTL: Finished " << dx << std::endl;
175  ASSERT(dx.validTime() == end);
176 
177  Log::trace() << "LinearModel<MODEL>::forecastTL done" << std::endl;
178 }
179 
180 // -----------------------------------------------------------------------------
181 
182 template<typename MODEL>
184  const util::Duration & len,
187  const bool idmodel) const {
188  Log::trace() << "LinearModel<MODEL>::forecastAD starting" << std::endl;
189 
190  const util::DateTime bgn(dx.validTime() - len);
191  const util::Duration tstep(linearmodel_->timeResolution());
192  Log::info() << "LinearModel<MODEL>::forecastAD: Starting " << dx << std::endl;
193  this->initializeAD(dx);
194  post.initialize(dx, bgn, tstep);
195  cost.initializeAD(dx, bgn, tstep);
196  if (idmodel) {
197  while (dx.validTime() > bgn) {
198  cost.processAD(dx);
199  dx.updateTime(-tstep);
200  post.process(dx);
201  }
202  } else {
203  while (dx.validTime() > bgn) {
204  cost.processAD(dx);
205  this->stepAD(dx, maux);
206  post.process(dx);
207  }
208  }
209  cost.processAD(dx);
210  post.process(dx);
211  cost.finalizeAD(dx);
212  post.finalize(dx);
213  this->finalizeAD(dx);
214  Log::info() << "LinearModel<MODEL>::forecastAD: Finished " << dx << std::endl;
215  ASSERT(dx.validTime() == bgn);
216 
217  Log::trace() << "LinearModel<MODEL>::forecastAD done" << std::endl;
218 }
219 
220 // -----------------------------------------------------------------------------
221 
222 template<typename MODEL>
224  Log::trace() << "LinearModel<MODEL>::initializeTL starting" << std::endl;
225  util::Timer timer(classname(), "initializeTL");
226  linearmodel_->initializeTL(dx);
227  Log::trace() << "LinearModel<MODEL>::initializeTL done" << std::endl;
228 }
229 
230 // -----------------------------------------------------------------------------
231 
232 template<typename MODEL>
233 void LinearModel<MODEL>::stepTL(Increment_ & dx, const ModelAuxInc_ & maux) const {
234  Log::trace() << "LinearModel<MODEL>::stepTL starting" << std::endl;
235  util::Timer timer(classname(), "stepTL");
236  linearmodel_->stepTL(dx, maux);
237  Log::trace() << "LinearModel<MODEL>::stepTL done" << std::endl;
238 }
239 
240 // -----------------------------------------------------------------------------
241 
242 template<typename MODEL>
244  Log::trace() << "LinearModel<MODEL>::finalizeTL starting" << std::endl;
245  util::Timer timer(classname(), "finalizeTL");
246  linearmodel_->finalizeTL(dx);
247  Log::trace() << "LinearModel<MODEL>::finalizeTL done" << std::endl;
248 }
249 
250 // -----------------------------------------------------------------------------
251 
252 template<typename MODEL>
254  Log::trace() << "LinearModel<MODEL>::initializeAD starting" << std::endl;
255  util::Timer timer(classname(), "initializeAD");
256  linearmodel_->initializeAD(dx);
257  Log::trace() << "LinearModel<MODEL>::initializeAD done" << std::endl;
258 }
259 
260 // -----------------------------------------------------------------------------
261 
262 template<typename MODEL>
264  Log::trace() << "LinearModel<MODEL>::stepAD starting" << std::endl;
265  util::Timer timer(classname(), "stepAD");
266  linearmodel_->stepAD(dx, maux);
267  Log::trace() << "LinearModel<MODEL>::stepAD done" << std::endl;
268 }
269 
270 // -----------------------------------------------------------------------------
271 
272 template<typename MODEL>
274  Log::trace() << "LinearModel<MODEL>::finalizeAD starting" << std::endl;
275  util::Timer timer(classname(), "finalizeAD");
276  linearmodel_->finalizeAD(dx);
277  Log::trace() << "LinearModel<MODEL>::finalizeAD done" << std::endl;
278 }
279 
280 // -----------------------------------------------------------------------------
281 
282 template<typename MODEL>
283 void LinearModel<MODEL>::print(std::ostream & os) const {
284  Log::trace() << "LinearModel<MODEL>::print starting" << std::endl;
285  util::Timer timer(classname(), "print");
286  os << *linearmodel_;
287  Log::trace() << "LinearModel<MODEL>::print done" << std::endl;
288 }
289 
290 // -----------------------------------------------------------------------------
291 
292 template<typename MODEL>
294  const ModelAuxCtl_ & maux) {
295  Log::trace() << "LinearModel<MODEL>::setTrajectory starting" << std::endl;
296  util::Timer timer(classname(), "setTrajectory");
297  linearmodel_->setTrajectory(xx, xtraj, maux);
298  Log::trace() << "LinearModel<MODEL>::setTrajectory done" << std::endl;
299 }
300 
301 // -----------------------------------------------------------------------------
302 
303 } // namespace oops
304 
305 #endif // OOPS_BASE_LINEARMODEL_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
Base class for generic implementations of the linearized forecasting models. Use this class as a base...
Abstract linear forecast model used by high level algorithms and applications.
Geometry< MODEL > Geometry_
void initializeTL(Increment_ &) const
Tangent linear forecast initialization, called before every run.
static const std::string classname()
LinearModelBase< MODEL > LinearModelBase_
void stepTL(Increment_ &, const ModelAuxInc_ &) const
Tangent linear forecast "step", called during run; updates increment to the next time.
void finalizeAD(Increment_ &) const
Adjoint forecast finalization; called after each run.
const util::Duration & timeResolution() const
Time step for running LinearModel's forecast in oops (frequency with which the State will be updated)
void setTrajectory(const State_ &, State_ &, const ModelAuxCtl_ &)
Set the trajectory for the linear model.
ModelAuxControl< MODEL > ModelAuxCtl_
void finalizeTL(Increment_ &) const
Tangent linear forecast finalization; called after each run.
ModelAuxIncrement< MODEL > ModelAuxInc_
void stepAD(Increment_ &, ModelAuxInc_ &) const
Adjoint forecast "step", called during run; updates increment to the next time.
const oops::Variables & variables() const
LinearModel variables (only used in 4DVar)
std::unique_ptr< LinearModelBase_ > linearmodel_
Pointer to the LinearModel implementation.
void forecastTL(Increment_ &dx, const ModelAuxInc_ &, const util::Duration &len, PostProcessor< Increment_ > post=PostProcessor< Increment_ >(), PostProcessorTLAD< MODEL > cost=PostProcessorTLAD< MODEL >(), const bool idmodel=false) const
Run the linear forecast from increment dx for len time, with post postprocessors. Does not need to be...
void initializeAD(Increment_ &) const
Adjoint forecast initialization, called before every run.
LinearModel(const Geometry_ &, const LinearModelParametersBase &)
void forecastAD(Increment_ &dx, ModelAuxInc_ &, const util::Duration &len, PostProcessor< Increment_ > post=PostProcessor< Increment_ >(), PostProcessorTLAD< MODEL > cost=PostProcessorTLAD< MODEL >(), const bool idmodel=false) const
Run the adjoint linear forecast from increment dx for len -time, with post postprocessors....
Increment< MODEL > Increment_
void print(std::ostream &) const
Print, used in logging.
Base class for classes storing linear model-specific parameters.
Contains a polymorphic parameter holding an instance of a subclass of LinearModelParametersBase.
Auxiliary state related to model (could be e.g. model bias), not used at the moment.
Auxiliary Increment related to model, not used at the moment.
Control model post processing.
Definition: PostProcessor.h:30
void process(const FLDS &xx)
Definition: PostProcessor.h:56
void finalize(const FLDS &xx)
Definition: PostProcessor.h:62
void initialize(const FLDS &xx, const util::DateTime &end, const util::Duration &step)
Definition: PostProcessor.h:49
Control model post processing.
void finalizeAD(Increment_ &dx)
void processTL(const Increment_ &dx)
void processAD(Increment_ &dx)
void finalizeTL(const Increment_ &dx)
void initializeTL(const Increment_ &dx, const util::DateTime &end, const util::Duration &step)
Tangent linear methods.
void initializeAD(Increment_ &dx, const util::DateTime &bgn, const util::Duration &step)
Adjoint methods.
State class used in oops; subclass of interface class interface::State.
void updateTime(const util::Duration &dt)
Updates this Increment's valid time by dt (used in PseudoModel)
const util::DateTime validTime() const
Accessor to the time of this Increment.
Definition: FieldL95.h:22
The namespace for the main oops code.