44 std::vector<T> res(in.size());
45 for (
size_t i = 0; i < in.size(); ++i) res[i] = gsl::narrow<T>(in[i]);
52 : var_(d), container_(container) {
63 H5I_type_t typ = H5Iget_type(
var_());
65 return (typ == H5I_DATASET);
87 std::vector<hsize_t> dims, dimsmax;
88 htri_t isSimple = H5Sis_simple(
space()());
89 if (isSimple < 0)
throw Exception(
"Dimension space parameter is invalid.",
ioda_Here(), errOpts);
90 if (isSimple == 0)
throw Exception(
"Dataspace is not simple. Unsupported case in code. "
91 "Complex dataspace support was not available in HDF5 when this function was written.",
93 hssize_t numPoints = H5Sget_simple_extent_npoints(
space()());
94 errOpts.
add(
"numPoints", numPoints);
97 int dimensionality = H5Sget_simple_extent_ndims(
space()());
98 errOpts.
add(
"dimensionality", dimensionality);
99 if (dimensionality < 0)
101 dims.resize(dimensionality);
102 dimsmax.resize(dimensionality);
103 if (H5Sget_simple_extent_dims(
space()(), dims.data(), dimsmax.data()) < 0)
106 ret.
numElements = gsl::narrow<decltype(Dimensions::numElements)>(numPoints);
107 ret.
dimensionality = gsl::narrow<decltype(Dimensions::dimensionality)>(dimensionality);
108 for (
const auto& d : dims) ret.
dimsCur.push_back(gsl::narrow<Dimensions_t>(d));
109 for (
const auto& d : dimsmax)
116 std::vector<hsize_t> hdims = convertToH5Length<hsize_t>(
newDims);
118 if (H5Dset_extent(
var_(), hdims.data()) < 0)
121 .
add(
"dimensionality", hdims.size());
123 return Variable{shared_from_this()};
129 errOpts.
add(
"DimensionNumber", DimensionNumber);
133 auto scaleBackendBase = scale.
get();
137 auto scaleBackendDerived = std::dynamic_pointer_cast<HH_Variable>(scaleBackendBase);
140 const herr_t res = H5DSattach_scale(
var_(), scaleBackendDerived->var_(), DimensionNumber);
141 if (res != 0)
throw Exception(
"Dimension scale attachment failed.",
ioda_Here(), errOpts);
143 return Variable{shared_from_this()};
144 }
catch (std::bad_cast) {
145 throw Exception(
"Cannot attach dimension scales across incompatible backends.",
153 errOpts.
add(
"DimensionNumber", DimensionNumber);
157 auto scaleBackendBase = scale.
get();
161 auto scaleBackendDerived = std::dynamic_pointer_cast<HH_Variable>(scaleBackendBase);
164 const herr_t res = H5DSdetach_scale(
var_(), scaleBackendDerived->var_(), DimensionNumber);
165 if (res != 0)
throw Exception(
"Dimension scale detachment failed",
ioda_Here(), errOpts);
167 return Variable{shared_from_this()};
168 }
catch (std::bad_cast) {
169 throw Exception(
"Cannot detach dimension scales across incompatible backends.",
175 const htri_t res = H5DSis_scale(
var_());
181 errOpts.
add(
"variable",
"unknown / bad id");
190 const htri_t res = H5DSset_scale(
var_(), dimensionScaleName.c_str());
193 errOpts.
add(
"dimensionScaleName", dimensionScaleName);
197 errOpts.
add(
"variable",
"unknown / bad id");
201 "Error returned from H5DSset_scale.",
ioda_Here(), errOpts);
203 return Variable{shared_from_this()};
207 constexpr
size_t max_label_size = 1000;
208 std::array<char, max_label_size> label{};
209 const ssize_t sz = H5DSget_scale_name(
var_(), label.data(), max_label_size);
215 errOpts.
add(
"variable",
"unknown / bad id");
222 label[max_label_size - 1] =
'\0';
223 res = std::string(label.data());
224 return Variable{std::make_shared<HH_Variable>(*
this)};
246 const std::vector<Named_Variable>& scalesToQueryAgainst,
bool firstOnly,
247 const std::vector<unsigned>& dimensionNumbers_)
const {
252 std::vector<std::pair<std::string, std::shared_ptr<HH_Variable>>> scales;
253 for (
const auto& scale : scalesToQueryAgainst) {
254 auto scaleBackendBase = scale.var.get();
255 auto scaleBackendDerived = std::dynamic_pointer_cast<HH_Variable>(scaleBackendBase);
257 {scale.name, scaleBackendDerived});
266 std::vector<unsigned> dimensionNumbers = dimensionNumbers_;
267 if (!dimensionNumbers.empty()) {
268 auto max_elem_it = std::max_element(dimensionNumbers.cbegin(), dimensionNumbers.cend());
272 dimensionNumbers.resize(datadims.dimensionality);
273 std::iota(dimensionNumbers.begin(), dimensionNumbers.end(), 0);
277 std::vector<std::vector<Named_Variable>> ret(
278 gsl::narrow<size_t>(datadims.dimensionality));
294 search_data_opts.
search_for =
"DIMENSION_LIST";
295 herr_t att_search_ret
296 = H5Aiterate2(
get()(),
301 reinterpret_cast<void*
>(&search_data_opts)
305 if (!search_data_opts.
success)
return ret;
307 hid_t found_att = H5Aopen_by_idx(
get()(),
".", iteration_type, H5_ITER_NATIVE,
308 search_data_opts.
idx, H5P_DEFAULT, H5P_DEFAULT);
323 Vlen_data buf((
size_t)datadims.dimensionality, vltyp, aDims_HH.
space());
325 if (H5Aread(aDims_HH.
get()(), vltyp.
get(),
reinterpret_cast<void*
>(buf.
buf.get())) < 0)
332 #if H5_VERSION_GE(1, 12, 0)
333 std::vector<H5O_info1_t> scale_infos(scales.size());
334 H5O_info1_t check_info;
336 std::vector<H5O_info_t> scale_infos(scales.size());
337 H5O_info_t check_info;
339 for (
size_t i = 0; i < scales.size(); ++i) {
340 #if H5_VERSION_GE(1, 10, 3)
341 if (H5Oget_info2(scales[i].second->get()(), &scale_infos[i], H5O_INFO_BASIC) < 0)
344 if (H5Oget_info(scales[i].second->get()(), &scale_infos[i], H5O_INFO_BASIC) < 0)
354 for (
const auto& curDim : dimensionNumbers) {
357 for (
size_t i = 0; i < buf.
buf[curDim].len; ++i) {
358 hobj_ref_t ref = ((hobj_ref_t*)buf.
buf[curDim].p)[i];
362 hid_t deref_scale_id = H5Rdereference2(
364 get()(), H5P_DEFAULT, H5R_OBJECT, &ref);
365 Expects(deref_scale_id >= 0);
370 #if H5_VERSION_GE(1, 10, 3)
371 if (H5Oget_info2(deref_scale.
get()(), &check_info, H5O_INFO_BASIC) < 0)
374 if (H5Oget_info(deref_scale.
get()(), &check_info, H5O_INFO_BASIC) < 0)
380 bool foundScale =
false;
381 for (
size_t j = 0; j < scale_infos.size(); ++j) {
382 if ((scale_infos[j].fileno == check_info.fileno)
383 && (scale_infos[j].addr == check_info.addr)) {
385 ret[curDim].push_back(scalesToQueryAgainst[j]);
393 if (firstOnly && foundScale)
break;
403 errOpts.
add(
"variable",
"unknown / bad id");
413 const std::vector<Named_Variable> scalesToQueryAgainst{{
"unused_param", scale}};
415 return !res[DimensionNumber].empty();
421 errOpts.
add(
"variable",
"unknown / bad id");
429 const std::list<Named_Variable>& scalesToQueryAgainst,
bool firstOnly)
const {
435 auto res = std::make_shared<HH_Selection>();
445 auto csel = std::dynamic_pointer_cast<HH_Selection>(concretized);
447 }
catch (std::bad_cast) {
458 if (!sel.
extent().empty()) {
459 if (H5Sset_extent_simple(spc(), gsl::narrow<int>(sel.
extent().size()),
460 convertToH5Length<hsize_t>(sel.
extent()).data(),
461 convertToH5Length<hsize_t>(sel.
extent()).data())
467 if (H5Sselect_all(spc()) < 0)
throw Exception(
"Dataspace selection failed.",
ioda_Here());
469 if (H5Sselect_none(spc()) < 0)
throw Exception(
"Dataspace selection failed.",
ioda_Here());
472 static const std::map<SelectionOperator, H5S_seloper_t> op_map
481 bool first_action =
true;
486 if (!s.points_.empty()) {
488 size_t dimensionality = s.points_.at(0).size();
490 std::vector<hsize_t> elems(dimensionality * s.points_.size());
493 for (
size_t i = 0; i < s.points_.size(); ++i)
495 if (s.points_[i].size() != dimensionality)
497 .
add(
"dimensionality", dimensionality)
498 .
add(
"s.points_[i].size()", s.points_[i].size())
500 for (
size_t j = 0; j < dimensionality; ++j)
501 elems[j + (dimensionality * i)] = s.points_[i][j];
505 chk = H5Sselect_elements(spc(), op_map.at(s.op_), s.points_.size(), elems.data());
506 }
else if (!s.dimension_indices_starts_.empty()) {
514 #if H5_VERSION_GE(1, 12, 0)
516 if(H5Sselect_none(cloned_space()) < 0)
throw Exception(
"Cannot copy space",
ioda_Here());
520 const size_t numSlabs = s.dimension_indices_starts_.size();
521 for (
size_t i = 0; i < numSlabs; ++i) {
523 std::vector<hsize_t> hstart;
524 if (sel.
extent().empty()) {
527 hstart.resize((
size_t)sel.
extent().size(), 0);
529 hstart[s.dimension_] = s.dimension_indices_starts_[i];
532 std::vector<hsize_t> hcount;
533 if (sel.
extent().empty()) {
534 hcount = convertToH5Length<hsize_t>(dims.
dimsCur);
536 hcount = convertToH5Length<hsize_t>(sel.
extent());
539 = (i < s.dimension_indices_counts_.size()) ? s.dimension_indices_counts_[i] : 1;
542 NULL, hcount.data(), NULL)
553 if (H5Sselect_copy(spc(), cloned_space()) < 0)
556 if (H5Smodify_select(spc(), op_map.at(s.op_), cloned_space()) < 0)
561 "The HDF5 engine needs to be backed by at least "
562 "HDF5 1.12.0 to do the requested selection properly. Older HDF5 versions "
563 "do not have the H5Smodify_select function.",
ioda_Here());
567 const auto hstart = convertToH5Length<hsize_t>(s.start_);
568 const auto hstride = convertToH5Length<hsize_t>(s.stride_);
569 const auto hcount = convertToH5Length<hsize_t>(s.count_);
570 const auto hblock = convertToH5Length<hsize_t>(s.block_);
572 chk = H5Sselect_hyperslab(spc(), op_map.at(s.op_), hstart.data(),
573 (s.stride_.size()) ? hstride.data() : NULL, hcount.data(),
574 (s.block_.size()) ? hblock.data() : NULL);
577 first_action =
false;
582 if (H5Soffset_simple(spc(), convertToH5Length<hssize_t>(sel.
getOffset()).data()) < 0)
586 auto res = std::make_shared<HH_Selection>();
594 auto typeBackend = std::dynamic_pointer_cast<HH_Type>(in_memory_dataType.
getBackend());
597 auto ret = H5Dwrite(
var_(),
598 typeBackend->handle(),
605 return Variable{shared_from_this()};
621 if (cls_my == H5T_STRING) {
633 if (H5Tis_variable_str(file_type.get()) > 0) {
638 auto ret = H5Dread(
var_(),
655 hssize_t sz = (memSpace.get() == H5S_ALL)
658 H5S_sel_type st = H5Sget_select_type(memSpace.get());
659 return (st == H5S_SEL_NONE) ? 0 : H5Sget_select_npoints(memSpace.get());
664 std::vector<char> tmp_buf(gsl::narrow<size_t>(sz));
665 auto ret = H5Dread(
var_(),
683 const size_t sz_each_str = H5Tget_size(file_type.get());
684 const size_t num_strs = tmp_buf.size() / sz_each_str;
689 char** reint_buf =
reinterpret_cast<char**
>(data.data());
690 for (
size_t i = 0; i < num_strs; ++i) {
691 std::string s(tmp_buf.data() + (sz_each_str * i), sz_each_str);
693 = (
char*)malloc(sz_each_str + 1);
699 auto typeBackend = std::dynamic_pointer_cast<HH_Type>(in_memory_dataType.
getBackend());
700 auto ret = H5Dread(
var_(),
701 typeBackend->handle(),
710 return Variable{std::make_shared<HH_Variable>(*
this)};
714 auto typeBackend = std::dynamic_pointer_cast<HH_Type>(lhs.
getBackend());
725 H5T_class_t cls_lhs = H5Tget_class(typeBackend->handle.get());
727 if (cls_lhs == H5T_STRING && cls_my == H5T_STRING)
return true;
732 if (cls_lhs != cls_my)
return false;
735 if (H5Tget_size(typeBackend->handle.get()) != H5Tget_size(
internalType()()))
return false;
738 if (cls_lhs == H5T_INTEGER)
739 if (H5Tget_sign(typeBackend->handle.get()) != H5Tget_sign(
internalType()()))
return false;
753 auto ret = H5Tequal(ttype(), otype());
755 return (ret > 0) ? true :
false;
759 H5D_fill_value_t fvstatus;
762 return (fvstatus != H5D_FILL_VALUE_UNDEFINED);
775 H5D_fill_value_t fvstatus;
779 res.
set_ = (fvstatus != H5D_FILL_VALUE_UNDEFINED);
789 auto fvp =
container_.lock()->getFillValuePolicy();
795 if (isA<std::string>())
796 assignFillValue<std::string>(res, FillValuePolicies::netCDF4_default<std::string>());
797 else if (isA<signed char>())
799 else if (isA<char>())
801 else if (isA<int16_t>())
803 else if (isA<int32_t>())
805 else if (isA<float>())
807 else if (isA<double>())
809 else if (isA<unsigned char>())
811 else if (isA<uint16_t>())
813 else if (isA<uint32_t>())
815 else if (isA<int64_t>())
817 else if (isA<uint64_t>())
820 assignFillValue<uint64_t>(res, 0);
827 H5T_class_t cls = H5Tget_class(hType());
829 const std::set<H5T_class_t> supported{H5T_INTEGER, H5T_FLOAT, H5T_STRING};
832 if (!supported.count(cls))
834 "HH's getFillValue function only supports "
835 "basic numeric and string data types. Any other types "
836 "will require enhancement to FillValueData_t::FillValueUnion_t.",
839 size_t szType_inBytes = H5Tget_size(hType());
843 std::vector<char> fvbuf(szType_inBytes, 0);
844 if (H5Pget_fill_value(create_plist.
get(), hType(),
845 reinterpret_cast<void*
>(fvbuf.data())) < 0)
853 if (cls == H5T_STRING) {
855 htri_t str_type = H5Tis_variable_str(hType());
859 const char** ccp = (
const char**)fvbuf.data();
865 if (H5free_memory(
const_cast<void*
>(
reinterpret_cast<const void*
>(ccp[0]))) < 0)
875 "The fill value in HDF5 is too large for the "
876 "fillValue_ union. ioda-engines currently only supports fill "
877 "values on fundamental types and strings.",
879 .
add(
"szType_inBytes", szType_inBytes)
899 H5D_layout_t layout = H5Pget_layout(create_plist.
get());
900 if (layout == H5D_CHUNKED) {
902 std::vector<hsize_t>
chunks(max_ndims);
903 if (H5Pget_chunk(create_plist.
get(), max_ndims,
chunks.data()) < 0)
905 std::vector<Dimensions_t> res;
906 res.reserve(
chunks.size());
907 for (
const auto& i :
chunks)
908 res.emplace_back(gsl::narrow<Dimensions_t>(
922 int nfilters = H5Pget_nfilters(create_plist.
get());
925 for (
unsigned i = 0; i < (unsigned)nfilters; ++i) {
930 const size_t cd_nelems_init = 16;
931 size_t cd_nelems = cd_nelems_init;
933 std::vector<unsigned> cd_values(cd_nelems_init);
934 const size_t namelen = 32;
935 std::vector<char>
name(namelen);
936 unsigned filter_config = 0;
938 H5Z_filter_t filt = H5Pget_filter2(create_plist.
get(), i, &flags, &cd_nelems, cd_values.data(),
939 namelen,
name.data(), &filter_config);
940 if (filt != H5Z_FILTER_DEFLATE)
continue;
944 return std::pair<bool, int>(
true, gsl::narrow<int>(cd_values[0]));
948 return std::pair<bool, int>(
false, 0);
958 int nfilters = H5Pget_nfilters(create_plist.
get());
961 for (
unsigned i = 0; i < (unsigned)nfilters; ++i) {
966 const size_t cd_nelems_init = 16;
967 size_t cd_nelems = cd_nelems_init;
969 std::vector<unsigned> cd_values(cd_nelems_init);
970 const size_t namelen = 32;
971 std::vector<char>
name(namelen);
972 unsigned filter_config = 0;
974 H5Z_filter_t filt = H5Pget_filter2(create_plist.
get(), i, &flags, &cd_nelems, cd_values.data(),
975 namelen,
name.data(), &filter_config);
976 if (filt != H5Z_FILTER_SZIP)
continue;
981 return std::tuple<bool, unsigned, unsigned>(
true, cd_values[0], cd_values[1]);
985 return std::tuple<bool, unsigned, unsigned>(
false, 0, 0);
1001 if (chunkinfo.size()) {
1009 if (std::get<0>(sz)) res.
compressWithSZIP(std::get<1>(sz), std::get<2>(sz));
Convenience classes for constructing ObsSpaces and setting up new Dimension Scales.
Describe the dimensions of a ioda::Attribute or ioda::Variable.
HDF5 engine implementation of Attribute.
HDF5 engine implementation of Has_Attributes.
HDF5 engine implementation of Has_Variables.
HDF5 engine implementation of ioda::detail::Type_Provider.
Utility functions for HDF5.
HDF5 engine implementation of Variable.
HDF5 resource handles in C++.
The ioda exception class.
Exception & add(const std::string &key, const T value)
Add a key-value pair to the error message.
This class exists inside of ioda::Group or ioda::Variable and provides the interface to manipulating ...
Quick and easy key-value container that stringifies all values.
Options & add(const std::string &key, const T &value)
Adds an option. Throws if the same name already exists.
A Selection represents the bounds of the data, in ioda or in userspace, that you are reading or writi...
Represents the "type" (i.e. integer, string, float) of a piece of data.
This is the implementation of Attributes using HDF5.
Dimensions getDimensions() const final
Get Attribute's dimensions.
HH_hid_t internalType() const
Get HDF5-internal type.
static HH_Type_Provider * instance()
This is the implementation of ioda::Type using HDF5. Do not use outside of IODA.
This is the implementation of Variables using HDF5.
Variable read(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection, const Selection &file_selection) const final
Read the Variable - as char array. Ordering is row-major.
Variable detachDimensionScale(unsigned int DimensionNumber, const Variable &scale) final
Detach a dimension scale.
std::vector< Dimensions_t > getChunkSizes() const final
Retrieve the chunking options for the Variable.
std::vector< std::vector< Named_Variable > > getDimensionScaleMappings(const std::vector< Named_Variable > &scalesToQueryAgainst, bool firstOnly, const std::vector< unsigned > &dimensionNumbers) const
std::pair< bool, int > getGZIPCompression() const final
Retrieve the GZIP compression options for the Variable.
std::tuple< bool, unsigned, unsigned > getSZIPCompression() const final
Retrieve the SZIP compression options for the Variable.
FillValueData_t getFillValue() const final
Retrieve the fill value.
Variable write(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection, const Selection &file_selection) final
The fundamental write function. Backends overload this function to implement all write operations.
Variable resize(const std::vector< Dimensions_t > &newDims) final
Resize the variable.
VariableCreationParameters getCreationParameters(bool doAtts=true, bool doDims=true) const final
HDF5-specific, performance-focused implementation.
bool hasFillValue() const final
Check if a variable has a fill value set.
Type getType() const final
Get HDF5-internal type, wrapped as a ioda::Type object.
Variable setIsDimensionScale(const std::string &dimensionScaleName) final
Designate this table as a dimension scale.
Variable attachDimensionScale(unsigned int DimensionNumber, const Variable &scale) final
Attach a dimension scale to this Variable.
Selections::SelectionBackend_t instantiateSelection(const Selection &sel) const final
Convert a selection into its backend representation.
detail::Type_Provider * getTypeProvider() const final
Query the backend and get the type provider.
std::weak_ptr< const HH_HasVariables > container_
HH_hid_t getSpaceWithSelection(const Selection &sel) const
bool isDimensionScaleAttached(unsigned int DimensionNumber, const Variable &scale) const final
HDF5-specific, performance-focused implementation.
bool isDimensionScale() const final
Is this Variable used as a dimension scale?
Dimensions getDimensions() const final
HH_hid_t internalType() const
Get HDF5-internal type.
A class to wrap HDF5's hid_t resource handles.
std::shared_ptr< Type_Backend > getBackend() const
Backends implement type providers in conjunction with Attributes, Has_Attributes, Variables and Has_V...
Has_Attributes atts
Attributes.
std::string getDimensionScaleName() const
Get the name of this Variable's defined dimension scale.
std::shared_ptr< Variable_Backend > get() const
Gets a handle to the underlying object that implements the backend functionality.
std::shared_ptr< InstantiatedSelection > SelectionBackend_t
Selection & extent(const VecDimensions_t &sz)
Provide the dimensions of the object that you are selecting from.
bool isConcretized() const
Is the selection already cached in the backend?
const std::vector< SingleSelection > & getActions() const
void invalidate() const
Ditch the concretized selection.
SelectionState getDefault() const
const VecDimensions_t & getOffset() const
Selections::SelectionBackend_t concretize(const Variable &) const
Talk to the backend and generate the appropriate selection object.
@ NETCDF4
Use NetCDF4 default fill values. This is the default option for ioda files.
IODA_HIDDEN std::string getNameFromIdentifier(hid_t obj_id)
Gets a variable / group / link name from an id. Useful for debugging.
std::unique_ptr< hvl_t[]> buf
IODA_HIDDEN herr_t iterate_find_attr(hid_t loc_id, const char *name, void *op_data)
Callback function for H5Aiterate / H5Aiterate2 / H5Aiterate1.
IODA_HIDDEN H5_index_t getAttrCreationOrder(hid_t obj, H5O_type_t objType)
Determine attribute creation order for a dataset.
char netCDF4_default< char >()
uint32_t netCDF4_default< uint32_t >()
int16_t netCDF4_default< int16_t >()
uint16_t netCDF4_default< uint16_t >()
unsigned char netCDF4_default< unsigned char >()
double netCDF4_default< double >()
signed char netCDF4_default< signed char >()
int64_t netCDF4_default< int64_t >()
float netCDF4_default< float >()
uint64_t netCDF4_default< uint64_t >()
int32_t netCDF4_default< int32_t >()
std::vector< T > convertToH5Length(const std::vector< Dimensions_t > &in)
IODA_DL size_t COMPAT_strncpy_s(char *dest, size_t destSz, const char *src, size_t srcSz)
Safe char array copy.
constexpr int Unlimited
Specifies that a dimension is resizable to infinity.
Describes the dimensions of an Attribute or Variable.
std::vector< Dimensions_t > dimsCur
The dimensions of the data.
Dimensions_t dimensionality
The dimensionality (rank) of the data.
std::vector< Dimensions_t > dimsMax
This must always equal dimsCur for Attribute.
Used to specify Variable creation-time properties.
void compressWithGZIP(int level=6)
detail::FillValueData_t fillValue_
std::vector< Dimensions_t > chunks
Manually specify the chunks. Never directly use. Use getChunks(...) instead.
void compressWithSZIP(unsigned PixelsPerBlock=16, unsigned options=4)
bool chunk
Do we chunk this variable? Required for extendible / compressible Variables.
static void CloseP(hid_t *h)
static void CloseP(hid_t *h)
static void CloseP(hid_t *h)
static void CloseP(hid_t *h)
static void CloseP(hid_t *h)
Data to pass to/from iterator classes.
Internal structure to encapsulate resources and prevent leaks.
Container used to store and manipulate fill values.
std::string stringFillValue_
union ioda::detail::FillValueData_t::FillValueUnion_t fillValue_