HighFive 3.0.0
HighFive - Header-only C++ HDF5 interface
Loading...
Searching...
No Matches
H5Attribute_misc.hpp
Go to the documentation of this file.
1/*
2 * Copyright (c), 2017, Ali Can Demiralp <ali.demiralp@rwth-aachen.de>
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 <H5Ipublic.h>
12#include <algorithm>
13#include <functional>
14#include <numeric>
15#include <sstream>
16#include <string>
17
18#include <H5Ppublic.h>
19
20#include "../H5DataSpace.hpp"
21#include "H5Converter_misc.hpp"
22#include "H5Inspector_misc.hpp"
23#include "H5ReadWrite_misc.hpp"
24#include "H5Utils.hpp"
25#include "h5a_wrapper.hpp"
26#include "h5d_wrapper.hpp"
27#include "squeeze.hpp"
29
30namespace HighFive {
31
32inline std::string Attribute::getName() const {
33 return details::get_name(
34 [&](char* buffer, size_t length) { return detail::h5a_get_name(_hid, length, buffer); });
35}
36
37inline size_t Attribute::getStorageSize() const {
38 if (!this->isValid()) {
39 throw AttributeException("Invalid call to `DataSet::getFile` for invalid object");
40 }
41
42 return static_cast<size_t>(detail::h5a_get_storage_size(_hid));
43}
44
46 DataType res;
47 res._hid = detail::h5a_get_type(_hid);
48 return res;
49}
50
52 DataSpace space;
53 space._hid = detail::h5a_get_space(_hid);
54 return space;
55}
56
58 return _mem_space.getId() == H5I_INVALID_HID ? getSpace() : _mem_space;
59}
60
61template <typename T>
62inline T Attribute::read() const {
63 T array;
64 read(array);
65 return array;
66}
67
68template <typename T>
69inline void Attribute::read(T& array) const {
70 const DataSpace& mem_space = getMemSpace();
71 auto file_datatype = getDataType();
72 const details::BufferInfo<T> buffer_info(
73 file_datatype,
74 [this]() -> std::string { return this->getName(); },
75 details::BufferInfo<T>::Operation::read);
76
77 if (!details::checkDimensions(mem_space, buffer_info.getMinRank(), buffer_info.getMaxRank())) {
78 std::ostringstream ss;
79 ss << "Impossible to read attribute of dimensions " << mem_space.getNumberDimensions()
80 << " into arrays of dimensions: " << buffer_info.getMinRank() << "(min) to "
81 << buffer_info.getMaxRank() << "(max)";
82 throw DataSpaceException(ss.str());
83 }
84 auto dims = mem_space.getDimensions();
85
86 if (mem_space.getElementCount() == 0) {
87 details::inspector<T>::prepare(array, dims);
88 return;
89 }
90
91 auto r = details::data_converter::get_reader<T>(dims, array, file_datatype);
92 read_raw(r.getPointer(), buffer_info.data_type);
93 // re-arrange results
94 r.unserialize(array);
95
96 auto t = buffer_info.data_type;
97 auto c = t.getClass();
98
99 if (c == DataTypeClass::VarLen || t.isVariableStr()) {
100#if H5_VERSION_GE(1, 12, 0)
101 // This one have been created in 1.12.0
102 (void) detail::h5t_reclaim(t.getId(), mem_space.getId(), H5P_DEFAULT, r.getPointer());
103#else
104 // This one is deprecated since 1.12.0
105 (void) detail::h5d_vlen_reclaim(t.getId(), mem_space.getId(), H5P_DEFAULT, r.getPointer());
106#endif
107 }
108}
109
110template <typename T>
111inline void Attribute::read_raw(T* array, const DataType& mem_datatype) const {
112 static_assert(!std::is_const<T>::value,
113 "read() requires a non-const structure to read data into");
114
115 detail::h5a_read(getId(), mem_datatype.getId(), static_cast<void*>(array));
116}
117
118template <typename T>
119inline void Attribute::read_raw(T* array) const {
120 using element_type = typename details::inspector<T>::base_type;
121 const DataType& mem_datatype = create_and_check_datatype<element_type>();
122
123 read_raw(array, mem_datatype);
124}
125
126template <typename T>
127inline void Attribute::write(const T& buffer) {
128 const DataSpace& mem_space = getMemSpace();
129 auto dims = mem_space.getDimensions();
130
131 if (mem_space.getElementCount() == 0) {
132 return;
133 }
134
135 auto file_datatype = getDataType();
136
137 const details::BufferInfo<T> buffer_info(
138 file_datatype,
139 [this]() -> std::string { return this->getName(); },
140 details::BufferInfo<T>::Operation::write);
141
142 if (!details::checkDimensions(mem_space, buffer_info.getMinRank(), buffer_info.getMaxRank())) {
143 std::ostringstream ss;
144 ss << "Impossible to write attribute of dimensions " << mem_space.getNumberDimensions()
145 << " into arrays of dimensions: " << buffer_info.getMinRank() << "(min) to "
146 << buffer_info.getMaxRank() << "(max)";
147 throw DataSpaceException(ss.str());
148 }
149 auto w = details::data_converter::serialize<T>(buffer, dims, file_datatype);
150 write_raw(w.getPointer(), buffer_info.data_type);
151}
152
153template <typename T>
154inline void Attribute::write_raw(const T* buffer, const DataType& mem_datatype) {
155 detail::h5a_write(getId(), mem_datatype.getId(), buffer);
156}
157
158template <typename T>
159inline void Attribute::write_raw(const T* buffer) {
160 using element_type = typename details::inspector<T>::base_type;
161 const auto& mem_datatype = create_and_check_datatype<element_type>();
162
163 write_raw(buffer, mem_datatype);
164}
165
166inline Attribute Attribute::squeezeMemSpace(const std::vector<size_t>& axes) const {
167 auto mem_dims = this->getMemSpace().getDimensions();
168 auto squeezed_dims = detail::squeeze(mem_dims, axes);
169
170 auto attr = *this;
171 attr._mem_space = DataSpace(squeezed_dims);
172 return attr;
173}
174
175inline Attribute Attribute::reshapeMemSpace(const std::vector<size_t>& new_dims) const {
176 detail::assert_compatible_spaces(this->getMemSpace(), new_dims);
177
178 auto attr = *this;
179 attr._mem_space = DataSpace(new_dims);
180 return attr;
181}
182
183} // namespace HighFive
Exception specific to HighFive Attribute interface.
Definition H5Exception.hpp:121
Class representing an Attribute of a DataSet or Group.
Definition H5Attribute.hpp:47
DataSpace getSpace() const
Get a copy of the DataSpace of the current Attribute.
Definition H5Attribute_misc.hpp:51
DataType getDataType() const
Get the DataType of the Attribute.
Definition H5Attribute_misc.hpp:45
void read_raw(T *array, const DataType &mem_datatype) const
Read the attribute into a pre-allocated buffer.
Definition H5Attribute_misc.hpp:111
std::string getName() const
Get the name of the current Attribute.
Definition H5Attribute_misc.hpp:32
T read() const
Get the value of the Attribute.
Definition H5Attribute_misc.hpp:62
Attribute reshapeMemSpace(const std::vector< size_t > &dims) const
Return a Attribute with a simple memspace with dims.
Definition H5Attribute_misc.hpp:175
DataSpace getMemSpace() const
Get the memory DataSpace of the current Attribute.
Definition H5Attribute_misc.hpp:57
void write(const T &value)
Write the value into the Attribute.
Definition H5Attribute_misc.hpp:127
Attribute squeezeMemSpace(const std::vector< size_t > &axes) const
Return an Attribute with axes squeezed from the memspace.
Definition H5Attribute_misc.hpp:166
void write_raw(const T *buffer, const DataType &mem_datatype)
Write from a raw pointer.
Definition H5Attribute_misc.hpp:154
size_t getStorageSize() const
The number of bytes required to store the attribute in the HDF5 file.
Definition H5Attribute_misc.hpp:37
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
size_t getElementCount() const
Return the number of elements in this DataSpace.
Definition H5Dataspace_misc.hpp:112
std::vector< size_t > getDimensions() const
Returns the size of the dataset in each dimension.
Definition H5Dataspace_misc.hpp:104
HDF5 Data Type.
Definition H5DataType.hpp:61
hid_t getId() const noexcept
getId
Definition H5Object_misc.hpp:63
bool isValid() const noexcept
isValid
Definition H5Object_misc.hpp:59
hid_t _hid
Definition H5Object.hpp:86
Definition assert_compatible_spaces.hpp:15