IODA
Marshalling.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 /*! \addtogroup ioda_cxx_types
9  *
10  * @{
11  * \file Marshalling.h
12  * \brief Classes and functions that implement the type system and allow
13  * for frontend/backend communication.
14  */
15 #include <complex>
16 #include <cstring>
17 #include <exception>
18 #include <memory>
19 #include <stdexcept>
20 #include <string>
21 #include <type_traits>
22 #include <vector>
23 
24 #include "ioda/Exception.h"
26 #include "ioda/defs.h"
27 
28 namespace ioda {
29 
30 /// \ingroup ioda_internals_engines_types
31 template <class DataType, bool FreeOnClose>
32 void FreeType(DataType, typename std::enable_if<!std::is_pointer<DataType>::value>::type* = 0) {}
33 template <class DataType, bool FreeOnClose>
34 void FreeType(DataType, typename std::enable_if<std::is_pointer<DataType>::value>::type* = 0,
35  typename std::enable_if<!FreeOnClose>::type* = 0) {}
36 template <class DataType, bool FreeOnClose>
37 void FreeType(DataType d, typename std::enable_if<std::is_pointer<DataType>::value>::type* = 0,
38  typename std::enable_if<FreeOnClose>::type* = 0) {
39  free(d);
40 }
41 
42 /// \brief Structure used to pass data between the frontend and the backend engine.
43 /// \ingroup ioda_internals_engines_types
44 template <class T, class value_type = T, bool FreeOnClose = false>
46  std::vector<value_type> DataPointers;
49  : pointerOwner_(pointerOwner) {}
52  if (FreeOnClose)
53  for (auto& p : DataPointers) FreeType<value_type, FreeOnClose>(p);
54  }
55 };
56 
57 namespace detail {
58 /// \note We want character streams, and these void* types are horribly hacked :-(
59 /// Using void* because we want to preserve a semantic difference between
60 /// serialized / deserialized data.
61 /// \note By default, we are using the POD accessor. Valid for simple data types,
62 /// where multiple objects are in the same dataspace, and each object is a
63 /// singular instance of the base data type.
64 /// \ingroup ioda_internals_engines_types
65 template <class DataType, class value_type = DataType>
67  typedef typename std::remove_const<DataType>::type mutable_DataType;
68  typedef typename std::remove_const<value_type>::type mutable_value_type;
69 
70  typedef std::shared_ptr<Marshalled_Data<DataType, mutable_DataType>> serialized_type;
71  typedef std::shared_ptr<const Marshalled_Data<DataType, mutable_DataType>> const_serialized_type;
72 
74 
75 public:
77  : pointerOwner_(pointerOwner) {}
78  /// \brief Converts an object into a void* byte stream.
79  /// \note The shared_ptr takes care of "deallocation" when we no longer need the "buffer".
80  const_serialized_type serialize(::gsl::span<const DataType> d) {
81  auto res = std::make_shared<Marshalled_Data<DataType, mutable_DataType>>();
82  res->DataPointers = std::vector<mutable_DataType>(d.size());
83  // Forcible memset of the data to zero. Needed for long double type, which is typically 80
84  // bits but takes up 96 or 128 bits of storage space. This triggers a Valgrind warning for
85  // uninitialized memory access by HDF5.
86  // See https://en.wikipedia.org/wiki/Long_double
87  memset(res->DataPointers.data(), 0, sizeof(mutable_DataType) * d.size());
88  for (size_t i = 0; i < (size_t)d.size(); ++i) res->DataPointers[i] = d[i];
89  // res->DataPointers = std::vector<mutable_value_type>(d.begin(), d.end()); //return (const
90  // void*)d.data();
91  return res;
92  }
93  /// \brief Construct an object from a byte stream,
94  /// and deallocate any temporary buffer.
95  /// \note For trivial (POD) objects, there is no need to do anything.
96  serialized_type prep_deserialize(size_t numObjects) {
97  auto res = std::make_shared<typename serialized_type::element_type>(pointerOwner_);
98  res->DataPointers = std::vector<mutable_DataType>(numObjects);
99  return res;
100  }
101  /// Unpack the data. For POD, nothing special here.
102  void deserialize(serialized_type p, gsl::span<DataType> data) {
103  const size_t ds = data.size(), dp = p->DataPointers.size();
104  if (ds != dp) throw Exception("ds != dp", ioda_Here());
105  for (size_t i = 0; i < (size_t)data.size(); ++i) {
106  data[i] = p->DataPointers[i];
107  }
108  }
109 };
110 
111 /// \ingroup ioda_internals_engines_types
112 template <class DataType, class value_type = std::remove_pointer<std::decay<DataType>>>
114  typedef typename std::remove_const<DataType>::type mutable_DataType;
115  typedef typename std::remove_const<value_type>::type mutable_value_type;
116  typedef std::shared_ptr<Marshalled_Data<DataType, mutable_DataType>> serialized_type;
117  typedef std::shared_ptr<const Marshalled_Data<DataType, mutable_DataType>> const_serialized_type;
119 
120 public:
122  : pointerOwner_(pointerOwner) {}
123  /// \brief Converts an object into a void* byte stream.
124  /// \note The shared_ptr takes care of "deallocation" when we no longer need the "buffer".
125  const_serialized_type serialize(::gsl::span<const DataType> d) {
126  auto res = std::make_shared<Marshalled_Data<DataType, mutable_DataType>>();
127  res->DataPointers = std::vector<mutable_DataType>(d.size());
128  std::copy_n(reinterpret_cast<char*>(d.data()), d.size_bytes(),
129  reinterpret_cast<char*>(res->DataPointers.data()));
130  // Cannot do this for int[2].
131  // res->DataPointers = std::vector<mutable_value_type>(d.begin(), d.end());
132 
133  // for (size_t i = 0; i < (size_t)d.size(); ++i)
134  // res->DataPointers[i] = d[i];
135  // res->DataPointers = std::vector<mutable_value_type>(d.begin(), d.end()); //return (const
136  // void*)d.data();
137  return res;
138  }
139  /// \brief Construct an object from a byte stream,
140  /// and deallocate any temporary buffer.
141  /// \note For trivial (POD) objects, there is no need to do anything.
142  serialized_type prep_deserialize(size_t numObjects) {
143  auto res = std::make_shared<typename serialized_type::element_type>(pointerOwner_);
144  res->DataPointers = std::vector<mutable_DataType>(numObjects);
145  return res;
146  }
147  /// Unpack the data. For POD, nothing special here.
148  void deserialize(serialized_type p, gsl::span<DataType> data) {
149  const size_t ds = data.size(), dp = p->DataPointers.size();
150  if (ds != dp) throw Exception("ds != dp", ioda_Here());
151  std::copy_n(reinterpret_cast<char*>(p->DataPointers.data()), data.size_bytes(),
152  reinterpret_cast<char*>(data.data()));
153 
154  // for (size_t i = 0; i < (size_t)data.size(); ++i)
155  // data[i] = p->DataPointers[i];
156  }
157 };
158 
159 /// \ingroup ioda_internals_engines_types
160 template <class DataType, class value_type = DataType*>
162  typedef typename std::remove_const<DataType>::type mutable_DataType;
163  typedef typename std::remove_const<value_type>::type mutable_value_type;
164  typedef std::shared_ptr<const Marshalled_Data<DataType, mutable_value_type, false>>
166  typedef std::shared_ptr<Marshalled_Data<DataType, mutable_value_type, true>> serialized_type;
168 
169 public:
172  : pointerOwner_(pointerOwner) {}
173  const_serialized_type serialize(::gsl::span<const DataType> d) {
174  auto res = std::make_shared<Marshalled_Data<DataType, mutable_value_type, false>>();
175  for (const auto& i : d) {
176  res->DataPointers.push_back(const_cast<mutable_value_type>(i.data()));
177  }
178  return res;
179  }
180  serialized_type prep_deserialize(size_t numObjects) {
181  auto res = std::make_shared<typename serialized_type::element_type>(pointerOwner_);
182  res->DataPointers = std::vector<mutable_value_type>(numObjects);
183  return res;
184  }
185  void deserialize(serialized_type p, gsl::span<DataType> data) {
186  const size_t ds = data.size(), dp = p->DataPointers.size();
187  if (ds != dp) throw Exception("ds != dp", ioda_Here());
188  for (size_t i = 0; i < ds; ++i) {
189  if (p->DataPointers[i]) // Odd Valgrind detection. Maybe a false positive.
190  data[i] = p->DataPointers[i];
191  }
192  }
193 };
194 
195 /// \ingroup ioda_internals_engines_types
196 template <class DataType, class value_type = DataType*>
198  typedef typename std::remove_const<DataType>::type mutable_DataType;
199  typedef typename std::remove_const<value_type>::type mutable_value_type;
200  typedef std::shared_ptr<const Marshalled_Data<DataType, mutable_value_type, false>>
202  typedef std::shared_ptr<Marshalled_Data<DataType, mutable_value_type, true>> serialized_type;
204 
205 public:
208  : pointerOwner_(pointerOwner) {}
209  const_serialized_type serialize(::gsl::span<const DataType> d) {
210  auto res = std::make_shared<Marshalled_Data<DataType, mutable_value_type, false>>();
211  for (const auto& i : d) {
212  res->DataPointers.push_back(const_cast<mutable_value_type>(&i[0]));
213  }
214  return res;
215  }
216  serialized_type prep_deserialize(size_t numObjects) {
217  auto res = std::make_shared<typename serialized_type::element_type>(pointerOwner_);
218  res->DataPointers = std::vector<mutable_value_type>(numObjects);
219  return res;
220  }
221  void deserialize(serialized_type p, gsl::span<DataType> data) {
222  const size_t ds = data.size(), dp = p->DataPointers.size();
223  if (ds != dp) throw Exception("ds != dp", ioda_Here());
224  for (size_t i = 0; i < (size_t)data.size(); ++i) {
225  data[i] = p->DataPointers[i];
226  }
227  }
228 };
229 
230 /// \ingroup ioda_cxx_types
231 template <typename T>
234 };
235 /// \ingroup ioda_cxx_types
236 template <>
237 struct Object_AccessorTypedef<std::string> {
239 };
240 
241 // Used in an example
242 template <>
243 struct Object_AccessorTypedef<int[2]> {
244  /// \todo Make a fixed-length array type
246 };
247 
248 // Used in an example
249 template <>
250 struct Object_AccessorTypedef<std::array<int, 2>> {
251  /// \todo Make a fixed-length array type
253 };
254 } // namespace detail
255 
256 /// \ingroup ioda_cxx_types
257 template <typename DataType>
259 
260 } // namespace ioda
261 
262 /// @}
IODA's error system.
Frontend/backend bindings for the type system.
The ioda exception class.
Definition: Exception.h:54
Common preprocessor definitions used throughout IODA.
typename detail::Object_AccessorTypedef< DataType >::type Object_Accessor
Definition: Marshalling.h:258
PointerOwner
Who owns (and should free) pointers passed across the frontend / backend interface?
Definition: Type_Provider.h:27
void FreeType(DataType, typename std::enable_if<!std::is_pointer< DataType >::value >::type *=0)
Definition: Marshalling.h:32
@ Caller
The user has to free pointers.
@ Engine
The backend engine frees pointers that it provides.
#define ioda_Here()
Structure used to pass data between the frontend and the backend engine.
Definition: Marshalling.h:45
Marshalled_Data(detail::PointerOwner pointerOwner=detail::PointerOwner::Caller)
Definition: Marshalling.h:48
std::vector< value_type > DataPointers
Definition: Marshalling.h:46
detail::PointerOwner pointerOwner_
Definition: Marshalling.h:47
Object_Accessor_Fixed_Array(detail::PointerOwner pointerOwner=detail::PointerOwner::Caller)
Definition: Marshalling.h:121
std::remove_const< value_type >::type mutable_value_type
Definition: Marshalling.h:115
const_serialized_type serialize(::gsl::span< const DataType > d)
Converts an object into a void* byte stream.
Definition: Marshalling.h:125
serialized_type prep_deserialize(size_t numObjects)
Construct an object from a byte stream, and deallocate any temporary buffer.
Definition: Marshalling.h:142
std::shared_ptr< Marshalled_Data< DataType, mutable_DataType > > serialized_type
Definition: Marshalling.h:116
void deserialize(serialized_type p, gsl::span< DataType > data)
Unpack the data. For POD, nothing special here.
Definition: Marshalling.h:148
std::remove_const< DataType >::type mutable_DataType
Definition: Marshalling.h:114
std::shared_ptr< const Marshalled_Data< DataType, mutable_DataType > > const_serialized_type
Definition: Marshalling.h:117
Object_Accessor_Regular(detail::PointerOwner pointerOwner=detail::PointerOwner::Caller)
Definition: Marshalling.h:76
std::shared_ptr< Marshalled_Data< DataType, mutable_DataType > > serialized_type
Definition: Marshalling.h:70
std::remove_const< value_type >::type mutable_value_type
Definition: Marshalling.h:68
std::remove_const< DataType >::type mutable_DataType
Definition: Marshalling.h:67
serialized_type prep_deserialize(size_t numObjects)
Construct an object from a byte stream, and deallocate any temporary buffer.
Definition: Marshalling.h:96
const_serialized_type serialize(::gsl::span< const DataType > d)
Converts an object into a void* byte stream.
Definition: Marshalling.h:80
std::shared_ptr< const Marshalled_Data< DataType, mutable_DataType > > const_serialized_type
Definition: Marshalling.h:71
void deserialize(serialized_type p, gsl::span< DataType > data)
Unpack the data. For POD, nothing special here.
Definition: Marshalling.h:102
detail::PointerOwner pointerOwner_
Definition: Marshalling.h:73
Object_Accessor_Variable_Array_With_Data_Method(detail::PointerOwner pointerOwner=detail::PointerOwner::Caller)
Definition: Marshalling.h:170
std::shared_ptr< Marshalled_Data< DataType, mutable_value_type, true > > serialized_type
Definition: Marshalling.h:166
void deserialize(serialized_type p, gsl::span< DataType > data)
Definition: Marshalling.h:185
std::remove_const< value_type >::type mutable_value_type
Definition: Marshalling.h:163
std::remove_const< DataType >::type mutable_DataType
Definition: Marshalling.h:162
const_serialized_type serialize(::gsl::span< const DataType > d)
Definition: Marshalling.h:173
std::shared_ptr< const Marshalled_Data< DataType, mutable_value_type, false > > const_serialized_type
Definition: Marshalling.h:165
std::shared_ptr< Marshalled_Data< DataType, mutable_value_type, true > > serialized_type
Definition: Marshalling.h:202
Object_Accessor_Variable_Raw_Array(detail::PointerOwner pointerOwner=detail::PointerOwner::Caller)
Definition: Marshalling.h:206
void deserialize(serialized_type p, gsl::span< DataType > data)
Definition: Marshalling.h:221
std::remove_const< DataType >::type mutable_DataType
Definition: Marshalling.h:198
std::remove_const< value_type >::type mutable_value_type
Definition: Marshalling.h:199
serialized_type prep_deserialize(size_t numObjects)
Definition: Marshalling.h:216
const_serialized_type serialize(::gsl::span< const DataType > d)
Definition: Marshalling.h:209
std::shared_ptr< const Marshalled_Data< DataType, mutable_value_type, false > > const_serialized_type
Definition: Marshalling.h:201
Object_Accessor_Fixed_Array< int[2]> type
Definition: Marshalling.h:245
Object_Accessor_Fixed_Array< std::array< int, 2 >, int > type
Definition: Marshalling.h:252
Object_Accessor_Variable_Array_With_Data_Method< std::string, char * > type
Definition: Marshalling.h:238
Object_Accessor_Regular< T > type
Definition: Marshalling.h:233