IODA
HH-variablecreation.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-variablecreation.cpp
12  * \brief HDF5 engine Variable creation and property lists.
13  */
14 
16 
17 #include <hdf5.h>
18 
19 #include <algorithm>
20 #include <numeric>
21 #include <set>
22 
23 #include "./HH/HH-Filters.h"
24 #include "./HH/HH-hasvariables.h"
25 #include "./HH/HH-types.h"
26 #include "./HH/Handles.h"
28 
29 namespace ioda {
30 namespace detail {
31 namespace Engines {
32 namespace HH {
34  const Vec_t& max_dims, std::shared_ptr<HH_Type> data_type) {
35  // This constructor generates and stores the dataset creation property list.
36  // We only have to do this once per variable, so we might as well do it here.
37 
38  dcp_ = HH_hid_t(H5Pcreate(H5P_DATASET_CREATE), Handles::Closers::CloseHDF5PropertyList::CloseP);
39  if (!dcp_.isValid()) throw;
40 
41  // Convert dims and max_dims to HDF5 equivalents
42  // Data dimensions and max dimensions
43  for (const auto& d : dims) dims_.push_back(gsl::narrow<hsize_t>(d));
44  // Deliberately not an equality comparison. max_dims may be unspecified.
45  if (dims.size() < max_dims.size()) throw; // TODO(ryan): Separate PR for ioda Exceptions class.
46  for (size_t i = 0; i < max_dims.size(); ++i) {
47  if (max_dims[i] == ioda::Unlimited)
48  max_dims_.push_back(H5S_UNLIMITED);
49  else if (max_dims[i] == ioda::Unspecified)
50  max_dims_.push_back(gsl::narrow<hsize_t>(dims[i]));
51  else
52  max_dims_.push_back(gsl::narrow<hsize_t>(max_dims[i]));
53  }
54 
55  // Chunking
56  if (p.chunk) {
57  // Either the user can specify chunk sizes manually, or we calculate these by
58  // taking hints from the dimensions that the variable will be attached to.
59  // Those hints are determined using a function.
60 
61  /// \todo Revise chunking calculation.
62  // TODO(ryan): Revise and combine with the dimension scale hint.
63  // Bit of an awkward call. If chunks are manually set, it uses those. Otherwise,
64  // it uses the initial variable size as a hint. Problematic because we get to override it
65  // if zero.
66 
67  auto chunksizes = p.getChunks(dims);
68  std::vector<hsize_t> hcs(chunksizes.size()); // chunksizes converted to hsize_t.
69  for (size_t i = 0; i < chunksizes.size(); ++i) {
70  hcs[i] = gsl::narrow<hsize_t>( // Always narrow to hsize_t.
71  (chunksizes[i] > 0) ? chunksizes[i] : dims_[i]);
72  if (hcs[i] == 0) throw; // We need a hint at this point.
73  if (max_dims_[i] >= 0) {
74  if (hcs[i] > max_dims_[i]) hcs[i] = max_dims_[i];
75  }
76  }
77  final_chunks_ = hcs;
78 
79  if (H5Pset_chunk(dcp_(), static_cast<int>(hcs.size()), hcs.data()) < 0) throw;
80  }
81 
82  // Filters (compression, shuffle, scale-offset)
83  //
84  // We only expose basic compression options for now, but could add support for
85  // more advanced compression plugins and additional filters in the future. This
86  // depends on user needs and a redesign of how the user specifies these options when
87  // creating a variable. The current method is already rather complex.
88  {
89  if ((p.gzip_ || p.szip_) && !p.chunk)
90  throw; // Compression filters are only allowed when chunking is used.
91 
92  Filters filt(dcp_.get());
93  if (p.gzip_) filt.setGZIP(p.gzip_level_);
95  }
96 
97  // Initial fill value
98  if (p.fillValue_.set_) {
99  // finalization makes sure that strings and other odd types are handled properly.
101  if (0 > H5Pset_fill_value(dcp_(), data_type->handle(), &(fvdata))) throw;
102  }
103 }
104 
106 
108  hid_t space = (dims_.empty()) ? H5Screate(H5S_SCALAR)
109  : H5Screate_simple(gsl::narrow<int>(dims_.size()), dims_.data(),
110  max_dims_.data());
111  if (space < 0) throw;
112 
114  return res;
115 }
116 
118  static HH_hid_t res(H5P_DEFAULT, Handles::Closers::DoNotClose::CloseP);
119  return res;
120 }
121 
123  // Always default to create intermediate groups if these do not already exist.
124  HH_hid_t res(H5Pcreate(H5P_LINK_CREATE), Handles::Closers::CloseHDF5PropertyList::CloseP);
125  if (H5Pset_create_intermediate_group(res.get(), 1) < 0) throw;
126  return res;
127 }
128 } // namespace HH
129 } // namespace Engines
130 } // namespace detail
131 } // namespace ioda
132 
133 /// @}
Convenience classes for constructing ObsSpaces and setting up new Dimension Scales.
HDF5 filters.
HDF5 engine implementation of Has_Variables.
HDF5 engine implementation of ioda::detail::Type_Provider.
HDF5 engine variable creation parameters.
HDF5 resource handles in C++.
A class to wrap HDF5's hid_t resource handles.
Definition: Handles.h:92
static HH_hid_t datasetAccessPlist()
The default dataset access property list. Currently a nullop.
VariableCreation(const VariableCreationParameters &p, const Vec_t &dims, const Vec_t &max_dims, std::shared_ptr< HH_Type > data_type)
Manages property lists for HDF5 variable creation.
HH_hid_t datasetCreationPlist() const
Generates a dataset creation property list, which encodes the chunking options, compression,...
static HH_hid_t linkCreationPlist()
The ioda-default link creation property list. @detail This just sets a property to create missing int...
HH_hid_t dataspace() const
Generate a dataspace for the constructor-provided dimensions and max dimensions.
std::vector< ioda::Dimensions_t > Vec_t
constexpr int Unlimited
Specifies that a dimension is resizable to infinity.
constexpr int Unspecified
Specifies that a dimension has no specified size. Tells ioda to figure it out from elsewhere.
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 > getChunks(const std::vector< Dimensions_t > &cur_dims) const
Definition: Has_Variables.h:93
bool chunk
Do we chunk this variable? Required for extendible / compressible Variables.
Definition: Has_Variables.h:84
Order-obeying filter insertions and replacements.
Definition: HH-Filters.h:54
void setSZIP(unsigned int optm, unsigned int ppb)
Definition: HH-Filters.cpp:157
void setGZIP(unsigned int level)
Definition: HH-Filters.cpp:172
FillValueUnion_t finalize() const
Definition: Fill.cpp:16