11 #ifndef OOPS_ASSIMILATION_IPCG_H_
12 #define OOPS_ASSIMILATION_IPCG_H_
18 #include "oops/util/dot_product.h"
19 #include "oops/util/formats.h"
20 #include "oops/util/Logger.h"
62 template <
typename VECTOR,
typename AMATRIX,
typename PMATRIX>
63 double IPCG(VECTOR & x,
const VECTOR & b,
64 const AMATRIX & A,
const PMATRIX & precond,
65 const int maxiter,
const double tolerance ) {
75 std::vector<VECTOR> vVEC;
76 std::vector<VECTOR> zVEC;
78 vVEC.reserve(maxiter+1);
79 zVEC.reserve(maxiter+1);
83 double xnrm2 = dot_product(x, x);
90 precond.multiply(r, s);
92 double rnorm0 = sqrt(dot_product(r, r));
93 double dotSr0 = dot_product(r, s);
94 double normReduction = 1.0;
95 double rdots_old = dotSr0;
96 double rdots = dotSr0;
106 Log::info() << std::endl;
107 for (
int jiter = 0; jiter < maxiter; ++jiter) {
108 Log::info() <<
" IPCG Starting Iteration " << jiter+1 << std::endl;
115 double beta = dot_product(s, w)/rdots_old;
124 double alpha = rdots/dot_product(p, ap);
130 for (
int iiter = 0; iiter < jiter; ++iiter) {
131 double proj = dot_product(r, zVEC[iiter]);
132 r.axpy(-proj, vVEC[iiter]);
135 precond.multiply(r, s);
138 rdots = dot_product(r, s);
147 double rnorm = sqrt(dot_product(r, r));
148 normReduction = rnorm/rnorm0;
149 Log::info() <<
"IPCG end of iteration " << jiter+1 << std::endl;
152 if (normReduction < tolerance) {
153 Log::info() <<
"IPCG: Achieved required reduction in residual norm." << std::endl;
158 Log::info() <<
"IPCG: end" << std::endl;
160 return normReduction;
The namespace for the main oops code.
double IPCG(VECTOR &x, const VECTOR &b, const AMATRIX &A, const PMATRIX &precond, const int maxiter, const double tolerance)
void printNormReduction(int iteration, const double &grad, const double &norm)