IODA
HH-variables.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2017-2020 Ryan Honeyager (ryan@honeyager.info)
3  * (C) Copyright 2020-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-variables.cpp
12  * \brief HDF5 engine implementation of Variable.
13  */
14 
15 #include "./HH/HH-variables.h"
16 
17 #include <hdf5_hl.h>
18 
19 #include <algorithm>
20 #include <exception>
21 #include <numeric>
22 #include <set>
23 
24 #include "./HH/HH-Filters.h"
25 #include "./HH/HH-attributes.h"
26 #include "./HH/HH-hasattributes.h"
27 #include "./HH/HH-hasvariables.h"
28 #include "./HH/HH-types.h"
29 #include "./HH/HH-util.h"
30 #include "./HH/Handles.h"
31 #include "ioda/Exception.h"
33 #include "ioda/Misc/Dimensions.h"
34 #include "ioda/Misc/StringFuncs.h"
35 
36 namespace ioda {
37 namespace detail {
38 namespace Engines {
39 namespace HH {
40 
41 
42 template <class T>
43 std::vector<T> convertToH5Length(const std::vector<Dimensions_t>& in) {
44  std::vector<T> res(in.size());
45  for (size_t i = 0; i < in.size(); ++i) res[i] = gsl::narrow<T>(in[i]);
46  return res;
47 }
48 
49 HH_Variable::HH_Variable() = default;
50 HH_Variable::~HH_Variable() = default;
51 HH_Variable::HH_Variable(HH_hid_t d, std::shared_ptr<const HH_HasVariables> container)
52  : var_(d), container_(container) {
53  atts = Has_Attributes(std::make_shared<HH_HasAttributes>(d));
54  // auto has_atts_backend = Variable_Backend::_getHasAttributesBackend(backend_->atts);
55  // auto hh_atts_backend = std::dynamic_pointer_cast<HH_HasAttributes>(has_atts_backend);
56  // atts = Has_Attributes(hh_atts_backend);
57  // atts = Has_Attributes(std::make_shared<HH_HasAttributes>(backend_->atts));
58 }
59 
60 HH_hid_t HH_Variable::get() const { return var_; }
61 
63  H5I_type_t typ = H5Iget_type(var_());
64  if (typ == H5I_BADID) throw Exception("Cannot determine object type", ioda_Here());
65  return (typ == H5I_DATASET);
66 }
67 
69 
72 }
73 
75  return Type{std::make_shared<HH_Type>(internalType()), typeid(HH_Type)};
76 }
77 
80 }
81 
83  Dimensions ret;
84  Options errOpts; // Used for tracking parameters that can show up in the error message.
85  errOpts.add("variable", getNameFromIdentifier(var_()));
86 
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.",
92  ioda_Here(), errOpts);
93  hssize_t numPoints = H5Sget_simple_extent_npoints(space()());
94  errOpts.add("numPoints", numPoints);
95  if (numPoints < 0)
96  throw Exception("H5Sget_simple_extent_npoints error.", ioda_Here(), errOpts);
97  int dimensionality = H5Sget_simple_extent_ndims(space()());
98  errOpts.add("dimensionality", dimensionality);
99  if (dimensionality < 0)
100  throw Exception("H5Sget_simple_extent_ndims error.", ioda_Here(), errOpts);
101  dims.resize(dimensionality);
102  dimsmax.resize(dimensionality);
103  if (H5Sget_simple_extent_dims(space()(), dims.data(), dimsmax.data()) < 0)
104  throw Exception("H5Sget_simple_extent_dims error.", ioda_Here(), errOpts);
105 
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)
110  ret.dimsMax.push_back((d == H5S_UNLIMITED) ? ioda::Unlimited : gsl::narrow<Dimensions_t>(d));
111 
112  return ret;
113 }
114 
115 Variable HH_Variable::resize(const std::vector<Dimensions_t>& newDims) {
116  std::vector<hsize_t> hdims = convertToH5Length<hsize_t>(newDims);
117 
118  if (H5Dset_extent(var_(), hdims.data()) < 0)
119  throw Exception("Failure to resize a Variable with the HDF5 backend.", ioda_Here())
120  .add("variable", getNameFromIdentifier(var_()))
121  .add("dimensionality", hdims.size());
122 
123  return Variable{shared_from_this()};
124 }
125 
126 Variable HH_Variable::attachDimensionScale(unsigned int DimensionNumber, const Variable& scale) {
127  Options errOpts;
128  errOpts.add("variable", getNameFromIdentifier(var_()));
129  errOpts.add("DimensionNumber", DimensionNumber);
130 
131  try {
132  // We are extracting the backend object.
133  auto scaleBackendBase = scale.get();
134  // If the backend object is an HH object, then we can attach this scale.
135  // Otherwise, throw an error because you can't mix Variables from different
136  // backends.
137  auto scaleBackendDerived = std::dynamic_pointer_cast<HH_Variable>(scaleBackendBase);
138  errOpts.add("scale", getNameFromIdentifier(scaleBackendDerived->var_()));
139 
140  const herr_t res = H5DSattach_scale(var_(), scaleBackendDerived->var_(), DimensionNumber);
141  if (res != 0) throw Exception("Dimension scale attachment failed.", ioda_Here(), errOpts);
142 
143  return Variable{shared_from_this()};
144  } catch (std::bad_cast) {
145  throw Exception("Cannot attach dimension scales across incompatible backends.",
146  ioda_Here(), errOpts);
147  }
148 }
149 
150 Variable HH_Variable::detachDimensionScale(unsigned int DimensionNumber, const Variable& scale) {
151  Options errOpts;
152  errOpts.add("variable", getNameFromIdentifier(var_()));
153  errOpts.add("DimensionNumber", DimensionNumber);
154 
155  try {
156  // We are extracting the backend object.
157  auto scaleBackendBase = scale.get();
158  // If the backend object is an HH object, then we can attach this scale.
159  // Otherwise, throw an error because you can't mix Variables from different
160  // backends.
161  auto scaleBackendDerived = std::dynamic_pointer_cast<HH_Variable>(scaleBackendBase);
162  errOpts.add("scale", getNameFromIdentifier(scaleBackendDerived->var_()));
163 
164  const herr_t res = H5DSdetach_scale(var_(), scaleBackendDerived->var_(), DimensionNumber);
165  if (res != 0) throw Exception("Dimension scale detachment failed", ioda_Here(), errOpts);
166 
167  return Variable{shared_from_this()};
168  } catch (std::bad_cast) {
169  throw Exception("Cannot detach dimension scales across incompatible backends.",
170  ioda_Here(), errOpts);
171  }
172 }
173 
175  const htri_t res = H5DSis_scale(var_());
176  if (res < 0) {
177  Options errOpts;
178  try {
179  errOpts.add("variable", getNameFromIdentifier(var_()));
180  } catch (...) {
181  errOpts.add("variable", "unknown / bad id");
182  }
183 
184  throw Exception("Error returned from H5DSis_scale.", ioda_Here(), errOpts);
185  }
186  return (res > 0);
187 }
188 
189 Variable HH_Variable::setIsDimensionScale(const std::string& dimensionScaleName) {
190  const htri_t res = H5DSset_scale(var_(), dimensionScaleName.c_str());
191  if (res != 0) {
192  Options errOpts;
193  errOpts.add("dimensionScaleName", dimensionScaleName);
194  try {
195  errOpts.add("variable", getNameFromIdentifier(var_()));
196  } catch (...) {
197  errOpts.add("variable", "unknown / bad id");
198  }
199 
200  throw Exception(
201  "Error returned from H5DSset_scale.", ioda_Here(), errOpts);
202  }
203  return Variable{shared_from_this()};
204 }
205 
207  constexpr size_t max_label_size = 1000;
208  std::array<char, max_label_size> label{}; // Value-initialized to nulls.
209  const ssize_t sz = H5DSget_scale_name(var_(), label.data(), max_label_size);
210  if (sz < 0) {
211  Options errOpts;
212  try {
213  errOpts.add("variable", getNameFromIdentifier(var_()));
214  } catch (...) {
215  errOpts.add("variable", "unknown / bad id");
216  }
217 
218  throw Exception("Error returned from H5DSget_scale_name.", ioda_Here(), errOpts);
219  }
220  // sz is the size of the label. The HDF5 documentation does not include whether the label is
221  // null-terminated, so I am terminating it manually.
222  label[max_label_size - 1] = '\0';
223  res = std::string(label.data());
224  return Variable{std::make_shared<HH_Variable>(*this)};
225 }
226 
227 
228 /** \details This function is byzantine, it is performance-critical, and it cannot be split apart.
229  *
230  * It serves as the common calling point for both the regular getDimensionScaleMappings function and
231  * isDimensionScaleAttached. They share a lot of complex code.
232  *
233  * The code is detailed because we are working around deficiencies in HDF5's dimension scales API.
234  * The key problem is that dimension scale mappings are bi-directional, and H5DSis_attached
235  * repeatedly re-opens the **scale's** list of variables that it acts as a dimension for.
236  * It then has to de-reference and re-open each variable when verifying attachment, and this
237  * performs horribly when you have hundreds or thousands of variables.
238  *
239  * So, the logic here simplifies H5DSis_attached to verify only a uni-directional mapping
240  * so see if a Variable is attached to a scale (and not the other way around).
241  *
242  * Also note: different HDF5 versions use slightly different structs and function calls,
243  * hence the #ifdefs.
244  **/
245 std::vector<std::vector<Named_Variable>> HH_Variable::getDimensionScaleMappings(
246  const std::vector<Named_Variable>& scalesToQueryAgainst, bool firstOnly,
247  const std::vector<unsigned>& dimensionNumbers_) const {
248  try {
249  // Extract all of the scalesToQueryAgainst and convert them into the appropriate backend
250  // objects. If the backend objects are not from this engine, then this is an error (no mixing
251  // Variables and scales across different backends).
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);
256  scales.push_back(
257  {scale.name, scaleBackendDerived}); // NOLINT: macos oddly is missing emplace_back here.
258  }
259 
260  // The logic here roughly follows H5DSis_attached, but with extra optimizations and added
261  // loops to avoid variable repeated variable {open and close} operations.
262 
263  // Check that the dimensionality is sufficient to have a
264  // DimensionNumber element.
265  auto datadims = getDimensions();
266  std::vector<unsigned> dimensionNumbers = dimensionNumbers_;
267  if (!dimensionNumbers.empty()) {
268  auto max_elem_it = std::max_element(dimensionNumbers.cbegin(), dimensionNumbers.cend());
269  if (max_elem_it == dimensionNumbers.cend()) throw Exception(ioda_Here());
270  if (datadims.dimensionality <= *max_elem_it) throw Exception(ioda_Here());
271  } else {
272  dimensionNumbers.resize(datadims.dimensionality);
273  std::iota(dimensionNumbers.begin(), dimensionNumbers.end(), 0);
274  }
275 
276  // The return value. Give it the correct size (the dimensionality of the variable).
277  std::vector<std::vector<Named_Variable>> ret(
278  gsl::narrow<size_t>(datadims.dimensionality));
279 
280  // Iterate over all attributes to get the DIMENSION_LIST attribute. Then,
281  // attempt to read the reference list inside the DIMENSION_LIST attribute and link
282  // dimension references to dimension scales.
283 
284  // NOTE: This code does not use the regular atts.open("DIMENSION_LIST") call
285  // for performance reasons on large IASI-like files where we have to repeat this
286  // call for tens of thousands of variables. We instead do a creation-order-preferred
287  // search.
288 
289  // Get search order
290  H5_index_t iteration_type = getAttrCreationOrder(get()(), H5O_TYPE_DATASET);
291  // H5Aiterate2 exists in v1.8 and up.
292  hsize_t pos = 0;
293  Iterator_find_attr_data_t search_data_opts;
294  search_data_opts.search_for = "DIMENSION_LIST";
295  herr_t att_search_ret
296  = H5Aiterate2(get()(), // Search on this dataset
297  iteration_type, // Iterate by name or creation order
298  H5_ITER_NATIVE, // Fastest ordering possible
299  &pos, // Initial (and current) index
300  iterate_find_attr, // C-style search function
301  reinterpret_cast<void*>(&search_data_opts) // Data passed to/from the C-style search function
302  );
303  if (att_search_ret < 0) throw Exception(ioda_Here());
304 
305  if (!search_data_opts.success) return ret; // Fallthrough returning a vector of empty vectors.
306 
307  hid_t found_att = H5Aopen_by_idx(get()(), ".", iteration_type, H5_ITER_NATIVE,
308  search_data_opts.idx, H5P_DEFAULT, H5P_DEFAULT);
309  HH_Attribute aDims_HH(
311 
312  // Attempt to read the reference list for our variable's
313  // DIMENSION_LIST attribute.
314  //if (!atts.exists("DIMENSION_LIST"))
315  // return ret; // Fallthrough returning a vector of empty vectors.
316  //Attribute aDims = atts.open("DIMENSION_LIST");
317  //auto aDims_backend = _getAttributeBackend<ioda::detail::Attribute_Base<Attribute>>(aDims);
318  //auto aDims_HH = std::dynamic_pointer_cast<HH_Attribute>(aDims_backend);
319 
320  auto vltyp = aDims_HH.internalType();
321  auto vldims = aDims_HH.getDimensions();
322 
323  Vlen_data buf((size_t)datadims.dimensionality, vltyp, aDims_HH.space());
324 
325  if (H5Aread(aDims_HH.get()(), vltyp.get(), reinterpret_cast<void*>(buf.buf.get())) < 0)
326  throw Exception("Attribute read failure", ioda_Here());
327 
328  // We now have the list of object references.
329  // We need to query the scale information.
330 
331  // Get the information for all of the scales.
332 #if H5_VERSION_GE(1, 12, 0)
333  std::vector<H5O_info1_t> scale_infos(scales.size());
334  H5O_info1_t check_info;
335 #else
336  std::vector<H5O_info_t> scale_infos(scales.size());
337  H5O_info_t check_info;
338 #endif
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)
342  throw Exception("H5Oget_info2 failure", ioda_Here());
343 #else
344  if (H5Oget_info(scales[i].second->get()(), &scale_infos[i], H5O_INFO_BASIC) < 0)
345  throw Exception("H5Oget_info failure", ioda_Here());
346 #endif
347  }
348 
349  // Iterate over each dimension (in the set), and iterate along each scale.
350  // See which scales are attached to which dimensions.
351 
352  // For each dimension. The remaining dimensions in the returned vector (i.e. entries not in
353  // dimensionNumbers) are unfilled in the output.
354  for (const auto& curDim : dimensionNumbers) {
355  // For each *scale reference* listed in the variable along a particular dimension
356  // Note well: this is NOT *each scale that the user passed*.
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]; // NOLINT: type conversions
359 
360  // Dereference the scale variable stored in DIMENSION_LIST.
361  // This opens it.
362  hid_t deref_scale_id = H5Rdereference2(
363  // First parameter is any object id in the same file
364  get()(), H5P_DEFAULT, H5R_OBJECT, &ref);
365  Expects(deref_scale_id >= 0); // Die on failure. Would have to clean up memory otherwise.
366  HH_hid_t encap_id(deref_scale_id, Handles::Closers::CloseHDF5Dataset::CloseP);
367  HH_Variable deref_scale(encap_id, nullptr); // Move into managed memory
368 
369  // Get deref_scale's info and compare to scale_info.
370 #if H5_VERSION_GE(1, 10, 3)
371  if (H5Oget_info2(deref_scale.get()(), &check_info, H5O_INFO_BASIC) < 0)
372  throw Exception("H5Oget_info2 failure", ioda_Here());
373 #else
374  if (H5Oget_info(deref_scale.get()(), &check_info, H5O_INFO_BASIC) < 0)
375  throw Exception("H5Oget_info failure", ioda_Here());
376 #endif
377 
378  // Iterate over each scalesToQueryAgainst
379  // I.e. for each *scale that the user passed*.
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)) {
384  // Success! We matched a scale!
385  ret[curDim].push_back(scalesToQueryAgainst[j]);
386 
387  foundScale = true;
388  break; // No need to check this *scale reference* against the remaining known scales.
389  }
390  }
391  // If we are asking for only the first matched scale in each dimension,
392  // and if we just found the first match, then continue on to the next dimension.
393  if (firstOnly && foundScale) break;
394  }
395  }
396 
397  return ret;
398  } catch (...) {
399  Options errOpts;
400  try {
401  errOpts.add("variable", getNameFromIdentifier(var_()));
402  } catch (...) {
403  errOpts.add("variable", "unknown / bad id");
404  }
405 
406  std::throw_with_nested(Exception("Caught an exception.", ioda_Here(), errOpts));
407  }
408 }
409 
410 bool HH_Variable::isDimensionScaleAttached(unsigned int DimensionNumber,
411  const Variable& scale) const {
412  try {
413  const std::vector<Named_Variable> scalesToQueryAgainst{{"unused_param", scale}};
414  auto res = getDimensionScaleMappings(scalesToQueryAgainst, true, {DimensionNumber});
415  return !res[DimensionNumber].empty();
416  } catch (...) {
417  Options errOpts;
418  try {
419  errOpts.add("variable", getNameFromIdentifier(var_()));
420  } catch (...) {
421  errOpts.add("variable", "unknown / bad id");
422  }
423 
424  std::throw_with_nested(Exception("Caught an exception.", ioda_Here(), errOpts));
425  }
426 }
427 
428 std::vector<std::vector<Named_Variable>> HH_Variable::getDimensionScaleMappings(
429  const std::list<Named_Variable>& scalesToQueryAgainst, bool firstOnly) const {
430  return getDimensionScaleMappings({scalesToQueryAgainst.begin(), scalesToQueryAgainst.end()},
431  firstOnly, {});
432 }
433 
435  auto res = std::make_shared<HH_Selection>();
436  res->sel = getSpaceWithSelection(sel);
437  return res;
438 }
439 
441  if (sel.isConcretized()) {
442  auto concretized = sel.concretize();
443  try {
444  // Only return the concretized selection if this is the correct backend.
445  auto csel = std::dynamic_pointer_cast<HH_Selection>(concretized);
446  return csel->sel;
447  } catch (std::bad_cast) {
448  sel.invalidate();
449  }
450  }
451 
452  if (sel.getDefault() == SelectionState::ALL)
453  if (sel.getActions().empty()) return HH_hid_t(H5S_ALL);
454 
456  if (spc() < 0) throw Exception("Cannot copy dataspace.", ioda_Here());
457 
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())
462  < 0)
463  throw Exception("Cannot set dataspace extent.", ioda_Here());
464  }
465 
466  if (sel.getDefault() == SelectionState::ALL) {
467  if (H5Sselect_all(spc()) < 0) throw Exception("Dataspace selection failed.", ioda_Here());
468  } else if (sel.getDefault() == SelectionState::NONE) {
469  if (H5Sselect_none(spc()) < 0) throw Exception("Dataspace selection failed.", ioda_Here());
470  }
471 
472  static const std::map<SelectionOperator, H5S_seloper_t> op_map
473  = {{SelectionOperator::SET, H5S_SELECT_SET},
474  {SelectionOperator::OR, H5S_SELECT_OR},
475  {SelectionOperator::AND, H5S_SELECT_AND},
476  {SelectionOperator::XOR, H5S_SELECT_XOR},
477  {SelectionOperator::NOT_B, H5S_SELECT_NOTB},
478  {SelectionOperator::NOT_A, H5S_SELECT_NOTA},
479  {SelectionOperator::APPEND, H5S_SELECT_APPEND},
480  {SelectionOperator::PREPEND, H5S_SELECT_PREPEND}};
481  bool first_action = true;
482  for (const auto& s : sel.getActions()) {
483  if (!op_map.count(s.op_)) throw Exception("Unimplemented map value.", ioda_Here());
484  herr_t chk = 0;
485  // Is this a hyperslab or a single point selection?
486  if (!s.points_.empty()) { // Single point selection
487 
488  size_t dimensionality = s.points_.at(0).size();
489 
490  std::vector<hsize_t> elems(dimensionality * s.points_.size());
491 
492  // Waiting for std::ranges in C++20
493  for (size_t i = 0; i < s.points_.size(); ++i) // const auto& p : s.points_)
494  {
495  if (s.points_[i].size() != dimensionality)
496  throw Exception("Points have inconsistent dimensionalities.", ioda_Here())
497  .add("dimensionality", dimensionality)
498  .add("s.points_[i].size()", s.points_[i].size())
499  .add("i", i);
500  for (size_t j = 0; j < dimensionality; ++j)
501  elems[j + (dimensionality * i)] = s.points_[i][j];
502  // std::copy_n(p.data(), dimensionality, elems.data() + (i * dimensionality));
503  }
504 
505  chk = H5Sselect_elements(spc(), op_map.at(s.op_), s.points_.size(), elems.data());
506  } else if (!s.dimension_indices_starts_.empty()) {
507  // This is a variant of the hyperslab selection code.
508  // We have index ranges, and we then convert to the usual
509  // start and count.
510  // SelectionOperator is a problem here, because the selections
511  // that we are making are not commutative. So, we do a two-part selection.
512  // First, clone the dataspace and select everything.
513  // Then, apply this bulk selection to the actual space.
514 #if H5_VERSION_GE(1, 12, 0)
515  HH_hid_t cloned_space(H5Scopy(spc()), Handles::Closers::CloseHDF5Dataspace::CloseP);
516  if(H5Sselect_none(cloned_space()) < 0) throw Exception("Cannot copy space", ioda_Here());
517 
519  Expects(s.dimension_ < (size_t)dims.dimensionality);
520  const size_t numSlabs = s.dimension_indices_starts_.size();
521  for (size_t i = 0; i < numSlabs; ++i) {
522  // Fill with zeros, and set the right starting dimension
523  std::vector<hsize_t> hstart;
524  if (sel.extent().empty()) {
525  hstart.resize((size_t)dims.dimensionality, 0);
526  } else {
527  hstart.resize((size_t)sel.extent().size(), 0);
528  }
529  hstart[s.dimension_] = s.dimension_indices_starts_[i];
530 
531  // Fill with the total size, and then set the right extent
532  std::vector<hsize_t> hcount;
533  if (sel.extent().empty()) {
534  hcount = convertToH5Length<hsize_t>(dims.dimsCur);
535  } else {
536  hcount = convertToH5Length<hsize_t>(sel.extent());
537  }
538  hcount[s.dimension_]
539  = (i < s.dimension_indices_counts_.size()) ? s.dimension_indices_counts_[i] : 1;
540 
541  if (H5Sselect_hyperslab(cloned_space(), op_map.at(SelectionOperator::OR), hstart.data(),
542  NULL, hcount.data(), NULL)
543  < 0)
544  throw Exception("Sub-space selection failed.", ioda_Here());
545  }
546 
547  // Once we have looped through then we apply the actual selection operator to our
548  // compound data. If on the first action, space will be either an ALL or NONE
549  // selection type, which will cause the H5Smodify_select to fail. H5Smodify_select
550  // wants both spaces to be HYPERSLAB spaces. So, on the first action, do a select copy
551  // instead of a select modify.
552  if (first_action) {
553  if (H5Sselect_copy(spc(), cloned_space()) < 0)
554  throw Exception("Space copy selection failed", ioda_Here());
555  } else {
556  if (H5Smodify_select(spc(), op_map.at(s.op_), cloned_space()) < 0)
557  throw Exception("Space modify selection failed", ioda_Here());
558  }
559 #else
560  throw Exception(
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());
564 #endif
565  } else { // Hyperslab selection
566 
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_);
571 
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);
575  }
576  if (chk < 0) throw Exception("Space selection failed.", ioda_Here());
577  first_action = false; // NOLINT: Compilers inconsistently complain about use/unuse of
578  // first_action. Not our bug.
579  }
580 
581  if (!sel.getOffset().empty()) {
582  if (H5Soffset_simple(spc(), convertToH5Length<hssize_t>(sel.getOffset()).data()) < 0)
583  throw Exception("Problem applying offset to space.", ioda_Here());
584  }
585 
586  auto res = std::make_shared<HH_Selection>();
587  res->sel = spc;
588  sel.concretize(res);
589  return spc;
590 }
591 
592 Variable HH_Variable::write(gsl::span<char> data, const Type& in_memory_dataType,
593  const Selection& mem_selection, const Selection& file_selection) {
594  auto typeBackend = std::dynamic_pointer_cast<HH_Type>(in_memory_dataType.getBackend());
595  auto memSpace = getSpaceWithSelection(mem_selection);
596  auto fileSpace = getSpaceWithSelection(file_selection);
597  auto ret = H5Dwrite(var_(), // dataset id
598  typeBackend->handle(), // mem_type_id
599  memSpace(), // mem_space_id
600  fileSpace(), // file_space_id
601  H5P_DEFAULT, // xfer_plist_id
602  data.data() // data
603  );
604  if (ret < 0) throw Exception(ioda_Here());
605  return Variable{shared_from_this()};
606 }
607 
608 Variable HH_Variable::read(gsl::span<char> data, const Type& in_memory_dataType,
609  const Selection& mem_selection, const Selection& file_selection) const {
610  auto memSpace = getSpaceWithSelection(mem_selection);
611  auto fileSpace = getSpaceWithSelection(file_selection);
612 
613  // Override for old-format ioda files:
614  // Unfortunately, v0 ioda files have an odd mixture of ascii vs unicode strings, as well as
615  // fixed and variable-length strings. We try and fix a few of these issues here.
616 
617  // Apologies for the complexity of what would otherwise be a straightforward function.
618 
619  // Is this a string of any type?
620  H5T_class_t cls_my = H5Tget_class(internalType().get());
621  if (cls_my == H5T_STRING) {
622  // The type *is* a string type.
623 
624  // We always load strings using the hdf5-provided
625  // type. This way, we ignore character set flags
626  // and other string properties that do not matter
627  // in ioda.
628  auto file_type = internalType();
629 
630  // However, fixed and variable-length strings need
631  // slightly different read calls.
632 
633  if (H5Tis_variable_str(file_type.get()) > 0) {
634  // Variable-length string.
635  // No special read is needed.
636  // Character set is specified correctly by
637  // matching file_type above.
638  auto ret = H5Dread(var_(), // dataset id
639  file_type(), // mem_type_id
640  memSpace(), // mem_space_id
641  fileSpace(), // file_space_id
642  H5P_DEFAULT, // xfer_plist_id
643  data.data() // data
644  );
645  if (ret < 0) throw Exception("Failure in H5Dread", ioda_Here());
646 
647  } else {
648  // Fixed-length string
649  // Special read is needed.
650 
651  // Figure out the size of the data being read.
652  // The data space selections can be a space or numeric identifier H5S_ALL.
653  // If H5S_ALL, then return the maximum size.
654  // Otherwise, call H5Sget_select_type, H5Sget_select_npoints to compute size.
655  hssize_t sz = (memSpace.get() == H5S_ALL)
656  ? gsl::narrow<hssize_t>(getDimensions().numElements)
657  : [&memSpace]() {
658  H5S_sel_type st = H5Sget_select_type(memSpace.get());
659  return (st == H5S_SEL_NONE) ? 0 : H5Sget_select_npoints(memSpace.get());
660  }();
661  if (sz < 0) throw Exception(ioda_Here());
662 
663  // This is all of the strings concatenated.
664  std::vector<char> tmp_buf(gsl::narrow<size_t>(sz));
665  auto ret = H5Dread(var_(), // dataset id
666  file_type(), // mem_type_id
667  memSpace(), // mem_space_id
668  fileSpace(), // file_space_id
669  H5P_DEFAULT, // xfer_plist_id
670  tmp_buf.data() // data
671  );
672  if (ret < 0) throw Exception(ioda_Here());
673 
674  // From here, we must "fake" the structure that we fill so that
675  // it looks like a variable-length string. Basically, we want to present a
676  // uniform view to the caller.
677 
678  // We have a set of fixed-length strings that we read
679  // as characters. We can turn this into strings.
680 
681  // Get the size of the enclosed strings, in bytes.
682  // This leaves off the terminating NULL.
683  const size_t sz_each_str = H5Tget_size(file_type.get());
684  const size_t num_strs = tmp_buf.size() / sz_each_str;
685 
686  // Reinterpret the input buffer as a sequence of char*s.
687  // Assign each string to the appropriate value.
688 
689  char** reint_buf = reinterpret_cast<char**>(data.data()); // NOLINT: casting
690  for (size_t i = 0; i < num_strs; ++i) {
691  std::string s(tmp_buf.data() + (sz_each_str * i), sz_each_str);
692  reint_buf[i]
693  = (char*)malloc(sz_each_str + 1); // NOLINT: quite a deliberate call to malloc here.
694  ioda::detail::COMPAT_strncpy_s(reint_buf[i], sz_each_str + 1, s.data(), s.size() + 1);
695  }
696  }
697  } else {
698  // The type *is not* a string type.
699  auto typeBackend = std::dynamic_pointer_cast<HH_Type>(in_memory_dataType.getBackend());
700  auto ret = H5Dread(var_(), // dataset id
701  typeBackend->handle(), // mem_type_id
702  memSpace(), // mem_space_id
703  fileSpace(), // file_space_id
704  H5P_DEFAULT, // xfer_plist_id
705  data.data() // data
706  );
707  if (ret < 0) throw Exception(ioda_Here());
708  }
709 
710  return Variable{std::make_shared<HH_Variable>(*this)};
711 }
712 
713 bool HH_Variable::isA(Type lhs) const {
714  auto typeBackend = std::dynamic_pointer_cast<HH_Type>(lhs.getBackend());
715 
716  // Override for old-format ioda files:
717  // Unfortunately, v0 ioda files have an odd mixture
718  // of ascii vs unicode strings, as well as
719  // fixed and variable-length strings.
720  // We try and fix a few of these issues here.
721 
722  // Do we expect a string of any type?
723  // Is the object a string of any type?
724  // If both are true, then we just return true.
725  H5T_class_t cls_lhs = H5Tget_class(typeBackend->handle.get());
726  H5T_class_t cls_my = H5Tget_class(internalType()());
727  if (cls_lhs == H5T_STRING && cls_my == H5T_STRING) return true;
728 
729  // Another issue: are the types equivalent but not
730  // exactly the same? This happens across platforms. Endianness
731  // can change.
732  if (cls_lhs != cls_my) return false;
733  // Now the data are either both integers or floats.
734  // For both, are the size (in bytes) the same?
735  if (H5Tget_size(typeBackend->handle.get()) != H5Tget_size(internalType()())) return false;
736 
737  // For integers, are they both signed or unsigned?
738  if (cls_lhs == H5T_INTEGER)
739  if (H5Tget_sign(typeBackend->handle.get()) != H5Tget_sign(internalType()())) return false;
740 
741  // Ignored:
742  // Are the precisions the same?
743  // Are the offsets the same?
744  // Is the padding the same?
745  // Basically everything in https://support.hdfgroup.org/HDF5/doc/H5.user/Datatypes.html.
746 
747  return true;
748  // return backend_.isOfType(typeBackend->handle);
749 }
750 
752  HH_hid_t otype = internalType();
753  auto ret = H5Tequal(ttype(), otype());
754  if (ret < 0) throw Exception(ioda_Here());
755  return (ret > 0) ? true : false;
756 }
757 
759  H5D_fill_value_t fvstatus; // NOLINT: HDF5 C interface
760  if (H5Pfill_value_defined(create_plist.get(), &fvstatus) < 0) throw Exception(ioda_Here());
761  // if H5D_FILL_VALUE_UNDEFINED, return false. In all other cases, return true.
762  return (fvstatus != H5D_FILL_VALUE_UNDEFINED);
763 }
764 
766  HH_hid_t create_plist(H5Dget_create_plist(var_()),
768  return hasFillValue(create_plist);
769 }
770 
772  try {
774 
775  H5D_fill_value_t fvstatus; // NOLINT: HDF5 C interface
776  if (H5Pfill_value_defined(create_plist.get(), &fvstatus) < 0) throw Exception(ioda_Here());
777 
778  // if H5D_FILL_VALUE_UNDEFINED, false. In all other cases, true.
779  res.set_ = (fvstatus != H5D_FILL_VALUE_UNDEFINED);
780 
781  // There are two types of fill values that we encounter.
782  // Those set to fixed values and those set to a "default" value. Unfortunately,
783  // netCDF4-written files set the "default" value but use a default that does
784  // not match HDF5, and this causes major problems with file compatability.
785  // So, we have to override the behavior here to return the fill values expected by
786  // either NetCDF4 or HDF5.
787 
788  // Query the containing HH_Has_Variables to get the policy.
789  auto fvp = container_.lock()->getFillValuePolicy();
790  if ((fvstatus == H5D_FILL_VALUE_DEFAULT) && (fvp == FillValuePolicy::NETCDF4)) {
791  // H5D_FILL_VALUE_DEFAULT with NETCDF4 fill value policy
792  //
793  // This is ugly but necessary since we can't use template magic at this level in this
794  // direction. Catchall is to assign a zero which is the typical HDF5 default value.
795  if (isA<std::string>())
796  assignFillValue<std::string>(res, FillValuePolicies::netCDF4_default<std::string>());
797  else if (isA<signed char>())
798  assignFillValue<signed char>(res, FillValuePolicies::netCDF4_default<signed char>());
799  else if (isA<char>())
800  assignFillValue<char>(res, FillValuePolicies::netCDF4_default<char>());
801  else if (isA<int16_t>())
802  assignFillValue<int16_t>(res, FillValuePolicies::netCDF4_default<int16_t>());
803  else if (isA<int32_t>())
804  assignFillValue<int32_t>(res, FillValuePolicies::netCDF4_default<int32_t>());
805  else if (isA<float>())
806  assignFillValue<float>(res, FillValuePolicies::netCDF4_default<float>());
807  else if (isA<double>())
808  assignFillValue<double>(res, FillValuePolicies::netCDF4_default<double>());
809  else if (isA<unsigned char>())
810  assignFillValue<unsigned char>(res, FillValuePolicies::netCDF4_default<unsigned char>());
811  else if (isA<uint16_t>())
812  assignFillValue<uint16_t>(res, FillValuePolicies::netCDF4_default<uint16_t>());
813  else if (isA<uint32_t>())
814  assignFillValue<uint32_t>(res, FillValuePolicies::netCDF4_default<uint32_t>());
815  else if (isA<int64_t>())
816  assignFillValue<int64_t>(res, FillValuePolicies::netCDF4_default<int64_t>());
817  else if (isA<uint64_t>())
818  assignFillValue<uint64_t>(res, FillValuePolicies::netCDF4_default<uint64_t>());
819  else
820  assignFillValue<uint64_t>(res, 0);
821  } else {
822  // H5D_FILL_VALUE_DEFAULT with HDF5 fill value policy
823  // H5D_FILL_VALUE_USER_DEFINED regardless of fill value policy
824  auto hType = internalType(); // Get type as
825  if (!hType.isValid()) throw Exception(ioda_Here());
826 
827  H5T_class_t cls = H5Tget_class(hType()); // NOLINT: HDF5 C interface
828  // Check types for support in this function
829  const std::set<H5T_class_t> supported{H5T_INTEGER, H5T_FLOAT, H5T_STRING};
830  // Unsupported for now: H5T_BITFIELD, H5T_OPAQUE, H5T_COMPOUND,
831  // H5T_REFERENCE, H5T_ENUM, H5T_VLEN, H5T_ARRAY.
832  if (!supported.count(cls))
833  throw Exception(
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.",
837  ioda_Here());
838 
839  size_t szType_inBytes = H5Tget_size(hType());
840 
841  // Basic types and string pointers fit in the union. Fixed-length string
842  // types do not, which is why we create a special buffer to accommodate.
843  std::vector<char> fvbuf(szType_inBytes, 0);
844  if (H5Pget_fill_value(create_plist.get(), hType(),
845  reinterpret_cast<void*>(fvbuf.data())) < 0)
846  throw Exception(ioda_Here());
847 
848  // When recovering the fill value, we need to distinguish between
849  // strings and the other types.
850 
851  // We do this by checking the type.
852 
853  if (cls == H5T_STRING) {
854  // Need to distinguish between variable-length and fixed-width data types.
855  htri_t str_type = H5Tis_variable_str(hType());
856  if (str_type < 0) throw Exception(ioda_Here());
857  if (str_type > 0) {
858  // Variable-length string
859  const char** ccp = (const char**)fvbuf.data(); // NOLINT: Casting with HDF5
860  // A fill value for a string should always be at the zero element.
861  // It makes no sense to have a multidimensional fill.
862  if (ccp[0]) {
863  res.stringFillValue_ = std::string(ccp[0]);
864  // Do proper deallocation of the HDF5-returned string array.
865  if (H5free_memory(const_cast<void*>(reinterpret_cast<const void*>(ccp[0]))) < 0)
866  throw Exception(ioda_Here());
867  }
868  } else {
869  // Fixed-length string
870  res.stringFillValue_ = std::string(fvbuf.data(), fvbuf.size());
871  }
872  } else {
873  if (szType_inBytes > sizeof(res.fillValue_))
874  throw Exception(
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.",
878  ioda_Here())
879  .add("szType_inBytes", szType_inBytes)
880  .add("sizeof(res.fillValue_)", sizeof(res.fillValue_));
881  // Copy the buffer to the fvdata object
882  memcpy(&(res.fillValue_.ui64), fvbuf.data(),
883  fvbuf.size()); // NOLINT: Accessing this union member deliberately.
884  }
885  }
886  return res;
887  } catch (...) {
888  std::throw_with_nested(Exception("Caught an exception.", ioda_Here()));
889  }
890 }
891 
893  HH_hid_t create_plist(H5Dget_create_plist(var_()),
895  return getFillValue(create_plist);
896 }
897 
898 std::vector<Dimensions_t> HH_Variable::getChunkSizes(HH_hid_t create_plist, const Dimensions &dims) {
899  H5D_layout_t layout = H5Pget_layout(create_plist.get());
900  if (layout == H5D_CHUNKED) {
901  int max_ndims = gsl::narrow<int>(dims.dimensionality);
902  std::vector<hsize_t> chunks(max_ndims);
903  if (H5Pget_chunk(create_plist.get(), max_ndims, chunks.data()) < 0)
904  throw Exception(ioda_Here());
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>(
909  i)); // NOLINT: 'res' has space reserved. No need to rewrite the loop.
910  return res;
911  }
912  return {};
913 }
914 
915 std::vector<Dimensions_t> HH_Variable::getChunkSizes() const {
916  HH_hid_t create_plist(H5Dget_create_plist(var_()),
918  return getChunkSizes(create_plist, getDimensions());
919 }
920 
921 std::pair<bool, int> HH_Variable::getGZIPCompression(HH_hid_t create_plist) {
922  int nfilters = H5Pget_nfilters(create_plist.get());
923  if (nfilters < 0) throw Exception(ioda_Here());
924 
925  for (unsigned i = 0; i < (unsigned)nfilters; ++i) {
926  // See https://support.hdfgroup.org/HDF5/doc/RM/RM_H5P.html#Property-GetFilter2 for the function
927  // signature.
928 
929  unsigned flags = 0; // Unused.
930  const size_t cd_nelems_init = 16; // Size of array
931  size_t cd_nelems = cd_nelems_init; // Pass size of array to function. Overwritten with number
932  // of values actually read.
933  std::vector<unsigned> cd_values(cd_nelems_init); // Data for filter.
934  const size_t namelen = 32; // Unused
935  std::vector<char> name(namelen); // Unused
936  unsigned filter_config = 0; // Unused
937 
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;
941 
942  if (!cd_nelems) throw Exception(ioda_Here());
943 
944  return std::pair<bool, int>(true, gsl::narrow<int>(cd_values[0]));
945  }
946 
947  // Fallthrough. No GZIP compression was specified.
948  return std::pair<bool, int>(false, 0);
949 }
950 
951 std::pair<bool, int> HH_Variable::getGZIPCompression() const {
952  HH_hid_t create_plist(H5Dget_create_plist(var_()),
954  return getGZIPCompression(create_plist);
955 }
956 
957 std::tuple<bool, unsigned, unsigned> HH_Variable::getSZIPCompression(HH_hid_t create_plist) {
958  int nfilters = H5Pget_nfilters(create_plist.get());
959  if (nfilters < 0) throw Exception(ioda_Here());
960 
961  for (unsigned i = 0; i < (unsigned)nfilters; ++i) {
962  // See https://support.hdfgroup.org/HDF5/doc/RM/RM_H5P.html#Property-GetFilter2 for the function
963  // signature.
964 
965  unsigned flags = 0; // Unused.
966  const size_t cd_nelems_init = 16; // Size of array
967  size_t cd_nelems = cd_nelems_init; // Pass size of array to function. Overwritten with number
968  // of values actually read.
969  std::vector<unsigned> cd_values(cd_nelems_init); // Data for filter.
970  const size_t namelen = 32; // Unused
971  std::vector<char> name(namelen); // Unused
972  unsigned filter_config = 0; // Unused
973 
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;
977 
978  if (cd_nelems < 2) throw Exception(ioda_Here());
979 
980  // cd_nelems is actually 4, but the options do not match the H5Pset_szip flags!
981  return std::tuple<bool, unsigned, unsigned>(true, cd_values[0], cd_values[1]);
982  }
983 
984  // Fallthrough. No SZIP compression was specified.
985  return std::tuple<bool, unsigned, unsigned>(false, 0, 0);
986 }
987 
988 std::tuple<bool, unsigned, unsigned> HH_Variable::getSZIPCompression() const {
989  HH_hid_t create_plist(H5Dget_create_plist(var_()),
991  return getSZIPCompression(create_plist);
992 }
993 
995  HH_hid_t create_plist(H5Dget_create_plist(var_()),
998 
999  // Get chunking
1000  auto chunkinfo = getChunkSizes(create_plist, getDimensions());
1001  if (chunkinfo.size()) {
1002  res.chunk = true;
1003  res.chunks = chunkinfo;
1004  }
1005  // Get compression
1006  auto gz = getGZIPCompression(create_plist);
1007  if (gz.first) res.compressWithGZIP(gz.second);
1008  auto sz = getSZIPCompression(create_plist);
1009  if (std::get<0>(sz)) res.compressWithSZIP(std::get<1>(sz), std::get<2>(sz));
1010  // Get fill value
1011  res.fillValue_ = getFillValue(create_plist);
1012  // Attributes (optional)
1013  if (doAtts) {
1014  throw Exception("Unimplemented doAtts", ioda_Here());
1015  }
1016  // Dimensions (optional)
1017  if (doDims) {
1018  throw Exception("Unimplemented doDims", ioda_Here());
1019  }
1020 
1021  return res;
1022 }
1023 
1024 HH_Selection::~HH_Selection() = default;
1025 
1026 } // namespace HH
1027 } // namespace Engines
1028 } // namespace detail
1029 } // namespace ioda
1030 
1031 /// @}
Convenience classes for constructing ObsSpaces and setting up new Dimension Scales.
Describe the dimensions of a ioda::Attribute or ioda::Variable.
IODA's error system.
HDF5 filters.
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.
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 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.
Definition: Options.h:23
Options & add(const std::string &key, const T &value)
Adds an option. Throws if the same name already exists.
Definition: Options.h:73
A Selection represents the bounds of the data, in ioda or in userspace, that you are reading or writi...
Definition: Selection.h:48
Represents the "type" (i.e. integer, string, float) of a piece of data.
Definition: Type.h:123
Variables store data!
Definition: Variable.h:680
This is the implementation of Attributes using HDF5.
Definition: HH-attributes.h:30
Dimensions getDimensions() const final
Get Attribute's dimensions.
HH_hid_t internalType() const
Get HDF5-internal type.
static HH_Type_Provider * instance()
Definition: HH-types.cpp:36
This is the implementation of ioda::Type using HDF5. Do not use outside of IODA.
Definition: HH-types.h:40
This is the implementation of Variables using HDF5.
Definition: HH-variables.h:40
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_
Definition: HH-variables.h:42
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.
Definition: Handles.h:92
std::shared_ptr< Type_Backend > getBackend() const
Definition: Type.h:103
Backends implement type providers in conjunction with Attributes, Has_Attributes, Variables and Has_V...
Definition: Type_Provider.h:36
Has_Attributes atts
Attributes.
Definition: Variable.h:71
std::string getDimensionScaleName() const
Get the name of this Variable's defined dimension scale.
Definition: Variable.h:217
std::shared_ptr< Variable_Backend > get() const
Gets a handle to the underlying object that implements the backend functionality.
Definition: Variable.cpp:21
std::shared_ptr< InstantiatedSelection > SelectionBackend_t
Definition: Selection.h:35
Selection & extent(const VecDimensions_t &sz)
Provide the dimensions of the object that you are selecting from.
Definition: Selection.h:111
bool isConcretized() const
Is the selection already cached in the backend?
Definition: Selection.h:127
const std::vector< SingleSelection > & getActions() const
Definition: Selection.h:108
void invalidate() const
Ditch the concretized selection.
Definition: Selection.h:129
SelectionState getDefault() const
Definition: Selection.h:109
const VecDimensions_t & getOffset() const
Definition: Selection.h:101
Selections::SelectionBackend_t concretize(const Variable &) const
Talk to the backend and generate the appropriate selection object.
Definition: Selection.cpp:18
@ 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.
Definition: HH-util.cpp:263
std::unique_ptr< hvl_t[]> buf
Definition: HH-util.h:168
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 H5_index_t getAttrCreationOrder(hid_t obj, H5O_type_t objType)
Determine attribute creation order for a dataset.
Definition: HH-util.cpp:55
list newDims
Definition: 05-ObsGroup.py:95
char netCDF4_default< char >()
Definition: FillPolicy.h:64
uint32_t netCDF4_default< uint32_t >()
Definition: FillPolicy.h:92
int16_t netCDF4_default< int16_t >()
Definition: FillPolicy.h:68
uint16_t netCDF4_default< uint16_t >()
Definition: FillPolicy.h:88
unsigned char netCDF4_default< unsigned char >()
Definition: FillPolicy.h:84
double netCDF4_default< double >()
Definition: FillPolicy.h:80
signed char netCDF4_default< signed char >()
Definition: FillPolicy.h:60
int64_t netCDF4_default< int64_t >()
Definition: FillPolicy.h:96
float netCDF4_default< float >()
Definition: FillPolicy.h:76
uint64_t netCDF4_default< uint64_t >()
Definition: FillPolicy.h:100
int32_t netCDF4_default< int32_t >()
Definition: FillPolicy.h:72
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.
Definition: Type.cpp:25
constexpr int Unlimited
Specifies that a dimension is resizable to infinity.
#define ioda_Here()
Describes the dimensions of an Attribute or Variable.
Definition: Dimensions.h:22
std::vector< Dimensions_t > dimsCur
The dimensions of the data.
Definition: Dimensions.h:23
Dimensions_t numElements
Definition: Dimensions.h:26
Dimensions_t dimensionality
The dimensionality (rank) of the data.
Definition: Dimensions.h:25
std::vector< Dimensions_t > dimsMax
This must always equal dimsCur for Attribute.
Definition: Dimensions.h:24
Used to specify Variable creation-time properties.
Definition: Has_Variables.h:57
detail::FillValueData_t fillValue_
Definition: Has_Variables.h:66
std::vector< Dimensions_t > chunks
Manually specify the chunks. Never directly use. Use getChunks(...) instead.
Definition: Has_Variables.h:87
void compressWithSZIP(unsigned PixelsPerBlock=16, unsigned options=4)
bool chunk
Do we chunk this variable? Required for extendible / compressible Variables.
Definition: Has_Variables.h:84
Data to pass to/from iterator classes.
Definition: HH-util.h:48
Internal structure to encapsulate resources and prevent leaks.
Definition: HH-util.h:167
Container used to store and manipulate fill values.
Definition: Fill.h:35
std::string stringFillValue_
Definition: Fill.h:53
union ioda::detail::FillValueData_t::FillValueUnion_t fillValue_