IODA
04-VariablesAttributesAndDimensions.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_4 Ex 4: Variables, Attributes, and Dimension Scales
12  * \brief Variables, Attributes, and Dimension Scales using the C interface
13  * \details This example parallels the C++ examples.
14  * \see 04-VariablesAttributesAndDimensions.cpp for comments and the walkthrough.
15  *
16  * @{
17  *
18  * \file 04-VariablesAttributesAndDimensions.c
19  * \brief Variables, Attributes, and Dimension Scales using the C interface
20  * \see 04-VariablesAttributesAndDimensions.cpp for comments and the walkthrough.
21  **/
22 
23 #include <stdio.h>
24 #include <stdlib.h>
25 #include <string.h>
26 
27 #include "ioda/C/ioda_c.h"
28 #include "ioda/defs.h" // Always include this first.
29 
30 #define sslin(x) #x
31 #define slin(x) sslin(x)
32 #define doErr \
33  { \
34  errlin = "Error in file " __FILE__ " at line " slin(__LINE__) ".\n"; \
35  goto hadError; \
36  }
37 
38 int main(int argc, char** argv) {
39  int errval = 0;
40  const char* errlin = NULL;
41  struct c_ioda ioda = use_c_ioda();
42  struct ioda_group* g = NULL;
43  struct ioda_has_variables* gvars = NULL;
44  struct ioda_variable *dim_location = NULL, *dim_channel = NULL;
45  struct ioda_variable_creation_parameters *params_default = NULL, *params_dimchannel = NULL,
46  *p1 = NULL;
47  int* dim_location_data = NULL;
48  int* dim_channel_data = NULL;
49  struct ioda_variable *var_longitude = NULL, *var_latitude = NULL;
50  struct ioda_has_attributes *var_lon_atts = NULL, *var_lat_atts = NULL;
51  struct ioda_attribute *v_lon_units = NULL, *v_lon_valid_range = NULL, *v_lon_long_name = NULL;
52  struct ioda_attribute *v_lat_units = NULL, *v_lat_valid_range = NULL, *v_lat_long_name = NULL;
53  struct ioda_variable* var_tb = NULL;
54  struct ioda_has_attributes* var_tb_atts = NULL;
55  struct ioda_attribute *v_tb_units = NULL, *v_tb_valid_range = NULL, *v_tb_long_name = NULL;
56  struct ioda_variable* var_sza = NULL;
57  struct ioda_has_attributes* var_sza_atts = NULL;
58  struct ioda_attribute *v_sza_units = NULL, *v_sza_valid_range = NULL;
59 
60  g = ioda.Engines.constructFromCmdLine(argc, argv, "Example-04-C.hdf5");
61  if (!g) doErr;
62 
63  gvars = ioda.Group.getVars(g);
64  if (!gvars) doErr;
65 
66  // C++ line 70
67 
68  const long num_locs = 3000;
69  const long num_channels = 23;
70 
71  params_default = ioda.VariableCreationParams.create();
72  if (!params_default) doErr;
73 
75  = ioda.Has_Variables.create_int(gvars, 8, "Location", 1, &num_locs, &num_locs, params_default);
76  if (!dim_location) doErr;
77  if (!ioda.Variable.setIsDimensionScale(dim_location, 8, "Location")) doErr;
78 
79  dim_location_data = (int*)malloc(sizeof(int) * num_locs);
80  if (!dim_location_data) doErr;
81  for (long i = 0; i < num_locs; ++i) dim_location_data[i] = (int)(i + 1);
82  if (!ioda.Variable.write_full_int(dim_location, num_locs, dim_location_data)) doErr;
83 
84  params_dimchannel = ioda.VariableCreationParams.create();
85  if (!params_dimchannel) doErr;
86 
87  dim_channel = ioda.Has_Variables.create_int(gvars, 12, "ATMS Channel", 1, &num_channels,
88  &num_channels, params_dimchannel);
89  if (!dim_channel) doErr;
90  if (!ioda.Variable.setIsDimensionScale(dim_channel, 19, "ATMS Channel Number")) doErr;
92 
93  const char* check_str_dimscale_name = "ATMS Channel Number";
94  const int buf_sz = 60;
95  char buf[60] = {"\n"};
96  size_t sz = ioda.Variable.getDimensionScaleName(dim_channel, buf_sz, buf);
97  if (sz != 20) doErr;
98  if (0 != strncmp(buf, check_str_dimscale_name, buf_sz)) doErr;
99 
100  dim_channel_data = (int*)malloc(sizeof(int) * num_channels);
101  if (!dim_channel_data) doErr;
102  for (long i = 0; i < num_channels; ++i) dim_channel_data[i] = (int)(i + 1);
103  if (!ioda.Variable.write_full_int(dim_channel, num_channels, dim_channel_data)) doErr;
104 
105  // C++ line 98
106  var_longitude = ioda.Has_Variables.create_float(gvars, 9, "Longitude", 1, &num_locs, &num_locs,
107  params_default);
108  if (!var_longitude) doErr;
109  const struct ioda_variable* dims_loc[] = {dim_location};
110  if (!ioda.Variable.setDimScale(var_longitude, 1, dims_loc)) doErr;
111  var_lon_atts = ioda.Variable.getAtts(var_longitude);
112  if (!var_lon_atts) doErr;
113 
114  const char* lon_units = "degrees_east";
115  const char* lon_long_name = "Longitude";
116  const long valid_range_dims[] = {2};
117  const float lon_valid_range[2] = {-180, 180};
118  const long str_dims[] = {1};
119 
120  v_lon_valid_range
121  = ioda.Has_Attributes.create_float(var_lon_atts, 11, "valid_range", 1, valid_range_dims);
122  if (!v_lon_valid_range) doErr;
123  if (!ioda.Attribute.write_float(v_lon_valid_range, 2, lon_valid_range)) doErr;
124 
125  v_lon_units = ioda.Has_Attributes.create_str(var_lon_atts, 5, "units", 1, str_dims);
126  if (!v_lon_units) doErr;
127  if (!ioda.Attribute.write_str(v_lon_units, 1, &lon_units)) doErr;
128 
129  v_lon_long_name = ioda.Has_Attributes.create_str(var_lon_atts, 9, "long_name", 1, str_dims);
130  if (!v_lon_long_name) doErr;
131  if (!ioda.Attribute.write_str(v_lon_long_name, 1, &lon_long_name)) doErr;
132 
133  // C++ line 110
134  var_latitude = ioda.Has_Variables.create_float(gvars, 8, "Latitude", 1, &num_locs, &num_locs,
135  params_default);
136  if (!var_latitude) doErr;
137  if (!ioda.Variable.setDimScale(var_latitude, 1, dims_loc)) doErr;
138  var_lat_atts = ioda.Variable.getAtts(var_latitude);
139  if (!var_lat_atts) doErr;
140 
141  const char* lat_units = "degrees_north";
142  const char* lat_long_name = "Latitude";
143  const float lat_valid_range[2] = {-90, 90};
144 
145  v_lat_valid_range
146  = ioda.Has_Attributes.create_float(var_lat_atts, 11, "valid_range", 1, valid_range_dims);
147  if (!v_lat_valid_range) doErr;
148  if (!ioda.Attribute.write_float(v_lat_valid_range, 2, lat_valid_range)) doErr;
149 
150  v_lat_units = ioda.Has_Attributes.create_str(var_lat_atts, 5, "units", 1, str_dims);
151  if (!v_lat_units) doErr;
152  if (!ioda.Attribute.write_str(v_lat_units, 1, &lat_units)) doErr;
153 
154  v_lat_long_name = ioda.Has_Attributes.create_str(var_lat_atts, 9, "long_name", 1, str_dims);
155  if (!v_lat_long_name) doErr;
156  if (!ioda.Attribute.write_str(v_lat_long_name, 1, &lat_long_name)) doErr;
157 
158  // C++ line 117
159  const long sz_tb[] = {num_locs, num_channels};
160  var_tb = ioda.Has_Variables.create_float(gvars, 22, "Brightness Temperature", 2, sz_tb, sz_tb,
161  params_default);
162  if (!var_tb) doErr;
165  var_tb_atts = ioda.Variable.getAtts(var_tb);
166  if (!var_tb_atts) doErr;
167 
169 
170  const char* tb_units = "K";
171  const char* tb_long_name = "ATMS Observed (Uncorrected) Brightness Temperature";
172  const float tb_valid_range[2] = {100, 400};
173 
174  v_tb_valid_range
175  = ioda.Has_Attributes.create_float(var_tb_atts, 11, "valid_range", 1, valid_range_dims);
176  if (!v_tb_valid_range) doErr;
177  if (!ioda.Attribute.write_float(v_tb_valid_range, 2, tb_valid_range)) doErr;
178 
179  v_tb_units = ioda.Has_Attributes.create_str(var_tb_atts, 5, "units", 1, str_dims);
180  if (!v_tb_units) doErr;
181  if (!ioda.Attribute.write_str(v_tb_units, 1, &tb_units)) doErr;
182 
183  v_tb_long_name = ioda.Has_Attributes.create_str(var_tb_atts, 9, "long_name", 1, str_dims);
184  if (!v_tb_long_name) doErr;
185  if (!ioda.Attribute.write_str(v_tb_long_name, 1, &tb_long_name)) doErr;
186 
187  // C++ line 155
188  p1 = ioda.VariableCreationParams.create();
189  if (!p1) doErr;
190  ioda.VariableCreationParams.setFillValue_float(p1, -999);
191  const ptrdiff_t p1_chunks[] = {100};
192  ioda.VariableCreationParams.chunking(p1, true, 1, p1_chunks);
193  ioda.VariableCreationParams.compressWithGZIP(p1, 6);
194  const struct ioda_variable* sza_dimscale[] = {dim_location};
195 
196  // C++ line 166
197  var_sza
198  = ioda.Has_Variables.create_float(gvars, 18, "Solar Zenith Angle", 1, &num_locs, &num_locs, p1);
199  if (!var_sza) doErr;
200  if (!ioda.Variable.setDimScale(var_sza, 1, sza_dimscale)) doErr;
203  doErr; // Just trying to check...
207  var_sza_atts = ioda.Variable.getAtts(var_sza);
208  if (!var_sza_atts) doErr;
209 
210  const char* sza_units = "degrees";
211  const float sza_valid_range[2] = {-90, 90};
212 
213  v_sza_valid_range
214  = ioda.Has_Attributes.create_float(var_sza_atts, 11, "valid_range", 1, valid_range_dims);
215  if (!v_sza_valid_range) doErr;
216  if (!ioda.Attribute.write_float(v_sza_valid_range, 2, sza_valid_range)) doErr;
217 
218  v_sza_units = ioda.Has_Attributes.create_str(var_sza_atts, 5, "units", 1, str_dims);
219  if (!v_sza_units) doErr;
220  if (!ioda.Attribute.write_str(v_sza_units, 1, &sza_units)) doErr;
221 
222 
223  goto cleanup;
224 
225 hadError:
226  printf("%s", (errlin) ? errlin : "An unknown error has occurred somewhere.");
227  errval = 1;
228 
229 cleanup:
230  if (g) ioda.Group.destruct(g);
231  if (gvars) ioda.Has_Variables.destruct(gvars);
232  if (dim_location) ioda.Variable.destruct(dim_location);
233  if (dim_channel) ioda.Variable.destruct(dim_channel);
234  if (params_default) ioda.VariableCreationParams.destruct(params_default);
235  if (params_dimchannel) ioda.VariableCreationParams.destruct(params_dimchannel);
236  if (p1) ioda.VariableCreationParams.destruct(p1);
237  if (dim_location_data) free((void*)dim_location_data);
238  if (dim_channel_data) free((void*)dim_channel_data);
239  if (var_longitude) ioda.Variable.destruct(var_longitude);
240  if (var_latitude) ioda.Variable.destruct(var_latitude);
241  if (var_lon_atts) ioda.Has_Attributes.destruct(var_lon_atts);
242  if (var_lat_atts) ioda.Has_Attributes.destruct(var_lat_atts);
243  if (v_lon_units) ioda.Attribute.destruct(v_lon_units);
244  if (v_lon_valid_range) ioda.Attribute.destruct(v_lon_valid_range);
245  if (v_lon_long_name) ioda.Attribute.destruct(v_lon_long_name);
246  if (v_lat_units) ioda.Attribute.destruct(v_lat_units);
247  if (v_lat_valid_range) ioda.Attribute.destruct(v_lat_valid_range);
248  if (v_lat_long_name) ioda.Attribute.destruct(v_lat_long_name);
249  if (var_tb) ioda.Variable.destruct(var_tb);
250  if (var_tb_atts) ioda.Has_Attributes.destruct(var_tb_atts);
251  if (v_tb_units) ioda.Attribute.destruct(v_tb_units);
252  if (v_tb_valid_range) ioda.Attribute.destruct(v_tb_valid_range);
253  if (v_tb_long_name) ioda.Attribute.destruct(v_tb_long_name);
254  if (var_sza) ioda.Variable.destruct(var_sza);
255  if (var_sza_atts) ioda.Has_Attributes.destruct(var_sza_atts);
256  if (v_sza_units) ioda.Attribute.destruct(v_sza_units);
257  if (v_sza_valid_range) ioda.Attribute.destruct(v_sza_valid_range);
258 
259  return errval;
260 }
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
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
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
std::string getDimensionScaleName() const
Get the name of this Variable's defined dimension scale.
Definition: Variable.h:217
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)
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