OOPS
oops/base/Localization.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
3  * (C) Copyright 2020-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_LOCALIZATION_H_
13 #define OOPS_BASE_LOCALIZATION_H_
14 
15 #include <memory>
16 #include <string>
17 
18 #include <boost/noncopyable.hpp>
19 
20 #include "oops/base/Geometry.h"
21 #include "oops/base/Increment.h"
23 #include "oops/mpi/mpi.h"
24 #include "oops/util/Logger.h"
25 #include "oops/util/ObjectCounter.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 model-space localization class used by high level algorithms
36 /// and applications.
37 ///
38 /// Note: to see methods that need to be implemented in a generic Localization
39 /// implementation, see LocalizationBase class in generic/LocalizationBase.h.
40 /// To see methods that need to be implemented in a MODEL-specific Localization
41 /// implementation, see interface::LocalizationBase class in
42 /// interface/LocalizationBase.h.
43 template <typename MODEL>
44 class Localization : public util::Printable,
45  private boost::noncopyable,
46  private util::ObjectCounter<Localization<MODEL> > {
50 
51  public:
52  static const std::string classname() {return "oops::Localization";}
53 
54  /// Set up Localization for \p geometry, configured with \p conf
55  Localization(const Geometry_ & geometry,
56  const eckit::Configuration & conf);
57  ~Localization();
58 
59  /// Randomize \p dx and apply 4D localization. All 3D blocks of the 4D localization
60  /// matrix are the same (and defined by 3D localization loc_)
61  virtual void randomize(Increment_ & dx) const;
62  /// Apply 4D localization. All 3D blocks of the 4D localization matrix are the same
63  /// (and defined by 3D localization loc_)
64  virtual void multiply(Increment_ & dx) const;
65 
66  private:
67  /// Print, used in logging
68  void print(std::ostream &) const override;
69 
70  /// Pointer to the Localization implementation
71  std::unique_ptr<LocBase_> loc_;
72 };
73 
74 // -----------------------------------------------------------------------------
75 
76 template<typename MODEL>
78  const eckit::Configuration & conf)
79  : loc_(LocalizationFactory<MODEL>::create(geometry, conf))
80 {
81  Log::trace() << "Localization<MODEL>::Localization done" << std::endl;
82 }
83 
84 // -----------------------------------------------------------------------------
85 
86 template<typename MODEL>
88  Log::trace() << "Localization<MODEL>::~Localization starting" << std::endl;
89  util::Timer timer(classname(), "~Localization");
90  loc_.reset();
91  Log::trace() << "Localization<MODEL>::~Localization done" << std::endl;
92 }
93 
94 // -----------------------------------------------------------------------------
95 
96 template <typename MODEL>
98  Log::trace() << "Localization<MODEL>::randomize starting" << std::endl;
99  util::Timer timer(classname(), "randomize");
100  const eckit::mpi::Comm & comm = dx.timeComm();
101  static int tag = 23456;
102  size_t nslots = comm.size();
103  int mytime = comm.rank();
104 
105  if (mytime > 0) {
106  util::DateTime dt = dx.validTime(); // Save original time value
107  dx.zero();
108  oops::mpi::receive(comm, dx, 0, tag);
109  dx.updateTime(dt - dx.validTime()); // Set time back to original value
110  } else {
111  // Apply 3D localization
112  loc_->randomize(dx);
113 
114  // Copy result to all timeslots
115  for (size_t jj = 1; jj < nslots; ++jj) {
116  oops::mpi::send(comm, dx, jj, tag);
117  }
118  }
119  ++tag;
120  Log::trace() << "Localization<MODEL>::randomize done" << std::endl;
121 }
122 
123 // -----------------------------------------------------------------------------
124 
125 template <typename MODEL>
127  Log::trace() << "Localization<MODEL>::multiply starting" << std::endl;
128  util::Timer timer(classname(), "multiply");
129  const eckit::mpi::Comm & comm = dx.timeComm();
130  static int tag = 23456;
131  size_t nslots = comm.size();
132  int mytime = comm.rank();
133 
134  // Use Mark Buehner's trick to save CPU when applying the same 3D localization for all
135  // 3D blocks of the 4D localization matrix:
136  // L_4D = ( L_3D L_3D L_3D ) = ( Id ) L_3D ( Id Id Id )
137  // ( L_3D L_3D L_3D ) ( Id )
138  // ( L_3D L_3D L_3D ) ( Id )
139  // so if :
140  // x_4D = ( x_1 )
141  // ( x_2 )
142  // ( x_3 )
143  // then:
144  // L_4D x_4D = (Id) L_3D (Id Id Id) (x_1) = (Id) L_3D (x_1+x_2+x_3) = (L_3D ( x_1 + x_2 + x_3 ))
145  // (Id) (x_2) (Id) (L_3D ( x_1 + x_2 + x_3 ))
146  // (Id) (x_3) (Id) (L_3D ( x_1 + x_2 + x_3 ))
147  // Reference in section 3.4.2. of https://rmets.onlinelibrary.wiley.com/doi/full/10.1002/qj.2325.
148  if (mytime > 0) {
149  util::DateTime dt = dx.validTime(); // Save original time value
150  oops::mpi::send(comm, dx, 0, tag);
151  dx.zero();
152  oops::mpi::receive(comm, dx, 0, tag);
153  dx.updateTime(dt - dx.validTime()); // Set time back to original value
154  } else {
155  // Sum over timeslots
156  for (size_t jj = 1; jj < nslots; ++jj) {
157  Increment_ dxtmp(dx);
158  oops::mpi::receive(comm, dxtmp, jj, tag);
159  dx.axpy(1.0, dxtmp, false);
160  }
161 
162  // Apply 3D localization
163  loc_->multiply(dx);
164 
165  // Copy result to all timeslots
166  for (size_t jj = 1; jj < nslots; ++jj) {
167  oops::mpi::send(comm, dx, jj, tag);
168  }
169  }
170  ++tag;
171 
172  Log::trace() << "Localization<MODEL>::multiply done" << std::endl;
173 }
174 
175 // -----------------------------------------------------------------------------
176 
177 template <typename MODEL>
178 void Localization<MODEL>::print(std::ostream & os) const {
179  Log::trace() << "Localization<MODEL>::print starting" << std::endl;
180  util::Timer timer(classname(), "print");
181  os << *loc_;
182  Log::trace() << "Localization<MODEL>::print done" << std::endl;
183 }
184 
185 // -----------------------------------------------------------------------------
186 
187 } // namespace oops
188 
189 #endif // OOPS_BASE_LOCALIZATION_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
const eckit::mpi::Comm & timeComm() const
Accessor to the time communicator.
Base class for generic implementations of model-space localization. Use this class as a base class fo...
Abstract model-space localization class used by high level algorithms and applications.
static const std::string classname()
virtual void randomize(Increment_ &dx) const
Localization(const Geometry_ &geometry, const eckit::Configuration &conf)
Set up Localization for geometry, configured with conf.
Geometry< MODEL > Geometry_
Increment< MODEL > Increment_
LocalizationBase< MODEL > LocBase_
virtual void multiply(Increment_ &dx) const
void print(std::ostream &) const override
Print, used in logging.
std::unique_ptr< LocBase_ > loc_
Pointer to the Localization implementation.
void axpy(const double &w, const Increment &dx, const bool check=true)
void updateTime(const util::Duration &dt)
Updates this Increment's valid time by dt (used in PseudoModel)
void zero()
Zero out this Increment.
const util::DateTime validTime() const
Accessor to the time of this Increment.
Definition: FieldL95.h:22
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.