OOPS
oops/interface/State.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_STATE_H_
13 #define OOPS_INTERFACE_STATE_H_
14 
15 #include <memory>
16 #include <string>
17 #include <vector>
18 
19 #include "eckit/config/Configuration.h"
20 #include "oops/base/Geometry.h"
22 #include "oops/base/Variables.h"
23 #include "oops/util/DateTime.h"
24 #include "oops/util/gatherPrint.h"
25 #include "oops/util/ObjectCounter.h"
26 #include "oops/util/Printable.h"
27 #include "oops/util/Serializable.h"
28 #include "oops/util/Timer.h"
29 
30 namespace oops {
31 
32 namespace interface {
33 
34 /// Encapsulates the model state
35 
36 // -----------------------------------------------------------------------------
37 
38 template <typename MODEL>
39 class State : public util::Printable,
40  public util::Serializable,
41  private util::ObjectCounter<State<MODEL> > {
42  typedef typename MODEL::State State_;
44 
45  public:
46  static const std::string classname() {return "oops::State";}
47 
48  /// Constructor for specified \p resol, with \p vars, valid at \p time
49  State(const Geometry_ & resol, const Variables & vars, const util::DateTime & time);
50  /// Constructor for specified \p resol and files read from \p conf
51  State(const Geometry_ & resol, const eckit::Configuration & conf);
52  /// Copies \p other State, changing its resolution to \p geometry
53  State(const Geometry_ & resol, const State & other);
54  /// Copy constructor
55  State(const State &);
56  /// Destructor (defined explicitly for timing and tracing)
57  ~State();
58  /// Assignment operator
59  State & operator =(const State &);
60 
61  /// Accessor
62  State_ & state() {return *state_;}
63  /// const accessor
64  const State_ & state() const {return *state_;}
65 
66  /// Accessor to the time of this State
67  const util::DateTime validTime() const {return state_->validTime();}
68  /// Update this State's valid time by \p dt
69  void updateTime(const util::Duration & dt) {state_->updateTime(dt);}
70 
71  /// Read this State from file
72  void read(const eckit::Configuration &);
73  /// Write this State out to file
74  void write(const eckit::Configuration &) const;
75  /// Norm (used in tests)
76  double norm() const;
77 
78  /// Accessor to geometry associated with this State
79  Geometry_ geometry() const;
80  /// Accessor to variables associated with this State
81  const Variables & variables() const;
82 
83  /// Zero out this State
84  void zero();
85  /// Accumulate (add \p w * \p x to the state)
86  void accumul(const double & w, const State & x);
87 
88  /// Serialize and deserialize (used in 4DEnVar, weak-constraint 4DVar and Block-Lanczos minimizer)
89  size_t serialSize() const override;
90  void serialize(std::vector<double> &) const override;
91  void deserialize(const std::vector<double> &, size_t &) override;
92 
93  private:
94  std::unique_ptr<State_> state_;
95  void print(std::ostream &) const override;
96 };
97 
98 // =============================================================================
99 
100 template<typename MODEL>
101 State<MODEL>::State(const Geometry_ & resol, const Variables & vars,
102  const util::DateTime & time) : state_()
103 {
104  Log::trace() << "State<MODEL>::State starting" << std::endl;
105  util::Timer timer(classname(), "State");
106  state_.reset(new State_(resol.geometry(), vars, time));
107  this->setObjectSize(state_->serialSize()*sizeof(double));
108  Log::trace() << "State<MODEL>::State done" << std::endl;
109 }
110 
111 // -----------------------------------------------------------------------------
112 
113 template<typename MODEL>
114 State<MODEL>::State(const Geometry_ & resol, const eckit::Configuration & conf)
115  : state_()
116 {
117  Log::trace() << "State<MODEL>::State read starting" << std::endl;
118  util::Timer timer(classname(), "State");
119 
120  eckit::LocalConfiguration myconf;
121  if (conf.has("states")) {
122 // Parallel 4D state:
123  std::vector<eckit::LocalConfiguration> confs;
124  conf.get("states", confs);
125  ASSERT(confs.size() == resol.timeComm().size());
126  myconf = confs[resol.timeComm().rank()];
127  } else {
128 // 3D state:
129  myconf = eckit::LocalConfiguration(conf);
130  }
131 
132  state_.reset(new State_(resol.geometry(), myconf));
133  this->setObjectSize(state_->serialSize()*sizeof(double));
134  Log::trace() << "State<MODEL>::State read done" << std::endl;
135 }
136 
137 // -----------------------------------------------------------------------------
138 
139 template<typename MODEL>
140 State<MODEL>::State(const Geometry_ & resol, const State & other)
141  : state_()
142 {
143  Log::trace() << "State<MODEL>::State interpolated starting" << std::endl;
144  util::Timer timer(classname(), "State");
145  state_.reset(new State_(resol.geometry(), *other.state_));
146  this->setObjectSize(state_->serialSize()*sizeof(double));
147  Log::trace() << "State<MODEL>::State interpolated done" << std::endl;
148 }
149 
150 // -----------------------------------------------------------------------------
151 
152 template<typename MODEL>
153 State<MODEL>::State(const State & other) : state_()
154 {
155  Log::trace() << "State<MODEL>::State starting copy" << std::endl;
156  util::Timer timer(classname(), "State");
157  state_.reset(new State_(*other.state_));
158  this->setObjectSize(state_->serialSize()*sizeof(double));
159  Log::trace() << "State<MODEL>::State copy done" << std::endl;
160 }
161 
162 // -----------------------------------------------------------------------------
163 
164 template<typename MODEL>
166  Log::trace() << "State<MODEL>::~State starting" << std::endl;
167  util::Timer timer(classname(), "~State");
168  state_.reset();
169  Log::trace() << "State<MODEL>::~State done" << std::endl;
170 }
171 
172 // -----------------------------------------------------------------------------
173 
174 template<typename MODEL>
176  Log::trace() << "State<MODEL>::operator= starting" << std::endl;
177  util::Timer timer(classname(), "operator=");
178  *state_ = *rhs.state_;
179  Log::trace() << "State<MODEL>::operator= done" << std::endl;
180  return *this;
181 }
182 
183 // -----------------------------------------------------------------------------
184 
185 template<typename MODEL>
186 void State<MODEL>::read(const eckit::Configuration & conf) {
187  Log::trace() << "State<MODEL>::read starting" << std::endl;
188  util::Timer timer(classname(), "read");
189  state_->read(conf);
190  Log::trace() << "State<MODEL>::read done" << std::endl;
191 }
192 
193 // -----------------------------------------------------------------------------
194 
195 template<typename MODEL>
196 void State<MODEL>::write(const eckit::Configuration & conf) const {
197  Log::trace() << "State<MODEL>::write starting" << std::endl;
198  util::Timer timer(classname(), "write");
199  state_->write(conf);
200  Log::trace() << "State<MODEL>::write done" << std::endl;
201 }
202 
203 // -----------------------------------------------------------------------------
204 
205 template<typename MODEL>
206 double State<MODEL>::norm() const {
207  Log::trace() << "State<MODEL>::norm starting" << std::endl;
208  util::Timer timer(classname(), "norm");
209  double zz = state_->norm();
210  Log::trace() << "State<MODEL>::norm done" << std::endl;
211  return zz;
212 }
213 
214 // -----------------------------------------------------------------------------
215 
216 template<typename MODEL>
218  Log::trace() << "State<MODEL>::geometry starting" << std::endl;
219  util::Timer timer(classname(), "geometry");
220  oops::Geometry<MODEL> geom(state_->geometry());
221  Log::trace() << "State<MODEL>::geometry done" << std::endl;
222  return geom;
223 }
224 
225 // -----------------------------------------------------------------------------
226 
227 template<typename MODEL>
229  Log::trace() << "State<MODEL>::variables starting" << std::endl;
230  util::Timer timer(classname(), "variables");
231  return state_->variables();
232 }
233 
234 // -----------------------------------------------------------------------------
235 
236 template<typename MODEL>
237 size_t State<MODEL>::serialSize() const {
238  Log::trace() << "State<MODEL>::serialSize" << std::endl;
239  util::Timer timer(classname(), "serialSize");
240  return state_->serialSize();
241 }
242 
243 // -----------------------------------------------------------------------------
244 
245 template<typename MODEL>
246 void State<MODEL>::serialize(std::vector<double> & vect) const {
247  Log::trace() << "State<MODEL>::serialize starting" << std::endl;
248  util::Timer timer(classname(), "serialize");
249  state_->serialize(vect);
250  Log::trace() << "State<MODEL>::serialize done" << std::endl;
251 }
252 
253 // -----------------------------------------------------------------------------
254 
255 template<typename MODEL>
256 void State<MODEL>::deserialize(const std::vector<double> & vect, size_t & current) {
257  Log::trace() << "State<MODEL>::State deserialize starting" << std::endl;
258  util::Timer timer(classname(), "deserialize");
259  state_->deserialize(vect, current);
260  Log::trace() << "State<MODEL>::State deserialize done" << std::endl;
261 }
262 
263 // -----------------------------------------------------------------------------
264 
265 template<typename MODEL>
266 void State<MODEL>::print(std::ostream & os) const {
267  Log::trace() << "State<MODEL>::print starting" << std::endl;
268  util::Timer timer(classname(), "print");
269  os << *state_;
270  Log::trace() << "State<MODEL>::print done" << std::endl;
271 }
272 
273 // -----------------------------------------------------------------------------
274 
275 template<typename MODEL>
277  Log::trace() << "State<MODEL>::zero starting" << std::endl;
278  util::Timer timer(classname(), "zero");
279  state_->zero();
280  Log::trace() << "State<MODEL>::zero done" << std::endl;
281 }
282 
283 // -----------------------------------------------------------------------------
284 
285 template<typename MODEL>
286 void State<MODEL>::accumul(const double & zz, const State & xx) {
287  Log::trace() << "State<MODEL>::accumul starting" << std::endl;
288  util::Timer timer(classname(), "accumul");
289  state_->accumul(zz, *xx.state_);
290  Log::trace() << "State<MODEL>::accumul done" << std::endl;
291 }
292 
293 // -----------------------------------------------------------------------------
294 
295 } // namespace interface
296 
297 } // namespace oops
298 
299 #endif // OOPS_INTERFACE_STATE_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
const eckit::mpi::Comm & timeComm() const
Accessor to the MPI communicator for distribution in time.
const Geometry_ & geometry() const
Encapsulates the model state.
void print(std::ostream &) const override
Geometry_ geometry() const
Accessor to geometry associated with this State.
State & operator=(const State &)
Assignment operator.
oops::Geometry< MODEL > Geometry_
void zero()
Zero out this State.
State(const Geometry_ &resol, const Variables &vars, const util::DateTime &time)
Constructor for specified resol, with vars, valid at time.
const util::DateTime validTime() const
Accessor to the time of this State.
static const std::string classname()
const Variables & variables() const
Accessor to variables associated with this State.
State_ & state()
Accessor.
size_t serialSize() const override
Serialize and deserialize (used in 4DEnVar, weak-constraint 4DVar and Block-Lanczos minimizer)
const State_ & state() const
const accessor
void updateTime(const util::Duration &dt)
Update this State's valid time by dt.
~State()
Destructor (defined explicitly for timing and tracing)
void accumul(const double &w, const State &x)
Accumulate (add w * x to the state)
void serialize(std::vector< double > &) const override
std::unique_ptr< State_ > state_
void read(const eckit::Configuration &)
Read this State from file.
void write(const eckit::Configuration &) const
Write this State out to file.
void deserialize(const std::vector< double > &, size_t &) override
double norm() const
Norm (used in tests)
The namespace for the main oops code.