OOPS
ChangeVariables.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018-2021 UCAR
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  */
7 
8 #ifndef OOPS_INTERFACE_CHANGEVARIABLES_H_
9 #define OOPS_INTERFACE_CHANGEVARIABLES_H_
10 
11 #include <memory>
12 #include <string>
13 
14 #include "eckit/config/LocalConfiguration.h"
15 #include "oops/base/Geometry.h"
16 #include "oops/base/State.h"
17 #include "oops/base/Variables.h"
19 #include "oops/util/Logger.h"
20 #include "oops/util/ObjectCounter.h"
21 #include "oops/util/Printable.h"
22 #include "oops/util/Timer.h"
23 
24 namespace oops {
25 
26 /// \brief Encapsulates the nonlinear variable change
27 /// There should not be a factory for ChangeVariable, it should be a trait class.
28 /// This is a temporary solution to get around it until we can fix the models.
29 template <typename MODEL>
30 class ChangeVariables : public util::Printable,
31  private util::ObjectCounter<ChangeVariables<MODEL> > {
35 
36  public:
37  static const std::string classname() {return "oops::ChangeVariables";}
38 
39  ChangeVariables(const eckit::Configuration &, const Geometry_ &,
40  const Variables &, const Variables &);
41  virtual ~ChangeVariables();
42  ChangeVariables(const ChangeVariables &) = delete;
44  const ChangeVariables & operator=(const ChangeVariables &) = delete;
46 
47  /// change variable from state \p xin to \p xout
48  void changeVar(const State_ & xin, State_ & xout) const;
49  /// inverse of changeVar, change variables back from \p xout to \p xin
50  void changeVarInverse(const State_ & xout, State_ & xin) const;
51 
52  private:
53  void print(std::ostream &) const override;
54 
55  std::unique_ptr<VariableChange_> chvar_; /// pointer to the VariableChange implementation
56 };
57 
58 // =============================================================================
59 
60 template<typename MODEL>
61 ChangeVariables<MODEL>::ChangeVariables(const eckit::Configuration & conf, const Geometry_ & geom,
62  const Variables & varin, const Variables & varout)
63  : chvar_()
64 {
65  Log::trace() << "ChangeVariables<MODEL>::ChangeVariables starting" << std::endl;
66 // Comment timer for now to avoid double counting
67 // util::Timer timer(classname(), "ChangeVariables");
68  eckit::LocalConfiguration chconf(conf);
69  if (!chconf.has("variable change")) chconf.set("variable change", "default");
70  chconf.set("input variables", varin.variables());
71  chconf.set("output variables", varout.variables());
72  chvar_.reset(new VariableChange_(geom, chconf));
73  Log::trace() << "ChangeVariables<MODEL>::ChangeVariables done" << std::endl;
74 }
75 
76 // -----------------------------------------------------------------------------
77 
78 template<typename MODEL>
80  Log::trace() << "ChangeVariables<MODEL>::~ChangeVariables starting" << std::endl;
81 // util::Timer timer(classname(), "~ChangeVariables");
82  chvar_.reset();
83  Log::trace() << "ChangeVariables<MODEL>::~ChangeVariables done" << std::endl;
84 }
85 
86 // -----------------------------------------------------------------------------
87 
88 template<typename MODEL>
89 void ChangeVariables<MODEL>::changeVar(const State_ & x1, State_ & x2) const {
90  Log::trace() << "ChangeVariables<MODEL>::changeVar starting" << std::endl;
91 // util::Timer timer(classname(), "changeVar");
92  chvar_->changeVar(x1, x2);
93  Log::trace() << "ChangeVariables<MODEL>::changeVar done" << std::endl;
94 }
95 
96 // -----------------------------------------------------------------------------
97 
98 template<typename MODEL>
100  Log::trace() << "ChangeVariables<MODEL>::changeVarInverse starting" << std::endl;
101 // util::Timer timer(classname(), "changeVarInverse");
102  chvar_->changeVarInverse(x1, x2);
103  Log::trace() << "ChangeVariables<MODEL>::changeVarInverse done" << std::endl;
104 }
105 
106 // -----------------------------------------------------------------------------
107 
108 template<typename MODEL>
109 void ChangeVariables<MODEL>::print(std::ostream & os) const {
110  Log::trace() << "ChangeVariables<MODEL>::print starting" << std::endl;
111 // util::Timer timer(classname(), "print");
112  os << *chvar_;
113  Log::trace() << "ChangeVariables<MODEL>::print done" << std::endl;
114 }
115 
116 // -----------------------------------------------------------------------------
117 
118 } // namespace oops
119 
120 #endif // OOPS_INTERFACE_CHANGEVARIABLES_H_
Encapsulates the nonlinear variable change There should not be a factory for ChangeVariable,...
void changeVarInverse(const State_ &xout, State_ &xin) const
inverse of changeVar, change variables back from xout to xin
const ChangeVariables & operator=(const ChangeVariables &)=delete
Geometry< MODEL > Geometry_
ChangeVariables & operator=(ChangeVariables &&)=default
std::unique_ptr< VariableChange_ > chvar_
ChangeVariables(ChangeVariables &&)=default
ChangeVariables(const eckit::Configuration &, const Geometry_ &, const Variables &, const Variables &)
ChangeVariables(const ChangeVariables &)=delete
void changeVar(const State_ &xin, State_ &xout) const
change variable from state xin to xout
static const std::string classname()
State< MODEL > State_
VariableChange< MODEL > VariableChange_
void print(std::ostream &) const override
Geometry class used in oops; subclass of interface class interface::Geometry.
State class used in oops; subclass of interface class interface::State.
Encapsulates the nonlinear variable change Note: to see methods that need to be implemented in the im...
const std::vector< std::string > & variables() const
The namespace for the main oops code.