OOPS
oops/interface/Increment.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
3  * (C) Copyright 2017-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  * In applying this licence, ECMWF does not waive the privileges and immunities
8  * granted to it by virtue of its status as an intergovernmental organisation nor
9  * does it submit to any jurisdiction.
10  */
11 
12 #ifndef OOPS_INTERFACE_INCREMENT_H_
13 #define OOPS_INTERFACE_INCREMENT_H_
14 
15 #include <memory>
16 #include <string>
17 #include <vector>
18 
19 #include "atlas/field.h"
20 
22 #include "oops/base/Geometry.h"
24 #include "oops/base/State.h"
25 #include "oops/base/Variables.h"
27 #include "oops/util/DateTime.h"
28 #include "oops/util/Duration.h"
29 #include "oops/util/ObjectCounter.h"
30 #include "oops/util/Serializable.h"
31 #include "oops/util/Timer.h"
32 
33 namespace oops {
34 
35 namespace interface {
36 
37 /// Increment: Difference between two model states.
38 /// Some fields that are present in a State may not be present in an Increment.
39 template <typename MODEL>
41  public util::Serializable,
42  private util::ObjectCounter<Increment<MODEL> > {
43  typedef typename MODEL::Increment Increment_;
47 
48  public:
49  static const std::string classname() {return "oops::Increment";}
50 
51  /// Constructor for specified \p geometry, with \p variables, valid on \p date
52  Increment(const Geometry_ & geometry, const Variables & variables, const util::DateTime & date);
53  /// Copies \p other increment, changing its resolution to \p geometry
54  Increment(const Geometry_ & geometry, const Increment & other);
55  /// Creates Increment with the same geometry and variables as \p other.
56  /// Copies \p other if \p copy is true, otherwise creates zero increment
57  Increment(const Increment &, const bool copy = true);
58 
59  /// Destructor (defined explicitly for timing and tracing)
60  virtual ~Increment();
61 
62  /// Set this Increment to be difference between \p state1 and \p state2
63  void diff(const State_ & state1, const State_ & state2);
64 
65  /// Accessor to the time of this Increment
66  const util::DateTime validTime() const {return increment_->validTime();}
67  /// Updates this Increment's valid time by \p dt (used in PseudoModel)
68  void updateTime(const util::Duration & dt) {increment_->updateTime(dt);}
69 
70  /// Zero out this Increment
71  void zero();
72  /// Zero out this Increment and set its date to \p date
73  void zero(const util::DateTime & date);
74  /// Set this Increment to ones (used in tests)
75  void ones();
76  /// Set Increment according to the configuration (used in Dirac application)
77  void dirac(const eckit::Configuration &);
78 
79  /// Assignment operator
80  Increment & operator =(const Increment &);
81  /// Linear algebra operators
82  Increment & operator+=(const Increment &);
83  Increment & operator-=(const Increment &);
84  Increment & operator*=(const double &);
85  /// Add \p w * \p dx to the Increment. If \p check is set, check whether this and \p dx's
86  /// dates are the same
87  void axpy(const double & w, const Increment & dx, const bool check = true);
88  /// Compute dot product of this Increment with \p other
89  double dot_product_with(const Increment & other) const;
90  /// Compute Schur product of this Increment with \p other, assign to this Increment
91  void schur_product_with(const Increment & other);
92 
93  /// Randomize the Increment (used in tests)
94  void random();
95  /// Accumulate (add \p w * \p x to the increment), used in WeightedDiff with Accumulator
96  void accumul(const double & w, const State_ & x);
97 
98  /// Read this Increment from file
99  void read(const eckit::Configuration &);
100  /// Write this Increment out to file
101  void write(const eckit::Configuration &) const;
102  /// Norm (used in tests)
103  double norm() const;
104 
105  /// Get local (at \p iter local volume) increment (used in LocalEnsembleSolver)
106  LocalIncrement getLocal(const GeometryIterator_ & iter) const;
107  /// Set local (at \p iter local volume) increment to be \p gp (used in LocalEnsembleSolver)
108  void setLocal(const LocalIncrement & gp, const GeometryIterator_ & iter);
109 
110  /// Accessor to geometry associated with this Increment
111  Geometry_ geometry() const;
112 
113  /// ATLAS FieldSet interface (used to communicate data with SABER)
114  /// For models that are not using ATLAS fields for their own Increment data:
115  /// - "setAtlas" allocates the ATLAS fields based on the variables present in the Increment.
116  /// - "toAtlas" allocates the ATLAS fields if necessary and copies Increment data into ATLAS
117  /// fields.
118  /// - "fromAtlas" copies ATLAS fields data into the Increment.
119  /// For models that are using ATLAS fields for their own Incerment data
120  /// - "setAtlas" copies ATLAS fields pointers from the Increment to the ATLAS fieldset.
121  /// - "toAtlas" copies data if ATLAS fields pointers in the Increment and the ATLAS fieldset are
122  /// different, and does nothing if pointers are the same.
123  /// - "fromAtlas" does nothing.
124  void setAtlas(atlas::FieldSet *) const;
125  void toAtlas(atlas::FieldSet *) const;
126  void fromAtlas(atlas::FieldSet *);
127 
128  /// Serialize and deserialize (used in 4DEnVar, weak-constraint 4DVar and Block-Lanczos minimizer)
129  size_t serialSize() const override;
130  void serialize(std::vector<double> &) const override;
131  void deserialize(const std::vector<double> &, size_t &) override;
132 
133  /// Accessor to MODEL::Increment, used in the other interface classes in oops.
134  /// Does not need to be implemented.
135  const Increment_ & increment() const {return *this->increment_;}
136  Increment_ & increment() {return *this->increment_;}
137 
138  protected:
139  std::unique_ptr<Increment_> increment_; /// pointer to the Increment implementation
140 
141  private:
142  void print(std::ostream &) const override;
143 };
144 
145 // -----------------------------------------------------------------------------
146 
147 template<typename MODEL>
148 Increment<MODEL>::Increment(const Geometry_ & resol, const Variables & vars,
149  const util::DateTime & time)
150  : increment_()
151 {
152  Log::trace() << "Increment<MODEL>::Increment starting" << std::endl;
153  util::Timer timer(classname(), "Increment");
154  increment_.reset(new Increment_(resol.geometry(), vars, time));
155  this->setObjectSize(increment_->serialSize()*sizeof(double));
156  Log::trace() << "Increment<MODEL>::Increment done" << std::endl;
157 }
158 
159 // -----------------------------------------------------------------------------
160 
161 template<typename MODEL>
162 Increment<MODEL>::Increment(const Geometry_ & resol, const Increment & other)
163  : increment_()
164 {
165  Log::trace() << "Increment<MODEL>::Increment starting" << std::endl;
166  util::Timer timer(classname(), "Increment");
167  increment_.reset(new Increment_(resol.geometry(), *other.increment_));
168  this->setObjectSize(increment_->serialSize()*sizeof(double));
169  Log::trace() << "Increment<MODEL>::Increment done" << std::endl;
170 }
171 
172 // -----------------------------------------------------------------------------
173 
174 template<typename MODEL>
175 Increment<MODEL>::Increment(const Increment & other, const bool copy)
176  : increment_()
177 {
178  Log::trace() << "Increment<MODEL>::Increment copy starting" << std::endl;
179  util::Timer timer(classname(), "Increment");
180  increment_.reset(new Increment_(*other.increment_, copy));
181  this->setObjectSize(increment_->serialSize()*sizeof(double));
182  Log::trace() << "Increment<MODEL>::Increment copy done" << std::endl;
183 }
184 
185 // -----------------------------------------------------------------------------
186 
187 template<typename MODEL>
189  Log::trace() << "Increment<MODEL>::~Increment starting" << std::endl;
190  util::Timer timer(classname(), "~Increment");
191  increment_.reset();
192  Log::trace() << "Increment<MODEL>::~Increment done" << std::endl;
193 }
194 
195 // -----------------------------------------------------------------------------
196 
197 template<typename MODEL>
198 void Increment<MODEL>::diff(const State_ & x1, const State_ & x2) {
199  Log::trace() << "Increment<MODEL>::diff starting" << std::endl;
200  util::Timer timer(classname(), "diff");
201  increment_->diff(x1.state(), x2.state());
202  Log::trace() << "Increment<MODEL>::diff done" << std::endl;
203 }
204 
205 // -----------------------------------------------------------------------------
206 
207 template<typename MODEL>
209  Log::trace() << "Increment<MODEL>::zero starting" << std::endl;
210  util::Timer timer(classname(), "zero");
211  increment_->zero();
212  Log::trace() << "Increment<MODEL>::zero done" << std::endl;
213 }
214 
215 // -----------------------------------------------------------------------------
216 
217 template<typename MODEL>
218 void Increment<MODEL>::zero(const util::DateTime & tt) {
219  Log::trace() << "Increment<MODEL>::zero starting" << std::endl;
220  util::Timer timer(classname(), "zero");
221  increment_->zero(tt);
222  Log::trace() << "Increment<MODEL>::zero done" << std::endl;
223 }
224 
225 // -----------------------------------------------------------------------------
226 
227 template<typename MODEL>
229  Log::trace() << "Increment<MODEL>::ones starting" << std::endl;
230  util::Timer timer(classname(), "ones");
231  increment_->ones();
232  Log::trace() << "Increment<MODEL>::ones done" << std::endl;
233 }
234 
235 // -----------------------------------------------------------------------------
236 
237 template<typename MODEL>
238 void Increment<MODEL>::dirac(const eckit::Configuration & config) {
239  Log::trace() << "Increment<MODEL>::dirac starting" << std::endl;
240  util::Timer timer(classname(), "dirac");
241  increment_->dirac(config);
242  Log::trace() << "Increment<MODEL>::dirac done" << std::endl;
243 }
244 
245 // -----------------------------------------------------------------------------
246 
247 template<typename MODEL>
249  Log::trace() << "Increment<MODEL>::operator= starting" << std::endl;
250  util::Timer timer(classname(), "operator=");
251  *increment_ = *rhs.increment_;
252  Log::trace() << "Increment<MODEL>::operator= done" << std::endl;
253  return *this;
254 }
255 
256 // -----------------------------------------------------------------------------
257 
258 template<typename MODEL>
260  Log::trace() << "Increment<MODEL>::operator+= starting" << std::endl;
261  util::Timer timer(classname(), "operator+=");
262  *increment_ += *rhs.increment_;
263  Log::trace() << "Increment<MODEL>::operator+= done" << std::endl;
264  return *this;
265 }
266 
267 // -----------------------------------------------------------------------------
268 
269 template<typename MODEL>
271  Log::trace() << "Increment<MODEL>::operator-= starting" << std::endl;
272  util::Timer timer(classname(), "operator-=");
273  *increment_ -= *rhs.increment_;
274  Log::trace() << "Increment<MODEL>::operator-= done" << std::endl;
275  return *this;
276 }
277 
278 // -----------------------------------------------------------------------------
279 
280 template<typename MODEL>
282  Log::trace() << "Increment<MODEL>::operator*= starting" << std::endl;
283  util::Timer timer(classname(), "operator*=");
284  *increment_ *= zz;
285  Log::trace() << "Increment<MODEL>::operator*= done" << std::endl;
286  return *this;
287 }
288 
289 // -----------------------------------------------------------------------------
290 
291 template<typename MODEL>
292 void Increment<MODEL>::axpy(const double & zz, const Increment & dx, const bool check) {
293  Log::trace() << "Increment<MODEL>::axpy starting" << std::endl;
294  util::Timer timer(classname(), "axpy");
295  increment_->axpy(zz, *dx.increment_, check);
296  Log::trace() << "Increment<MODEL>::axpy done" << std::endl;
297 }
298 
299 // -----------------------------------------------------------------------------
300 
301 template<typename MODEL>
303  Log::trace() << "Increment<MODEL>::dot_product_with starting" << std::endl;
304  util::Timer timer(classname(), "dot_product_with");
305  double zz = increment_->dot_product_with(*dx.increment_);
306  Log::trace() << "Increment<MODEL>::dot_product_with done" << std::endl;
307  return zz;
308 }
309 
310 // -----------------------------------------------------------------------------
311 
312 template<typename MODEL>
314  Log::trace() << "Increment<MODEL>::schur_product_with starting" << std::endl;
315  util::Timer timer(classname(), "schur_product_with");
316  increment_->schur_product_with(*dx.increment_);
317  Log::trace() << "Increment<MODEL>::schur_product_with done" << std::endl;
318 }
319 
320 // -----------------------------------------------------------------------------
321 
322 template<typename MODEL>
324  Log::trace() << "Increment<MODEL>::random starting" << std::endl;
325  util::Timer timer(classname(), "random");
326  increment_->random();
327  Log::trace() << "Increment<MODEL>::random done" << std::endl;
328 }
329 
330 // -----------------------------------------------------------------------------
331 
332 template<typename MODEL>
333 void Increment<MODEL>::accumul(const double & zz, const State_ & xx) {
334  Log::trace() << "Increment<MODEL>::accumul starting" << std::endl;
335  util::Timer timer(classname(), "accumul");
336  increment_->accumul(zz, xx.state());
337  Log::trace() << "Increment<MODEL>::accumul done" << std::endl;
338 }
339 
340 // -----------------------------------------------------------------------------
341 
342 template<typename MODEL>
344  Log::trace() << "Increment<MODEL>::getLocal starting" << std::endl;
345  util::Timer timer(classname(), "getLocal");
346  LocalIncrement gp = increment_->getLocal(iter.geometryiter());
347  Log::trace() << "Increment<MODEL>::getLocal done" << std::endl;
348  return gp;
349 }
350 
351 // -----------------------------------------------------------------------------
352 template<typename MODEL>
354  const GeometryIterator_ & iter) {
355  Log::trace() << "Increment<MODEL>::setLocal starting" << std::endl;
356  util::Timer timer(classname(), "setLocal");
357  increment_->setLocal(gp, iter.geometryiter());
358  Log::trace() << "Increment<MODEL>::setLocal done" << std::endl;
359 }
360 
361 // -----------------------------------------------------------------------------
362 
363 template<typename MODEL>
364 void Increment<MODEL>::read(const eckit::Configuration & conf) {
365  Log::trace() << "Increment<MODEL>::read starting" << std::endl;
366  util::Timer timer(classname(), "read");
367  increment_->read(conf);
368  Log::trace() << "Increment<MODEL>::read done" << std::endl;
369 }
370 
371 // -----------------------------------------------------------------------------
372 
373 template<typename MODEL>
374 void Increment<MODEL>::write(const eckit::Configuration & conf) const {
375  Log::trace() << "Increment<MODEL>::write starting" << std::endl;
376  util::Timer timer(classname(), "write");
377  increment_->write(conf);
378  Log::trace() << "Increment<MODEL>::write done" << std::endl;
379 }
380 
381 // -----------------------------------------------------------------------------
382 
383 template<typename MODEL>
384 double Increment<MODEL>::norm() const {
385  Log::trace() << "Increment<MODEL>::norm starting" << std::endl;
386  util::Timer timer(classname(), "norm");
387  double zz = increment_->norm();
388  Log::trace() << "Increment<MODEL>::norm done" << std::endl;
389  return zz;
390 }
391 
392 // -----------------------------------------------------------------------------
393 
394 template<typename MODEL>
396  Log::trace() << "Increment<MODEL>::geometry starting" << std::endl;
397  util::Timer timer(classname(), "geometry");
398  oops::Geometry<MODEL> geom(increment_->geometry());
399  Log::trace() << "Increment<MODEL>::geometry done" << std::endl;
400  return geom;
401 }
402 
403 // -----------------------------------------------------------------------------
404 
405 template<typename MODEL>
406 void Increment<MODEL>::setAtlas(atlas::FieldSet * atlasFieldSet) const {
407  Log::trace() << "Increment<MODEL>::setAtlas starting" << std::endl;
408  util::Timer timer(classname(), "setAtlas");
409  increment_->setAtlas(atlasFieldSet);
410  Log::trace() << "Increment<MODEL>::setAtlas done" << std::endl;
411 }
412 
413 // -----------------------------------------------------------------------------
414 
415 template<typename MODEL>
416 void Increment<MODEL>::toAtlas(atlas::FieldSet * atlasFieldSet) const {
417  Log::trace() << "Increment<MODEL>::toAtlas starting" << std::endl;
418  util::Timer timer(classname(), "toAtlas");
419  increment_->toAtlas(atlasFieldSet);
420  Log::trace() << "Increment<MODEL>::toAtlas done" << std::endl;
421 }
422 
423 // -----------------------------------------------------------------------------
424 
425 template<typename MODEL>
426 void Increment<MODEL>::fromAtlas(atlas::FieldSet * atlasFieldSet) {
427  Log::trace() << "Increment<MODEL>::fromAtlas starting" << std::endl;
428  util::Timer timer(classname(), "fromAtlas");
429  increment_->fromAtlas(atlasFieldSet);
430  Log::trace() << "Increment<MODEL>::fromAtlas done" << std::endl;
431 }
432 
433 // -----------------------------------------------------------------------------
434 
435 template<typename MODEL>
437  Log::trace() << "Increment<MODEL>::serialSize" << std::endl;
438  util::Timer timer(classname(), "serialSize");
439  return increment_->serialSize();
440 }
441 
442 // -----------------------------------------------------------------------------
443 
444 template<typename MODEL>
445 void Increment<MODEL>::serialize(std::vector<double> & vect) const {
446  Log::trace() << "Increment<MODEL>::serialize starting" << std::endl;
447  util::Timer timer(classname(), "serialize");
448  increment_->serialize(vect);
449  Log::trace() << "Increment<MODEL>::serialize done" << std::endl;
450 }
451 
452 // -----------------------------------------------------------------------------
453 
454 template<typename MODEL>
455 void Increment<MODEL>::deserialize(const std::vector<double> & vect, size_t & current) {
456  Log::trace() << "Increment<MODEL>::Increment deserialize starting" << std::endl;
457  util::Timer timer(classname(), "deserialize");
458  increment_->deserialize(vect, current);
459  Log::trace() << "Increment<MODEL>::Increment deserialize done" << std::endl;
460 }
461 
462 // -----------------------------------------------------------------------------
463 
464 template<typename MODEL>
465 void Increment<MODEL>::print(std::ostream & os) const {
466  Log::trace() << "Increment<MODEL>::print starting" << std::endl;
467  util::Timer timer(classname(), "print");
468  os << *increment_;
469  Log::trace() << "Increment<MODEL>::print done" << std::endl;
470 }
471 
472 // -----------------------------------------------------------------------------
473 
474 } // namespace interface
475 
476 } // namespace oops
477 
478 #endif // OOPS_INTERFACE_INCREMENT_H_
Abstract base class for quantities.
Geometry class used in oops; subclass of interface class interface::Geometry.
const GeometryIterator_ & geometryiter() const
Interfacing.
State class used in oops; subclass of interface class interface::State.
const Geometry_ & geometry() const
void setLocal(const LocalIncrement &gp, const GeometryIterator_ &iter)
Set local (at iter local volume) increment to be gp (used in LocalEnsembleSolver)
void axpy(const double &w, const Increment &dx, const bool check=true)
const Increment_ & increment() const
void deserialize(const std::vector< double > &, size_t &) override
Increment & operator*=(const double &)
void dirac(const eckit::Configuration &)
Set Increment according to the configuration (used in Dirac application)
void updateTime(const util::Duration &dt)
Updates this Increment's valid time by dt (used in PseudoModel)
static const std::string classname()
void schur_product_with(const Increment &other)
Compute Schur product of this Increment with other, assign to this Increment.
void fromAtlas(atlas::FieldSet *)
void setAtlas(atlas::FieldSet *) const
Increment(const Geometry_ &geometry, const Variables &variables, const util::DateTime &date)
Constructor for specified geometry, with variables, valid on date.
Increment & operator+=(const Increment &)
Linear algebra operators.
void random()
Randomize the Increment (used in tests)
virtual ~Increment()
Destructor (defined explicitly for timing and tracing)
LocalIncrement getLocal(const GeometryIterator_ &iter) const
Get local (at iter local volume) increment (used in LocalEnsembleSolver)
void read(const eckit::Configuration &)
Read this Increment from file.
Geometry_ geometry() const
Accessor to geometry associated with this Increment.
Increment & operator-=(const Increment &)
GeometryIterator< MODEL > GeometryIterator_
void toAtlas(atlas::FieldSet *) const
void diff(const State_ &state1, const State_ &state2)
Set this Increment to be difference between state1 and state2.
double dot_product_with(const Increment &other) const
Compute dot product of this Increment with other.
double norm() const
Norm (used in tests)
void serialize(std::vector< double > &) const override
oops::Geometry< MODEL > Geometry_
void accumul(const double &w, const State_ &x)
Accumulate (add w * x to the increment), used in WeightedDiff with Accumulator.
std::unique_ptr< Increment_ > increment_
void print(std::ostream &) const override
pointer to the Increment implementation
void write(const eckit::Configuration &) const
Write this Increment out to file.
Increment & operator=(const Increment &)
Assignment operator.
void zero()
Zero out this Increment.
void ones()
Set this Increment to ones (used in tests)
size_t serialSize() const override
Serialize and deserialize (used in 4DEnVar, weak-constraint 4DVar and Block-Lanczos minimizer)
const util::DateTime validTime() const
Accessor to the time of this Increment.
State_ & state()
Accessor.
The namespace for the main oops code.