IODA
HH.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020-2021 UCAR
3  *
4  * This software is licensed under the terms of the Apache Licence Version 2.0
5  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
6  */
7 /*! \addtogroup ioda_cxx_engines_pub_HH
8  *
9  * @{
10  * \file HH.cpp
11  * \brief HDF5 engine interface to the rest of ioda.
12  */
13 
14 #include "ioda/Engines/HH.h"
15 
16 #include <mutex>
17 #include <random>
18 #include <sstream>
19 
20 #include "./HH/HH-attributes.h"
21 #include "./HH/HH-groups.h"
22 #include "./HH/Handles.h"
23 #include "ioda/Exception.h"
24 #include "ioda/Group.h"
25 #include "ioda/defs.h"
26 
27 namespace ioda {
28 namespace Engines {
29 namespace HH {
30 unsigned int random_char() {
31  // Adapted from https://lowrey.me/guid-generation-in-c-11/
32  static std::random_device rd;
33  static std::mt19937 gen(rd());
34  static std::uniform_int_distribution<> dis(0, 255);
35  return dis(gen);
36 }
37 
38 std::string generate_hex(const unsigned int len) {
39  std::stringstream ss;
40  for (unsigned i = 0; i < len; i++) {
41  const auto rc = random_char();
42  std::stringstream hexstream;
43  hexstream << std::hex << rc;
44  auto hex = hexstream.str();
45  ss << (hex.length() < 2 ? '0' + hex : hex);
46  }
47  return ss.str();
48 }
49 
50 std::string genUniqueName() {
51  // GUIDs look like: {CD1A91C6-9C1B-454E-AD1C-977F4C72A01C}.
52  // We use these for the file name because they are quite unique.
53  // HDF5 needs unique names, otherwise it might open the same memory file twice.
54 
55  static std::mutex m;
56 
57  std::lock_guard<std::mutex> l{m};
58 
59  std::array<std::string, 5> uuid
61  std::string res
62  = uuid[0] + "-" + uuid[1] + "-" + uuid[2] + "-" + uuid[3] + "-" + uuid[4] + ".hdf5";
63  return res;
64 }
65 
66 const std::map<HDF5_Version, H5F_libver_t> map_h5ver
67  = {{HDF5_Version::Earliest, H5F_LIBVER_EARLIEST},
68  {HDF5_Version::V18, H5F_LIBVER_V18},
69 #if H5_VERSION_GE(1, 10, 0)
70  {HDF5_Version::V110, H5F_LIBVER_V110},
71 #endif
72 #if H5_VERSION_GE(1, 12, 0)
73  {HDF5_Version::V112, H5F_LIBVER_V112},
74 #endif
75  {HDF5_Version::Latest, H5F_LIBVER_LATEST}};
76 
78 #if H5_VERSION_GE(1, 10, 0)
80 #else
81  // Old HDF5 version fallthrough
83 #endif
84 }
85 
86 Group createMemoryFile(const std::string& filename, BackendCreateModes mode, bool flush_on_close,
87  size_t increment_len, HDF5_Version_Range compat) {
88  using namespace ioda::detail::Engines::HH;
89 
90  static const std::map<BackendCreateModes, unsigned int> m{
92  {BackendCreateModes::Fail_If_Exists, H5F_ACC_CREAT}};
93 
94  Options errOpts;
95  errOpts.add("filename", filename);
96  errOpts.add("mode", mode);
97  errOpts.add("flush_on_close", flush_on_close);
98  errOpts.add("increment_len", increment_len);
99  errOpts.add("compat", compat);
100 
101  hid_t plid = H5Pcreate(H5P_FILE_ACCESS);
102  Expects(plid >= 0);
103  HH_hid_t pl(plid, Handles::Closers::CloseHDF5PropertyList::CloseP);
104 
105  if (0 > H5Pset_fapl_core(pl.get(), increment_len, flush_on_close))
106  throw Exception("H5Pset_fapl_core failed", ioda_Here(), errOpts);
107  // H5F_LIBVER_V18, H5F_LIBVER_V110, H5F_LIBVER_V112, H5F_LIBVER_LATEST.
108  // Note: this propagates to any files flushed to disk.
109  if (0 > H5Pset_libver_bounds(pl.get(), map_h5ver.at(compat.first), map_h5ver.at(compat.second)))
110  throw Exception("H5Pset_libver_bounds failed", ioda_Here(), errOpts);
111 
112  HH_hid_t f = H5Fcreate(filename.c_str(), m.at(mode), H5P_DEFAULT, pl.get());
113  if (f() < 0) throw Exception("H5Fcreate failed", ioda_Here(), errOpts);
114 
115  auto backend
116  = std::make_shared<detail::Engines::HH::HH_Group>(f, getCapabilitiesInMemoryEngine(), f);
117  return ::ioda::Group{backend};
118 }
119 
120 Group createFile(const std::string& filename, BackendCreateModes mode, HDF5_Version_Range compat) {
121  using namespace ioda::detail::Engines::HH;
122 
123  static const std::map<BackendCreateModes, unsigned int> m{
125  {BackendCreateModes::Fail_If_Exists, H5F_ACC_CREAT}};
126 
127  Options errOpts;
128  errOpts.add("filename", filename);
129  errOpts.add("mode", mode);
130  errOpts.add("compat", compat);
131 
132  hid_t plid = H5Pcreate(H5P_FILE_ACCESS);
133  if (plid < 0) throw Exception("H5Pcreate failed", ioda_Here(), errOpts);
134  HH_hid_t pl(plid, Handles::Closers::CloseHDF5PropertyList::CloseP);
135  // H5F_LIBVER_V18, H5F_LIBVER_V110, H5F_LIBVER_V112, H5F_LIBVER_LATEST.
136  // Note: this propagates to any files flushed to disk.
137  if (0 > H5Pset_libver_bounds(pl.get(), map_h5ver.at(compat.first), map_h5ver.at(compat.second)))
138  throw Exception("H5Pset_libver_bounds failed", ioda_Here(), errOpts);
139 
140  HH_hid_t f = H5Fcreate(filename.c_str(), m.at(mode), H5P_DEFAULT, pl.get());
141  if (f() < 0) throw Exception("H5Fcreate failed", ioda_Here(), errOpts);
142 
143  auto backend = std::make_shared<detail::Engines::HH::HH_Group>(f, getCapabilitiesFileEngine(), f);
144  return ::ioda::Group{backend};
145 }
146 
147 Group openFile(const std::string& filename, BackendOpenModes mode, HDF5_Version_Range compat) {
148  using namespace ioda::detail::Engines::HH;
149  static const std::map<BackendOpenModes, unsigned int> m{
150  {BackendOpenModes::Read_Only, H5F_ACC_RDONLY}, {BackendOpenModes::Read_Write, H5F_ACC_RDWR}};
151 
152  Options errOpts;
153  errOpts.add("filename", filename);
154  errOpts.add("mode", mode);
155  errOpts.add("compat", compat);
156 
157  hid_t plid = H5Pcreate(H5P_FILE_ACCESS);
158  if (plid < 0) throw Exception("H5Pcreate failed", ioda_Here(), errOpts);
159  HH_hid_t pl(plid, Handles::Closers::CloseHDF5PropertyList::CloseP);
160  if (0 > H5Pset_libver_bounds(pl.get(), map_h5ver.at(compat.first), map_h5ver.at(compat.second)))
161  throw Exception("H5Pset_libver_bounds failed", ioda_Here(), errOpts);
162 
163  HH_hid_t f = H5Fopen(filename.c_str(), m.at(mode), pl.get());
164  if (f() < 0) throw Exception("H5Fopen failed", ioda_Here(), errOpts);
165 
166  auto backend = std::make_shared<detail::Engines::HH::HH_Group>(f, getCapabilitiesFileEngine(), f);
167 
168  return ::ioda::Group{backend};
169 }
170 
171 Group openMemoryFile(const std::string& filename, BackendOpenModes mode, bool flush_on_close,
172  size_t increment_len, HDF5_Version_Range compat) {
173  using namespace ioda::detail::Engines::HH;
174  static const std::map<BackendOpenModes, unsigned int> m{
175  {BackendOpenModes::Read_Only, H5F_ACC_RDONLY}, {BackendOpenModes::Read_Write, H5F_ACC_RDWR}};
176 
177  Options errOpts;
178  errOpts.add("filename", filename);
179  errOpts.add("mode", mode);
180  errOpts.add("flush_on_close", flush_on_close);
181  errOpts.add("increment_len", increment_len);
182  errOpts.add("compat", compat);
183 
184  hid_t plid = H5Pcreate(H5P_FILE_ACCESS);
185  Expects(plid >= 0);
186  HH_hid_t pl(plid, Handles::Closers::CloseHDF5PropertyList::CloseP);
187 
188  const auto h5Result = H5Pset_fapl_core(pl.get(), increment_len, flush_on_close);
189  if (h5Result < 0) throw Exception("H5Pset_fapl_core failed", ioda_Here(), errOpts);
190  if (0 > H5Pset_libver_bounds(pl.get(), map_h5ver.at(compat.first), map_h5ver.at(compat.second)))
191  throw Exception("H5Pset_libver_bounds failed", ioda_Here(), errOpts);
192 
193  HH_hid_t f = H5Fopen(filename.c_str(), m.at(mode), pl.get());
194  if (f() < 0) throw Exception("H5Fopen failed", ioda_Here(), errOpts);
195 
196  auto backend
197  = std::make_shared<detail::Engines::HH::HH_Group>(f, getCapabilitiesInMemoryEngine(), f);
198 
199  return ::ioda::Group{backend};
200 }
201 
203  static Capabilities caps;
204  static bool inited = false;
205  if (!inited) {
209 
210  bool canSZIP = false; //::HH::CanUseSZIP<int>();
213  }
214 
215  return caps;
216 }
217 
219  static Capabilities caps;
220  static bool inited = false;
221  if (!inited) {
225 
226  bool canSZIP = false; //::HH::CanUseSZIP<int>();
229  }
230 
231  return caps;
232 }
233 
234 std::ostream& operator<<(std::ostream& os, const ioda::Engines::HH::HDF5_Version& ver)
235 {
236  using namespace ioda::Engines::HH;
237  static const std::map<HDF5_Version, std::string> names {
238  {HDF5_Version::Earliest, "Earliest"},
239  {HDF5_Version::V18, "V18"},
240  {HDF5_Version::V110, "V110"},
241  {HDF5_Version::V112, "V112"},
242  {HDF5_Version::Latest, "Latest"}
243  };
244  if (names.count(ver) == 0) throw Exception("Unhandled HDF5 version", ioda_Here());
245  os << names.at(ver);
246 
247  // For Latest, get the current library version and add this to the output.
248  if (ver == HDF5_Version::Latest) {
249  unsigned maj = 0, min = 0, rel = 0;
250  if (H5get_libversion(&maj, &min, &rel) < 0) throw Exception("Bad HDF5 return value", ioda_Here());
251  os << " (" << maj << "." << min << "." << rel << ")";
252  }
253 
254  return os;
255 }
256 
257 std::ostream& operator<<(std::ostream& os, const ioda::Engines::HH::HDF5_Version_Range& range)
258 {
259  using namespace ioda::Engines::HH;
260  os << "HDF5_Version_Range: [" << range.first << ", " << range.second << "]";
261  return os;
262 }
263 
264 } // namespace HH
265 } // namespace Engines
266 } // namespace ioda
267 
268 /// @}
269 
270 
IODA's error system.
Interfaces for ioda::Group and related classes.
HDF5 engine implementation of Attribute.
HDF5 group interface.
HDF5 engine.
HDF5 resource handles in C++.
The ioda exception class.
Definition: Exception.h:54
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
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
Common preprocessor definitions used throughout IODA.
IODA_DL Group createFile(const std::string &filename, BackendCreateModes mode, HDF5_Version_Range compat=defaultVersionRange())
Create a ioda::Group backed by an HDF5 file.
Definition: HH.cpp:120
IODA_DL HDF5_Version_Range defaultVersionRange()
Definition: HH.cpp:77
IODA_DL Group openFile(const std::string &filename, BackendOpenModes mode, HDF5_Version_Range compat=defaultVersionRange())
Open a ioda::Group backed by an HDF5 file.
Definition: HH.cpp:147
IODA_DL Capabilities getCapabilitiesFileEngine()
Get capabilities of the HDF5 file-backed engine.
Definition: HH.cpp:202
IODA_DL std::string genUniqueName()
Convenience function to generate a random file name.
Definition: HH.cpp:50
HDF5_Version
HDF5 library format versions.
Definition: HH.h:34
IODA_DL Capabilities getCapabilitiesInMemoryEngine()
Get capabilities of the HDF5 memory-backed engine.
Definition: HH.cpp:218
IODA_DL Group openMemoryFile(const std::string &filename, BackendOpenModes mode=BackendOpenModes::Read_Only, bool flush_on_close=false, size_t increment_len_bytes=1000000, HDF5_Version_Range compat=defaultVersionRange())
Map an HDF5 file in memory and open a ioda::Group.
Definition: HH.cpp:171
IODA_DL Group createMemoryFile(const std::string &filename, BackendCreateModes mode, bool flush_on_close=false, size_t increment_len_bytes=1000000, HDF5_Version_Range compat=defaultVersionRange())
Create a ioda::Group backed by the HDF5 in-memory-store.
Definition: HH.cpp:86
std::pair< HDF5_Version, HDF5_Version > HDF5_Version_Range
Definition: HH.h:42
@ Earliest
Use the earliest possible HDF5 format for storing objects.
@ V18
Use the latest HDF5 v1.8 format for storing objects.
@ V110
Use the latest HDF5 v1.10 format for storing objects.
@ Latest
Use the latest possible HDF5 format for storing objects.
@ V112
Use the latest HDF5 v1.12 format for storing objects.
BackendCreateModes
Definition: Factory.h:44
@ Unsupported
The feature causes an exception if used.
@ Supported
The feature always works.
@ Fail_If_Exists
If the file already exists, fail with an error.
@ Truncate_If_Exists
If the file already exists, overwrite it.
@ Read_Write
Open the file in read-write mode.
@ Read_Only
Open the file in read-only mode.
Functions that are helpful in creating a new ioda::Group that is backed by HDF5.
Definition: HH.h:30
unsigned int random_char()
Definition: HH.cpp:30
std::string generate_hex(const unsigned int len)
Definition: HH.cpp:38
const std::map< HDF5_Version, H5F_libver_t > map_h5ver
Definition: HH.cpp:67
IODA_DL std::ostream & operator<<(std::ostream &os, const HDF5_Version &ver)
stream operator
Definition: HH.cpp:234
#define ioda_Here()
Struct defining what an engine can/cannot do.
Definition: Capabilities.h:47
Capability_Mask canCompressWithGZIP
Definition: Capabilities.h:49
Capability_Mask canChunk
Definition: Capabilities.h:48
Capability_Mask MPIaware
Definition: Capabilities.h:51
Capability_Mask canCompressWithSZIP
Definition: Capabilities.h:50