IODA
Type.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 /*! \defgroup ioda_cxx_types Type System
9  * \brief The data type system
10  * \ingroup ioda_cxx_api
11  *
12  * @{
13  * \file Type.h
14  * \brief Interfaces for ioda::Type and related classes. Implements the type system.
15  */
16 #include <array>
17 #include <cstring>
18 #include <functional>
19 #include <gsl/gsl-lite.hpp>
20 #include <memory>
21 #include <string>
22 #include <typeindex>
23 #include <typeinfo>
24 #include <vector>
25 
27 #include "ioda/Exception.h"
28 #include "ioda/defs.h"
29 
30 namespace ioda {
31 class Type;
32 
33 /// Basic pre-defined types (Python convenience wrappers)
34 /// \see py_ioda.cpp
35 /// \note Names here do not match the python equivalents. The
36 /// Python names match numpy's definitions.
37 enum class BasicTypes {
38  undefined_, ///< Internal use only
39  float_,
40  double_,
41  ldouble_,
42  char_,
43  short_,
44  ushort_,
45  int_,
46  uint_,
47  lint_,
48  ulint_,
49  llint_,
50  ullint_,
51  int32_,
52  uint32_,
53  int16_,
54  uint16_,
55  int64_,
56  uint64_,
57  bool_,
58  str_
59 };
60 
61 namespace detail {
62 IODA_DL size_t COMPAT_strncpy_s(char* dest, size_t destSz, const char* src, size_t srcSz);
63 
64 class Type_Backend;
65 
66 template <class Type_Implementation = Type>
67 class Type_Base {
68  friend class ::ioda::Type;
69  std::shared_ptr<Type_Backend> backend_;
70 
71 protected:
73 
74  /// @name General Functions
75  /// @{
76 
77  Type_Base(std::shared_ptr<Type_Backend> b, ::ioda::detail::Type_Provider* p)
78  : backend_(b), provider_(p) {}
79 
80  /// Get the type provider.
81  inline detail::Type_Provider* getTypeProvider() const { return provider_; }
82 
83  /*
84  /// \brief Convenience function to check a type.
85  /// \param DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
86  /// \returns True if the type matches
87  /// \returns False (0) if the type does not match
88  /// \throws if an error occurred.
89  template <class DataType>
90  bool isA() const {
91  Type templateType = Types::GetType_Wrapper<DataType>::GetType(getTypeProvider());
92 
93  return isA(templateType);
94  }
95  /// Hand-off to the backend to check equivalence
96  virtual bool isA(Type lhs) const;
97 
98  /// Python compatability function
99  inline bool isA(BasicTypes dataType) const { return isA(Type(dataType, getTypeProvider())); }
100  */
101 public:
102  virtual ~Type_Base() {}
103  std::shared_ptr<Type_Backend> getBackend() const { return backend_; }
104  bool isValid() const { return (backend_.use_count() > 0); }
105 
106  /// \brief Get the size of a single element of a type, in bytes.
107  /// \details This function is paired with the read and write functions to allow you to
108  /// read and write data in a type-agnostic manner.
109  /// This size report is a bit complicated when variable-length strings are encountered.
110  /// In these cases, the size of the string pointer is returned.
111  virtual size_t getSize() const;
112 
113  /// @}
114 };
115 } // namespace detail
116 
117 /// \brief Represents the "type" (i.e. integer, string, float) of a piece of data.
118 /// \ingroup ioda_cxx_types
119 ///
120 /// Generally, you do not have to use this class directly. Attributes and Variables have
121 /// templated functions that convert your type into the type used internally by ioda.
122 /// \see Types::GetType and Types::GetType_Wrapper for functions that produce these types.
123 class IODA_DL Type : public detail::Type_Base<> {
124 public:
125  Type();
126  Type(std::shared_ptr<detail::Type_Backend> b, std::type_index t);
127  Type(BasicTypes, gsl::not_null<::ioda::detail::Type_Provider*> t);
128 
129  virtual ~Type();
130 
131  /// @name Type-querying functions
132  /// @{
133 
134  /// @deprecated This function is problematic since we cannot query a type properly
135  /// when loading from a file.
136  std::type_index getType() const { return as_type_index_; }
137  inline std::type_index operator()() const { return getType(); }
138  inline std::type_index get() const { return getType(); }
139 
140  /// @}
141 private:
142  std::type_index as_type_index_;
143 };
144 
145 namespace detail {
146 
147 /// Backends inherit from this and they provide their own functions.
148 /// Lots of std::dynamic_cast, unfortunately.
149 class IODA_DL Type_Backend : public Type_Base<> {
150 public:
151  virtual ~Type_Backend();
152  size_t getSize() const override;
153 
154 protected:
155  Type_Backend();
156 };
157 
158 } // namespace detail
159 
160 /// \brief Defines the type system used for manipulating IODA objects.
161 namespace Types {
162 
163 // using namespace ioda::Handles;
164 
165 /// \brief Convenience struct to determine if a type can represent a string.
166 /// \ingroup ioda_cxx_types
167 /// \todo extend to UTF-8 strings, as HDF5 supports these. No support for UTF-16, but conversion
168 /// functions may be applied.
169 /// \todo Fix for "const std::string".
170 template <typename T>
171 struct is_string : public std::integral_constant<
172  bool, std::is_same<char*, typename std::decay<T>::type>::value
173  || std::is_same<const char*, typename std::decay<T>::type>::value> {};
174 /// \brief Convenience struct to determine if a type can represent a string.
175 /// \ingroup ioda_cxx_types
176 template <>
177 struct is_string<std::string> : std::true_type {};
178 
179 /// Useful compile-time definitions.
180 namespace constants {
181 /// \note Different than ObsSpace variable-length dimension. This is for a Type.
182 constexpr size_t _Variable_Length = 0;
183 // constexpr int _Not_An_Array_type = -3;
184 } // namespace constants
185 
186 /// \brief For fundamental, non-string types.
187 /// \ingroup ioda_cxx_types
188 template <class DataType, int Array_Type_Dimensionality = 0>
189 Type GetType(gsl::not_null<const ::ioda::detail::Type_Provider*> t,
190  std::initializer_list<Dimensions_t> Adims = {},
191  typename std::enable_if<!is_string<DataType>::value>::type* = 0) {
192  if (Array_Type_Dimensionality <= 0)
193  throw Exception(
194  "Bad assertion / unsupported fundamental type at the frontend side "
195  "of the ioda type system.", ioda_Here());
196  else
197  return t->makeArrayType(Adims, typeid(DataType[]), typeid(DataType));
198 }
199 /// \brief For fundamental string types. These are either constant or variable length arrays.
200 /// Separate handling elsewhere.
201 /// \ingroup ioda_cxx_types
202 template <class DataType, int String_Type_Length = constants::_Variable_Length>
203 Type GetType(gsl::not_null<const ::ioda::detail::Type_Provider*> t,
204  std::initializer_list<Dimensions_t> = {},
205  typename std::enable_if<is_string<DataType>::value>::type* = 0) {
206  return t->makeStringType(String_Type_Length, typeid(DataType));
207 }
208 
209 // This macro just repeats a long definition
210 
211 /// @def IODA_ADD_FUNDAMENTAL_TYPE
212 /// Macro that defines a "fundamental type" that needs to be supported
213 /// by the backend. These match C++11.
214 /// \ingroup ioda_cxx_types
215 /// \see https://en.cppreference.com/w/cpp/language/types
216 /// \since C++11: we use bool, short int, unsigned short int,
217 /// int, unsigned int, long int, unsigned long int,
218 /// long long int, unsigned long long int,
219 /// signed char, unsigned char, char,
220 /// wchar_t, char16_t, char32_t,
221 /// float, double, long double.
222 /// \since C++20: we also add char8_t.
223 #define IODA_ADD_FUNDAMENTAL_TYPE(x) \
224  template <> \
225  inline Type GetType<x, 0>(gsl::not_null<const ::ioda::detail::Type_Provider*> t, \
226  std::initializer_list<Dimensions_t>, void*) { \
227  return t->makeFundamentalType(typeid(x)); \
228  }
229 
232 IODA_ADD_FUNDAMENTAL_TYPE(unsigned short int);
236 IODA_ADD_FUNDAMENTAL_TYPE(unsigned long int);
238 IODA_ADD_FUNDAMENTAL_TYPE(unsigned long long int);
245 // IODA_ADD_FUNDAMENTAL_TYPE(char8_t); // C++20
249 
250 #undef IODA_ADD_FUNDAMENTAL_TYPE
251 
252 /*
253 /// Used in an example. Incomplete.
254 /// \todo Pop off std::array as a 1-D object
255 template<> inline Type GetType<std::array<int,2>, 0>
256  (gsl::not_null<const ::ioda::detail::Type_Provider*> t,
257  std::initializer_list<Dimensions_t>, void*) {
258  return t->makeArrayType({2}, typeid(std::array<int,2>), typeid(int)); }
259 
260 */
261 
262 /*
263 template <class DataType, int Array_Type_Dimensionality = 0>
264 Type GetType(
265  gsl::not_null<const ::ioda::detail::Type_Provider*> t,
266  std::initializer_list<Dimensions_t> Adims = {},
267  typename std::enable_if<!is_string<DataType>::value>::type* = 0);
268 template <class DataType, int String_Type_Length = constants::_Variable_Length>
269 Type GetType(
270  gsl::not_null<const ::ioda::detail::Type_Provider*> t,
271  typename std::enable_if<is_string<DataType>::value>::type* = 0);
272  */
273 
274 /// \brief Wrapper struct to call GetType. Needed because of C++ template rules.
275 /// \ingroup ioda_cxx_types
276 /// \see ioda::Attribute, ioda::Has_Attributes, ioda::Variable, ioda::Has_Variables
277 template <class DataType,
278  int Length = 0> //, typename = std::enable_if_t<!is_string<DataType>::value>>
280  static Type GetType(gsl::not_null<const ::ioda::detail::Type_Provider*> t) {
281  /// \note Currently breaks array types, but these are not yet used.
282  return ::ioda::Types::GetType<DataType, Length>(t, {Length});
283  }
284 };
285 /// \ingroup ioda_cxx_types
286 typedef std::function<Type(gsl::not_null<const ::ioda::detail::Type_Provider*>)>
288 /*
289 template <class DataType, int Length = 0, typename = std::enable_if_t<is_string<DataType>::value>>
290 struct GetType_Wrapper {
291  Type GetType(gsl::not_null<const ::ioda::detail::Type_Provider*> t) const {
292  // string split
293  return ::ioda::Types::GetType<DataType, Length>(t);
294  }
295 };
296 */
297 
298 // inline Encapsulated_Handle GetTypeFixedString(Dimensions_t sz);
299 } // namespace Types
300 } // namespace ioda
301 
302 /// @}
IODA's error system.
Frontend/backend bindings for the type system.
The ioda exception class.
Definition: Exception.h:54
Represents the "type" (i.e. integer, string, float) of a piece of data.
Definition: Type.h:123
std::type_index as_type_index_
Definition: Type.h:142
std::type_index getType() const
Definition: Type.h:136
virtual ~Type()
std::type_index operator()() const
Definition: Type.h:137
std::type_index get() const
Definition: Type.h:138
virtual size_t getSize() const
Get the size of a single element of a type, in bytes.
Definition: Type.cpp:67
detail::Type_Provider * getTypeProvider() const
Get the type provider.
Definition: Type.h:81
::ioda::detail::Type_Provider * provider_
Definition: Type.h:72
std::shared_ptr< Type_Backend > getBackend() const
Definition: Type.h:103
std::shared_ptr< Type_Backend > backend_
Definition: Type.h:69
Type_Base(std::shared_ptr< Type_Backend > b, ::ioda::detail::Type_Provider *p)
Definition: Type.h:77
bool isValid() const
Definition: Type.h:104
virtual ~Type_Base()
Definition: Type.h:102
Backends implement type providers in conjunction with Attributes, Has_Attributes, Variables and Has_V...
Definition: Type_Provider.h:36
Common preprocessor definitions used throughout IODA.
#define IODA_DL
A preprocessor tag that indicates that a symbol is to be exported/imported.
Definition: defs.h:110
std::function< Type(gsl::not_null< const ::ioda::detail::Type_Provider * >)> TypeWrapper_function
Definition: Type.h:287
Type GetType(gsl::not_null< const ::ioda::detail::Type_Provider * > t, std::initializer_list< Dimensions_t > Adims={}, typename std::enable_if<!is_string< DataType >::value >::type *=0)
For fundamental, non-string types.
Definition: Type.h:189
constexpr size_t _Variable_Length
Definition: Type.h:182
IODA_ADD_FUNDAMENTAL_TYPE(bool)
IODA_DL size_t COMPAT_strncpy_s(char *dest, size_t destSz, const char *src, size_t srcSz)
Safe char array copy.
Definition: Type.cpp:25
BasicTypes
Definition: Type.h:37
@ undefined_
Internal use only.
#define ioda_Here()
Wrapper struct to call GetType. Needed because of C++ template rules.
Definition: Type.h:279
static Type GetType(gsl::not_null< const ::ioda::detail::Type_Provider * > t)
Definition: Type.h:280
Convenience struct to determine if a type can represent a string.
Definition: Type.h:173