OOPS
DolphChebyshev.cc
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 
12 
13 #include <cmath>
14 #include <vector>
15 
16 #include "eckit/config/Configuration.h"
17 #include "eckit/exception/Exceptions.h"
18 #include "oops/util/DateTime.h"
19 #include "oops/util/Duration.h"
20 
21 namespace oops {
22 
23 // -----------------------------------------------------------------------------
24 
25 DolphChebyshev::DolphChebyshev(const eckit::Configuration & config) {
26  tau_ = util::Duration(config.getString("cutoff"));
27 }
28 
29 // -----------------------------------------------------------------------------
30 
31 std::map< util::DateTime, double > DolphChebyshev::setWeights(const util::DateTime & bgn,
32  const util::DateTime & end,
33  const util::Duration & dt) {
34  const double pi = 4.0*std::atan(1.0);
35  const util::Duration window(end-bgn);
36  const int nstep = window.toSeconds() / dt.toSeconds();
37  ASSERT(window.toSeconds() == dt.toSeconds()*nstep);
38 
39  const int M = nstep/2;
40  const int N = 2*M + 1;
41 
42  const int tt = tau_.toSeconds() / dt.toSeconds();
43  ASSERT(tau_.toSeconds() == dt.toSeconds()*tt);
44  ASSERT(tt > 1);
45  const double thetas = 2.0 * pi / tt;
46  const double x0 = 1.0 / std::cos(thetas/2.0);
47  const double rr = 1.0 / std::cosh(nstep * std::acosh(x0));
48 
49  std::vector<double> w(M+1);
50  for (int n = 0; n <= M; ++n) {
51  double tn = 2.0 * pi * n / N;
52  double sum = 0.0;
53  for (int m = 1; m <= M; ++m) {
54  double xx = x0 * std::cos(pi * m / N);
55  double t2m = 0.0;
56  double tnm2 = 1.0;
57  double tnm1 = xx;
58  for (int kk = 2; kk <= 2 * M; ++kk) {
59  t2m = 2.0 * xx * tnm1 - tnm2;
60  tnm2 = tnm1;
61  tnm1 = t2m;
62  }
63  sum += t2m * std::cos(m*tn);
64  }
65  w[n] = (1.0 + 2.0 * rr * sum) / N;
66  }
67 
68  std::map< util::DateTime, double > weights;
69  double checksum = 0.0;
70  util::DateTime now(bgn);
71  for (int jj = -M; jj <= M; ++jj) {
72  int n = std::abs(jj);
73  weights[now] = w[n];
74  checksum += w[n];
75  now += dt;
76  }
77  ASSERT(now == end+dt);
78  ASSERT(std::abs(checksum-1.0) < 1.0e-8);
79 
80  return weights;
81 }
82 
83 // -----------------------------------------------------------------------------
84 
85 } // namespace oops
86 
std::map< util::DateTime, double > setWeights(const util::DateTime &, const util::DateTime &, const util::Duration &)
DolphChebyshev(const eckit::Configuration &)
util::Duration tau_
The namespace for the main oops code.
real(kind_real), parameter, public pi
Pi.