IODA
Variable_c.h
Go to the documentation of this file.
1 #pragma once
2 /*
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 /*! \defgroup ioda_variable Variables
9  * \brief Provides the C-style interface for ioda::Variable objects.
10  * \ingroup ioda_c_api
11  *
12  * @{
13  * \file Variable_c.h
14  * \brief @link ioda_variable C bindings @endlink for ioda::Variable
15  */
16 
17 #include <stdbool.h>
18 
19 #include "../defs.h"
20 #include "./Has_Attributes_c.h"
21 #include "./String_c.h"
22 #include "./c_binding_macros.h"
23 
24 #ifdef __cplusplus
25 extern "C" {
26 #endif
27 
28 struct ioda_has_attributes;
29 struct ioda_variable;
30 struct ioda_dimensions;
31 struct c_has_attributes;
32 
33 /// \brief Deallocates an variable.
34 /// \param var is the variable to be destructed.
36 
37 /// \brief Access a variable's attributes.
38 /// \see ioda::Group::atts.
39 /// \param[in] var is the variable whose attributes are being accessed.
40 /// \returns A handle to the Has_Attributes container on success (you must free after use).
41 /// \returns NULL on failure.
43 
44 /// \brief Gets an variable's dimensions.
45 /// \param[in] var is the variable.
46 /// \returns A dimension object that contains the variable's dimensions. Must be freed when no
47 /// longer used.
49 
50 /// \brief Resize a variable.
51 /// \param[in] var is the variable.
52 /// \param N is the dimensionality of newDims. Must match current dimensionality.
53 /// \param[in] newDims is a sequence of dimension sizes.
54 /// \returns true on success.
55 /// \returns false on failure.
56 /// \pre var must be valid.
57 /// \pre N and newDims must match the variable's current dimensionality. There is no way to add
58 /// more dimensions to an existing variable.
59 IODA_DL bool ioda_variable_resize(struct ioda_variable* var, size_t N, const long* newDims);
60 
61 /// \brief Attach a dimension scale to a variable.
62 /// \param[in] var is the variable.
63 /// \param DimensionNumber denotes the dimension which you are adding a scale to.
64 /// Counts start at zero.
65 /// \param[in] scale is the dimension scale that will be attached.
66 /// \returns true on success.
67 /// \returns false on failure.
68 /// \pre var and scale must both be valid.
69 /// \pre DimensionNumber must be within the variable's dimensionality.
70 /// \pre scale must not already be attached at the same dimension number.
71 /// \pre var and scale must share the same backend instance.
73  unsigned int DimensionNumber,
74  const struct ioda_variable* scale);
75 
76 /// \brief Detach a dimension scale from a variable.
77 /// \param[in] var is the variable.
78 /// \param DimensionNumber denotes the dimension of var which contains the scale that will be
79 /// detached.
80 /// \param[in] scale is the dimension scale that will be attached.
81 /// \returns true on success.
82 /// \returns false on failure.
83 /// \pre var and scale must both be valid.
84 /// \pre DimensionNumber must be within the variable's dimensionality.
85 /// \pre scale must already be attached at the specified dimension number.
86 /// \pre var and scale must share the same backend instance.
88  unsigned int DimensionNumber,
89  const struct ioda_variable* scale);
90 
91 /// \brief Convenience function to set a sequence of scales on a variable.
92 /// \param[in] var is the variable.
93 /// \param n_dims is the size of the dims array.
94 /// \param[in] dims is a sequence of dimension scales that will be attached to var.
95 /// dims[0] will be attached to var along dimension 0,
96 /// dims[1] will be attached to var along dimension 1, and so on.
97 /// \returns true on success.
98 /// \returns false on failure.
99 /// \pre var must be valid.
100 /// \pre dims must be non-null, and each scale in dims must be valid and share the same backend
101 /// instance as var.
102 /// \pre The scales in dims should not already be attached to var at their expected places.
103 /// \pre n_dims must be less than or equal to the dimensionality of var.
104 IODA_DL bool ioda_variable_setDimScale(struct ioda_variable* var, size_t n_dims,
105  const struct ioda_variable* const* dims);
106 
107 /// \brief Check if a variable acts as a dimension scale.
108 /// \param[in] var is the variable.
109 /// \returns 1 if yes.
110 /// \returns 0 if no.
111 /// \returns -1 on error.
112 /// \pre var must be valid.
114 
115 /// \brief Convert a variable into a dimension scale.
116 /// \param[in] var is the variable that will become a scale.
117 /// \param sz_name is strlen(dimensionScaleName). Fortran compatability.
118 /// \param[in] dimensionScaleName is the "name" of the dimension scale. This name does
119 /// not need to correspond to the variable's name, and acts as a convenience label
120 /// when reading data.
121 /// \returns true on success.
122 /// \returns false on failure.
123 /// \pre var must be valid.
124 /// \pre dimensionScaleName must be valid. If unused, it should be set to an empty string and
125 /// not NULL.
126 IODA_DL bool ioda_variable_setIsDimensionScale(struct ioda_variable* var, size_t sz_name,
127  const char* dimensionScaleName);
128 
129 /**
130  * \brief Get the name of the dimension scale.
131  * \param[in] var is the dimension scale.
132  * \param[out] out is the output buffer that will hold the name of the dimension scale.
133  * This will always be a null-terminated string. If len_out is smaller than the length
134  * of the dimension scale's name, then out will be truncated to fit.
135  * \param len_out is the length (size) of the output buffer, in bytes.
136  * \returns The minimum size of an output buffer needed to fully read the scale name.
137  * Callers should check that the return value is less than len_out. If it is not, then
138  * the output buffer is too small and should be expanded. The output buffer is
139  * always at least one byte in size (the null byte).
140  * \returns 0 if an error has occurred.
141  * \pre var must be valid.
142  * \pre out must be valid (not null!)
143  **/
144 IODA_DL size_t ioda_variable_getDimensionScaleName(const struct ioda_variable* var, size_t len_out,
145  char* out);
146 
147 /**
148  * \brief Is the variable "scale" attached as dimension "DimensionNumber" to variable "var"?
149  * \param[in] var is the variable.
150  * \param DimensionNumber is the dimension number. Numbering starts at zero.
151  * \param[in] scale is the candidate dimension scale.
152  * \returns 1 if attached.
153  * \returns 0 if not attached.
154  * \returns -1 on error.
155  * \pre var and scale must both be valid and from the same backend engine instance.
156  * \pre DimensionNumber must be less than the variable's dimensionality.
157  **/
159  unsigned int DimensionNumber,
160  const struct ioda_variable* scale);
161 
162 /*!
163  * \defgroup ioda_variable_isa ioda_variable_isa
164  * \brief Checks a variable's type.
165  * \details This is documentation for a series of functions in C that attempt to emulate C++
166  * templates using macro magic. The template parameter SUFFIX is written into the function
167  * name. Ex:, to check if a variable is an integer, call
168  * ```ioda_variable_isa_int```. To check if a variable is a float, try
169  * ```ioda_variable_isa_float```.
170  * \tparam SUFFIX is the type (int, long, int64_t) that is appended
171  * to this function name in the C interface.
172  * \param var is the variable.
173  * \return an integer denoting yes (> 0), no (== 0), or failure (< 0).
174  * @{
175  */
176 
177 // isA - int ioda_variable_isa_char(const ioda_variable*);
178 /// \def IODA_VARIABLE_ISA_TEMPLATE
179 /// \brief See @link ioda_variable_isa ioda_variable_isa @endlink
180 /// \see ioda_variable_isa
181 #define IODA_VARIABLE_ISA_TEMPLATE(funcnamestr, junk) \
182  IODA_DL int funcnamestr(const struct ioda_variable* var);
184 
185 /*!
186  * @}
187  * \defgroup ioda_variable_write_full ioda_variable_write_full
188  * \brief Write an entire variable.
189  * \details This is documentation for a series of functions in C that attempt to emulate C++
190  * templates using macro magic. The template parameter SUFFIX is written into the function
191  * name. Ex:, to write an integer variable, call
192  * ```ioda_variable_write_full_int```. To write a float variable, try
193  * ```ioda_variable_write_full_float```.
194  * \tparam SUFFIX is the type (int, long, int64_t) that is appended to this function name
195  * in the C interface.
196  * \param var is the variable.
197  * \return true on success, false on failure.
198  * @{
199  */
200 
201 // write_full - bool ioda_variable_write_full_char(ioda_variable*, size_t, const char*);
202 /// \def IODA_VARIABLE_WRITE_FULL_TEMPLATE
203 /// \brief See @link ioda_variable_write_full ioda_variable_write_full @endlink
204 /// \see ioda_variable_write_full
205 #define IODA_VARIABLE_WRITE_FULL_TEMPLATE(funcnamestr, Type) \
206  IODA_DL bool funcnamestr(struct ioda_variable* var, size_t sz, const Type* vals);
208 
209 /// \brief Write a sequence of strings to a variable.
210 /// \param[in] var is the variable.
211 /// \param sz is the number of strings to write. For ioda_variable_write_full_str, this
212 /// must match var's current number of elements.
213 /// \param[in] vals is a sequence of sz NULL-terminated strings.
214 /// \returns true on success.
215 /// \returns false on failure.
216 /// \pre var must be a valid string-typed variable.
217 /// \pre sz must equal the variable's current number of elements.
218 /// \pre vals must be valid and have at least length sz.
219 /// \pre Each val in vals must be a NULL-terminated string. Each val must not be NULL itself.
220 IODA_DL bool ioda_variable_write_full_str(struct ioda_variable* var, size_t sz,
221  const char* const* vals);
222 
223 /*!
224  * @}
225  * \defgroup ioda_variable_read_full ioda_variable_read_full
226  * \brief Read an entire variable.
227  * \details This is documentation for a series of functions in C that attempt to emulate C++
228  * templates using macro magic. The template parameter SUFFIX is written into the function
229  * name. Ex:, to read an integer variable, call
230  * ```ioda_variable_read_full_int```. To read a float variable, try
231  * ```ioda_variable_read_full_float```.
232  * \tparam SUFFIX is the type (int, long, int64_t) that is appended to this function name
233  * in the C interface.
234  * \param[in] var is the variable.
235  * \param sz is the size of the output buffer vals. Must match the number of elements in var.
236  * \param[out] vals is the output buffer.
237  * \return true on success, false on failure.
238  * \pre sz must match the current number of elements in the variable.
239  * \pre var and vals must be valid.
240  * @{
241  */
242 
243 // read_full - void ioda_variable_read_full_char(const ioda_variable*, size_t, char*);
244 /// \def IODA_VARIABLE_READ_FULL_TEMPLATE
245 /// \brief See @link ioda_variable_read_full ioda_variable_read_full @endlink
246 /// \see ioda_variable_read_full
247 #define IODA_VARIABLE_READ_FULL_TEMPLATE(funcnamestr, Type) \
248  IODA_DL bool funcnamestr(const struct ioda_variable* var, size_t sz, Type* vals);
250 
251 /// \brief Read strings from a variable.
252 /// \param[in] var is the variable.
253 /// \returns a sequence of strings. These should be freed by the caller.
254 /// \returns NULL on failure.
255 /// \pre var must be valid.
256 /// \pre var must contain strings. Other types cannot be transparently converted.
258 
259 /*! @}
260  * @brief Class-like encapsulation of C variable functions.
261  * @see c_ioda for an example.
262  * @see use_c_ioda for an example.
263  */
264 struct c_variable {
265  void (*destruct)(struct ioda_variable*);
266  struct ioda_has_attributes* (*getAtts)(const struct ioda_variable*);
267  struct ioda_dimensions* (*getDimensions)(const struct ioda_variable*);
268  bool (*resize)(struct ioda_variable*, size_t, const long*);
269  bool (*attachDimensionScale)(struct ioda_variable*, unsigned int, const struct ioda_variable*);
270  bool (*detachDimensionScale)(struct ioda_variable*, unsigned int, const struct ioda_variable*);
271  bool (*setDimScale)(struct ioda_variable*, size_t, const struct ioda_variable* const*);
272  int (*isDimensionScale)(const struct ioda_variable*);
273  bool (*setIsDimensionScale)(struct ioda_variable*, size_t, const char*);
274  size_t (*getDimensionScaleName)(const struct ioda_variable*, size_t, char*);
275  int (*isDimensionScaleAttached)(const struct ioda_variable*, unsigned int,
276  const struct ioda_variable*);
277 
278  // bool isA_char(const ioda_attribute*);
279 #define IODA_VARIABLE_ISA_FUNC_TEMPLATE(shortnamestr, basenamestr) \
280  int (*shortnamestr)(const struct ioda_variable*);
282 
283  // void write_full_char(ioda_variable*, size_t, const Type*);
284 #define IODA_VARIABLE_WRITE_FULL_FUNC_TEMPLATE(shortnamestr, basenamestr, Type) \
285  bool (*shortnamestr)(struct ioda_variable*, size_t, const Type*);
286  C_TEMPLATE_FUNCTION_DECLARATION_4_NOSTR(write_full, ioda_variable_write_full,
288  bool (*write_full_str)(struct ioda_variable*, size_t, const char* const*);
289 
290  // void read_full_char(const ioda_variable*, size_t, char*);
291 #define IODA_VARIABLE_READ_FULL_FUNC_TEMPLATE(shortnamestr, basenamestr, Type) \
292  bool (*shortnamestr)(const struct ioda_variable*, size_t, Type*);
293  C_TEMPLATE_FUNCTION_DECLARATION_4_NOSTR(read_full, ioda_variable_read_full,
295  struct ioda_string_ret_t* (*read_full_str)(const struct ioda_variable*);
296 
297  struct c_has_attributes atts;
298 };
299 
300 #ifdef __cplusplus
301 }
302 #endif
303 
304 /// @} // End Doxygen block
C bindings for ioda::Has_Attributes
C bindings .
#define IODA_VARIABLE_ISA_FUNC_TEMPLATE(shortnamestr, basenamestr)
Definition: Variable_c.h:279
#define IODA_VARIABLE_READ_FULL_FUNC_TEMPLATE(shortnamestr, basenamestr, Type)
Definition: Variable_c.h:291
#define IODA_VARIABLE_WRITE_FULL_FUNC_TEMPLATE(shortnamestr, basenamestr, Type)
Definition: Variable_c.h:284
C bindings interface to templated C++ ioda classes and functions.
#define IODA_DL
A preprocessor tag that indicates that a symbol is to be exported/imported.
Definition: defs.h:110
#define IODA_VARIABLE_ISA_TEMPLATE(funcnamestr, junk)
See ioda_variable_isa .
Definition: Variable_c.h:181
C_TEMPLATE_FUNCTION_DECLARATION(ioda_variable_isa, IODA_VARIABLE_ISA_TEMPLATE)
IODA_DL struct ioda_string_ret_t * ioda_variable_read_full_str(const struct ioda_variable *var)
Read strings from a variable.
#define IODA_VARIABLE_READ_FULL_TEMPLATE(funcnamestr, Type)
See ioda_variable_read_full .
Definition: Variable_c.h:247
C_TEMPLATE_FUNCTION_DEFINITION_NOSTR(ioda_variable_write_full, IODA_VARIABLE_WRITE_FULL_TEMPLATE)
#define IODA_VARIABLE_WRITE_FULL_TEMPLATE(funcnamestr, Type)
See ioda_variable_write_full .
Definition: Variable_c.h:205
IODA_DL bool ioda_variable_write_full_str(struct ioda_variable *var, size_t sz, const char *const *vals)
Write a sequence of strings to a variable.
Definition: Variable_c.cpp:157
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_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
IODA_DL struct ioda_dimensions * ioda_variable_get_dimensions(const struct ioda_variable *)
Gets an variable's dimensions.
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
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 struct ioda_has_attributes * ioda_variable_atts(const struct ioda_variable *var)
Access a variable's attributes.
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
list newDims
Definition: 05-ObsGroup.py:95
Class-like encapsulation of C has_attributes functions.
Class-like encapsulation of C variable functions.
Definition: Variable_c.h:264
int(* isDimensionScale)(const struct ioda_variable *)
Definition: Variable_c.h:272
bool(* write_full_str)(struct ioda_variable *, size_t, const char *const *)
Definition: Variable_c.h:288
void(* destruct)(struct ioda_variable *)
Definition: Variable_c.h:265
bool(* resize)(struct ioda_variable *, size_t, const long *)
Definition: Variable_c.h:268
C_TEMPLATE_FUNCTION_DECLARATION_3(isA, ioda_variable_isa, IODA_VARIABLE_ISA_FUNC_TEMPLATE)
size_t(* getDimensionScaleName)(const struct ioda_variable *, size_t, char *)
Definition: Variable_c.h:274
bool(* setDimScale)(struct ioda_variable *, size_t, const struct ioda_variable *const *)
Definition: Variable_c.h:271
bool(* attachDimensionScale)(struct ioda_variable *, unsigned int, const struct ioda_variable *)
Definition: Variable_c.h:269
int(* isDimensionScaleAttached)(const struct ioda_variable *, unsigned int, const struct ioda_variable *)
Definition: Variable_c.h:275
struct c_has_attributes atts
Definition: Variable_c.h:297
bool(* detachDimensionScale)(struct ioda_variable *, unsigned int, const struct ioda_variable *)
Definition: Variable_c.h:270
bool(* setIsDimensionScale)(struct ioda_variable *, size_t, const char *)
Definition: Variable_c.h:273
C_TEMPLATE_FUNCTION_DECLARATION_4_NOSTR(write_full, ioda_variable_write_full, IODA_VARIABLE_WRITE_FULL_FUNC_TEMPLATE)
Return type when arrays of strings are encountered.
Definition: String_c.h:24