IODA
HH-attributes.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-attributes.cpp
12  * \brief HDF5 engine implementation of Attribute.
13  */
14 
15 #include "./HH/HH-attributes.h"
16 
17 #include "./HH/HH-types.h"
18 #include "ioda/Exception.h"
19 #include "ioda/Misc/Dimensions.h"
20 
21 namespace ioda {
22 namespace detail {
23 namespace Engines {
24 namespace HH {
25 HH_Attribute::HH_Attribute() : attr_(HH_hid_t::dummy()) {}
27 HH_Attribute::~HH_Attribute() = default;
28 
31 }
32 
33 HH_hid_t HH_Attribute::get() const { return attr_; }
34 
36  H5I_type_t typ = H5Iget_type(attr_());
37  if (typ == H5I_BADID) throw Exception("H5Iget_type failed.", ioda_Here());
38  return (typ == H5I_ATTR);
39 }
40 
41 std::string HH_Attribute::getName() const {
42  ssize_t sz = H5Aget_name(attr_(), 0, nullptr);
43  if (sz < 0) throw Exception("H5Aget_name failed.", ioda_Here());
44  auto s = gsl::narrow<size_t>(sz);
45  std::vector<char> v(s + 1, '\0');
46  sz = H5Aget_name(attr_(), v.size(), v.data());
47  if (sz < 0) throw Exception("H5Aget_name failed.", ioda_Here());
48  return std::string(v.data());
49 }
50 
51 void HH_Attribute::write(gsl::span<const char> data, HH_hid_t in_memory_dataType) {
52  if (H5Awrite(attr_(), in_memory_dataType(), data.data()) < 0)
53  throw Exception("H5Awrite failed.", ioda_Here());
54 }
55 
56 Attribute HH_Attribute::write(gsl::span<char> data, const Type& in_memory_dataType) {
57  auto typeBackend = std::dynamic_pointer_cast<HH_Type>(in_memory_dataType.getBackend());
58  write(data, typeBackend->handle);
59  return Attribute{shared_from_this()};
60 }
61 
62 void HH_Attribute::read(gsl::span<char> data, HH_hid_t in_memory_dataType) const {
63  herr_t ret = H5Aread(attr_(), in_memory_dataType(), static_cast<void*>(data.data()));
64  if (ret < 0) throw Exception("H5Aread failed.", ioda_Here());
65 }
66 
67 Attribute HH_Attribute::read(gsl::span<char> data, const Type& in_memory_dataType) const {
68  auto typeBackend = std::dynamic_pointer_cast<HH_Type>(in_memory_dataType.getBackend());
69  read(data, typeBackend->handle);
70 
71  return Attribute{
72  std::make_shared<HH_Attribute>(*this)}; // For const-ness instead of shared_from_this.
73 }
74 
75 bool HH_Attribute::isA(HH_hid_t ttype) const {
76  HH_hid_t otype = internalType();
77  auto ret = H5Tequal(ttype(), otype());
78  if (ret < 0) throw Exception("H5Tequal failed.", ioda_Here());
79  return (ret > 0) ? true : false;
80 }
81 
82 bool HH_Attribute::isA(Type lhs) const {
83  try {
84  auto typeBackend = std::dynamic_pointer_cast<HH_Type>(lhs.getBackend());
85 
86  // Override for old-format ioda files:
87  // Unfortunately, v0 ioda files have an odd mixture
88  // of ascii vs unicode strings, as well as
89  // fixed and variable-length strings.
90  // We try and fix a few of these issues here.
91 
92  // Do we expect a string of any type?
93  // Is the object a string of any type?
94  // If both are true, then we just return true.
95  H5T_class_t cls_lhs = H5Tget_class(typeBackend->handle.get());
96  H5T_class_t cls_my = H5Tget_class(internalType()());
97  if (cls_lhs == H5T_STRING && cls_my == H5T_STRING) return true;
98 
99  return isA(typeBackend->handle);
100  } catch (std::bad_cast) {
101  std::throw_with_nested(Exception("lhs is not an HH_Type.", ioda_Here()));
102  }
103 }
104 
107 }
108 
110  return Type{std::make_shared<HH_Type>(internalType()), typeid(HH_Type)};
111 }
112 
115 }
116 
118  Dimensions ret;
119 
120  std::vector<hsize_t> dims;
121  if (H5Sis_simple(space()()) < 0) throw Exception("H5Sis_simple failed.", ioda_Here());
122  hssize_t numPoints = H5Sget_simple_extent_npoints(space()());
123  int dimensionality = H5Sget_simple_extent_ndims(space()());
124  if (dimensionality < 0) throw Exception("H5Sget_simple_extent_ndims failed.", ioda_Here());
125  dims.resize(dimensionality);
126  if (H5Sget_simple_extent_dims(space()(), dims.data(), nullptr) < 0)
127  throw Exception("H5Sget_simple_extent_dims failed.", ioda_Here());
128 
129  ret.numElements = gsl::narrow<decltype(Dimensions::numElements)>(numPoints);
130  ret.dimensionality = gsl::narrow<decltype(Dimensions::dimensionality)>(dimensionality);
131  for (const auto& d : dims) ret.dimsCur.push_back(gsl::narrow<Dimensions_t>(d));
132  for (const auto& d : dims) ret.dimsMax.push_back(gsl::narrow<Dimensions_t>(d));
133 
134  return ret;
135 }
136 } // namespace HH
137 } // namespace Engines
138 } // namespace detail
139 } // namespace ioda
140 
141 /// @}
Describe the dimensions of a ioda::Attribute or ioda::Variable.
IODA's error system.
HDF5 engine implementation of Attribute.
HDF5 engine implementation of ioda::detail::Type_Provider.
This class represents attributes, which may be attached to both Variables and Groups.
Definition: Attribute.h:493
The ioda exception class.
Definition: Exception.h:54
Represents the "type" (i.e. integer, string, float) of a piece of data.
Definition: Type.h:123
bool isA() const
Convenience function to check an Attribute's storage type.
Definition: Attribute.h:447
DataType read() const
Read a single value (convenience function).
Definition: Attribute.h:319
detail::Type_Provider * getTypeProvider() const final
Query the backend and get the type provider.
Type getType() const final
Get HDF5-internal type, wrapped as a ioda::Type object.
Dimensions getDimensions() const final
Get Attribute's dimensions.
HH_hid_t internalType() const
Get HDF5-internal type.
Attribute write(gsl::span< char > data, const Type &in_memory_dataType) final
The fundamental write function. Backends overload this function to implement all write operations.
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
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
#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