HighFive 3.0.0
HighFive - Header-only C++ HDF5 interface
Loading...
Searching...
No Matches
H5Slice_traits_misc.hpp
Go to the documentation of this file.
1/*
2 * Copyright (c), 2017, Adrien Devresse <adrien.devresse@epfl.ch>
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#pragma once
10
11#include <algorithm>
12#include <cassert>
13#include <functional>
14#include <numeric>
15#include <sstream>
16#include <string>
17
18#include "h5d_wrapper.hpp"
19#include "h5s_wrapper.hpp"
20
21#include "H5ReadWrite_misc.hpp"
22#include "H5Converter_misc.hpp"
23#include "squeeze.hpp"
26
27namespace HighFive {
28
29namespace details {
30
31// map the correct reference to the dataset depending of the layout
32// dataset -> itself
33// subselection -> parent dataset
34inline const DataSet& get_dataset(const Selection& sel) {
35 return sel.getDataset();
36}
37
38inline const DataSet& get_dataset(const DataSet& ds) {
39 return ds;
40}
41
42// map the correct memspace identifier depending of the layout
43// dataset -> entire memspace
44// selection -> resolve space id
45inline hid_t get_memspace_id(const Selection& ptr) {
46 return ptr.getMemSpace().getId();
47}
48
49inline hid_t get_memspace_id(const DataSet&) {
50 return H5S_ALL;
51}
52} // namespace details
53
54inline ElementSet::ElementSet(std::initializer_list<std::size_t> list)
55 : _ids(list) {}
56
57inline ElementSet::ElementSet(std::initializer_list<std::vector<std::size_t>> list)
58 : ElementSet(std::vector<std::vector<std::size_t>>(list)) {}
59
60inline ElementSet::ElementSet(const std::vector<std::size_t>& element_ids)
61 : _ids(element_ids) {}
62
63inline ElementSet::ElementSet(const std::vector<std::vector<std::size_t>>& element_ids) {
64 for (const auto& vec: element_ids) {
65 std::copy(vec.begin(), vec.end(), std::back_inserter(_ids));
66 }
67}
68
69namespace detail {
70class HyperCube {
71 public:
72 HyperCube(size_t rank)
73 : offset(rank)
74 , count(rank) {}
75
76 void cross(const std::array<size_t, 2>& range, size_t axis) {
77 offset[axis] = range[0];
78 count[axis] = range[1] - range[0];
79 }
80
81 RegularHyperSlab asSlab() {
82 return RegularHyperSlab(offset, count);
83 }
84
85 private:
86 std::vector<size_t> offset;
87 std::vector<size_t> count;
88};
89
90inline void build_hyper_slab(HyperSlab& slab, size_t /* axis */, HyperCube& cube) {
91 slab |= cube.asSlab();
92}
93
94template <class... Slices>
95inline void build_hyper_slab(HyperSlab& slab,
96 size_t axis,
97 HyperCube& cube,
98 const std::array<size_t, 2>& slice,
99 const Slices&... higher_slices) {
100 cube.cross(slice, axis);
101 build_hyper_slab(slab, axis + 1, cube, higher_slices...);
102}
103
104template <class... Slices>
105inline void build_hyper_slab(HyperSlab& slab,
106 size_t axis,
107 HyperCube& cube,
108 const std::vector<std::array<size_t, 2>>& slices,
109 const Slices&... higher_slices) {
110 for (const auto& slice: slices) {
111 build_hyper_slab(slab, axis, cube, slice, higher_slices...);
112 }
113}
114
115template <class... Slices>
116inline void build_hyper_slab(HyperSlab& slab,
117 size_t axis,
118 HyperCube& cube,
119 const std::vector<size_t>& ids,
120 const Slices&... higher_slices) {
121 for (const auto& id: ids) {
122 auto slice = std::array<size_t, 2>{id, id + 1};
123 build_hyper_slab(slab, axis, cube, slice, higher_slices...);
124 }
125}
126
127template <class... Slices>
128inline void build_hyper_slab(HyperSlab& slab,
129 size_t axis,
130 HyperCube& cube,
131 size_t id,
132 const Slices&... higher_slices) {
133 auto slice = std::array<size_t, 2>{id, id + 1};
134 build_hyper_slab(slab, axis, cube, slice, higher_slices...);
135}
136
137inline void compute_squashed_shape(size_t /* axis */, std::vector<size_t>& /* shape */) {
138 // assert(axis == shape.size());
139}
140
141template <class... Slices>
142inline void compute_squashed_shape(size_t axis,
143 std::vector<size_t>& shape,
144 const std::array<size_t, 2>& slice,
145 const Slices&... higher_slices);
146
147template <class... Slices>
148inline void compute_squashed_shape(size_t axis,
149 std::vector<size_t>& shape,
150 const std::vector<size_t>& points,
151 const Slices&... higher_slices);
152
153template <class... Slices>
154inline void compute_squashed_shape(size_t axis,
155 std::vector<size_t>& shape,
156 size_t point,
157 const Slices&... higher_slices);
158
159template <class... Slices>
160inline void compute_squashed_shape(size_t axis,
161 std::vector<size_t>& shape,
162 const std::vector<std::array<size_t, 2>>& slices,
163 const Slices&... higher_slices);
164
165template <class... Slices>
166inline void compute_squashed_shape(size_t axis,
167 std::vector<size_t>& shape,
168 const std::array<size_t, 2>& slice,
169 const Slices&... higher_slices) {
170 shape[axis] = slice[1] - slice[0];
171 compute_squashed_shape(axis + 1, shape, higher_slices...);
172}
173
174template <class... Slices>
175inline void compute_squashed_shape(size_t axis,
176 std::vector<size_t>& shape,
177 const std::vector<size_t>& points,
178 const Slices&... higher_slices) {
179 shape[axis] = points.size();
180 compute_squashed_shape(axis + 1, shape, higher_slices...);
181}
182
183template <class... Slices>
184inline void compute_squashed_shape(size_t axis,
185 std::vector<size_t>& shape,
186 const std::vector<std::array<size_t, 2>>& slices,
187 const Slices&... higher_slices) {
188 shape[axis] = 0;
189 for (const auto& slice: slices) {
190 shape[axis] += slice[1] - slice[0];
191 }
192 compute_squashed_shape(axis + 1, shape, higher_slices...);
193}
194
195template <class... Slices>
196inline void compute_squashed_shape(size_t axis,
197 std::vector<size_t>& shape,
198 size_t /* point */,
199 const Slices&... higher_slices) {
200 shape[axis] = 1;
201 compute_squashed_shape(axis + 1, shape, higher_slices...);
202}
203} // namespace detail
204
205template <class... Slices>
206inline ProductSet::ProductSet(const Slices&... slices) {
207 auto rank = sizeof...(slices);
208 detail::HyperCube cube(rank);
209 detail::build_hyper_slab(slab, 0, cube, slices...);
210
211 shape = std::vector<size_t>(rank, size_t(0));
212 detail::compute_squashed_shape(0, shape, slices...);
213}
214
215
216template <typename Derivate>
218 const DataSpace& memspace) const {
219 // Note: The current limitation are that memspace must describe a
220 // packed memspace.
221 //
222 // The reason for this is that we're unable to unpack general
223 // hyperslabs when the memory is not contiguous, e.g.
224 // `std::vector<std::vector<double>>`.
225 const auto& slice = static_cast<const Derivate&>(*this);
226 auto filespace = hyperslab.apply(slice.getSpace());
227
228 return detail::make_selection(memspace, filespace, details::get_dataset(slice));
229}
230
231template <typename Derivate>
232inline Selection SliceTraits<Derivate>::select(const HyperSlab& hyper_slab) const {
233 const auto& slice = static_cast<const Derivate&>(*this);
234 auto filespace = slice.getSpace();
235 filespace = hyper_slab.apply(filespace);
236
237 auto n_elements = detail::h5s_get_select_npoints(filespace.getId());
238 auto memspace = DataSpace(std::array<size_t, 1>{size_t(n_elements)});
239
240 return detail::make_selection(memspace, filespace, details::get_dataset(slice));
241}
242
243
244template <typename Derivate>
245inline Selection SliceTraits<Derivate>::select(const std::vector<size_t>& offset,
246 const std::vector<size_t>& count,
247 const std::vector<size_t>& stride,
248 const std::vector<size_t>& block) const {
249 auto slab = HyperSlab(RegularHyperSlab(offset, count, stride, block));
250 auto memspace = DataSpace(count);
251 return select(slab, memspace);
252}
253
254template <typename Derivate>
255inline Selection SliceTraits<Derivate>::select(const std::vector<size_t>& columns) const {
256 const auto& slice = static_cast<const Derivate&>(*this);
257 const DataSpace& space = slice.getSpace();
258 std::vector<size_t> dims = space.getDimensions();
259
260 if (dims.empty()) {
261 throw DataSpaceException(
262 "Invalid, zero-dimensional (scalar) dataspace encountered when "
263 "selecting columns; must be atleast 1-dimensional.");
264 }
265
266 std::vector<size_t> counts = dims;
267 counts.back() = 1;
268
269 std::vector<size_t> offsets(dims.size(), 0);
270
271 HyperSlab slab;
272 for (const auto& column: columns) {
273 offsets.back() = column;
274 slab |= RegularHyperSlab(offsets, counts);
275 }
276
277 std::vector<size_t> memdims = dims;
278 memdims.back() = columns.size();
279
280 return select(slab, DataSpace(memdims));
281}
282
283template <typename Derivate>
285 const auto& slice = static_cast<const Derivate&>(*this);
286 const hsize_t* data = nullptr;
287 const DataSpace space = slice.getSpace().clone();
288 const std::size_t length = elements._ids.size();
289 if (length % space.getNumberDimensions() != 0) {
290 throw DataSpaceException(
291 "Number of coordinates in elements picking "
292 "should be a multiple of the dimensions.");
293 }
294 const std::size_t num_elements = length / space.getNumberDimensions();
295 std::vector<hsize_t> raw_elements;
296
297 // optimised at compile time
298 // switch for data conversion on 32bits platforms
299 if (std::is_same<std::size_t, hsize_t>::value) {
300 // `if constexpr` can't be used, thus a reinterpret_cast is needed.
301 data = reinterpret_cast<const hsize_t*>(&(elements._ids[0]));
302 } else {
303 raw_elements.resize(length);
304 std::copy(elements._ids.begin(), elements._ids.end(), raw_elements.begin());
305 data = raw_elements.data();
306 }
307
308 detail::h5s_select_elements(space.getId(), H5S_SELECT_SET, num_elements, data);
309
310 return detail::make_selection(DataSpace(num_elements), space, details::get_dataset(slice));
311}
312
313template <typename Derivate>
314inline Selection SliceTraits<Derivate>::select(const ProductSet& product_set) const {
315 return this->select(product_set.slab, DataSpace(product_set.shape));
316}
317
318
319template <typename Derivate>
320template <typename T>
321inline T SliceTraits<Derivate>::read(const DataTransferProps& xfer_props) const {
322 T array;
323 read(array, xfer_props);
324 return array;
325}
326
327
328template <typename Derivate>
329template <typename T>
330inline void SliceTraits<Derivate>::read(T& array, const DataTransferProps& xfer_props) const {
331 const auto& slice = static_cast<const Derivate&>(*this);
332 const DataSpace& mem_space = slice.getMemSpace();
333
334 auto file_datatype = slice.getDataType();
335
336 const details::BufferInfo<T> buffer_info(
337 file_datatype,
338 [&slice]() -> std::string { return details::get_dataset(slice).getPath(); },
339 details::BufferInfo<T>::Operation::read);
340
341 if (!details::checkDimensions(mem_space, buffer_info.getMinRank(), buffer_info.getMaxRank())) {
342 std::ostringstream ss;
343 ss << "Impossible to read DataSet of dimensions " << mem_space.getNumberDimensions()
344 << " into arrays of dimensions: " << buffer_info.getMinRank() << "(min) to "
345 << buffer_info.getMaxRank() << "(max)";
346 throw DataSpaceException(ss.str());
347 }
348 auto dims = mem_space.getDimensions();
349
350 auto r = details::data_converter::get_reader<T>(dims, array, file_datatype);
351 read_raw(r.getPointer(), buffer_info.data_type, xfer_props);
352 // re-arrange results
353 r.unserialize(array);
354
355 auto t = buffer_info.data_type;
356 auto c = t.getClass();
357 if (c == DataTypeClass::VarLen || t.isVariableStr()) {
358#if H5_VERSION_GE(1, 12, 0)
359 // This one have been created in 1.12.0
360 (void)
361 detail::h5t_reclaim(t.getId(), mem_space.getId(), xfer_props.getId(), r.getPointer());
362#else
363 // This one is deprecated since 1.12.0
364 (void) detail::h5d_vlen_reclaim(t.getId(),
365 mem_space.getId(),
366 xfer_props.getId(),
367 r.getPointer());
368#endif
369 }
370}
371
372
373template <typename Derivate>
374template <typename T>
376 const DataType& mem_datatype,
377 const DataTransferProps& xfer_props) const {
378 static_assert(!std::is_const<T>::value,
379 "read() requires a non-const structure to read data into");
380
381 const auto& slice = static_cast<const Derivate&>(*this);
382
383 detail::h5d_read(details::get_dataset(slice).getId(),
384 mem_datatype.getId(),
385 details::get_memspace_id(slice),
386 slice.getSpace().getId(),
387 xfer_props.getId(),
388 static_cast<void*>(array));
389}
390
391
392template <typename Derivate>
393template <typename T>
394inline void SliceTraits<Derivate>::read_raw(T* array, const DataTransferProps& xfer_props) const {
395 using element_type = typename details::inspector<T>::base_type;
396 const DataType& mem_datatype = create_and_check_datatype<element_type>();
397
398 read_raw(array, mem_datatype, xfer_props);
399}
400
401
402template <typename Derivate>
403template <typename T>
404inline void SliceTraits<Derivate>::write(const T& buffer, const DataTransferProps& xfer_props) {
405 const auto& slice = static_cast<const Derivate&>(*this);
406 const DataSpace& mem_space = slice.getMemSpace();
407 auto dims = mem_space.getDimensions();
408
409 auto file_datatype = slice.getDataType();
410
411 const details::BufferInfo<T> buffer_info(
412 file_datatype,
413 [&slice]() -> std::string { return details::get_dataset(slice).getPath(); },
414 details::BufferInfo<T>::Operation::write);
415
416 if (!details::checkDimensions(mem_space, buffer_info.getMinRank(), buffer_info.getMaxRank())) {
417 std::ostringstream ss;
418 ss << "Impossible to write buffer with dimensions n = " << buffer_info.getRank(buffer)
419 << "into dataset with dimensions " << details::format_vector(mem_space.getDimensions())
420 << ".";
421 throw DataSpaceException(ss.str());
422 }
423 auto w = details::data_converter::serialize<T>(buffer, dims, file_datatype);
424 write_raw(w.getPointer(), buffer_info.data_type, xfer_props);
425}
426
427
428template <typename Derivate>
429template <typename T>
430inline void SliceTraits<Derivate>::write_raw(const T* buffer,
431 const DataType& mem_datatype,
432 const DataTransferProps& xfer_props) {
433 const auto& slice = static_cast<const Derivate&>(*this);
434
435 detail::h5d_write(details::get_dataset(slice).getId(),
436 mem_datatype.getId(),
437 details::get_memspace_id(slice),
438 slice.getSpace().getId(),
439 xfer_props.getId(),
440 static_cast<const void*>(buffer));
441}
442
443
444template <typename Derivate>
445template <typename T>
446inline void SliceTraits<Derivate>::write_raw(const T* buffer, const DataTransferProps& xfer_props) {
447 using element_type = typename details::inspector<T>::base_type;
448 const auto& mem_datatype = create_and_check_datatype<element_type>();
449
450 write_raw(buffer, mem_datatype, xfer_props);
451}
452
453namespace detail {
454inline const DataSet& getDataSet(const Selection& selection) {
455 return selection.getDataset();
456}
457
458inline const DataSet& getDataSet(const DataSet& dataset) {
459 return dataset;
460}
461
462} // namespace detail
464template <typename Derivate>
465inline Selection SliceTraits<Derivate>::squeezeMemSpace(const std::vector<size_t>& axes) const {
466 auto slice = static_cast<const Derivate&>(*this);
467 auto mem_dims = slice.getMemSpace().getDimensions();
468 auto squeezed_dims = detail::squeeze(mem_dims, axes);
469
470 return detail::make_selection(DataSpace(squeezed_dims),
471 slice.getSpace(),
472 detail::getDataSet(slice));
474
475template <typename Derivate>
476inline Selection SliceTraits<Derivate>::reshapeMemSpace(const std::vector<size_t>& new_dims) const {
477 auto slice = static_cast<const Derivate&>(*this);
478
479 detail::assert_compatible_spaces(slice.getMemSpace(), new_dims);
480 return detail::make_selection(DataSpace(new_dims), slice.getSpace(), detail::getDataSet(slice));
481}
482
483} // namespace HighFive
Class representing a dataset.
Definition H5DataSet.hpp:30
Exception specific to HighFive DataSpace interface.
Definition H5Exception.hpp:112
Class representing the space (dimensions) of a DataSet.
Definition H5DataSpace.hpp:39
size_t getNumberDimensions() const
Returns the number of dimensions of a DataSpace.
Definition H5Dataspace_misc.hpp:100
std::vector< size_t > getDimensions() const
Returns the size of the dataset in each dimension.
Definition H5Dataspace_misc.hpp:104
DataSpace clone() const
Create a copy of the DataSpace which will have different id.
Definition H5Dataspace_misc.hpp:94
HDF5 Data Type.
Definition H5DataType.hpp:61
Definition H5Slice_traits.hpp:23
ElementSet(std::initializer_list< std::size_t > list)
Create a list of points of N-dimension for selection.
Definition H5Slice_traits_misc.hpp:54
Definition H5Slice_traits.hpp:111
DataSpace apply(const DataSpace &space_) const
Definition H5Slice_traits.hpp:164
hid_t getId() const noexcept
getId
Definition H5Object_misc.hpp:63
Selects the Cartesian product of slices.
Definition H5Slice_traits.hpp:405
ProductSet(const Slices &... slices)
Definition H5Slice_traits_misc.hpp:206
HDF5 property Lists.
Definition H5PropertyList.hpp:158
Selection: represent a view on a slice/part of a dataset.
Definition H5Selection.hpp:27
DataSpace getMemSpace() const
getMemSpace
Definition H5Selection_misc.hpp:24
DataSet & getDataset()
getDataSet
Definition H5Selection_misc.hpp:28
DataSpace getSpace() const
getSpace
Definition H5Selection_misc.hpp:20
Selection squeezeMemSpace(const std::vector< size_t > &axes) const
Return a Selection with axes squeezed from the memspace.
Definition H5Slice_traits_misc.hpp:465
T read(const DataTransferProps &xfer_props=DataTransferProps()) const
Definition H5Slice_traits_misc.hpp:321
void write(const T &buffer, const DataTransferProps &xfer_props=DataTransferProps())
Definition H5Slice_traits_misc.hpp:404
Selection select(const HyperSlab &hyperslab) const
Select an hyperslab in the current Slice/Dataset.
Definition H5Slice_traits_misc.hpp:232
Selection reshapeMemSpace(const std::vector< size_t > &dims) const
Return a Selection with a simple memspace with dims.
Definition H5Slice_traits_misc.hpp:476
void read_raw(T *array, const DataType &dtype, const DataTransferProps &xfer_props=DataTransferProps()) const
Definition H5Slice_traits_misc.hpp:375
void write_raw(const T *buffer, const DataType &mem_datatype, const DataTransferProps &xfer_props=DataTransferProps())
Definition H5Slice_traits_misc.hpp:430
Definition assert_compatible_spaces.hpp:15
Definition H5Slice_traits.hpp:63