IODA
Variable_c.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_variable
8  *
9  * @{
10  * \file Variable_c.cpp
11  * \brief @link ioda_variable C bindings @endlink for ioda::Variable
12  */
13 
14 #include "ioda/C/Variable_c.h"
15 
16 #include "./structs_c.h"
18 #include "ioda/Exception.h"
19 #include "ioda/Types/Type.h"
21 
22 extern "C" {
23 
25  C_TRY;
26  Expects(var != nullptr);
27  delete var;
29 }
30 
32  ioda_has_attributes* res = nullptr;
33  C_TRY;
34  Expects(var != nullptr);
35  res = new ioda_has_attributes;
36  res->atts = var->var.atts;
37  C_CATCH_RETURN_FREE(res, NULL, res);
38 }
39 
41  ioda_dimensions* res = nullptr;
42  C_TRY;
43  Expects(var != nullptr);
44  res = new ioda_dimensions;
45  Expects(res != nullptr);
46  res->d = var->var.getDimensions();
47  C_CATCH_RETURN_FREE(res, nullptr, res);
48 }
49 
50 bool ioda_variable_resize(struct ioda_variable* var, size_t N, const long* newDims) {
51  C_TRY;
52  Expects(var != nullptr);
54  for (size_t i = 0; i < N; ++i)
55  nd[i] = gsl::narrow<ioda::Selection::VecDimensions_t::value_type>(newDims[i]);
56  var->var.resize(nd);
57  C_CATCH_AND_RETURN(true, false);
58 }
59 
60 bool ioda_variable_attachDimensionScale(struct ioda_variable* var, unsigned int DimensionNumber,
61  const struct ioda_variable* scale) {
62  C_TRY;
63  Expects(var != nullptr);
64  Expects(scale != nullptr);
65  var->var.attachDimensionScale(DimensionNumber, scale->var);
66  C_CATCH_AND_RETURN(true, false);
67 }
68 
69 bool ioda_variable_detachDimensionScale(struct ioda_variable* var, unsigned int DimensionNumber,
70  const struct ioda_variable* scale) {
71  C_TRY;
72  Expects(var != nullptr);
73  Expects(scale != nullptr);
74  var->var.detachDimensionScale(DimensionNumber, scale->var);
75  C_CATCH_AND_RETURN(true, false);
76 }
77 
78 bool ioda_variable_setDimScale(struct ioda_variable* var, size_t N,
79  const struct ioda_variable* const* dims) {
80  C_TRY;
81  Expects(var != nullptr);
82  Expects(dims != nullptr);
83  std::vector<ioda::Variable> newDims;
84  newDims.reserve(N);
85  for (size_t i = 0; i < N; ++i) {
86  Expects(dims[i] != nullptr);
87  newDims.push_back(dims[i]->var);
88  }
89  var->var.setDimScale(newDims);
90  C_CATCH_AND_RETURN(true, false);
91 }
92 
94  C_TRY;
95  Expects(var != nullptr);
96  bool res = var->var.isDimensionScale();
97  C_CATCH_AND_RETURN((res) ? 1 : 0, -1);
98 }
99 
101  const char* dimensionScaleName) {
102  C_TRY;
103  Expects(var != nullptr);
104  Expects(dimensionScaleName != nullptr);
105  var->var.setIsDimensionScale(std::string(dimensionScaleName, sz));
106  C_CATCH_AND_RETURN(true, false);
107 }
108 
109 size_t ioda_variable_getDimensionScaleName(const struct ioda_variable* var, size_t N, char* out) {
110  C_TRY;
111  Expects(var != nullptr);
112  std::string name;
114  if (name.size() == SIZE_MAX) throw ioda::Exception(
115  "Dimension scale name is too large.", ioda_Here());
116  if (!out) return name.size() + 1;
117  Expects(out != nullptr);
118  ioda::detail::COMPAT_strncpy_s(out, N, name.data(), name.size() + 1);
119 
120  C_CATCH_AND_RETURN(name.size() + 1, 0);
121 }
122 
124  unsigned int DimensionNumber,
125  const struct ioda_variable* scale) {
126  C_TRY;
127  Expects(var != nullptr);
128  Expects(scale != nullptr);
129  bool ret = var->var.isDimensionScaleAttached(DimensionNumber, scale->var);
130  C_CATCH_AND_RETURN((ret) ? 1 : 0, -1);
131 }
132 
133 // isA
134 #define IODA_VARIABLE_ISA_IMPL(funcnamestr, Type) \
135  IODA_DL int funcnamestr(const ioda_variable* var) { \
136  C_TRY; \
137  Expects(var != nullptr); \
138  bool ret = var->var.isA<Type>(); \
139  C_CATCH_AND_RETURN((ret) ? 1 : 0, -1); \
140  }
141 
143 
144 // write
145 
146 #define IODA_VARIABLE_WRITE_FULL(funcnamestr, Type) \
147  IODA_DL bool funcnamestr(ioda_variable* var, size_t sz, const Type* vals) { \
148  C_TRY; \
149  Expects(var != nullptr); \
150  Expects(vals != nullptr); \
151  var->var.write<Type>(gsl::span<const Type>(vals, sz)); \
152  C_CATCH_AND_RETURN(true, false); \
153  }
154 
156 
157 IODA_DL bool ioda_variable_write_full_str(ioda_variable* var, size_t sz, const char* const* vals) {
158  C_TRY;
159  Expects(var != nullptr);
160  Expects(vals != nullptr);
161  std::vector<std::string> vdata(sz);
162  for (size_t i = 0; i < sz; ++i) vdata[i] = std::string(vals[i]);
163  var->var.write<std::string>(vdata);
164  C_CATCH_AND_RETURN(true, false);
165 }
166 
167 // read
168 
169 #define IODA_VARIABLE_READ_FULL(funcnamestr, Type) \
170  IODA_DL bool funcnamestr(const ioda_variable* var, size_t sz, Type* vals) { \
171  C_TRY; \
172  Expects(var != nullptr); \
173  Expects(vals != nullptr); \
174  var->var.read<Type>(gsl::span<Type>(vals, sz)); \
175  C_CATCH_AND_RETURN(true, false); \
176  }
177 
179 
181  C_TRY;
182  Expects(var != nullptr);
183  std::vector<std::string> vdata;
184  var->var.read<std::string>(vdata);
185 
186  C_CATCH_AND_RETURN(create_str_vector_c(vdata), nullptr);
187 }
188 }
189 
190 /// @}
IODA's error system.
Interfaces for ioda::Type and related classes.
Interfaces for ioda::Variable and related classes.
C bindings for ioda::Variable
C bindings interface to templated C++ ioda classes and functions.
The ioda exception class.
Definition: Exception.h:54
virtual Variable attachDimensionScale(unsigned int DimensionNumber, const Variable &scale)
Attach a dimension scale to this Variable.
Definition: Variable.cpp:184
Variable setDimScale(const std::vector< Variable > &dims)
Set dimensions (convenience function to several invocations of attachDimensionScale).
Definition: Variable.cpp:212
virtual Variable setIsDimensionScale(const std::string &dimensionScaleName)
Designate this table as a dimension scale.
Definition: Variable.cpp:264
Has_Attributes atts
Attributes.
Definition: Variable.h:71
virtual bool isDimensionScaleAttached(unsigned int DimensionNumber, const Variable &scale) const
Is a dimension scale attached to this Variable in a certain position?
Definition: Variable.cpp:288
virtual bool isDimensionScale() const
Is this Variable used as a dimension scale?
Definition: Variable.cpp:251
virtual Variable detachDimensionScale(unsigned int DimensionNumber, const Variable &scale)
Detach a dimension scale.
Definition: Variable.cpp:198
virtual Dimensions getDimensions() const
Definition: Variable.cpp:160
std::string getDimensionScaleName() const
Get the name of this Variable's defined dimension scale.
Definition: Variable.h:217
virtual Variable read(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all) const
Read the Variable - as char array. Ordering is row-major.
Definition: Variable.cpp:330
virtual Variable write(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all)
The fundamental write function. Backends overload this function to implement all write operations.
Definition: Variable.cpp:317
virtual Variable resize(const std::vector< Dimensions_t > &newDims)
Resize the variable.
Definition: Variable.cpp:172
#define C_CATCH_AND_RETURN(retval_on_success, retval_on_error)
This macro catches C++ exceptions.
#define C_CATCH_AND_TERMINATE
Catch C++ exceptions before they go across code boundaries.
#define C_CATCH_RETURN_FREE(retval_on_success, retval_on_error, freeable)
Like C_CATCH_AND_RETURN, but free any in-function allocated C resource before returning to avoid memo...
#define C_TRY
Goes with C_CATCH_AND_TERMINATE.
#define IODA_DL
A preprocessor tag that indicates that a symbol is to be exported/imported.
Definition: defs.h:110
std::vector< Dimensions_t > VecDimensions_t
Definition: Selection.h:50
IODA_DL size_t ioda_variable_getDimensionScaleName(const struct ioda_variable *var, size_t len_out, char *out)
Get the name of the dimension scale.
Definition: Variable_c.cpp:109
IODA_DL bool ioda_variable_write_full_str(ioda_variable *var, size_t sz, const char *const *vals)
Write a sequence of strings to a variable.
Definition: Variable_c.cpp:157
#define IODA_VARIABLE_WRITE_FULL(funcnamestr, Type)
Definition: Variable_c.cpp:146
IODA_DL bool ioda_variable_setDimScale(struct ioda_variable *var, size_t n_dims, const struct ioda_variable *const *dims)
Convenience function to set a sequence of scales on a variable.
Definition: Variable_c.cpp:78
IODA_DL int ioda_variable_isDimensionScale(const struct ioda_variable *var)
Check if a variable acts as a dimension scale.
Definition: Variable_c.cpp:93
#define IODA_VARIABLE_READ_FULL(funcnamestr, Type)
Definition: Variable_c.cpp:169
IODA_DL bool ioda_variable_detachDimensionScale(struct ioda_variable *var, unsigned int DimensionNumber, const struct ioda_variable *scale)
Detach a dimension scale from a variable.
Definition: Variable_c.cpp:69
IODA_DL bool ioda_variable_setIsDimensionScale(struct ioda_variable *var, size_t sz_name, const char *dimensionScaleName)
Convert a variable into a dimension scale.
Definition: Variable_c.cpp:100
IODA_DL void ioda_variable_destruct(struct ioda_variable *var)
Deallocates an variable.
Definition: Variable_c.cpp:24
C_TEMPLATE_FUNCTION_DEFINITION_NOSTR(ioda_variable_write_full, IODA_VARIABLE_WRITE_FULL)
IODA_DL bool ioda_variable_resize(struct ioda_variable *var, size_t N, const long *newDims)
Resize a variable.
Definition: Variable_c.cpp:50
IODA_DL ioda_string_ret_t * ioda_variable_read_full_str(const ioda_variable *var)
Definition: Variable_c.cpp:180
#define IODA_VARIABLE_ISA_IMPL(funcnamestr, Type)
Definition: Variable_c.cpp:134
C_TEMPLATE_FUNCTION_DEFINITION(ioda_variable_isa, IODA_VARIABLE_ISA_IMPL)
ioda_has_attributes * ioda_variable_atts(const ioda_variable *var)
Definition: Variable_c.cpp:31
IODA_DL int ioda_variable_isDimensionScaleAttached(const struct ioda_variable *var, unsigned int DimensionNumber, const struct ioda_variable *scale)
Is the variable "scale" attached as dimension "DimensionNumber" to variable "var"?
Definition: Variable_c.cpp:123
IODA_DL bool ioda_variable_attachDimensionScale(struct ioda_variable *var, unsigned int DimensionNumber, const struct ioda_variable *scale)
Attach a dimension scale to a variable.
Definition: Variable_c.cpp:60
ioda_dimensions * ioda_variable_get_dimensions(const ioda_variable *var)
Definition: Variable_c.cpp:40
list newDims
Definition: 05-ObsGroup.py:95
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
#define ioda_Here()
ioda::Dimensions d
Definition: structs_c.h:36
ioda::Has_Attributes atts
Definition: structs_c.h:24
Return type when arrays of strings are encountered.
Definition: String_c.h:24
ioda::Variable var
Definition: structs_c.h:33
C wrappers for ioda classes and structures. Private header. Can have C++!
ioda_string_ret_t * create_str_vector_c(const std::vector< std::string > &vdata) noexcept
Definition: structs_c.h:43