IODA
DimensionScales.h
Go to the documentation of this file.
1 #pragma once
2 /*
3  * (C) Copyright 2020-2021 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  */
8 /// \file DimensionScales.h
9 /// \brief Convenience classes for constructing ObsSpaces and setting up new Dimension Scales.
10 #include <memory>
11 #include <string>
12 #include <typeindex>
13 #include <vector>
14 
15 #include "../defs.h"
16 #include "Dimensions.h"
17 #include "../Types/Type.h"
18 #include "../Variables/Variable.h"
19 
20 namespace ioda {
21 class Type;
22 class Variable;
23 
24 /// Specifies that a dimension is resizable to infinity.
25 constexpr int Unlimited = -1;
26 /// Specifies that a dimension has no specified size. Tells ioda to figure it out from elsewhere.
27 constexpr int Unspecified = -2;
28 
29 struct ScaleSizes {
30  /// Initial size of the new dimension.
31  Dimensions_t size_ = Unspecified;
32  /// Maximum size of the new dimension.
33  Dimensions_t maxSize_ = Unspecified;
34  /// Chunking size of the new dimension.
35  Dimensions_t chunkingSize_ = Unspecified;
36 
37  ScaleSizes(Dimensions_t size = Unspecified, Dimensions_t maxSize = Unspecified,
38  Dimensions_t chunkingSize = Unspecified)
39  : size_(size), maxSize_(maxSize), chunkingSize_(chunkingSize) {}
40 };
41 
42 /// \brief Used to specify a new dimension scale when making an ObsSpace.
43 /// \note Instantiation stored in ObsSpace.cpp.
44 /// \note shared_from_this for pybind11. Can't pass in a list
45 /// of unique_ptrs.
46 struct IODA_DL NewDimensionScale_Base : std::enable_shared_from_this<NewDimensionScale_Base> {
47  /// Name of the dimension. Scan position, scan line, latitude, ...
48  std::string name_;
49  /// Type of the new dimension. Int, char, etc. Used if a type is not passed directly.
50  std::type_index dataType_;
51  /// Type of the new dimension. Used if a type is passed directly.
53  /// Initial size of the new dimension.
54  Dimensions_t size_;
55  /// Maximum size of the new dimension. Unlimited (< 0) by default.
56  Dimensions_t maxSize_;
57  /// \brief Chunking size of the new dimension. May be used as a
58  /// hint when creating new Variables based on this dimension.
59  /// \details Matches size by default, but will throw an error if the size is zero.
60  Dimensions_t chunkingSize_;
61 
62  /// \note Not pure virtual to avoid pybind11 headaches with adding a trampoline class.
63  /// The base class should never be invoked directly.
64  /// \see https://pybind11.readthedocs.io/en/stable/advanced/classes.html
65  virtual void writeInitialData(Variable&) const {}
66 
68 
69  /// \note This should not be used directly. Keeping it public because of cross-language bindings.
70  /// \see NewDimensionScale
71  NewDimensionScale_Base(const std::string& name, const std::type_index& dataType,
72  Dimensions_t size, Dimensions_t maxSize,
73  Dimensions_t chunkingSize)
74  : name_(name),
75  dataType_(dataType),
76  size_(size),
77  maxSize_(maxSize),
78  chunkingSize_(chunkingSize) {}
79 
80  NewDimensionScale_Base(const std::string& name, const Type& dataType,
81  Dimensions_t size, Dimensions_t maxSize, Dimensions_t chunkingSize)
82  : name_(name),
83  dataTypeKnown_(dataType),
84  dataType_(typeid(void)),
85  size_(size),
86  maxSize_(maxSize),
87  chunkingSize_(chunkingSize) {}
88 };
89 typedef std::vector<std::shared_ptr<NewDimensionScale_Base>> NewDimensionScales_t;
90 
91 /// \brief Used to specify a new dimension scale when making an ObsSpace.
92 /// Templated version of NewDimensionScale_Base.
93 template <class DataType>
96 
97  std::vector<DataType> initdata_;
98 
99  NewDimensionScale_Object(const std::string& name, Dimensions_t size,
100  Dimensions_t maxSize, Dimensions_t chunkingSize)
101  : NewDimensionScale_Base(name, typeid(DataType), size, maxSize, chunkingSize),
102  initdata_(gsl::narrow<size_t>(size)) {
103  for (size_t i = 0; i < initdata_.size(); ++i) initdata_[i] = gsl::narrow<DataType>(i + 1);
104  }
105 
106  void writeInitialData(Variable& v) const override { v.write<DataType>(initdata_); }
107 
108  std::shared_ptr<NewDimensionScale_Object<DataType>> getShared() const {
109  return std::make_shared<NewDimensionScale_Object<DataType>>(*this);
110  }
111 };
112 
113 /// @brief Wrapper function used when listing new dimension scales to construct.
114 /// @tparam DataType is the type of data used in the scale.
115 /// @param name is the new scale's name.
116 /// @param size is the initial size (in elements).
117 /// @param maxSize is the maximum size, in elements.
118 /// ioda::Unspecified sets the max dimension size to the initial size.
119 /// ioda::Unlimited specifies an unlimited dimension, which should be seldom used.
120 /// @param chunkingSize is a "hint" parameter that guides how data are grouped in memory. New
121 /// variables using this scale take this hint to specify their chunk sizes.
122 /// @return A shared_ptr to a NewDimensionScale_Object, which can be inserted into a
123 /// NewDimensionScales_t container.
124 template <class DataType>
125 inline std::shared_ptr<NewDimensionScale_Object<DataType>> NewDimensionScale(
126  const std::string& name, Dimensions_t size, Dimensions_t maxSize = Unspecified,
127  Dimensions_t chunkingSize = Unspecified) {
128  if (maxSize == Unspecified) maxSize = size;
129  if (chunkingSize == Unspecified) chunkingSize = maxSize;
130  return std::make_shared<NewDimensionScale_Object<DataType>>(name, size, maxSize, chunkingSize);
131 }
132 
133 template <class DataType>
134 inline std::shared_ptr<NewDimensionScale_Object<DataType>> NewDimensionScale(
135  const std::string& name, ScaleSizes sizes) {
136  return NewDimensionScale<DataType>(name, sizes.size_, sizes.maxSize_, sizes.chunkingSize_);
137 }
138 
139 IODA_DL std::shared_ptr<NewDimensionScale_Base> NewDimensionScale(
140  const std::string& name, const Type& t, Dimensions_t size, Dimensions_t maxSize = Unspecified,
141  Dimensions_t chunkingSize = Unspecified);
142 
143 IODA_DL std::shared_ptr<NewDimensionScale_Base> NewDimensionScale(
144  const std::string& name, const Variable& scale, const ScaleSizes &overrides = ScaleSizes());
145 
146 } // namespace ioda
Describe the dimensions of a ioda::Attribute or ioda::Variable.
Represents the "type" (i.e. integer, string, float) of a piece of data.
Definition: Type.h:123
Variables store data!
Definition: Variable.h:680
virtual Variable write(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all)
The fundamental write function. Backends overload this function to implement all write operations.
Definition: Variable.cpp:317
#define IODA_DL
A preprocessor tag that indicates that a symbol is to be exported/imported.
Definition: defs.h:110
constexpr int Unlimited
Specifies that a dimension is resizable to infinity.
std::vector< std::shared_ptr< NewDimensionScale_Base > > NewDimensionScales_t
std::shared_ptr< NewDimensionScale_Object< DataType > > NewDimensionScale(const std::string &name, Dimensions_t size, Dimensions_t maxSize=Unspecified, Dimensions_t chunkingSize=Unspecified)
Wrapper function used when listing new dimension scales to construct.
constexpr int Unspecified
Specifies that a dimension has no specified size. Tells ioda to figure it out from elsewhere.
Used to specify a new dimension scale when making an ObsSpace.
Dimensions_t size_
Initial size of the new dimension.
Type dataTypeKnown_
Type of the new dimension. Used if a type is passed directly.
NewDimensionScale_Base(const std::string &name, const std::type_index &dataType, Dimensions_t size, Dimensions_t maxSize, Dimensions_t chunkingSize)
Dimensions_t chunkingSize_
Chunking size of the new dimension. May be used as a hint when creating new Variables based on this d...
Dimensions_t maxSize_
Maximum size of the new dimension. Unlimited (< 0) by default.
NewDimensionScale_Base(const std::string &name, const Type &dataType, Dimensions_t size, Dimensions_t maxSize, Dimensions_t chunkingSize)
std::string name_
Name of the dimension. Scan position, scan line, latitude, ...
virtual void writeInitialData(Variable &) const
std::type_index dataType_
Type of the new dimension. Int, char, etc. Used if a type is not passed directly.
Used to specify a new dimension scale when making an ObsSpace. Templated version of NewDimensionScale...
void writeInitialData(Variable &v) const override
std::vector< DataType > initdata_
NewDimensionScale_Object(const std::string &name, Dimensions_t size, Dimensions_t maxSize, Dimensions_t chunkingSize)
std::shared_ptr< NewDimensionScale_Object< DataType > > getShared() const
ScaleSizes(Dimensions_t size=Unspecified, Dimensions_t maxSize=Unspecified, Dimensions_t chunkingSize=Unspecified)
Dimensions_t size_
Initial size of the new dimension.
Dimensions_t chunkingSize_
Chunking size of the new dimension.
Dimensions_t maxSize_
Maximum size of the new dimension.