OOPS
TLML95.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 
11 #include "lorenz95/TLML95.h"
12 
13 #include "eckit/config/LocalConfiguration.h"
14 #include "eckit/exception/Exceptions.h"
15 
16 #include "oops/util/abor1_cpp.h"
17 #include "oops/util/DateTime.h"
18 #include "oops/util/Duration.h"
19 #include "oops/util/Logger.h"
20 
21 #include "lorenz95/FieldL95.h"
22 #include "lorenz95/IncrementL95.h"
23 #include "lorenz95/L95Traits.h"
24 #include "lorenz95/ModelBias.h"
26 #include "lorenz95/ModelL95.h"
28 #include "lorenz95/Resolution.h"
29 #include "lorenz95/StateL95.h"
30 
31 
32 namespace lorenz95 {
33 // -----------------------------------------------------------------------------
35 // -----------------------------------------------------------------------------
36 TLML95::TLML95(const Resolution & resol, const Parameters_ & params)
37  : resol_(resol), tstep_(params.tstep),
38  dt_(tstep_.toSeconds()/432000.0), traj_(),
39  lrmodel_(resol_, params.trajectory),
40  vars_()
41 {
42  oops::Log::info() << "TLML95: resol = " << resol_ << ", tstep = " << tstep_ << std::endl;
43  oops::Log::trace() << "TLML95::TLML95 created" << std::endl;
44 }
45 // -----------------------------------------------------------------------------
47  for (trajIter jtra = traj_.begin(); jtra != traj_.end(); ++jtra) {
48  delete jtra->second;
49  }
50  oops::Log::trace() << "TLML95::~TLML95 destructed" << std::endl;
51 }
52 // -----------------------------------------------------------------------------
53 void TLML95::setTrajectory(const StateL95 & xx, StateL95 &, const ModelBias & bias) {
54  ASSERT(traj_.find(xx.validTime()) == traj_.end());
55  ModelTrajectory * traj = new ModelTrajectory();
56 // Interpolate xx to xlr here
57  FieldL95 zz(xx.getField());
58  lrmodel_.stepRK(zz, bias, *traj);
59  traj_[xx.validTime()] = traj;
60 }
61 // -----------------------------------------------------------------------------
62 const ModelTrajectory * TLML95::getTrajectory(const util::DateTime & tt) const {
63  trajICst itra = traj_.find(tt);
64  if (itra == traj_.end()) {
65  oops::Log::error() << "TLML95: trajectory not available at time " << tt << std::endl;
66  ABORT("TLML95: trajectory not available");
67  }
68  return itra->second;
69 }
70 // -----------------------------------------------------------------------------
71 /// Run TLM and its adjoint
72 // -----------------------------------------------------------------------------
77 // -----------------------------------------------------------------------------
78 void TLML95::stepTL(IncrementL95 & xx, const ModelBiasCorrection & bias) const {
79  FieldL95 dx(xx.getField(), false);
80  FieldL95 zz(xx.getField(), false);
81  FieldL95 dz(xx.getField(), false);
82  const ModelTrajectory * traj = this->getTrajectory(xx.validTime());
83 
84  zz = xx.getField();
85  this->tendenciesTL(zz, bias.bias(), traj->get(1), dz);
86  dx = dz;
87 
88  zz = xx.getField();
89  zz.axpy(0.5, dz);
90  this->tendenciesTL(zz, bias.bias(), traj->get(2), dz);
91  dx.axpy(2.0, dz);
92 
93  zz = xx.getField();
94  zz.axpy(0.5, dz);
95  this->tendenciesTL(zz, bias.bias(), traj->get(3), dz);
96  dx.axpy(2.0, dz);
97 
98  zz = xx.getField();
99  zz += dz;
100  this->tendenciesTL(zz, bias.bias(), traj->get(4), dz);
101  dx += dz;
102 
103  const double zt = 1.0/6.0;
104  xx.getField().axpy(zt, dx);
105  xx.validTime() += tstep_;
106 }
107 // -----------------------------------------------------------------------------
109  FieldL95 dx(xx.getField(), false);
110  FieldL95 zz(xx.getField(), false);
111  FieldL95 dz(xx.getField(), false);
112 
113  xx.validTime() -= tstep_;
114  const ModelTrajectory * traj = this->getTrajectory(xx.validTime());
115 
116  const double zt = 1.0/6.0;
117  dx = xx.getField();
118  dx *= zt;
119 
120  dz = dx;
121  this->tendenciesAD(zz, bias.bias(), traj->get(4), dz);
122  xx.getField() += zz;
123  dz = zz;
124 
125  dz.axpy(2.0, dx);
126  this->tendenciesAD(zz, bias.bias(), traj->get(3), dz);
127  xx.getField() += zz;
128  dz = zz;
129  dz *= 0.5;
130 
131  dz.axpy(2.0, dx);
132  this->tendenciesAD(zz, bias.bias(), traj->get(2), dz);
133  xx.getField() += zz;
134  dz = zz;
135  dz *= 0.5;
136 
137  dz += dx;
138  this->tendenciesAD(zz, bias.bias(), traj->get(1), dz);
139  xx.getField() += zz;
140 }
141 // -----------------------------------------------------------------------------
142 // intel 19 tries to aggressive optimize these functions in a way that leads
143 // to memory violations. So, turn off optimizations.
144 #ifdef __INTEL_COMPILER
145 #pragma optimize("", off)
146 #endif
147 void TLML95::tendenciesTL(const FieldL95 & xx, const double & bias,
148  const FieldL95 & xtraj, FieldL95 & dx) const {
149  const int nn = resol_.npoints();
150  for (int jj = 0; jj < nn; ++jj) {
151  int jm2 = jj - 2;
152  int jm1 = jj - 1;
153  int jp1 = jj + 1;
154  if (jm2 < 0) jm2 += nn;
155  if (jm1 < 0) jm1 += nn;
156  if (jp1 >= nn) jp1 -= nn;
157  const double dxdt = - xx[jm2] * xtraj[jm1] - xtraj[jm2] * xx[jm1]
158  + xx[jm1] * xtraj[jp1] + xtraj[jm1] * xx[jp1]
159  - xx[jj] - bias;
160  dx[jj] = dt_ * dxdt;
161  }
162 }
163 // -----------------------------------------------------------------------------
164 void TLML95::tendenciesAD(FieldL95 & xx, double & bias,
165  const FieldL95 & xtraj, const FieldL95 & dx) const {
166  const int nn = resol_.npoints();
167  xx.zero();
168  for (int jj = 0; jj < nn; ++jj) {
169  int jm2 = jj - 2;
170  int jm1 = jj - 1;
171  int jp1 = jj + 1;
172  if (jm2 < 0) jm2 += nn;
173  if (jm1 < 0) jm1 += nn;
174  if (jp1 >= nn) jp1 -= nn;
175  const double dxdt = dt_ * dx[jj];
176  xx[jm2] -= dxdt * xtraj[jm1];
177  xx[jm1] -= dxdt * xtraj[jm2];
178  xx[jm1] += dxdt * xtraj[jp1];
179  xx[jp1] += dxdt * xtraj[jm1];
180  xx[jj] -= dxdt;
181  bias -= dxdt;
182  }
183 }
184 #ifdef __INTEL_COMPILER
185 #pragma optimize("", on)
186 #endif
187 // -----------------------------------------------------------------------------
188 void TLML95::print(std::ostream & os) const {
189  os << "TLML95: resol = " << resol_ << ", tstep = " << tstep_ << std::endl;
190  os << "L95 Model Trajectory, nstep=" << traj_.size() << std::endl;
191  typedef std::map< util::DateTime, ModelTrajectory * >::const_iterator trajICst;
192  if (traj_.size() > 0) {
193  os << "L95 Model Trajectory: times are:";
194  for (trajICst jtra = traj_.begin(); jtra != traj_.end(); ++jtra) {
195  os << " " << jtra->first;
196  }
197  }
198 }
199 // -----------------------------------------------------------------------------
200 } // namespace lorenz95
Class to represent fields for the L95 model.
Definition: FieldL95.h:34
void axpy(const double &, const FieldL95 &)
Definition: FieldL95.cc:130
void zero()
Linear algebra.
Definition: FieldL95.cc:57
Increment Class: Difference between two states.
Definition: IncrementL95.h:58
const util::DateTime & validTime() const
Definition: IncrementL95.h:92
const FieldL95 & getField() const
Access to data.
Definition: IncrementL95.h:100
Model error for Lorenz 95 model.
void stepRK(FieldL95 &, const ModelBias &, ModelTrajectory &) const
Definition: ModelL95.cc:56
L95 model trajectory.
const FieldL95 & get(const int) const
Get trajectory.
Handles resolution.
Definition: Resolution.h:43
int npoints() const
Definition: Resolution.h:53
L95 model state.
Definition: StateL95.h:53
const FieldL95 & getField() const
Definition: StateL95.h:69
const util::DateTime & validTime() const
Definition: StateL95.h:79
const ModelTrajectory * getTrajectory(const util::DateTime &) const
Definition: TLML95.cc:62
void setTrajectory(const StateL95 &, StateL95 &, const ModelBias &) override
Model trajectory computation.
Definition: TLML95.cc:53
std::map< util::DateTime, ModelTrajectory * > traj_
Definition: TLML95.h:97
const Resolution resol_
Definition: TLML95.h:94
void stepTL(IncrementL95 &, const ModelBiasCorrection &) const override
Definition: TLML95.cc:78
void print(std::ostream &) const override
Print, used in logging.
Definition: TLML95.cc:188
void initializeAD(IncrementL95 &) const override
Definition: TLML95.cc:75
void finalizeAD(IncrementL95 &) const override
Definition: TLML95.cc:76
void stepAD(IncrementL95 &, ModelBiasCorrection &) const override
Definition: TLML95.cc:108
void initializeTL(IncrementL95 &) const override
Run TLM and its adjoint.
Definition: TLML95.cc:73
void tendenciesTL(const FieldL95 &, const double &, const FieldL95 &, FieldL95 &) const
Definition: TLML95.cc:147
void tendenciesAD(FieldL95 &, double &, const FieldL95 &, const FieldL95 &) const
Definition: TLML95.cc:164
const util::Duration tstep_
Definition: TLML95.h:95
std::map< util::DateTime, ModelTrajectory * >::const_iterator trajICst
Definition: TLML95.h:91
const ModelL95 lrmodel_
Definition: TLML95.h:98
TLML95(const Resolution &, const Parameters_ &)
Definition: TLML95.cc:36
std::map< util::DateTime, ModelTrajectory * >::iterator trajIter
Definition: TLML95.h:90
void finalizeTL(IncrementL95 &) const override
Definition: TLML95.cc:74
const double dt_
Definition: TLML95.h:96
A subclass of LinearModelFactory able to create instances of T (a concrete subclass of interface::Lin...
int error
Definition: compare.py:168
The namespace for the L95 model.
static oops::interface::LinearModelMaker< L95Traits, TLML95 > makerTLML95_("L95TLM")