HighFive 3.0.0
HighFive - Header-only C++ HDF5 interface
Loading...
Searching...
No Matches
H5ReadWrite_misc.hpp
Go to the documentation of this file.
1/*
2 * Copyright (c) 2020 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#pragma once
10
11#include <H5Tpublic.h>
12#include "H5Inspector_misc.hpp"
13#include "H5Utils.hpp"
14
15namespace HighFive {
16
17namespace details {
18
19template <typename T>
20using unqualified_t = typename std::remove_const<typename std::remove_reference<T>::type>::type;
21
22// Find the type of an eventual char array, otherwise void
23template <typename T>
24struct type_char_array {
25 using type = typename std::conditional<
26 std::is_same<typename inspector<T>::base_type, std::string>::value,
27 std::string,
28 void>::type;
29 static constexpr bool is_char_array = false;
30};
31
32template <typename T>
33struct type_char_array<T*> {
34 using type = typename std::conditional<std::is_same<unqualified_t<T>, char>::value,
35 char*,
36 typename type_char_array<T>::type>::type;
37 static constexpr bool is_char_array = true;
38};
39
40template <typename T, std::size_t N>
41struct type_char_array<T[N]> {
42 using type = typename std::conditional<std::is_same<unqualified_t<T>, char>::value,
43 char[N],
44 typename type_char_array<T>::type>::type;
45 static constexpr bool is_char_array = true;
46};
47
48template <typename T>
49struct BufferInfo {
50 using type_no_const = typename std::remove_const<T>::type;
51 using elem_type = typename details::inspector<type_no_const>::base_type;
52 using char_array_t = typename details::type_char_array<type_no_const>::type;
53 static constexpr bool is_char_array = details::type_char_array<type_no_const>::is_char_array;
54
55 enum class Operation { read, write };
56 const Operation op;
57
58 template <class F>
59 BufferInfo(const DataType& dtype, F getName, Operation _op);
60
61 size_t getRank(const T& array) const;
62 size_t getMinRank() const;
63 size_t getMaxRank() const;
64
65 // member data for info depending on the destination dataset type
66 const bool is_fixed_len_string;
67 const DataType data_type;
68 const size_t rank_correction;
69};
70
71// details implementation
72template <typename SrcStrT>
73struct string_type_checker {
74 static DataType getDataType(const DataType&, const DataType&);
75};
76
77inline void enforce_ascii_hack(const DataType& dst, const DataType& src) {
78 // Note: constness only refers to constness of the DataType object, which
79 // is just an ID, we can/will change properties of `dst`.
80
81 // TEMP. CHANGE: Ensure that the character set is properly configured to prevent
82 // converter issues on HDF5 <=v1.12.0 when loading ASCII strings first.
83 // See https://github.com/HDFGroup/hdf5/issues/544 for further information.
84
85 bool is_dst_string = detail::h5t_get_class(dst.getId()) == H5T_STRING;
86 bool is_src_string = detail::h5t_get_class(src.getId()) == H5T_STRING;
87
88 if (is_dst_string && is_src_string) {
89 if (detail::h5t_get_cset(src.getId()) == H5T_CSET_ASCII) {
90 detail::h5t_set_cset(dst.getId(), H5T_CSET_ASCII);
91 }
92 }
93}
94
95template <>
96struct string_type_checker<void> {
97 inline static DataType getDataType(const DataType& element_type, const DataType& dtype) {
98 if (detail::h5t_get_class(element_type.getId()) == H5T_STRING) {
99 enforce_ascii_hack(element_type, dtype);
100 }
101 return element_type;
102 }
103};
104
105template <>
106struct string_type_checker<std::string> {
107 inline static DataType getDataType(const DataType&, const DataType& file_datatype) {
108 // The StringBuffer ensures that the data is transformed such that it
109 // matches the datatype of the dataset, i.e. `file_datatype` and
110 // `mem_datatype` are the same.
111 return file_datatype;
112 }
113};
114
115template <std::size_t FixedLen>
116struct string_type_checker<char[FixedLen]> {
117 inline static DataType getDataType(const DataType& element_type, const DataType& dtype) {
118 DataType return_type = (dtype.isFixedLenStr()) ? AtomicType<char[FixedLen]>()
119 : element_type;
120 enforce_ascii_hack(return_type, dtype);
121 return return_type;
122 }
123};
124
125template <>
126struct string_type_checker<char*> {
127 inline static DataType getDataType(const DataType&, const DataType& dtype) {
128 if (dtype.isFixedLenStr()) {
129 throw DataSetException("Can't output variable-length to fixed-length strings");
130 }
131 DataType return_type = AtomicType<std::string>();
132 enforce_ascii_hack(return_type, dtype);
133 return return_type;
134 }
135};
136
137template <typename T>
138template <class F>
139BufferInfo<T>::BufferInfo(const DataType& file_data_type, F getName, Operation _op)
140 : op(_op)
141 , is_fixed_len_string(file_data_type.isFixedLenStr())
142 // In case we are using Fixed-len strings we need to subtract one dimension
143 , data_type(string_type_checker<char_array_t>::getDataType(create_datatype<elem_type>(),
144 file_data_type))
145 , rank_correction((is_fixed_len_string && is_char_array) ? 1 : 0) {
146 // We warn. In case they are really not convertible an exception will rise on read/write
147 if (file_data_type.getClass() != data_type.getClass()) {
148 HIGHFIVE_LOG_WARN(getName() + "\": data and hdf5 dataset have different types: " +
149 data_type.string() + " -> " + file_data_type.string());
150 } else if ((file_data_type.getClass() & data_type.getClass()) == DataTypeClass::Float) {
152 (op == Operation::read) && (file_data_type.getSize() > data_type.getSize()),
153 getName() + "\": hdf5 dataset has higher floating point precision than data on read: " +
154 file_data_type.string() + " -> " + data_type.string());
155
157 (op == Operation::write) && (file_data_type.getSize() < data_type.getSize()),
158 getName() +
159 "\": data has higher floating point precision than hdf5 dataset on write: " +
160 data_type.string() + " -> " + file_data_type.string());
161 }
162}
163
164template <typename T>
165size_t BufferInfo<T>::getRank(const T& array) const {
166 return details::inspector<type_no_const>::getRank(array) - rank_correction;
167}
168
169template <typename T>
170size_t BufferInfo<T>::getMinRank() const {
171 return details::inspector<T>::min_ndim - rank_correction;
172}
173
174template <typename T>
175size_t BufferInfo<T>::getMaxRank() const {
176 return details::inspector<T>::max_ndim - rank_correction;
177}
178
179} // namespace details
180
181} // namespace HighFive
#define HIGHFIVE_LOG_WARN_IF(cond, message)
Definition H5Utility.hpp:193
#define HIGHFIVE_LOG_WARN(message)
Definition H5Utility.hpp:189
Definition assert_compatible_spaces.hpp:15
DataType create_datatype()
Create a DataType instance representing type T.
Definition H5DataType_misc.hpp:427
typename std::remove_const< typename std::remove_reference< T >::type >::type unqualified_t
Definition H5Inspector_decl.hpp:8