OOPS
InterpolatorAtlas.cc
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020- UCAR
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  */
7 
8 #include <ostream>
9 #include <string>
10 
11 #include "atlas/array.h"
12 #include "atlas/field.h"
13 #include "atlas/functionspace.h"
14 #include "atlas/interpolation.h"
15 #include "atlas/option.h"
16 #include "eckit/config/Configuration.h"
17 
19 #include "oops/util/abor1_cpp.h"
20 #include "oops/util/Logger.h"
21 
22 using atlas::array::make_datatype;
23 using atlas::array::make_shape;
24 using atlas::option::name;
25 using atlas::option::levels;
26 
27 namespace oops {
28 
30 
31 // -----------------------------------------------------------------------------
32 InterpolatorAtlas::InterpolatorAtlas(const eckit::Configuration & config,
33  const atlas::FunctionSpace & grid1,
34  const atlas::FunctionSpace & grid2,
35  const atlas::field::FieldSetImpl * masks) {
36  // for now, let's use the finite element interpolation, which is available
37  // for unstructured grids. We may wish to implement more options in the
38  // future. Also, atlas has not yet implemented masks so ignore that
39  // argument.
40  interpolator_.reset(new atlas::Interpolation(atlas::option::type( "finite-element" ),
41  grid1, grid2));
42 }
43 
44 // -----------------------------------------------------------------------------
45 void InterpolatorAtlas::apply(const atlas::Field & infield,
46  atlas::Field & outfield) {
47  interpolator_->execute(infield, outfield);
48 }
49 
50 // -----------------------------------------------------------------------------
51 void InterpolatorAtlas::apply(const atlas::FieldSet & infields,
52  atlas::FieldSet & outfields) {
53  // Allocate space for the output fields if the caller has not already done so
54  for (size_t ifield = 0; ifield < static_cast<size_t>(infields.size()); ++ifield) {
55  std::string fname = infields.field(ifield).name();
56  if (!outfields.has_field(fname)) {
57  oops::Log::info() << "Allocating output fields for Atlas Interpolation" << std::endl;
58 
59  atlas::Field outfield =
60  interpolator_->target().createField<double>(name(fname) |
61  levels(infields.field(ifield).levels()));
62  outfields.add(outfield);
63  }
64  }
65 
66  // apply interpolation
67  interpolator_->execute(infields, outfields);
68 }
69 
70 // -----------------------------------------------------------------------------
71 void InterpolatorAtlas::print(std::ostream & os) const {
72  os << " InterpolatorAtlas: print not implemented yet.";
73 }
74 // -----------------------------------------------------------------------------
75 } // namespace oops
InterpolatorAtlas(const eckit::Configuration &, const atlas::FunctionSpace &, const atlas::FunctionSpace &, const atlas::field::FieldSetImpl *=nullptr)
void apply(const atlas::Field &, atlas::Field &) override
std::unique_ptr< atlas::Interpolation > interpolator_
void print(std::ostream &) const override
The namespace for the main oops code.
static InterpolatorMaker< InterpolatorAtlas > makerAtlas_("atlas")