OOPS
oops/base/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_BASE_INCREMENT_H_
13 #define OOPS_BASE_INCREMENT_H_
14 
15 #include "atlas/field.h"
16 
17 #include "oops/base/Geometry.h"
18 #include "oops/base/State.h"
19 #include "oops/base/Variables.h"
21 #include "oops/mpi/mpi.h"
22 #include "oops/util/DateTime.h"
23 #include "oops/util/gatherPrint.h"
24 #include "oops/util/Timer.h"
25 
26 namespace oops {
27 
28 // -----------------------------------------------------------------------------
29 /// \brief Increment class used in oops
30 ///
31 /// \details
32 /// Adds extra methods that do not need to be implemented in the model implementations:
33 /// - timeComm() (accessor to the MPI communicator in time - collection of processes
34 /// holding the data needed to represent the state in a particular region
35 /// of space X_i and throughout the whole time interval for which DA is done)
36 /// - variables() (accessor to variables in this Increment)
37 /// - shift_forward
38 /// - shift_backward
39 /// - toAtlas, atlas
40 ///
41 /// Adds communication through time to the following Increment methods:
42 /// - dot_product_with
43 /// - norm
44 /// - print
45 
46 template <typename MODEL>
47 class Increment : public interface::Increment<MODEL> {
49 
50  public:
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 & other, const bool copy = true);
58 
59  /// Accessor to the time communicator
60  const eckit::mpi::Comm & timeComm() const {return *timeComm_;}
61  /// Accessor to Variables stored in this increment
62  const Variables & variables() const {return variables_;}
63 
64  /// Shift forward in time by \p dt
65  void shift_forward(const util::DateTime & dt);
66  /// Shift backward in time by \p dt
67  void shift_backward(const util::DateTime & dt);
68 
69  /// Set ATLAS fieldset associated with this Increment internally
70  void toAtlas();
71  /// Allow to access base class's method as well
73  /// Accessors to the ATLAS fieldset
74  atlas::FieldSet & atlas() {return atlasFieldSet_;}
75  const atlas::FieldSet & atlas() const {return atlasFieldSet_;}
76 
77  /// dot product with the \p other increment
78  double dot_product_with(const Increment & other) const;
79  /// Norm for diagnostics
80  double norm() const;
81 
82  private:
83  void print(std::ostream &) const override;
84 
85  Variables variables_; /// Variables stored in this Increment
86  const eckit::mpi::Comm * timeComm_; /// pointer to the MPI communicator in time
87  atlas::FieldSet atlasFieldSet_; /// Atlas fields associated with this Increment
88 };
89 
90 // -----------------------------------------------------------------------------
91 
92 template <typename MODEL>
93 Increment<MODEL>::Increment(const Geometry_ & geometry, const Variables & variables,
94  const util::DateTime & date):
95  interface::Increment<MODEL>(geometry, variables, date), variables_(variables),
96  timeComm_(&geometry.timeComm())
97 {}
98 
99 // -----------------------------------------------------------------------------
100 
101 template <typename MODEL>
102 Increment<MODEL>::Increment(const Geometry_ & geometry, const Increment & other):
103  interface::Increment<MODEL>(geometry, other), variables_(other.variables_),
104  timeComm_(other.timeComm_)
105 {}
106 
107 // -----------------------------------------------------------------------------
108 
109 template <typename MODEL>
110 Increment<MODEL>::Increment(const Increment & other, const bool copy):
111  interface::Increment<MODEL>(other, copy), variables_(other.variables_),
112  timeComm_(other.timeComm_)
113 {}
114 
115 // -----------------------------------------------------------------------------
116 
117 template<typename MODEL>
120  timeComm_->allReduceInPlace(zz, eckit::mpi::Operation::SUM);
121  return zz;
122 }
123 
124 // -----------------------------------------------------------------------------
125 
126 template<typename MODEL>
127 double Increment<MODEL>::norm() const {
128  double zz = interface::Increment<MODEL>::norm();
129  zz *= zz;
130  timeComm_->allReduceInPlace(zz, eckit::mpi::Operation::SUM);
131  zz = sqrt(zz);
132  return zz;
133 }
134 
135 // -----------------------------------------------------------------------------
136 
137 template<typename MODEL>
138 void Increment<MODEL>::shift_forward(const util::DateTime & begin) {
139  Log::trace() << "Increment<MODEL>::Increment shift_forward starting" << std::endl;
140  static int tag = 159357;
141  size_t mytime = timeComm_->rank();
142 
143 // Send values of M.dx_i at end of my subwindow to next subwindow
144  if (mytime + 1 < timeComm_->size()) {
145  oops::mpi::send(*timeComm_, *this, mytime+1, tag);
146  }
147 
148 // Receive values at beginning of my subwindow from previous subwindow
149  if (mytime > 0) {
150  oops::mpi::receive(*timeComm_, *this, mytime-1, tag);
151  } else {
152  this->zero(begin);
153  }
154 
155  ++tag;
156  Log::trace() << "Increment<MODEL>::Increment shift_forward done" << std::endl;
157 }
158 
159 // -----------------------------------------------------------------------------
160 
161 template<typename MODEL>
162 void Increment<MODEL>::shift_backward(const util::DateTime & end) {
163  Log::trace() << "Increment<MODEL>::Increment shift_backward starting" << std::endl;
164  static int tag = 30951;
165  size_t mytime = timeComm_->rank();
166 
167 // Send values of dx_i at start of my subwindow to previous subwindow
168  if (mytime > 0) {
169  oops::mpi::send(*timeComm_, *this, mytime-1, tag);
170  }
171 
172 // Receive values at end of my subwindow from next subwindow
173  if (mytime + 1 < timeComm_->size()) {
174  oops::mpi::receive(*timeComm_, *this, mytime+1, tag);
175  } else {
176  this->zero(end);
177  }
178 
179  ++tag;
180  Log::trace() << "Increment<MODEL>::Increment shift_backward done" << std::endl;
181 }
182 
183 
184 // -----------------------------------------------------------------------------
185 template<typename MODEL>
187  interface::Increment<MODEL>::setAtlas(&atlasFieldSet_);
188  interface::Increment<MODEL>::toAtlas(&atlasFieldSet_);
189  this->increment_.reset();
190 }
191 
192 // -----------------------------------------------------------------------------
193 
194 template<typename MODEL>
195 void Increment<MODEL>::print(std::ostream & os) const {
196  if (timeComm_->size() > 1) {
197  gatherPrint(os, this->increment(), *timeComm_);
198  } else {
199  os << this->increment();
200  }
201 }
202 
203 // -----------------------------------------------------------------------------
204 /// Add on \p dx incrment to model state \p xx
205 template <typename MODEL>
207  Log::trace() << "operator+=(State, Increment) starting" << std::endl;
208  util::Timer timer("oops::Increment", "operator+=(State, Increment)");
209  xx.state() += dx.increment();
210  Log::trace() << "operator+=(State, Increment) done" << std::endl;
211  return xx;
212 }
213 
214 
215 } // namespace oops
216 
217 #endif // OOPS_BASE_INCREMENT_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
void toAtlas()
Set ATLAS fieldset associated with this Increment internally.
void print(std::ostream &) const override
double dot_product_with(const Increment &other) const
dot product with the other increment
atlas::FieldSet & atlas()
Accessors to the ATLAS fieldset.
const Variables & variables() const
Accessor to Variables stored in this increment.
Geometry< MODEL > Geometry_
const eckit::mpi::Comm * timeComm_
Variables stored in this Increment.
atlas::FieldSet atlasFieldSet_
pointer to the MPI communicator in time
const eckit::mpi::Comm & timeComm() const
Accessor to the time communicator.
double norm() const
Norm for diagnostics.
Increment(const Geometry_ &geometry, const Variables &variables, const util::DateTime &date)
Constructor for specified geometry, with variables, valid on date.
const atlas::FieldSet & atlas() const
void shift_forward(const util::DateTime &dt)
Shift forward in time by dt.
void shift_backward(const util::DateTime &dt)
Shift backward in time by dt.
State class used in oops; subclass of interface class interface::State.
const Increment_ & increment() const
void setAtlas(atlas::FieldSet *) const
Geometry_ geometry() const
Accessor to geometry associated with this Increment.
void toAtlas(atlas::FieldSet *) const
double dot_product_with(const Increment &other) const
Compute dot product of this Increment with other.
double norm() const
Norm (used in tests)
State_ & state()
Accessor.
void send(const eckit::mpi::Comm &comm, const SERIALIZABLE &sendobj, const int dest, const int tag)
Extend eckit Comm for Serializable oops objects.
Definition: oops/mpi/mpi.h:44
void receive(const eckit::mpi::Comm &comm, SERIALIZABLE &recvobj, const int source, const int tag)
Definition: oops/mpi/mpi.h:55
The namespace for the main oops code.
State< MODEL > & operator+=(State< MODEL > &xx, const Increment< MODEL > &dx)
Add on dx incrment to model state xx.