OOPS
oops/interface/Geometry.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_INTERFACE_GEOMETRY_H_
12 #define OOPS_INTERFACE_GEOMETRY_H_
13 
14 #include <memory>
15 #include <ostream>
16 #include <string>
17 #include <vector>
18 
19 #include "atlas/field.h"
20 #include "atlas/functionspace.h"
21 
22 #include "oops/base/Variables.h"
24 #include "oops/mpi/mpi.h"
25 #include "oops/util/Logger.h"
26 #include "oops/util/ObjectCounter.h"
27 #include "oops/util/parameters/ConfigurationParameter.h"
28 #include "oops/util/parameters/GenericParameters.h"
29 #include "oops/util/parameters/HasParameters_.h"
30 #include "oops/util/parameters/Parameters.h"
31 #include "oops/util/parameters/ParametersOrConfiguration.h"
32 #include "oops/util/Printable.h"
33 #include "oops/util/Timer.h"
34 
35 namespace eckit {
36  class Configuration;
37 }
38 
39 namespace oops {
40 
41 namespace interface {
42 /// \brief Interface class for the geometry of the model/state space
43 ///
44 /// \details Can contain information about model resolution, gridpoints, MPI distribution
45 ///
46 /// Note: implementations of this interface can opt to extract their settings either from
47 /// a Configuration object or from a subclass of Parameters.
48 ///
49 /// In the former case, they should provide a constructor with the following signature:
50 ///
51 /// Geometry(const eckit::LocalConfiguration &, const eckit::mpi::Comm &);
52 ///
53 /// In the latter case, the implementer should first define a subclass of Parameters
54 /// holding the settings of the geometry in question. The implementation of the Geometry interface
55 /// should then typedef `Parameters_` to the name of that subclass and provide a constructor with
56 /// the following signature:
57 ///
58 /// Geometry(const Parameters_ &, const eckit::mpi::Comm &);
59 template <typename MODEL>
60 class Geometry : public util::Printable,
61  private util::ObjectCounter<Geometry<MODEL> > {
62  typedef typename MODEL::Geometry Geometry_;
64 
65  public:
66  /// Set to Geometry_::Parameters_ if Geometry_ provides a type called Parameters_
67  /// and to GenericParameters (a thin wrapper of an eckit::LocalConfiguration object) if not.
68  typedef TParameters_IfAvailableElseFallbackType_t<Geometry_, GenericParameters> Parameters_;
69 
70  static const std::string classname() {return "oops::Geometry";}
71 
72  /// Constructors from yaml (and mpi communicator), implement one (using Parameters preferred)
73  Geometry(const Parameters_ &, const eckit::mpi::Comm &);
74  Geometry(const eckit::Configuration &, const eckit::mpi::Comm &);
75  /// Constructor from pointer to the MODEL::Geometry (used in 1DVar filter)
76  explicit Geometry(std::shared_ptr<const Geometry_>);
77  /// Destructor (overridden for timer and log purposes)
78  virtual ~Geometry();
79 
80  /// Iterator to the first gridpoint of Geometry (only used in LocalEnsembleDA)
81  GeometryIterator_ begin() const;
82  /// Iterator to the past-the-end gridpoint of Geometry (only used in LocalEnsembleDA)
83  GeometryIterator_ end() const;
84  /// Values of vertical coordinate in units specified by string (only used in GETKF)
85  std::vector<double> verticalCoord(std::string &) const;
86 
87  /// Returns number of values required to store each of the passed model variables
88  /// \p vars at a single location (e.g. number of vertical levels).
89  /// The returned vector should be the same size as \p vars and contain number of
90  /// levels required for i-th variable in the i-th position.
91  std::vector<size_t> variableSizes(const Variables &) const;
92 
93  /// Accessor to the geometry communicator
94  const eckit::mpi::Comm & getComm() const {return geom_->getComm();}
95  atlas::FunctionSpace * atlasFunctionSpace() const {return geom_->atlasFunctionSpace();}
96  atlas::FieldSet * atlasFieldSet() const {return geom_->atlasFieldSet();}
97 
98  /// Accessor to MODEL::Geometry, used in the other interface classes in oops.
99  /// Does not need to be implemented.
100  const Geometry_ & geometry() const {return *this->geom_;}
101 
102  protected:
103  std::shared_ptr<const Geometry_> geom_; /// pointer to the Geometry implementation
104 
105  private:
106  void print(std::ostream &) const;
107 };
108 
109 // -----------------------------------------------------------------------------
110 
111 template <typename MODEL>
112 Geometry<MODEL>::Geometry(const eckit::Configuration & config,
113  const eckit::mpi::Comm & comm)
114  : Geometry(validateAndDeserialize<Parameters_>(config), comm)
115 {}
116 
117 // -----------------------------------------------------------------------------
118 
119 template <typename MODEL>
121  const eckit::mpi::Comm & comm): geom_() {
122  Log::trace() << "Geometry<MODEL>::Geometry starting" << std::endl;
123  util::Timer timer(classname(), "Geometry");
124  geom_.reset(new Geometry_(
125  parametersOrConfiguration<HasParameters_<Geometry_>::value>(parameters),
126  comm));
127  Log::trace() << "Geometry<MODEL>::Geometry done" << std::endl;
128 }
129 
130 // -----------------------------------------------------------------------------
131 
132 template <typename MODEL>
133 Geometry<MODEL>::Geometry(std::shared_ptr<const Geometry_> ptr)
134  : geom_(ptr)
135 {
136  Log::trace() << "Geometry<MODEL>::Geometry shared_ptr done" << std::endl;
137 }
138 
139 // -----------------------------------------------------------------------------
140 
141 template <typename MODEL>
143  Log::trace() << "Geometry<MODEL>::~Geometry starting" << std::endl;
144  util::Timer timer(classname(), "~Geometry");
145  geom_.reset();
146  Log::trace() << "Geometry<MODEL>::~Geometry done" << std::endl;
147 }
148 
149 // -----------------------------------------------------------------------------
150 
151 template <typename MODEL>
153  Log::trace() << "Geometry<MODEL>::begin starting" << std::endl;
154  util::Timer timer(classname(), "begin");
155  Log::trace() << "Geometry<MODEL>::begin done" << std::endl;
156  return GeometryIterator_(geom_->begin());
157 }
158 
159 // -----------------------------------------------------------------------------
160 
161 template <typename MODEL>
162 std::vector<double> Geometry<MODEL>::verticalCoord(std::string & str) const {
163  Log::trace() << "Geometry<MODEL>::verticalCoord starting" << std::endl;
164  util::Timer timer(classname(), "verticalCoord");
165  Log::trace() << "Geometry<MODEL>::verticalCoord done" << std::endl;
166  return geom_->verticalCoord(str);
167 }
168 
169 // -----------------------------------------------------------------------------
170 
171 template <typename MODEL>
172 std::vector<size_t> Geometry<MODEL>::variableSizes(const Variables & vars) const {
173  Log::trace() << "Geometry<MODEL>::variableSizes starting" << std::endl;
174  util::Timer timer(classname(), "variableSizes");
175  std::vector<size_t> sizes = geom_->variableSizes(vars);
176  Log::trace() << "Geometry<MODEL>::variableSizes done" << std::endl;
177  return sizes;
178 }
179 
180 // -----------------------------------------------------------------------------
181 
182 template <typename MODEL>
184  Log::trace() << "Geometry<MODEL>::end starting" << std::endl;
185  util::Timer timer(classname(), "end");
186  Log::trace() << "Geometry<MODEL>::end done" << std::endl;
187  return GeometryIterator_(geom_->end());
188 }
189 
190 // -----------------------------------------------------------------------------
191 
192 template <typename MODEL>
193 void Geometry<MODEL>::print(std::ostream & os) const {
194  Log::trace() << "Geometry<MODEL>::print starting" << std::endl;
195  util::Timer timer(classname(), "print");
196  os << *geom_;
197  Log::trace() << "Geometry<MODEL>::print done" << std::endl;
198 }
199 
200 } // namespace interface
201 
202 } // namespace oops
203 
204 #endif // OOPS_INTERFACE_GEOMETRY_H_
Interface class for the geometry of the model/state space.
GeometryIterator< MODEL > GeometryIterator_
atlas::FieldSet * atlasFieldSet() const
static const std::string classname()
atlas::FunctionSpace * atlasFunctionSpace() const
const eckit::mpi::Comm & getComm() const
Accessor to the geometry communicator.
GeometryIterator_ end() const
Iterator to the past-the-end gridpoint of Geometry (only used in LocalEnsembleDA)
std::vector< double > verticalCoord(std::string &) const
Values of vertical coordinate in units specified by string (only used in GETKF)
void print(std::ostream &) const
pointer to the Geometry implementation
TParameters_IfAvailableElseFallbackType_t< Geometry_, GenericParameters > Parameters_
const Geometry_ & geometry() const
std::vector< size_t > variableSizes(const Variables &) const
Geometry(const Parameters_ &, const eckit::mpi::Comm &)
Constructors from yaml (and mpi communicator), implement one (using Parameters preferred)
virtual ~Geometry()
Destructor (overridden for timer and log purposes)
GeometryIterator_ begin() const
Iterator to the first gridpoint of Geometry (only used in LocalEnsembleDA)
std::shared_ptr< const Geometry_ > geom_
Definition: FieldL95.h:22
The namespace for the main oops code.