IODA
03-VariablesIntro.c
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_c_ex
8  *
9  * @{
10  *
11  * \defgroup ioda_c_ex_3 Ex 3: Introduction to Variables
12  * \brief Basic usage of Variables using the C interface
13  * \details This example parallels the C++ examples.
14  * \see 03-VariablesIntro.cpp for comments and the walkthrough.
15  *
16  * @{
17  *
18  * \file 03-VariablesIntro.c
19  * \brief Basic usage of Variables using the C interface
20  * \see 03-VariablesIntro.cpp for comments and the walkthrough.
21  **/
22 
23 #include <stdio.h>
24 #include <string.h>
25 
26 #include "ioda/C/ioda_c.h"
27 #include "ioda/defs.h" // Always include this first.
28 
29 #define sslin(x) #x
30 #define slin(x) sslin(x)
31 #define doErr \
32  { \
33  errlin = "Error in " __FILE__ " at line " slin(__LINE__) ".\n"; \
34  goto hadError; \
35  }
36 
37 int main(int argc, char** argv) {
38  int errval = 0;
39  const char* errlin = NULL;
40  struct c_ioda ioda = use_c_ioda();
41  struct ioda_group* g = NULL;
42  struct ioda_has_variables* gvars = NULL;
43  struct ioda_has_attributes* v1atts = NULL;
44  struct ioda_attribute* v1a1 = NULL;
45  struct ioda_variable_creation_parameters *params_default = NULL, *p1 = NULL;
46  struct ioda_variable *var1 = NULL, *var2 = NULL, *var3 = NULL;
47  struct ioda_variable *mixed_int_float_1 = NULL, *removable_var1 = NULL;
48  struct ioda_variable *var1_reopened = NULL, *var2_reopened = NULL;
49  struct ioda_string_ret_t *list_of_vars = NULL, *str_list = NULL;
50  struct ioda_dimensions* dims = NULL;
51  struct ioda_variable* var_strs = NULL;
52 
53  // C++ line 46
54  g = ioda.Engines.constructFromCmdLine(argc, argv, "Example-03-C.hdf5");
55  if (!g) doErr;
56 
57  gvars = ioda.Group.getVars(g);
58  if (!gvars) doErr;
59 
60  // C++ line 61
61  params_default = ioda.VariableCreationParams.create();
62  if (!params_default) doErr;
63 
64  // We are creating a 2x3 array and writing it.
65  const size_t var1_dimensionality = 2;
66  long var1_dimsCur[2] = {2, 3};
67  long var1_dimsMax[2] = {2, 3};
68  var1 = ioda.Has_Variables.create_int(gvars, 5, "var-1", var1_dimensionality, var1_dimsCur,
69  var1_dimsMax, params_default);
70  if (!var1) doErr;
71 
72  int var1_data[6] = {1, 2, 3, 4, 5, 6};
73  // Note: The "6" here is the number of elements of var1_data that we are writing.
74  if (!ioda.Variable.write_full_int(var1, 6, var1_data)) doErr;
75 
76  v1atts = ioda.Variable.getAtts(var1);
77  const long s1sz = 1;
78  v1a1 = ioda.Has_Attributes.create_str(v1atts, 4, "Test", 1, &s1sz);
79  if (!v1a1) doErr;
80  const char* const v1a1_data[] = {"This is a test."};
81  // We are writing a single variable-length string, which is why we are passing a "1" here.
82  if (!ioda.Attribute.write_str(v1a1, 1, v1a1_data)) doErr;
83 
84  // C++ line 73
85  const size_t var2_dimensionality = 3;
86  long var2_dims[3] = {2, 3, 4};
87  var2 = ioda.Has_Variables.create_float(gvars, 5, "var-2", var2_dimensionality, var2_dims,
88  var2_dims, params_default);
89  if (!var2) doErr;
90 
91  float var2_data[24] = {1.1f, 2.2f, 3.14159f, 4, 5, 6, 7, 8, 9, 10, 11.5f, 12.6f,
92  13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24};
93  if (!ioda.Variable.write_full_float(var2, 24, var2_data)) doErr;
94 
95  // C++ line 85
96  p1 = ioda.VariableCreationParams.clone(params_default); // Slight difference in C++. Code cov.
97  if (!p1) doErr;
98  ptrdiff_t chunks[2] = {200, 3};
99  ioda.VariableCreationParams.chunking(p1, true, 2, chunks);
100  ioda.VariableCreationParams.setFillValue_int(p1, -999);
101  ioda.VariableCreationParams.compressWithSZIP(p1, 0, 16); // Turn on SZIP compression.
102  ioda.VariableCreationParams.noCompress(p1); // Turn off SZIP compression
103  ioda.VariableCreationParams.compressWithGZIP(p1, 6); // Turn on GZIP compression.
104 
105  const size_t var3_dimensionality = 2;
106  long var3_dimsCur[2] = {200, 3};
107  long var3_dimsMax[2] = {2000, 3};
108  var3 = ioda.Has_Variables.create_int(gvars, 5, "var-3", var3_dimensionality, var3_dimsCur,
109  var3_dimsMax, p1);
110  if (!var3) doErr;
111 
112  long var3_dimsNew[2] = {400, 3};
113  if (!ioda.Variable.resize(var3, 2, var3_dimsNew)) doErr;
114 
115  // Skip C++ lines 97 - 123. C hoes not have these containers.
116 
117  // C++ line 127
118  long mixed_int_float_1_dims[] = {1};
119  // mixed_int_float_1 = ioda.Group.vars.create_int(gvars, "bad-int-1", 1, mixed_int_float_1_dims,
120  // mixed_int_float_1_dims, params_default); if (!mixed_int_float_1) doErr; float
121  // mixed_int_float_1_test_var = 3.1f; bool mixed_int_float_1_res_write =
122  // ioda.Variable.write_full_float(mixed_int_float_1, 1, &mixed_int_float_1_test_var); if
123  // (!mixed_int_float_1_res_write) doErr;
124 
125  // Skip C++ lines 134-182.
126  // C does not have Eigen.
127 
128  // C++ line 184
129  list_of_vars = ioda.Group.vars.list(gvars);
130  if (!list_of_vars) doErr;
131  // if (list_of_vars->n != 4) doErr;
132  // list_of_vars->strings[0 - 3] are the strings!
133 
134  // C++ line 195
135  if (ioda.Group.vars.exists(gvars, 5, "var-2") <= 0) doErr;
136 
137  // C++ line 197
138  removable_var1
139  = ioda.Group.vars.create_int(gvars, 15, "removable-int-1", 1, mixed_int_float_1_dims,
140  mixed_int_float_1_dims, params_default);
141  if (!removable_var1) doErr;
142  ioda.Variable.destruct(removable_var1); // We have to release the variable before we delete it!
143  removable_var1 = NULL;
144  if (!ioda.Group.vars.remove(gvars, 15, "removable-int-1")) doErr;
145 
146  var1_reopened = ioda.Group.vars.open(gvars, 5, "var-1");
147  if (!var1_reopened) doErr;
148 
149  var2_reopened = ioda.Group.vars.open(gvars, 5, "var-2");
150  if (!var2_reopened) doErr;
151 
152  // C++ line 207
153  // TODO(rhoneyager): Add in error handling for
154  // forgivable exceptions.
155 
156  // C++ line 217
157  dims = ioda.Variable.getDimensions(var1_reopened);
158  if (!dims) doErr;
159  size_t dimensionality = 0;
160  ptrdiff_t dim = 0;
161  if (!ioda.Dimensions.getDimensionality(dims, &dimensionality)) doErr;
162  if (dimensionality != 2) doErr;
163  if (!ioda.Dimensions.getDimCur(dims, 0, &dim)) doErr;
164  if (dim != 2) doErr;
165  if (!ioda.Dimensions.getDimCur(dims, 1, &dim)) doErr;
166  if (dim != 3) doErr;
167  if (!ioda.Dimensions.getDimMax(dims, 0, &dim)) doErr;
168  if (dim != 2) doErr;
169  if (ioda.Variable.isA_int(var1_reopened) <= 0) doErr;
170 
171  // C++ line 236
172  int check_var1[6] = {0, 0, 0, 0, 0, 0};
173  if (!ioda.Variable.read_full_int(var1_reopened, 6, check_var1)) doErr;
174 
175  // C++ line 271
176  // double mixed_double_read[6] = { 0,0,0,0,0,0 };
177  // bool mixed_read_success = ioda.Variable.read_full_double(var1_reopened, 6, mixed_double_read);
178  // if (!mixed_read_success) doErr;
179 
180  // Some more stuff, not in the C++ example
181 
182  // Strings are a bit special, so we show how to read and write these separately.
183  // Writing strings
184  const char* strings[] = {"str-1", "string 2", "s3", "Hello, world!"};
185  const long n_strs = 4;
186  var_strs
187  = ioda.Has_Variables.create_str(gvars, 8, "var_strs", 1, &n_strs, &n_strs, params_default);
188  if (!var_strs) doErr;
189  if (!ioda.Variable.write_full_str(var_strs, n_strs, strings)) doErr;
190 
191  // String read test
192  // Read into str_list. Free when done.
193  str_list = ioda.Variable.read_full_str(var_strs);
194  if (!str_list) doErr;
195  if (str_list->n != 4) doErr;
196  // Compare each string, up to 20 chars. This size is bigger than the input strings.
197  for (size_t i = 0; i < 4; ++i)
198  if (strncmp(strings[i], str_list->strings[i], 20) != 0) doErr;
199 
200  goto cleanup;
201 
202 hadError:
203  printf("%s", (errlin) ? errlin : "An unknown error has occurred somewhere.");
204  errval = 1;
205 
206 cleanup:
207  if (g) ioda.Group.destruct(g);
208  if (gvars) ioda.Has_Variables.destruct(gvars);
209  if (v1atts) ioda.Has_Attributes.destruct(v1atts);
210  if (params_default) ioda.VariableCreationParams.destruct(params_default);
211  if (p1) ioda.VariableCreationParams.destruct(p1);
212  if (var1) ioda.Variable.destruct(var1);
213  if (v1a1) ioda.Attribute.destruct(v1a1);
214  if (var2) ioda.Variable.destruct(var2);
215  if (var3) ioda.Variable.destruct(var3);
216  if (mixed_int_float_1) ioda.Variable.destruct(mixed_int_float_1);
217  if (removable_var1) ioda.Variable.destruct(removable_var1);
218  if (var1_reopened) ioda.Variable.destruct(var1_reopened);
219  if (var2_reopened) ioda.Variable.destruct(var2_reopened);
220  if (list_of_vars) ioda.Strings.destruct(list_of_vars);
221  if (var_strs) ioda.Variable.destruct(var_strs);
222  if (str_list) ioda.Strings.destruct(str_list);
223  if (dims) ioda.Dimensions.destruct(dims);
224 
225  return errval;
226 }
This class represents attributes, which may be attached to both Variables and Groups.
Definition: Attribute.h:493
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
This class exists inside of ioda::Group or ioda::Variable and provides the interface to manipulating ...
This class exists inside of ioda::Group and provides the interface to manipulating Variables.
Variables store data!
Definition: Variable.h:680
Has_Variables vars
Use this to access variables.
Definition: Group.h:123
virtual Variable open(const std::string &name) const
Open a Variable by name.
virtual std::vector< std::string > list() const
virtual bool exists(const std::string &name) const
Does a Variable with the specified name exist?
virtual void remove(const std::string &name)
Delete an Attribute with the specified name.
virtual Dimensions getDimensions() const
Definition: Variable.cpp:160
virtual Variable resize(const std::vector< Dimensions_t > &newDims)
Resize the variable.
Definition: Variable.cpp:172
Common preprocessor definitions used throughout IODA.
IODA_DL struct c_ioda use_c_ioda()
Creates and returns a c_ioda struct that has all of the function pointers filled in.
Definition: ioda_c.cpp:18
int main(int argc, char **argv)
#define doErr
C bindings for ioda-engines.
The backends that implement the ioda-engines functionality.
Definition: Capabilities.h:21
A few chunking strategies for Variables.
Definition: Has_Variables.h:47
Definition: ioda_c.h:35
Describes the dimensions of an Attribute or Variable.
Definition: Dimensions.h:22
Return type when arrays of strings are encountered.
Definition: String_c.h:24