HighFive 3.0.0
HighFive - Header-only C++ HDF5 interface
Loading...
Searching...
No Matches
H5Inspector_misc.hpp
Go to the documentation of this file.
1/*
2 * Copyright (c) 2022 Blue Brain Project
3 *
4 * Distributed under the Boost Software License, Version 1.0.
5 * (See accompanying file LICENSE_1_0.txt or copy at
6 * http://www.boost.org/LICENSE_1_0.txt)
7 *
8 */
9
10#pragma once
11
12#include <type_traits>
13#include <cstring>
14#include <cassert>
15#include <vector>
16#include <array>
17#include <string>
18#include <numeric>
19
20#include "../H5Reference.hpp"
21
22#include "string_padding.hpp"
23
24#include "H5Inspector_decl.hpp"
25
26
27namespace HighFive {
28namespace details {
29
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) {
34 return true;
35 }
36
37
38 // Scalar values still support broadcasting
39 // into arrays with one element.
40 size_t n_elements = compute_total_size(dims);
41 return n_elements == 1 && min_dim_requested == 0;
42}
43
44} // namespace details
45
46
47/*****
48inspector<T> {
49 using type = T
50 // base_type is the base type inside c++ (e.g. std::vector<int> => int)
51 using base_type
52 // hdf5_type is the base read by hdf5 (c-type) (e.g. std::vector<std::string> => const char*)
53 using hdf5_type
54
55 // Is the inner type trivially copyable for optimisation
56 // If this value is true: data() is mandatory
57 // If this value is false: serialize, unserialize are mandatory
58 static constexpr bool is_trivially_copyable
59
60 // Is this type trivially nestable, i.e. is type[n] a contiguous
61 // array of `base_type[N]`?
62 static constexpr bool is_trivially_nestable
63
64 // Reading:
65 // Allocate the value following dims (should be recursive)
66 static void prepare(type& val, const std::vector<std::size_t> dims)
67 // Return a pointer of the first value of val (for reading)
68 static hdf5_type* data(type& val)
69 // Take a serialized vector 'in', some dims and copy value to val (for reading)
70 static void unserialize(const hdf5_type* in, const std::vector<size_t>&i, type& val)
71
72
73 // Writing:
74 // Return a point of the first value of val
75 static const hdf5_type* data(const type& val)
76 // Take a val and serialize it inside 'out'
77 static void serialize(const type& val, const std::vector<size_t>& dims, hdf5_type* out)
78 // Return an array of dimensions of the space needed for writing val
79 static std::vector<size_t> getDimensions(const type& val)
80}
81*****/
82
83
84namespace details {
85template <typename T>
86struct type_helper {
87 using type = unqualified_t<T>;
88 using base_type = unqualified_t<T>;
89 using hdf5_type = base_type;
90
91 static constexpr size_t ndim = 0;
92 static constexpr size_t min_ndim = ndim;
93 static constexpr size_t max_ndim = ndim;
94
95 static constexpr bool is_trivially_copyable = std::is_trivially_copyable<type>::value;
96 static constexpr bool is_trivially_nestable = is_trivially_copyable;
97
98 static size_t getRank(const type& /* val */) {
99 return ndim;
100 }
101
102 static std::vector<size_t> getDimensions(const type& /* val */) {
103 return {};
104 }
105
106 static void prepare(type& /* val */, const std::vector<size_t>& /* dims */) {}
107
108 static hdf5_type* data(type& val) {
109 static_assert(is_trivially_copyable, "The type is not trivially copyable");
110 return &val;
111 }
112
113 static const hdf5_type* data(const type& val) {
114 static_assert(is_trivially_copyable, "The type is not trivially copyable");
115 return &val;
116 }
117
118 static void serialize(const type& val, const std::vector<size_t>& /* dims*/, hdf5_type* m) {
119 static_assert(is_trivially_copyable, "The type is not trivially copyable");
120 *m = val;
121 }
122
123 static void unserialize(const hdf5_type* vec,
124 const std::vector<size_t>& /* dims */,
125 type& val) {
126 static_assert(is_trivially_copyable, "The type is not trivially copyable");
127 val = vec[0];
128 }
129};
130
131template <typename T>
132struct inspector: type_helper<T> {};
133
134enum class Boolean : int8_t {
135 HighFiveFalse = 0,
136 HighFiveTrue = 1,
137};
138
139template <>
140struct inspector<bool>: type_helper<bool> {
141 using base_type = Boolean;
142 using hdf5_type = int8_t;
143
144 static constexpr bool is_trivially_copyable = false;
145 static constexpr bool is_trivially_nestable = false;
146
147 static hdf5_type* data(type& /* val */) {
148 throw DataSpaceException("A boolean cannot be read directly.");
149 }
150
151 static const hdf5_type* data(const type& /* val */) {
152 throw DataSpaceException("A boolean cannot be written directly.");
153 }
154
155 static void unserialize(const hdf5_type* vec,
156 const std::vector<size_t>& /* dims */,
157 type& val) {
158 val = vec[0] != 0 ? true : false;
159 }
160
161 static void serialize(const type& val, const std::vector<size_t>& /* dims*/, hdf5_type* m) {
162 *m = val ? 1 : 0;
163 }
164};
165
166template <>
167struct inspector<std::string>: type_helper<std::string> {
168 using hdf5_type = const char*;
169
170 static hdf5_type* data(type& /* val */) {
171 throw DataSpaceException("A std::string cannot be read directly.");
172 }
173
174 static const hdf5_type* data(const type& /* val */) {
175 throw DataSpaceException("A std::string cannot be written directly.");
176 }
177
178 template <class It>
179 static void serialize(const type& val, const std::vector<size_t>& /* dims*/, It m) {
180 (*m).assign(val.data(), val.size(), StringPadding::NullTerminated);
181 }
182
183 template <class It>
184 static void unserialize(const It& vec, const std::vector<size_t>& /* dims */, type& val) {
185 const auto& view = *vec;
186 val.assign(view.data(), view.length());
187 }
188};
189
190template <>
191struct inspector<Reference>: type_helper<Reference> {
192 using hdf5_type = hobj_ref_t;
193
194 static constexpr bool is_trivially_copyable = false;
195 static constexpr bool is_trivially_nestable = false;
196
197 static hdf5_type* data(type& /* val */) {
198 throw DataSpaceException("A Reference cannot be read directly.");
199 }
200
201 static const hdf5_type* data(const type& /* val */) {
202 throw DataSpaceException("A Reference cannot be written directly.");
203 }
204
205 static void serialize(const type& val, const std::vector<size_t>& /* dims*/, hdf5_type* m) {
206 hobj_ref_t ref;
207 val.create_ref(&ref);
208 *m = ref;
209 }
210
211 static void unserialize(const hdf5_type* vec,
212 const std::vector<size_t>& /* dims */,
213 type& val) {
214 val = type{vec[0]};
215 }
216};
217
218template <typename 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;
224
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;
228
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;
232
233 static size_t getRank(const type& val) {
234 if (!val.empty()) {
235 return ndim + inspector<value_type>::getRank(val[0]);
236 } else {
237 return min_ndim;
238 }
239 }
240
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();
245 if (!val.empty()) {
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];
249 }
250 }
251 return sizes;
252 }
253
254 static void prepare(type& val, const std::vector<size_t>& dims) {
255 val.resize(dims[0]);
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);
259 }
260 }
261
262 static hdf5_type* data(type& val) {
263 return val.empty() ? nullptr : inspector<value_type>::data(val[0]);
264 }
265
266 static const hdf5_type* data(const type& val) {
267 return val.empty() ? nullptr : inspector<value_type>::data(val[0]);
268 }
269
270 template <class It>
271 static void serialize(const type& val, const std::vector<size_t>& dims, It m) {
272 if (!val.empty()) {
273 auto subdims = std::vector<size_t>(dims.begin() + 1, dims.end());
274 size_t subsize = compute_total_size(subdims);
275 for (auto&& e: val) {
276 inspector<value_type>::serialize(e, subdims, m);
277 m += subsize;
278 }
279 }
280 }
281
282 template <class It>
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());
285 size_t next_size = compute_total_size(next_dims);
286 for (size_t i = 0; i < dims[0]; ++i) {
287 inspector<value_type>::unserialize(vec_align + i * next_size, next_dims, val[i]);
288 }
289 }
290};
291
292template <>
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;
298
299 static constexpr size_t ndim = 1;
300 static constexpr size_t min_ndim = ndim;
301 static constexpr size_t max_ndim = ndim;
302
303 static constexpr bool is_trivially_copyable = false;
304 static constexpr bool is_trivially_nestable = false;
305
306 static size_t getRank(const type& /* val */) {
307 return ndim;
308 }
309
310 static std::vector<size_t> getDimensions(const type& val) {
311 std::vector<size_t> sizes{val.size()};
312 return sizes;
313 }
314
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.");
318 }
319 val.resize(dims[0]);
320 }
321
322 static hdf5_type* data(type& /* val */) {
323 throw DataSpaceException("A std::vector<bool> cannot be read directly.");
324 }
325
326 static const hdf5_type* data(const type& /* val */) {
327 throw DataSpaceException("A std::vector<bool> cannot be written directly.");
328 }
329
330 static void serialize(const type& val, const std::vector<size_t>& /* dims*/, hdf5_type* m) {
331 for (size_t i = 0; i < val.size(); ++i) {
332 m[i] = val[i] ? 1 : 0;
333 }
334 }
335
336 static void unserialize(const hdf5_type* vec_align,
337 const std::vector<size_t>& dims,
338 type& val) {
339 for (size_t i = 0; i < dims[0]; ++i) {
340 val[i] = vec_align[i] != 0 ? true : false;
341 }
342 }
343};
344
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;
351
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;
355
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;
360
361 static size_t getRank(const type& val) {
362 return ndim + inspector<value_type>::getRank(val[0]);
363 }
364
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());
369 return sizes;
370 }
371
372 static void prepare(type& val, const std::vector<size_t>& dims) {
373 if (dims[0] > N) {
374 std::ostringstream os;
375 os << "Size of std::array (" << N << ") is too small for dims (" << dims[0] << ").";
376 throw DataSpaceException(os.str());
377 }
378
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);
382 }
383 }
384
385 static hdf5_type* data(type& val) {
386 return inspector<value_type>::data(val[0]);
387 }
388
389 static const hdf5_type* data(const type& val) {
390 return inspector<value_type>::data(val[0]);
391 }
392
393 template <class It>
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());
396 size_t subsize = compute_total_size(subdims);
397 for (auto& e: val) {
398 inspector<value_type>::serialize(e, subdims, m);
399 m += subsize;
400 }
401 }
402
403 template <class It>
404 static void unserialize(const It& vec_align, const std::vector<size_t>& dims, type& val) {
405 if (dims[0] != N) {
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());
410 }
411 std::vector<size_t> next_dims(dims.begin() + 1, dims.end());
412 size_t next_size = compute_total_size(next_dims);
413 for (size_t i = 0; i < dims[0]; ++i) {
414 inspector<value_type>::unserialize(vec_align + i * next_size, next_dims, val[i]);
415 }
416 }
417};
418
419
420// Cannot be use for reading
421template <typename T>
422struct inspector<T*> {
423 using type = 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;
427
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;
431
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;
435
436 static size_t getRank(const type& val) {
437 if (val != nullptr) {
438 return ndim + inspector<value_type>::getRank(val[0]);
439 } else {
440 return min_ndim;
441 }
442 }
443
444 static std::vector<size_t> getDimensions(const type& /* val */) {
445 throw DataSpaceException("Not possible to have size of a T*");
446 }
447
448 static const hdf5_type* data(const type& val) {
449 return reinterpret_cast<const hdf5_type*>(val);
450 }
451
452 /* it works because there is only T[][][] currently
453 we will fix it one day */
454 static void serialize(const type& /* val */,
455 const std::vector<size_t>& /* dims*/,
456 hdf5_type* /* m */) {
457 throw DataSpaceException("Not possible to serialize a T*");
458 }
459};
460
461// Cannot be use for reading
462template <typename T, size_t N>
463struct inspector<T[N]> {
464 using type = 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;
468
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;
472
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;
476
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.");
480 }
481
482 if (dims[0] != N) {
483 throw DataSpaceException("Dimensions mismatch.");
484 }
485
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);
489 }
490 }
491
492 static size_t getRank(const type& val) {
493 return ndim + inspector<value_type>::getRank(val[0]);
494 }
495
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());
500 return sizes;
501 }
502
503 static const hdf5_type* data(const type& val) {
504 return inspector<value_type>::data(val[0]);
505 }
506
507 static hdf5_type* data(type& val) {
508 return inspector<value_type>::data(val[0]);
509 }
510
511 /* it works because there is only T[][][] currently
512 we will fix it one day */
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());
515 size_t subsize = compute_total_size(subdims);
516 for (size_t i = 0; i < N; ++i) {
517 inspector<value_type>::serialize(val[i], subdims, m + i * subsize);
518 }
519 }
520};
521
522
523} // namespace details
524} // namespace HighFive
Definition assert_compatible_spaces.hpp:15
size_t compute_total_size(const std::vector< size_t > &dims)
Definition compute_total_size.hpp:10