OOPS
CostFct4DEnsVar.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_ASSIMILATION_COSTFCT4DENSVAR_H_
12 #define OOPS_ASSIMILATION_COSTFCT4DENSVAR_H_
13 
14 #include <map>
15 #include <memory>
16 #include <string>
17 
18 #include "eckit/config/LocalConfiguration.h"
19 #include "eckit/mpi/Comm.h"
25 #include "oops/base/Geometry.h"
26 #include "oops/base/Increment.h"
29 #include "oops/base/State.h"
30 #include "oops/base/StateInfo.h"
32 #include "oops/base/Variables.h"
33 #include "oops/util/DateTime.h"
34 #include "oops/util/Duration.h"
35 #include "oops/util/Logger.h"
36 
37 namespace oops {
38 
39 /// 4D-Ens-Var Cost Function
40 /*!
41  * Although so far only used for 4D-Ens-Var this cost function can
42  * be interpreted more generally as a four dimensional 3D-Var in the
43  * sense that the control variable is 4D (like weak-constraint 4D-Var)
44  * but the observation operator is 3D (does not involve the forecast
45  * model).
46  */
47 
48 // -----------------------------------------------------------------------------
49 
50 template<typename MODEL, typename OBS> class CostFct4DEnsVar : public CostFunction<MODEL, OBS> {
57 
58  public:
59  CostFct4DEnsVar(const eckit::Configuration &, const eckit::mpi::Comm &);
61 
64  const bool idModel = false) const override;
67  const bool idModel = false) const override;
68  void zeroAD(CtrlInc_ &) const override;
69 
70  void runNL(CtrlVar_ &, PostProcessor<State_>&) const override;
71 
72  private:
73  void addIncr(CtrlVar_ &, const CtrlInc_ &, PostProcessor<Increment_>&) const override;
74 
75  CostJb4D<MODEL> * newJb(const eckit::Configuration &, const Geometry_ &,
76  const CtrlVar_ &) const override;
77  CostJo<MODEL, OBS> * newJo(const eckit::Configuration &) const override;
78  CostTermBase<MODEL, OBS> * newJc(const eckit::Configuration &, const Geometry_ &) const override;
79  void doLinearize(const Geometry_ &, const eckit::Configuration &,
80  const CtrlVar_ &, const CtrlVar_ &,
82  const Geometry_ & geometry() const override {return *resol_;}
83 
84  util::Duration subWinLength_;
85  util::DateTime subWinTime_;
86  util::DateTime subWinBegin_;
87  util::DateTime subWinEnd_;
88  size_t nsubwin_;
89  size_t mysubwin_;
90  std::unique_ptr<Geometry_> resol_;
92  eckit::mpi::Comm * commSpace_;
93  eckit::mpi::Comm * commTime_;
94 };
95 
96 // =============================================================================
97 
98 template<typename MODEL, typename OBS>
99 CostFct4DEnsVar<MODEL, OBS>::CostFct4DEnsVar(const eckit::Configuration & config,
100  const eckit::mpi::Comm & comm)
101  : CostFunction<MODEL, OBS>::CostFunction(config),
102  resol_(), ctlvars_(config, "analysis variables")
103 {
104  Log::trace() << "CostFct4DEnsVar::CostFct4DEnsVar start" << std::endl;
105  util::Duration windowLength(config.getString("window length"));
106  util::DateTime windowBegin(config.getString("window begin"));
107  util::DateTime windowEnd = windowBegin + windowLength;
108  subWinLength_ = util::Duration(config.getString("subwindow"));
109 
110  nsubwin_ = windowLength.toSeconds() / subWinLength_.toSeconds() + 1; // Not like WC
111  ASSERT(windowLength.toSeconds() == subWinLength_.toSeconds() * (int64_t)(nsubwin_ - 1));
112 
113  size_t ntasks = comm.size();
114  ASSERT(ntasks % nsubwin_ == 0);
115  size_t myrank = comm.rank();
116  size_t ntaskpslot = ntasks / nsubwin_;
117  size_t mysubwin_ = myrank / ntaskpslot;
118 
119 // Define local sub-window
120  subWinTime_ = windowBegin + mysubwin_ * subWinLength_;
123  if (mysubwin_ == 0) subWinBegin_ = subWinTime_;
125  ASSERT(subWinBegin_ >= windowBegin);
126  ASSERT(subWinEnd_ <= windowEnd);
127 
128 // Create a communicator for same sub-window, to be used for communications in space
129  std::string sgeom = "comm_geom_" + std::to_string(mysubwin_);
130  char const *geomName = sgeom.c_str();
131  commSpace_ = &comm.split(mysubwin_, geomName);
132 
133 // Create a communicator for same local area, to be used for communications in time
134  size_t myarea = commSpace_->rank();
135  std::string stime = "comm_time_" + std::to_string(myarea);
136  char const *timeName = stime.c_str();
137  commTime_ = &comm.split(myarea, timeName);
138  ASSERT(commTime_->size() == nsubwin_);
139 
140 // Now can setup the rest
141  resol_.reset(new Geometry_(eckit::LocalConfiguration(config, "geometry"),
142  *commSpace_, *commTime_));
143 
144  this->setupTerms(config);
145 
146  Log::trace() << "CostFct4DEnsVar::CostFct4DEnsVar done" << std::endl;
147 }
148 
149 // -----------------------------------------------------------------------------
150 
151 template <typename MODEL, typename OBS>
152 CostJb4D<MODEL> * CostFct4DEnsVar<MODEL, OBS>::newJb(const eckit::Configuration & jbConf,
153  const Geometry_ & resol,
154  const CtrlVar_ & xb) const {
155  Log::trace() << "CostFct4DEnsVar::newJb" << std::endl;
156  return new CostJb4D<MODEL>(jbConf, *commTime_, resol, ctlvars_, xb.state());
157 }
158 
159 // -----------------------------------------------------------------------------
160 
161 template <typename MODEL, typename OBS>
162 CostJo<MODEL, OBS> * CostFct4DEnsVar<MODEL, OBS>::newJo(const eckit::Configuration & joConf) const {
163  Log::trace() << "CostFct4DEnsVar::newJo" << std::endl;
164  return new CostJo<MODEL, OBS>(joConf, *commSpace_,
165  subWinBegin_, subWinEnd_, *commTime_);
166 }
167 
168 // -----------------------------------------------------------------------------
169 
170 template <typename MODEL, typename OBS>
171 CostTermBase<MODEL, OBS> * CostFct4DEnsVar<MODEL, OBS>::newJc(const eckit::Configuration & jcConf,
172  const Geometry_ & resol) const {
173  Log::trace() << "CostFct4DEnsVar::newJc" << std::endl;
174 // const eckit::LocalConfiguration jcdfi(jcConf, "jcdfi");
175 // const util::DateTime vt(subWinBegin_ + windowLength_/2);
176 // return new CostJcDFI<MODEL, OBS>(jcdfi, resol, vt, windowLength_, subWinLength_);
177  Log::warning() << "CostFct4DEnsVar::newJc NO Jc" << std::endl;
178  CostTermBase<MODEL, OBS> * pjc = 0;
179  return pjc;
180 }
181 
182 // -----------------------------------------------------------------------------
183 
184 template <typename MODEL, typename OBS>
186  Log::trace() << "CostFct4DEnsVar::runNL start" << std::endl;
187  ASSERT(xx.state().validTime() == subWinTime_);
188 
189  post.initialize(xx.state(), subWinTime_, subWinLength_);
190  post.process(xx.state());
191  post.finalize(xx.state());
192 
193  Log::info() << "CostFct4DEnsVar::runNL: " << xx << std::endl;
194  Log::trace() << "CostFct4DEnsVar::runNL done" << std::endl;
195 }
196 
197 // -----------------------------------------------------------------------------
198 
199 template<typename MODEL, typename OBS>
201  const eckit::Configuration & conf,
202  const CtrlVar_ &, const CtrlVar_ &,
204  PostProcessorTLAD<MODEL> & pptraj) {
205  Log::trace() << "CostFct4DEnsVar::doLinearize start" << std::endl;
206  pp.enrollProcessor(new TrajectorySaver<MODEL>(conf, resol, pptraj));
207  Log::trace() << "CostFct4DEnsVar::doLinearize done" << std::endl;
208 }
209 
210 // -----------------------------------------------------------------------------
211 
212 template <typename MODEL, typename OBS>
216  const bool) const {
217  Log::trace() << "CostFct4DEnsVar::runTLM start" << std::endl;
218  ASSERT(dx.state().validTime() == subWinTime_);
219 
220  cost.initializeTL(dx.state(), subWinTime_, subWinLength_);
221  post.initialize(dx.state(), subWinTime_, subWinLength_);
222 
223  cost.processTL(dx.state());
224  post.process(dx.state());
225 
226  cost.finalizeTL(dx.state());
227  post.finalize(dx.state());
228 
229  Log::info() << "CostFct4DEnsVar::runTLM: " << dx << std::endl;
230  Log::trace() << "CostFct4DEnsVar::runTLM done" << std::endl;
231 }
232 
233 // -----------------------------------------------------------------------------
234 
235 template <typename MODEL, typename OBS>
237  Log::trace() << "CostFct4DEnsVar::zeroAD start" << std::endl;
238  dx.state().zero(subWinTime_);
239  dx.modVar().zero();
240  dx.obsVar().zero();
241  Log::trace() << "CostFct4DEnsVar::zeroAD done" << std::endl;
242 }
243 
244 // -----------------------------------------------------------------------------
245 
246 template <typename MODEL, typename OBS>
250  const bool) const {
251  Log::trace() << "CostFct4DEnsVar::runADJ start" << std::endl;
252 
253  post.initialize(dx.state(), subWinTime_, subWinLength_);
254  cost.initializeAD(dx.state(), subWinTime_, subWinLength_);
255 
256  cost.processAD(dx.state());
257  post.process(dx.state());
258 
259  cost.finalizeAD(dx.state());
260  post.finalize(dx.state());
261 
262  Log::info() << "CostFct4DEnsVar::runADJ: " << dx << std::endl;
263  Log::trace() << "CostFct4DEnsVar::runADJ done" << std::endl;
264 }
265 
266 // -----------------------------------------------------------------------------
267 
268 template<typename MODEL, typename OBS>
270  PostProcessor<Increment_> &) const {
271  Log::trace() << "CostFct4DEnsVar::addIncr start" << std::endl;
272  ASSERT(xx.state().validTime() == subWinTime_);
273  ASSERT(dx.state().validTime() == subWinTime_);
274  xx.state() += dx.state();
275  Log::trace() << "CostFct4DEnsVar::addIncr done" << std::endl;
276 }
277 
278 // -----------------------------------------------------------------------------
279 
280 } // namespace oops
281 
282 #endif // OOPS_ASSIMILATION_COSTFCT4DENSVAR_H_
ObsAuxIncrs_ & obsVar()
Get augmented observation control variable.
Increment_ & state()
Get state control variable.
ModelAuxIncr_ & modVar()
Get augmented model control variable.
Control variable.
State_ & state()
Get state control variable.
4D-Ens-Var Cost Function
ControlVariable< MODEL, OBS > CtrlVar_
util::DateTime subWinEnd_
eckit::mpi::Comm * commTime_
ControlIncrement< MODEL, OBS > CtrlInc_
Increment< MODEL > Increment_
void addIncr(CtrlVar_ &, const CtrlInc_ &, PostProcessor< Increment_ > &) const override
void runADJ(CtrlInc_ &, PostProcessorTLAD< MODEL > &, PostProcessor< Increment_ >, const bool idModel=false) const override
void runNL(CtrlVar_ &, PostProcessor< State_ > &) const override
eckit::mpi::Comm * commSpace_
std::unique_ptr< Geometry_ > resol_
CostTermBase< MODEL, OBS > * newJc(const eckit::Configuration &, const Geometry_ &) const override
State< MODEL > State_
CostFct4DEnsVar(const eckit::Configuration &, const eckit::mpi::Comm &)
const Geometry_ & geometry() const override
CostFunction< MODEL, OBS > CostFct_
util::DateTime subWinTime_
CostJb4D< MODEL > * newJb(const eckit::Configuration &, const Geometry_ &, const CtrlVar_ &) const override
const Variables ctlvars_
CostJo< MODEL, OBS > * newJo(const eckit::Configuration &) const override
void zeroAD(CtrlInc_ &) const override
util::DateTime subWinBegin_
Geometry< MODEL > Geometry_
void runTLM(CtrlInc_ &, PostProcessorTLAD< MODEL > &, PostProcessor< Increment_ >, const bool idModel=false) const override
util::Duration subWinLength_
void doLinearize(const Geometry_ &, const eckit::Configuration &, const CtrlVar_ &, const CtrlVar_ &, PostProcessor< State_ > &, PostProcessorTLAD< MODEL > &) override
Cost Function.
Definition: CostFunction.h:53
void setupTerms(const eckit::Configuration &)
Definition: CostFunction.h:194
4D Jb Cost Function
Definition: CostJb4D.h:39
Jo Cost Function.
Definition: CostJo.h:52
Base Class for Cost Function Terms.
Definition: CostTermBase.h:36
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
void zero()
Zero out this ModelAuxIncrement.
Control model post processing.
Definition: PostProcessor.h:30
void enrollProcessor(PostBase_ *pp)
Definition: PostProcessor.h:38
void process(const FLDS &xx)
Definition: PostProcessor.h:56
void finalize(const FLDS &xx)
Definition: PostProcessor.h:62
void initialize(const FLDS &xx, const util::DateTime &end, const util::Duration &step)
Definition: PostProcessor.h:49
Control model post processing.
void finalizeAD(Increment_ &dx)
void processTL(const Increment_ &dx)
void processAD(Increment_ &dx)
void finalizeTL(const Increment_ &dx)
void initializeTL(const Increment_ &dx, const util::DateTime &end, const util::Duration &step)
Tangent linear methods.
void initializeAD(Increment_ &dx, const util::DateTime &bgn, const util::Duration &step)
Adjoint methods.
State class used in oops; subclass of interface class interface::State.
Save trajectory during forecast run.
void zero()
Zero out this Increment.
const util::DateTime validTime() const
Accessor to the time of this Increment.
const util::DateTime validTime() const
Accessor to the time of this State.
The namespace for the main oops code.