OOPS
LocalizationMatrixL95.cc
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
3  * (C) Copyright 2020-2020 UCAR
4  *
5  * This software is licensed under the terms of the Apache Licence Version 2.0
6  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7  * In applying this licence, ECMWF does not waive the privileges and immunities
8  * granted to it by virtue of its status as an intergovernmental organisation nor
9  * does it submit to any jurisdiction.
10  */
11 
13 
14 #include <unsupported/Eigen/FFT>
15 #include <algorithm>
16 #include <cmath>
17 
18 #include "eckit/config/Configuration.h"
19 #include "eckit/exception/Exceptions.h"
20 
21 #include "lorenz95/IncrementL95.h"
22 #include "lorenz95/Resolution.h"
23 
24 // -----------------------------------------------------------------------------
25 namespace lorenz95 {
26 // -----------------------------------------------------------------------------
28  const eckit::Configuration & config)
29  : resol_(resol.npoints()),
30  rscale_(1.0/config.getDouble("length_scale"))
31 {
32 // Gaussian structure function
33  unsigned int size = resol_/2+1;
34  std::vector<double> locfct(resol_);
35  for (unsigned int jj = 0; jj < resol_; ++jj) {
36  double zz = rscale_ * std::min(jj, resol_-jj);
37  locfct[jj] = std::exp(-0.5*zz*zz);
38  }
39 // Go to Fourier space
40  Eigen::FFT<double> fft;
41  std::vector<std::complex<double> > four(size);
42  fft.fwd(four, locfct);
43 // Save Fourier coefficients
44  coefs_.resize(size);
45  for (unsigned int jj = 0; jj < size; ++jj) {
46  coefs_[jj] = std::real(four[jj]);
47  }
48 }
49 // -----------------------------------------------------------------------------
51  dx.random();
52  unsigned int size = resol_/2+1;
53  Eigen::FFT<double> fft;
54  std::vector<std::complex<double> > four(size);
55  fft.fwd(four, dx.asVector());
56  for (unsigned int jj = 0; jj < size; ++jj) {
57  four[jj] *= std::sqrt(coefs_[jj]);
58  }
59  fft.inv(dx.asVector(), four);
60 }
61 // -----------------------------------------------------------------------------
63  unsigned int size = resol_/2+1;
64  Eigen::FFT<double> fft;
65  std::vector<std::complex<double> > four(size);
66  fft.fwd(four, dx.asVector());
67  for (unsigned int jj = 0; jj < size; ++jj) {
68  four[jj] *= coefs_[jj];
69  }
70  fft.inv(dx.asVector(), four);
71 }
72 // -----------------------------------------------------------------------------
73 void LocalizationMatrixL95::print(std::ostream & os) const {
74  os << "Localization with Gaussian, lengthscale = " << 1.0/rscale_;
75 }
76 // -----------------------------------------------------------------------------
77 
78 } // namespace lorenz95
Increment Class: Difference between two states.
Definition: IncrementL95.h:58
std::vector< double > & asVector()
Definition: IncrementL95.h:106
LocalizationMatrixL95(const Resolution &, const eckit::Configuration &)
void print(std::ostream &) const override
void multiply(IncrementL95 &) const override
void randomize(IncrementL95 &) const override
Handles resolution.
Definition: Resolution.h:43
The namespace for the L95 model.