20#include "../H5Reference.hpp"
30inline bool checkDimensions(
const std::vector<size_t>& dims,
31 size_t min_dim_requested,
32 size_t max_dim_requested) {
33 if (min_dim_requested <= dims.size() && dims.size() <= max_dim_requested) {
41 return n_elements == 1 && min_dim_requested == 0;
87 using type = unqualified_t<T>;
88 using base_type = unqualified_t<T>;
89 using hdf5_type = base_type;
91 static constexpr size_t ndim = 0;
92 static constexpr size_t min_ndim = ndim;
93 static constexpr size_t max_ndim = ndim;
95 static constexpr bool is_trivially_copyable = std::is_trivially_copyable<type>::value;
96 static constexpr bool is_trivially_nestable = is_trivially_copyable;
98 static size_t getRank(
const type& ) {
102 static std::vector<size_t> getDimensions(
const type& ) {
106 static void prepare(type& ,
const std::vector<size_t>& ) {}
108 static hdf5_type* data(type& val) {
109 static_assert(is_trivially_copyable,
"The type is not trivially copyable");
113 static const hdf5_type* data(
const type& val) {
114 static_assert(is_trivially_copyable,
"The type is not trivially copyable");
118 static void serialize(
const type& val,
const std::vector<size_t>& , hdf5_type* m) {
119 static_assert(is_trivially_copyable,
"The type is not trivially copyable");
123 static void unserialize(
const hdf5_type* vec,
124 const std::vector<size_t>& ,
126 static_assert(is_trivially_copyable,
"The type is not trivially copyable");
132struct inspector: type_helper<T> {};
134enum class Boolean : int8_t {
140struct inspector<bool>: type_helper<bool> {
141 using base_type = Boolean;
142 using hdf5_type = int8_t;
144 static constexpr bool is_trivially_copyable =
false;
145 static constexpr bool is_trivially_nestable =
false;
147 static hdf5_type* data(type& ) {
148 throw DataSpaceException(
"A boolean cannot be read directly.");
151 static const hdf5_type* data(
const type& ) {
152 throw DataSpaceException(
"A boolean cannot be written directly.");
155 static void unserialize(
const hdf5_type* vec,
156 const std::vector<size_t>& ,
158 val = vec[0] != 0 ? true :
false;
161 static void serialize(
const type& val,
const std::vector<size_t>& , hdf5_type* m) {
167struct inspector<std::string>: type_helper<std::string> {
168 using hdf5_type =
const char*;
170 static hdf5_type* data(type& ) {
171 throw DataSpaceException(
"A std::string cannot be read directly.");
174 static const hdf5_type* data(
const type& ) {
175 throw DataSpaceException(
"A std::string cannot be written directly.");
179 static void serialize(
const type& val,
const std::vector<size_t>& , It m) {
184 static void unserialize(
const It& vec,
const std::vector<size_t>& , type& val) {
185 const auto& view = *vec;
186 val.assign(view.data(), view.length());
191struct inspector<
Reference>: type_helper<Reference> {
192 using hdf5_type = hobj_ref_t;
194 static constexpr bool is_trivially_copyable =
false;
195 static constexpr bool is_trivially_nestable =
false;
197 static hdf5_type* data(type& ) {
198 throw DataSpaceException(
"A Reference cannot be read directly.");
201 static const hdf5_type* data(
const type& ) {
202 throw DataSpaceException(
"A Reference cannot be written directly.");
205 static void serialize(
const type& val,
const std::vector<size_t>& , hdf5_type* m) {
207 val.create_ref(&ref);
211 static void unserialize(
const hdf5_type* vec,
212 const std::vector<size_t>& ,
219struct inspector<std::vector<T>> {
220 using type = std::vector<T>;
221 using value_type = unqualified_t<T>;
222 using base_type =
typename inspector<value_type>::base_type;
223 using hdf5_type =
typename inspector<value_type>::hdf5_type;
225 static constexpr size_t ndim = 1;
226 static constexpr size_t min_ndim = ndim + inspector<value_type>::min_ndim;
227 static constexpr size_t max_ndim = ndim + inspector<value_type>::max_ndim;
229 static constexpr bool is_trivially_copyable = std::is_trivially_copyable<value_type>::value &&
230 inspector<value_type>::is_trivially_nestable;
231 static constexpr bool is_trivially_nestable =
false;
233 static size_t getRank(
const type& val) {
235 return ndim + inspector<value_type>::getRank(val[0]);
241 static std::vector<size_t> getDimensions(
const type& val) {
242 auto rank = getRank(val);
243 std::vector<size_t> sizes(rank, 1ul);
244 sizes[0] = val.size();
246 auto s = inspector<value_type>::getDimensions(val[0]);
247 for (
size_t i = 0; i < s.size(); ++i) {
248 sizes[i + ndim] = s[i];
254 static void prepare(type& val,
const std::vector<size_t>& dims) {
256 std::vector<size_t> next_dims(dims.begin() + 1, dims.end());
257 for (
auto&& e: val) {
258 inspector<value_type>::prepare(e, next_dims);
262 static hdf5_type* data(type& val) {
263 return val.empty() ? nullptr : inspector<value_type>::data(val[0]);
266 static const hdf5_type* data(
const type& val) {
267 return val.empty() ? nullptr : inspector<value_type>::data(val[0]);
271 static void serialize(
const type& val,
const std::vector<size_t>& dims, It m) {
273 auto subdims = std::vector<size_t>(dims.begin() + 1, dims.end());
275 for (
auto&& e: val) {
276 inspector<value_type>::serialize(e, subdims, m);
283 static void unserialize(
const It& vec_align,
const std::vector<size_t>& dims, type& val) {
284 std::vector<size_t> next_dims(dims.begin() + 1, dims.end());
286 for (
size_t i = 0; i < dims[0]; ++i) {
287 inspector<value_type>::unserialize(vec_align + i * next_size, next_dims, val[i]);
293struct inspector<std::vector<bool>> {
294 using type = std::vector<bool>;
295 using value_type = bool;
296 using base_type = Boolean;
297 using hdf5_type = uint8_t;
299 static constexpr size_t ndim = 1;
300 static constexpr size_t min_ndim = ndim;
301 static constexpr size_t max_ndim = ndim;
303 static constexpr bool is_trivially_copyable =
false;
304 static constexpr bool is_trivially_nestable =
false;
306 static size_t getRank(
const type& ) {
310 static std::vector<size_t> getDimensions(
const type& val) {
311 std::vector<size_t> sizes{val.size()};
315 static void prepare(type& val,
const std::vector<size_t>& dims) {
316 if (dims.size() > 1) {
317 throw DataSpaceException(
"std::vector<bool> is only 1 dimension.");
322 static hdf5_type* data(type& ) {
323 throw DataSpaceException(
"A std::vector<bool> cannot be read directly.");
326 static const hdf5_type* data(
const type& ) {
327 throw DataSpaceException(
"A std::vector<bool> cannot be written directly.");
330 static void serialize(
const type& val,
const std::vector<size_t>& , hdf5_type* m) {
331 for (
size_t i = 0; i < val.size(); ++i) {
332 m[i] = val[i] ? 1 : 0;
336 static void unserialize(
const hdf5_type* vec_align,
337 const std::vector<size_t>& dims,
339 for (
size_t i = 0; i < dims[0]; ++i) {
340 val[i] = vec_align[i] != 0 ? true :
false;
345template <
typename T,
size_t N>
346struct inspector<std::array<T, N>> {
347 using type = std::array<T, N>;
348 using value_type = unqualified_t<T>;
349 using base_type =
typename inspector<value_type>::base_type;
350 using hdf5_type =
typename inspector<value_type>::hdf5_type;
352 static constexpr size_t ndim = 1;
353 static constexpr size_t min_ndim = ndim + inspector<value_type>::min_ndim;
354 static constexpr size_t max_ndim = ndim + inspector<value_type>::max_ndim;
356 static constexpr bool is_trivially_copyable = std::is_trivially_copyable<value_type>::value &&
357 inspector<value_type>::is_trivially_nestable;
358 static constexpr bool is_trivially_nestable = (
sizeof(type) == N *
sizeof(T)) &&
359 is_trivially_copyable;
361 static size_t getRank(
const type& val) {
362 return ndim + inspector<value_type>::getRank(val[0]);
365 static std::vector<size_t> getDimensions(
const type& val) {
366 std::vector<size_t> sizes{N};
367 auto s = inspector<value_type>::getDimensions(val[0]);
368 sizes.insert(sizes.end(), s.begin(), s.end());
372 static void prepare(type& val,
const std::vector<size_t>& dims) {
374 std::ostringstream os;
375 os <<
"Size of std::array (" << N <<
") is too small for dims (" << dims[0] <<
").";
376 throw DataSpaceException(os.str());
379 std::vector<size_t> next_dims(dims.begin() + 1, dims.end());
380 for (
auto&& e: val) {
381 inspector<value_type>::prepare(e, next_dims);
385 static hdf5_type* data(type& val) {
386 return inspector<value_type>::data(val[0]);
389 static const hdf5_type* data(
const type& val) {
390 return inspector<value_type>::data(val[0]);
394 static void serialize(
const type& val,
const std::vector<size_t>& dims, It m) {
395 auto subdims = std::vector<size_t>(dims.begin() + 1, dims.end());
398 inspector<value_type>::serialize(e, subdims, m);
404 static void unserialize(
const It& vec_align,
const std::vector<size_t>& dims, type& val) {
406 std::ostringstream os;
407 os <<
"Impossible to pair DataSet with " << dims[0] <<
" elements into an array with "
408 << N <<
" elements.";
409 throw DataSpaceException(os.str());
411 std::vector<size_t> next_dims(dims.begin() + 1, dims.end());
413 for (
size_t i = 0; i < dims[0]; ++i) {
414 inspector<value_type>::unserialize(vec_align + i * next_size, next_dims, val[i]);
422struct inspector<T*> {
424 using value_type = unqualified_t<T>;
425 using base_type =
typename inspector<value_type>::base_type;
426 using hdf5_type =
typename inspector<value_type>::hdf5_type;
428 static constexpr size_t ndim = 1;
429 static constexpr size_t min_ndim = ndim + inspector<value_type>::min_ndim;
430 static constexpr size_t max_ndim = ndim + inspector<value_type>::max_ndim;
432 static constexpr bool is_trivially_copyable = std::is_trivially_copyable<value_type>::value &&
433 inspector<value_type>::is_trivially_nestable;
434 static constexpr bool is_trivially_nestable =
false;
436 static size_t getRank(
const type& val) {
437 if (val !=
nullptr) {
438 return ndim + inspector<value_type>::getRank(val[0]);
444 static std::vector<size_t> getDimensions(
const type& ) {
445 throw DataSpaceException(
"Not possible to have size of a T*");
448 static const hdf5_type* data(
const type& val) {
449 return reinterpret_cast<const hdf5_type*
>(val);
454 static void serialize(
const type& ,
455 const std::vector<size_t>& ,
457 throw DataSpaceException(
"Not possible to serialize a T*");
462template <
typename T,
size_t N>
463struct inspector<T[N]> {
465 using value_type = unqualified_t<T>;
466 using base_type =
typename inspector<value_type>::base_type;
467 using hdf5_type =
typename inspector<value_type>::hdf5_type;
469 static constexpr size_t ndim = 1;
470 static constexpr size_t min_ndim = ndim + inspector<value_type>::min_ndim;
471 static constexpr size_t max_ndim = ndim + inspector<value_type>::max_ndim;
473 static constexpr bool is_trivially_copyable = std::is_trivially_copyable<value_type>::value &&
474 inspector<value_type>::is_trivially_nestable;
475 static constexpr bool is_trivially_nestable = is_trivially_copyable;
477 static void prepare(type& val,
const std::vector<size_t>& dims) {
478 if (dims.size() < 1) {
479 throw DataSpaceException(
"Invalid 'dims', must be at least 1 dimensional.");
483 throw DataSpaceException(
"Dimensions mismatch.");
486 std::vector<size_t> next_dims(dims.begin() + 1, dims.end());
487 for (
size_t i = 0; i < dims[0]; ++i) {
488 inspector<value_type>::prepare(val[i], next_dims);
492 static size_t getRank(
const type& val) {
493 return ndim + inspector<value_type>::getRank(val[0]);
496 static std::vector<size_t> getDimensions(
const type& val) {
497 std::vector<size_t> sizes{N};
498 auto s = inspector<value_type>::getDimensions(val[0]);
499 sizes.insert(sizes.end(), s.begin(), s.end());
503 static const hdf5_type* data(
const type& val) {
504 return inspector<value_type>::data(val[0]);
507 static hdf5_type* data(type& val) {
508 return inspector<value_type>::data(val[0]);
513 static void serialize(
const type& val,
const std::vector<size_t>& dims, hdf5_type* m) {
514 auto subdims = std::vector<size_t>(dims.begin() + 1, dims.end());
516 for (
size_t i = 0; i < N; ++i) {
517 inspector<value_type>::serialize(val[i], subdims, m + i * subsize);
Definition assert_compatible_spaces.hpp:15
size_t compute_total_size(const std::vector< size_t > &dims)
Definition compute_total_size.hpp:10