IODA
HH-util.h
Go to the documentation of this file.
1 #pragma once
2 /*
3  * (C) Copyright 2021 UCAR
4  *
5  * This software is licensed under the terms of the Apache Licence Version 2.0
6  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7  */
8 /*! \addtogroup ioda_internals_engines_hh
9  *
10  * @{
11  * \file HH-util.h
12  * \brief Utility functions for HDF5.
13  */
14 
15 #include <string>
16 #include <utility>
17 #include <vector>
18 
19 #include <hdf5.h>
20 
21 #include "./Handles.h"
22 #include "ioda/defs.h"
23 #include "ioda/Exception.h"
24 
25 namespace ioda {
26 namespace detail {
27 namespace Engines {
28 namespace HH {
29 class HH_Attribute;
30 class HH_Variable;
31 
32 //#if H5_VERSION_GE(1, 12, 0)
33 //typedef H5R_ref_t ref_t;
34 //#else
35 typedef hobj_ref_t ref_t;
36 //#endif
37 
38 // brief Conveys information that a variable (or scale) is attached along a specified axis.
39 //typedef std::pair<ref_t, unsigned> ref_axis_t;
40 
41 /// @brief Duplicate the HDF5 dataset list structure for REFERENCE_LISTs.
42 struct ds_list_t {
43  hobj_ref_t ref; /* object reference */
44  unsigned int dim_idx; /* dimension index of the dataset */
45 };
46 
47 /// Data to pass to/from iterator classes.
49  std::string search_for;
50  hsize_t idx = 0;
51  bool success = false;
52 };
53 
54 /// Callback function for H5Aiterate / H5Aiterate2 / H5Aiterate1.
55 #if H5_VERSION_GE(1, 8, 0)
56 IODA_HIDDEN herr_t iterate_find_attr(hid_t loc_id, const char* name, const H5A_info_t* info,
57  void* op_data);
58 #else
59 IODA_HIDDEN herr_t iterate_find_attr(hid_t loc_id, const char* name, void* op_data);
60 #endif
61 
62 /*! @brief Determine attribute creation order for a dataset.
63  * @param obj is the dataset being queried
64  * @param objType is the type of object (Dataset or Group).
65  * @return H5_INDEX_CRT_ORDER if creation order is tracked.
66  * @return H5_INDEX_NAME if creation order is not tracked.
67  * @details Check if the variable has attribute creation order stored and/or indexed.
68  * This is not the default, but it can speed up object list accesses considerably.
69  */
70 IODA_HIDDEN H5_index_t getAttrCreationOrder(hid_t obj, H5O_type_t objType);
71 
72 /// @brief Linear search to find an attribute.
73 /// @param baseObject is the object that could contain the attribute.
74 /// @param attname is the name of the attribute.
75 /// @param iteration_type is the type of iteration for the search. See getAttrCreationOrder.
76 /// @return A pair of (success_flag, index).
77 IODA_HIDDEN std::pair<bool, hsize_t> iterativeAttributeSearch(hid_t baseObject,
78  const char* attname,
79  H5_index_t iteration_type);
80 
81 /// @brief Linear search to find and open an attribute, if it exists.
82 /// @param baseObject is the object that could contain the attribute.
83 /// @param objType is the type of object (Dataset or Group).
84 /// @param attname is the name of the attribute.
85 /// @return An open handle to the attribute, if it exists.
86 /// @return An invalid handle if the attribute does not exist or upon general failure.
87 /// @details This function is useful because it is faster than the regular attribute
88 /// open by name routine, which does not take advantage of attribute creation order
89 /// indexing. Performance is particularly good when there are few attributes attached
90 /// to the base object.
91 IODA_HIDDEN HH_Attribute iterativeAttributeSearchAndOpen(hid_t baseObject, H5O_type_t objType,
92  const char* attname);
93 
94 /*! @brief Attribute DIMENSION_LIST update function
95 *
96 * @details This function exists to update DIMENSION_LISTs without updating the
97 * mirrored REFERENCE_LIST entry in the variable's scales. This is done for
98 * performance reasons, as attaching dimension scales for hundreds of
99 * variables sequentially is very slow.
100 *
101 * NOTE: This code does not use the regular atts.open(...) call
102 * for performance reasons when we have to repeat this call for hundreds or
103 * thousands of variables. We instead do a creation-order-preferred search.
104 *
105 * @param var is the variable of interest.
106 * @param new_dim_list is the mapping of dimensions that should be added to the variable.
107 */
109  const std::vector<std::vector<ref_t>>& new_dim_list);
110 
111 /*! @brief Attribute REFERENCE_LIST update function
112 *
113 * @details This function exists to update REFERENCE_LISTs without updating the
114 * mirrored DIMENSION_LIST entry. This is done for
115 * performance reasons, as attaching dimension scales for hundreds of
116 * variables sequentially is very slow.
117 *
118 * NOTE: This code does not use the regular atts.open(...) call
119 * for performance reasons when we have to repeat this call for hundreds or
120 * thousands of variables. We instead do a creation-order-preferred search.
121 *
122 * @param scale is the scale of interest.
123 * @param ref_var_axis_list is the mapping of variables-dimension numbers that
124 * should be added to the scale's REFERENCE_LIST attribute.
125 */
127  const std::vector<ds_list_t>& ref_var_axis);
128 
129 
130 
131 /// @brief A "view" of hvl_t objects. Adds C++ conveniences to an otherwise troublesome class.
132 /// @tparam Inner is the datatype that we are really manipulating.
133 template <class Inner>
135  hvl_t &obj;
136  View_hvl_t(hvl_t& obj) : obj{obj} {}
137  size_t size() const { return obj.len; }
138  void resize(size_t newlen) {
139  if (newlen) {
140  obj.p = (obj.p) ? H5resize_memory(obj.p, newlen * sizeof(Inner))
141  : H5allocate_memory(newlen * sizeof(Inner), false);
142  if (!obj.p) throw Exception("Failed to allocate memory", ioda_Here());
143  }
144  else {
145  if (obj.p)
146  if (H5free_memory(obj.p) < 0) throw Exception("Failed to free memory", ioda_Here());
147  obj.p = nullptr;
148  }
149  obj.len = newlen;
150  }
151  void clear() { resize(0); }
152  Inner* at(size_t i) {
153  if (i >= obj.len) throw Exception("Out-of-bounds access", ioda_Here())
154  .add("i", i).add("obj.len", obj.len);
155  return operator[](i);
156  }
157  Inner* operator[](size_t i) {
158  return &(static_cast<Inner*>(obj.p)[i]);
159  }
160 };
161 
162 /*! Internal structure to encapsulate resources and prevent leaks.
163 *
164 * When reading dimension scales, calls to H5Aread return variable-length arrays
165 * that must be reclaimed. We use a custom object to encapsulate this.
166 */
168  std::unique_ptr<hvl_t[]> buf; // NOLINT: C array and visibility warnings.
169  HH_hid_t typ, space; // NOLINT: C visibility warnings.
170  size_t sz;
171  Vlen_data(size_t sz, HH_hid_t typ, HH_hid_t space)
172  : buf(new hvl_t[sz]), typ{typ}, space{space}, sz{sz} {
173  if (!buf) throw Exception("Failed to allocate buf", ioda_Here());
174  for (size_t i = 0; i < sz; i++) {
175  buf[i].len = 0;
176  buf[i].p = nullptr;
177  }
178  }
180  if (buf) {
181  /*
182  for (size_t i = 0; i < sz; i++) {
183  if (buf[i].p) delete[] buf[i].p;
184  buf[i].len = 0;
185  buf[i].p = nullptr;
186  }
187  */
188 
189  H5Dvlen_reclaim(typ.get(), space.get(), H5P_DEFAULT, reinterpret_cast<void*>(buf.get()));
190  }
191  }
192  Vlen_data(const Vlen_data&) = delete;
193  Vlen_data(Vlen_data&&) = delete;
194  Vlen_data operator=(const Vlen_data&) = delete;
196 
197  hvl_t& operator[](size_t idx) { return (buf).get()[idx]; }
198 };
199 
200 /// @brief Gets a variable / group / link name from an id. Useful for debugging.
201 /// @param obj_id is the object.
202 /// @return One of the possible object names.
203 /// @throws ioda::Exception if obj_id is invalid.
204 IODA_HIDDEN std::string getNameFromIdentifier(hid_t obj_id);
205 
206 } // namespace HH
207 } // namespace Engines
208 } // namespace detail
209 } // namespace ioda
IODA's error system.
HDF5 resource handles in C++.
The ioda exception class.
Definition: Exception.h:54
Exception & add(const std::string &key, const T value)
Add a key-value pair to the error message.
Definition: Exception.h:75
This is the implementation of Attributes using HDF5.
Definition: HH-attributes.h:30
This is the implementation of Variables using HDF5.
Definition: HH-variables.h:40
A class to wrap HDF5's hid_t resource handles.
Definition: Handles.h:92
Common preprocessor definitions used throughout IODA.
#define IODA_HIDDEN
A tag used to tell the compiler that a symbol should not be listed, but it may be referenced from oth...
Definition: defs.h:89
IODA_HIDDEN std::string getNameFromIdentifier(hid_t obj_id)
Gets a variable / group / link name from an id. Useful for debugging.
Definition: HH-util.cpp:263
IODA_HIDDEN std::pair< bool, hsize_t > iterativeAttributeSearch(hid_t baseObject, const char *attname, H5_index_t iteration_type)
Linear search to find an attribute.
Definition: HH-util.cpp:82
std::unique_ptr< hvl_t[]> buf
Definition: HH-util.h:168
Vlen_data(const Vlen_data &)=delete
IODA_HIDDEN herr_t iterate_find_attr(hid_t loc_id, const char *name, void *op_data)
Callback function for H5Aiterate / H5Aiterate2 / H5Aiterate1.
Definition: HH-util.cpp:37
IODA_HIDDEN void attr_update_reference_list(HH_Variable *scale, const std::vector< ds_list_t > &ref_var_axis)
Attribute REFERENCE_LIST update function.
Definition: HH-util.cpp:230
IODA_HIDDEN HH_Attribute iterativeAttributeSearchAndOpen(hid_t baseObject, H5O_type_t objType, const char *attname)
Linear search to find and open an attribute, if it exists.
Definition: HH-util.cpp:100
IODA_HIDDEN void attr_update_dimension_list(HH_Variable *var, const std::vector< std::vector< ref_t >> &new_dim_list)
Attribute DIMENSION_LIST update function.
Definition: HH-util.cpp:126
IODA_HIDDEN H5_index_t getAttrCreationOrder(hid_t obj, H5O_type_t objType)
Determine attribute creation order for a dataset.
Definition: HH-util.cpp:55
hvl_t & operator[](size_t idx)
Definition: HH-util.h:197
Vlen_data operator=(Vlen_data &&)=delete
Vlen_data(size_t sz, HH_hid_t typ, HH_hid_t space)
Definition: HH-util.h:171
Vlen_data operator=(const Vlen_data &)=delete
#define ioda_Here()
Data to pass to/from iterator classes.
Definition: HH-util.h:48
A "view" of hvl_t objects. Adds C++ conveniences to an otherwise troublesome class.
Definition: HH-util.h:134
Internal structure to encapsulate resources and prevent leaks.
Definition: HH-util.h:167
Duplicate the HDF5 dataset list structure for REFERENCE_LISTs.
Definition: HH-util.h:42