11 #ifndef OOPS_ASSIMILATION_DRIPCGMINIMIZER_H_
12 #define OOPS_ASSIMILATION_DRIPCGMINIMIZER_H_
25 #include "oops/util/dot_product.h"
26 #include "oops/util/formats.h"
27 #include "oops/util/Logger.h"
28 #include "oops/util/printRunStats.h"
82 const std::string
classname()
const override {
return "DRIPCGMinimizer";}
88 const double,
const double,
const int,
const double)
override;
95 template<
typename MODEL,
typename OBS>
102 template<
typename MODEL,
typename OBS>
105 const double costJ0Jb,
const double costJ0JoJc,
106 const int maxiter,
const double tolerance) {
107 util::printRunStats(
"DRIPCG start");
116 std::vector<CtrlInc_> vvecs;
117 std::vector<CtrlInc_> zvecs;
118 std::vector<double> scals;
120 vvecs.reserve(maxiter+1);
121 zvecs.reserve(maxiter+1);
122 scals.reserve(maxiter+1);
124 const double costJ0 = costJ0Jb + costJ0JoJc;
128 lmp_.multiply(rr, sh);
131 double rrnorm0 = sqrt(dot_product(rr, rr));
132 double dotSr0 = dot_product(rr, ss);
133 double normReduction = 1.0;
134 double rdots = dotSr0;
135 double rdots_old = dotSr0;
139 scals.push_back(1.0/dotSr0);
141 Log::info() << std::endl;
142 for (
int jiter = 0; jiter < maxiter; ++jiter) {
143 Log::info() <<
" DRIPCG Starting Iteration " << jiter+1 << std::endl;
144 util::printRunStats(
"DRIPCG iteration " + std::to_string(jiter+1));
151 double beta = -dot_product(ss, dr)/rdots_old;
165 double rho = dot_product(pp, ap);
166 double alpha = rdots/rho;
167 Log::info() <<
"DRIPCGMinimizer rho = " << rho <<
", alpha = " << alpha << std::endl;
174 double costJ = costJ0 - 0.5 * dot_product(xx, r0);
175 double costJb = costJ0Jb + 0.5 * dot_product(xx, xh);
176 double costJoJc = costJ - costJb;
179 for (
int jj = 0; jj < jiter; ++jj) {
180 double proj = scals[jj] * dot_product(rr, zvecs[jj]);
181 rr.
axpy(-proj, vvecs[jj]);
184 lmp_.multiply(rr, sh);
188 rdots = dot_product(rr, ss);
195 double rrnorm = sqrt(dot_product(rr, rr));
196 normReduction = rrnorm/rrnorm0;
198 Log::info() <<
"DRIPCG end of iteration " << jiter+1 << std::endl;
203 lmp_.push(pp, ph, ap, rho);
205 if (normReduction < tolerance) {
206 Log::info() <<
"DRIPCG: Achieved required reduction in residual norm." << std::endl;
212 scals.push_back(1.0/rdots);
218 util::printRunStats(
"DRIPCG end");
219 return normReduction;
void multiply(const CtrlInc_ &x, CtrlInc_ &bx) const
void axpy(const double, const ControlIncrement &)
Derber-Rosati IPCG Minimizer.
DRIPCGMinimizer(const eckit::Configuration &, const CostFct_ &)
BMatrix< MODEL, OBS > Bmat_
HtRinvHMatrix< MODEL, OBS > HtRinvH_
CostFunction< MODEL, OBS > CostFct_
const std::string classname() const override
ControlIncrement< MODEL, OBS > CtrlInc_
double solve(CtrlInc_ &, CtrlInc_ &, CtrlInc_ &, const Bmat_ &, const HtRinvH_ &, const double, const double, const int, const double) override
QNewtonLMP< CtrlInc_, Bmat_ > lmp_
DR (Derber and Rosati) Minimizers.
void multiply(const CtrlInc_ &dx, CtrlInc_ &dz) const
The namespace for the main oops code.
void printQuadraticCostFunction(int iteration, const double &costJ, const double &costJb, const double &costJoJc)
void printNormReduction(int iteration, const double &grad, const double &norm)