OOPS
LocationsQG.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 <memory>
12 #include <ostream>
13 #include <string>
14 #include <vector>
15 
16 #include "atlas/array.h"
17 #include "atlas/field.h"
18 #include "atlas/functionspace/PointCloud.h"
19 #include "eckit/config/Configuration.h"
20 #include "eckit/exception/Exceptions.h"
21 
22 #include "model/LocationsQG.h"
23 #include "model/QgFortran.h"
24 #include "oops/util/DateTime.h"
25 #include "oops/util/Duration.h"
26 #include "oops/util/Random.h"
27 
28 using atlas::array::make_datatype;
29 using atlas::array::make_shape;
30 using atlas::array::make_view;
31 using atlas::option::name;
32 
33 namespace qg {
34 
35 // -------------------------------------------------------------------------
36 /*! This constructor creates test locations based on settings in the config file */
37 LocationsQG::LocationsQG(const eckit::Configuration & config, const eckit::mpi::Comm &) {
38  std::vector<double> lons;
39  std::vector<double> lats;
40  std::vector<double> z;
41  unsigned int nrandom;
42 
43  if (config.has("lats") || config.has("nrandom")) {
44  /*! These are optional lists of locations that the user can
45  * specify in the config file for testing
46  */
47  lons = config.getDoubleVector("lons");
48  lats = config.getDoubleVector("lats");
49  z = config.getDoubleVector("z");
50 
51  ASSERT(lons.size() == lats.size());
52  ASSERT(lons.size() == z.size());
53 
54  /*! Instead of or in addition to the specific locations in the config file
55  * let the user specify a number of random locations
56  */
57  nrandom = config.getInt("nrandom", 0);
58 
59  } else {
60  /*! If the config file does not specify otherwise, default to 4 random locations */
61  nrandom = 4;
62  }
63 
64  if (nrandom > 0) {
65  /*! To create random locations, we need to know the dimensions of the
66  * computational domain
67  */
68  double lonmin, lonmax, latmin, latmax, zmax;
69  qg_geom_dimensions_f90(lonmin, lonmax, latmin, latmax, zmax);
70 
71  /*! Now create random locations
72  * use a specified random seed for reproducibility
73  */
74 
75  std::uint32_t rseed = 11;
76  util::UniformDistribution<double> randlons(nrandom, lonmin, lonmax, rseed);
77  util::UniformDistribution<double> randlats(nrandom, latmin, latmax);
78  util::UniformDistribution<double> randzs(nrandom, 0.0, zmax);
79  randlons.sort();
80 
81  for (std::size_t jj=0; jj < nrandom; ++jj) {
82  lons.push_back(randlons[jj]);
83  lats.push_back(randlats[jj]);
84  z.push_back(randzs[jj]);
85  }
86  }
87 
88  const unsigned int nlocs = lons.size();
89  ASSERT(nlocs > 0);
90 
91  /*! render lat, lon as an atlas functionspace */
92  atlas::Field field_lonlat("lonlat", make_datatype<double>(), make_shape(nlocs, 2));
93  auto lonlat = make_view<double, 2>(field_lonlat);
94  for ( unsigned int j = 0; j < nlocs; ++j ) {
95  lonlat(j, 0) = lons[j];
96  lonlat(j, 1) = lats[j];
97  }
98  pointcloud_.reset(new atlas::functionspace::PointCloud(field_lonlat));
99 
100  /*! render levels as an atlas field */
101  altitude_.reset(new atlas::Field("altitude", make_datatype<double>(),
102  make_shape(nlocs)));
103  auto zpc = make_view<double, 1>(*altitude_);
104  for (unsigned int j = 0; j < nlocs; ++j) {
105  zpc(j) = z[j];
106  }
107 
108  /*! Now get times */
109  if (config.has("window end")) {
110  util::DateTime winbgn(config.getString("window begin"));
111  util::DateTime winend(config.getString("window end"));
112  util::Duration window_len(winend - winbgn);
113  util::Duration dt = window_len / nlocs;
114 
115  times_.clear();
116  for (unsigned int j=0; j < nlocs; ++j) {
117  times_.push_back(winbgn + (j+1)*dt);
118  }
119  }
120 }
121 
122 // -------------------------------------------------------------------------
123 /*! Deep copy constructor */
125  pointcloud_.reset(new atlas::functionspace::PointCloud(other.lonlat()));
126  altitude_.reset(new atlas::Field(*other.altitude_));
127  times_ = other.times_;
128 }
129 // -------------------------------------------------------------------------
130 /*! Constructor from fields and times. These may be obtained from the obsdb,
131  * passed to C++ from Fortran
132  */
133 LocationsQG::LocationsQG(atlas::FieldSet & fields,
134  std::vector<util::DateTime> && times) {
135  pointcloud_.reset(new atlas::functionspace::PointCloud(fields.field("lonlat")));
136  if (fields.has_field("altitude")) {
137  altitude_.reset(new atlas::Field(fields.field("altitude")));
138  } else {
139  altitude_.reset(new atlas::Field(atlas::Field()));
140  }
141  times_ = times;
142 }
143 // -------------------------------------------------------------------------
144 void LocationsQG::print(std::ostream & os) const {
145  int nobs = pointcloud_->size();
146  atlas::Field field_lonlat = pointcloud_->lonlat();
147  auto lonlat = make_view<double, 2>(field_lonlat);
148  auto z = make_view<double, 1>(*altitude_);
149  for (size_t jj=0; jj < static_cast<size_t>(nobs); ++jj) {
150  os << "location " << jj << std::setprecision(2) << ": lon = " << lonlat(jj, 0)
151  << ", lat = " << lonlat(jj, 1) << ", z = " << z(jj) << std::endl;
152  }
153 }
154 // -------------------------------------------------------------------------
155 } // namespace qg
LocationsQG class to handle locations for QG model.
Definition: LocationsQG.h:36
std::vector< util::DateTime > times_
Definition: LocationsQG.h:57
atlas::Field lonlat() const
Definition: LocationsQG.h:49
std::unique_ptr< atlas::Field > altitude_
Definition: LocationsQG.h:56
LocationsQG(atlas::FieldSet &, std::vector< util::DateTime > &&)
Constructor from fields and times.
Definition: LocationsQG.cc:133
void print(std::ostream &) const
Definition: LocationsQG.cc:144
std::unique_ptr< atlas::functionspace::PointCloud > pointcloud_
Definition: LocationsQG.h:55
util::DateTime & times(size_t idx)
Definition: LocationsQG.h:51
integer, parameter rseed
Random seed (for reproducibility)
The namespace for the qg model.
void qg_geom_dimensions_f90(double &, double &, double &, double &, double &)