11 #ifndef OOPS_ASSIMILATION_DRMINIMIZER_H_
12 #define OOPS_ASSIMILATION_DRMINIMIZER_H_
19 #include "eckit/config/Configuration.h"
27 #include "oops/util/dot_product.h"
28 #include "oops/util/FloatCompare.h"
29 #include "oops/util/formats.h"
30 #include "oops/util/Logger.h"
63 const double,
const double,
64 const int,
const double) = 0;
68 std::vector<CtrlInc_>
dxh_;
74 template<
typename MODEL,
typename OBS>
77 int ninner = config.getInt(
"ninner");
78 double gnreduc = config.getDouble(
"gradient norm reduction");
80 bool runOnlineAdjTest = config.getBool(
"online diagnostics.online adj test",
false);
83 gradJb_.reset(
new CtrlInc_(J_.jb().resolution(), *gradJb_));
85 gradJb_.reset(
new CtrlInc_(J_.jb()));
88 Log::info() << std::endl;
89 Log::info() << classname() <<
": max iter = " << ninner
90 <<
", requested norm reduction = " << gnreduc << std::endl;
94 const HtRinvH_ HtRinvH(J_, runOnlineAdjTest);
106 if (config.has(
"fsoi")) {
107 const eckit::LocalConfiguration FcSensitivityConfig(config,
"fsoi.input forecast sensitivity");
108 rhs.
read(FcSensitivityConfig);
111 Log::info() << classname() <<
" rhs has forecast sensitivity" << std::endl;
113 J_.computeGradientFG(rhs);
114 J_.jb().addGradientFG(rhs, *gradJb_);
117 Log::info() << classname() <<
" rhs" << rhs << std::endl;
120 const double gnorm = dot_product(rhs, rhs);
121 const double epsilon = config.getDouble(
"epsilon", std::numeric_limits<double>::epsilon());
122 Log::info() <<
"Initial RHS squared norm = " << gnorm << std::endl;
123 if (gnorm < epsilon) {
124 Log::info() <<
"RHS smaller than " << epsilon <<
", returning." << std::endl;
129 const double costJ0Jb = costJ0Jb_;
130 const double costJ0JoJc = J_.getCostJoJc();
133 double reduc = this->solve(*dx, dxh, rhs, B, HtRinvH, costJ0Jb, costJ0JoJc, ninner, gnreduc);
135 Log::test() << classname() <<
": reduction in residual norm = " << reduc << std::endl;
136 Log::info() << classname() <<
" output increment:" << *dx << std::endl;
143 costJ0Jb_ += 0.5 * dot_product(*dx, dxh);
144 for (
unsigned int jouter = 1; jouter < dxh_.size(); ++jouter) {
146 costJ0Jb_ += dot_product(*dx, dxhtmp);
149 if (config.has(
"fsoi")) {
150 Log::info() << classname() <<
" Entering Observation Sensitivity Calculation" << std::endl;
158 const std::string osensname =
"ObsSensitivity";
161 bool runFSOIincTest = config.getBool(
"fsoi.increment test",
false);
162 if (runFSOIincTest) {
165 for (
unsigned jj = 0; jj < J_.nterms(); ++jj) {
166 std::unique_ptr<GeneralizedDepartures> ww(J_.jterm(jj).newGradientFG());
167 dp.
append(J_.jterm(jj).multiplyCovar(*ww));
171 double adj_tst_fwd = dot_product(dx0, dx0);
173 double adj_tst_bwd = dot_product(ys, dp);
175 Log::info() <<
"Online FSOI increment test: " << std::endl
176 << util::PrintAdjTest(adj_tst_fwd, adj_tst_bwd,
"K") << std::endl;
178 double fsoi_inctest_tolerance = config.getDouble(
"fsoi.increment test tolerance", 1.0e-5);
179 bool passed = oops::is_close_absolute(adj_tst_fwd, adj_tst_bwd, fsoi_inctest_tolerance);
181 Log::test() <<
"FSOI increment test within tolerance." << std::endl;
183 Log::test() <<
"FSOI increment test fails tolerance bound." << std::endl;
void zero()
Linear algebra operators.
Geometry_ geometry() const
Get geometry.
void read(const eckit::Configuration &)
I/O and diagnostics.
DR (Derber and Rosati) Minimizers.
std::unique_ptr< CtrlInc_ > gradJb_
CostFunction< MODEL, OBS > CostFct_
HtRinvHMatrix< MODEL, OBS > HtRinvH_
BMatrix< MODEL, OBS > Bmat_
ControlIncrement< MODEL, OBS > CtrlInc_
CtrlInc_ * doMinimize(const eckit::Configuration &) override
const std::string classname() const override=0
Minimizer< MODEL, OBS > Minimizer_
std::vector< CtrlInc_ > dxh_
virtual double solve(CtrlInc_ &, CtrlInc_ &, CtrlInc_ &, const Bmat_ &, const HtRinvH_ &, const double, const double, const int, const double)=0
RinvHMatrix< MODEL, OBS > RinvH_
DRMinimizer(const CostFct_ &J)
Container of dual space vectors for all terms of the cost function.
void append(std::unique_ptr< GeneralizedDepartures > &&)
void saveDep(const std::string &) const
A Minimizer knows how to minimize a cost function.
void multiply(const CtrlInc_ &dx, Dual_ &zz) const
The namespace for the main oops code.