11 #ifndef OOPS_ASSIMILATION_COSTFUNCTION_H_
12 #define OOPS_ASSIMILATION_COSTFUNCTION_H_
18 #include <boost/noncopyable.hpp>
19 #include <boost/ptr_container/ptr_vector.hpp>
21 #include "eckit/config/LocalConfiguration.h"
22 #include "eckit/mpi/Comm.h"
38 #include "oops/util/DateTime.h"
39 #include "oops/util/dot_product.h"
40 #include "oops/util/Duration.h"
41 #include "oops/util/Logger.h"
53 template<
typename MODEL,
typename OBS>
class CostFunction :
private boost::noncopyable {
68 const eckit::Configuration & config = eckit::LocalConfiguration(),
75 const bool idModel =
false)
const = 0;
78 const bool idModel =
false)
const = 0;
116 std::unique_ptr<const CtrlVar_>
xb_;
117 std::unique_ptr<JbTotal_>
jb_;
127 template <
typename MODEL,
typename OBS>
137 const eckit::mpi::Comm &) = 0;
138 static std::map < std::string, CostFactory<MODEL, OBS> * > &
getMakers() {
139 static std::map < std::string, CostFactory<MODEL, OBS> * > makers_;
144 template<
class MODEL,
class OBS,
class FCT>
148 const eckit::mpi::Comm & comm)
override
149 {
return new FCT(config, comm);}
159 template <
typename MODEL,
typename OBS>
161 if (getMakers().find(name) != getMakers().end()) {
162 throw std::runtime_error(name +
" already registered in cost function factory.");
164 getMakers()[name] =
this;
169 template <
typename MODEL,
typename OBS>
171 const eckit::mpi::Comm & comm) {
172 std::string
id = config.getString(
"cost type");
173 Log::trace() <<
"Variational Assimilation Type=" <<
id << std::endl;
174 typename std::map<std::string, CostFactory<MODEL, OBS>*>::iterator j = getMakers().find(
id);
175 if (j == getMakers().end()) {
176 throw std::runtime_error(
id +
" does not exist in cost function factory.");
178 Log::trace() <<
"CostFactory::create found cost function type" << std::endl;
179 return (*j).second->make(config, comm);
186 template<
typename MODEL,
typename OBS>
193 template<
typename MODEL,
typename OBS>
195 Log::trace() <<
"CostFunction::setupTerms start" << std::endl;
198 eckit::LocalConfiguration obsconf(config,
"observations");
200 jterms_.push_back(jo);
201 Log::trace() <<
"CostFunction::setupTerms Jo added" << std::endl;
205 jb_.reset(
new JbTotal_(*xb_, this->newJb(config, this->geometry(), *xb_),
206 config, this->geometry(), jo->
obspaces()));
207 Log::trace() <<
"CostFunction::setupTerms Jb added" << std::endl;
210 std::vector<eckit::LocalConfiguration> jcs;
211 config.get(
"constraints", jcs);
212 for (
size_t jj = 0; jj < jcs.size(); ++jj) {
214 jterms_.push_back(jc);
216 Log::trace() <<
"CostFunction::setupTerms done" << std::endl;
221 template<
typename MODEL,
typename OBS>
223 const eckit::Configuration & config,
225 Log::trace() <<
"CostFunction::evaluate start" << std::endl;
227 jb_->initialize(fguess);
229 for (
size_t jj = 0; jj < jterms_.size(); ++jj) {
230 jterms_[jj].setPostProc(fguess, config, pp);
235 this->runNL(mfguess, pp);
239 costJb_ = jb_->finalize(mfguess);
242 for (
size_t jj = 0; jj < jterms_.size(); ++jj) {
243 costJoJc_ += jterms_[jj].computeCost();
246 Log::test() <<
"CostFunction: Nonlinear J = " << zzz << std::endl;
247 Log::trace() <<
"CostFunction::evaluate done" << std::endl;
253 template<
typename MODEL,
typename OBS>
255 const eckit::Configuration & innerConf,
257 Log::trace() <<
"CostFunction::linearize start" << std::endl;
259 const eckit::LocalConfiguration resConf(innerConf,
"geometry");
260 const Geometry_ lowres(resConf, this->geometry().getComm(), this->geometry().timeComm());
265 for (
size_t jj = 0; jj < jterms_.size(); ++jj) {
266 jterms_[jj].setPostProcTraj(fguess, innerConf, lowres, pptraj);
270 this->doLinearize(lowres, innerConf, *xb_, fguess, post, pptraj);
273 double zzz = this->evaluate(fguess, innerConf, post);
277 for (
size_t jj = 0; jj < jterms_.size(); ++jj) {
278 jterms_[jj].computeCostTraj();
281 Log::trace() <<
"CostFunction::linearize done" << std::endl;
287 template<
typename MODEL,
typename OBS>
289 Log::trace() <<
"CostFunction::computeGradientFG start" << std::endl;
294 for (
size_t jj = 0; jj < jterms_.size(); ++jj) {
295 std::shared_ptr<const GeneralizedDepartures> tmp(jterms_[jj].newGradientFG());
296 jterms_[jj].computeCostAD(tmp, grad, costad);
299 this->runADJ(grad, costad, pp);
301 for (
size_t jj = 0; jj < jterms_.size(); ++jj) {
302 jterms_[jj].setPostProcAD();
305 Log::info() <<
"CostFunction::computeGradientFG: gradient:" << grad << std::endl;
306 Log::trace() <<
"CostFunction::computeGradientFG done" << std::endl;
311 template<
typename MODEL,
typename OBS>
314 Log::trace() <<
"CostFunction::addIncrement start" << std::endl;
315 Log::info() <<
"CostFunction::addIncrement: First guess:" << xx << std::endl;
316 Log::info() <<
"CostFunction::addIncrement: Increment:" << dx << std::endl;
320 this->addIncr(xx, dx, post);
322 Log::info() <<
"CostFunction::addIncrement: Analysis: " << xx << std::endl;
323 Log::test() <<
"CostFunction::addIncrement: Analysis: " << xx << std::endl;
324 Log::trace() <<
"CostFunction::addIncrement done" << std::endl;
329 template<
typename MODEL,
typename OBS>
331 Log::trace() <<
"CostFunction::resetLinearization start" << std::endl;
332 for (
size_t jj = 0; jj < jterms_.size(); ++jj) {
333 jterms_[jj].resetLinearization();
335 Log::trace() <<
"CostFunction::resetLinearization done" << std::endl;
ObsAuxIncrs_ & obsVar()
Get augmented observation control variable.
ModelAuxIncr_ & modVar()
Get augmented model control variable.
ObsAuxCtrls_ & obsVar()
Get augmented observation control variable.
ModelAux_ & modVar()
Get augmented model control variable.
virtual CostFunction< MODEL, OBS > * make(const eckit::Configuration &, const eckit::mpi::Comm &)=0
virtual ~CostFactory()=default
CostFactory(const std::string &)
static CostFunction< MODEL, OBS > * create(const eckit::Configuration &, const eckit::mpi::Comm &)
static std::map< std::string, CostFactory< MODEL, OBS > * > & getMakers()
CostFunction(const eckit::Configuration &)
double evaluate(const CtrlVar_ &, const eckit::Configuration &config=eckit::LocalConfiguration(), PostProcessor< State_ > post=PostProcessor< State_ >())
const JbTotal_ & jb() const
Access .
virtual void runTLM(CtrlInc_ &, PostProcessorTLAD< MODEL > &, PostProcessor< Increment_ > post=PostProcessor< Increment_ >(), const bool idModel=false) const =0
virtual void runNL(CtrlVar_ &, PostProcessor< State_ > &) const =0
ControlIncrement< MODEL, OBS > CtrlInc_
JqTermTLAD< MODEL > JqTermTLAD_
virtual void runADJ(CtrlInc_ &, PostProcessorTLAD< MODEL > &, PostProcessor< Increment_ > post=PostProcessor< Increment_ >(), const bool idModel=false) const =0
CostJbTotal< MODEL, OBS > JbTotal_
const CostBase_ & jterm(const size_t ii) const
Access terms of the cost function other than .
Geometry< MODEL > Geometry_
ControlVariable< MODEL, OBS > CtrlVar_
double getCostJoJc() const
void setupTerms(const eckit::Configuration &)
boost::ptr_vector< CostBase_ > jterms_
virtual CostJo< MODEL, OBS > * newJo(const eckit::Configuration &) const =0
virtual const Geometry_ & geometry() const =0
virtual CostJbState< MODEL > * newJb(const eckit::Configuration &, const Geometry_ &, const CtrlVar_ &) const =0
void addIncrement(CtrlVar_ &, const CtrlInc_ &, PostProcessor< Increment_ > post=PostProcessor< Increment_ >()) const
CostTermBase< MODEL, OBS > CostBase_
double linearize(const CtrlVar_ &, const eckit::Configuration &, PostProcessor< State_ > post=PostProcessor< State_ >())
std::unique_ptr< const CtrlVar_ > xb_
virtual CostTermBase< MODEL, OBS > * newJc(const eckit::Configuration &, const Geometry_ &) const =0
virtual void zeroAD(CtrlInc_ &) const =0
virtual void doLinearize(const Geometry_ &, const eckit::Configuration &, const CtrlVar_ &, const CtrlVar_ &, PostProcessor< State_ > &, PostProcessorTLAD< MODEL > &)=0
void computeGradientFG(CtrlInc_ &) const
Compute cost function gradient at first guess (without Jb).
void resetLinearization()
virtual void addIncr(CtrlVar_ &, const CtrlInc_ &, PostProcessor< Increment_ > &) const =0
const CtrlVar_ & background() const
std::unique_ptr< JbTotal_ > jb_
Increment< MODEL > Increment_
Jb Cost Function Base Class.
Control variable increment.
const ObsSpaces_ & obspaces() const
Accessor...
CostMaker(const std::string &name)
CostFunction< MODEL, OBS > * make(const eckit::Configuration &config, const eckit::mpi::Comm &comm) override
Base Class for Cost Function Terms.
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
Control model post processing.
Control model post processing.
void initializeTraj(const State_ &xx, const util::DateTime &end, const util::Duration &step)
Set linearization state.
State class used in oops; subclass of interface class interface::State.
The namespace for the main oops code.