IODA
VarAttrStore.hpp
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_internals_engines_obsstore
8  *
9  * @{
10  * \file VarAttrStore.hpp
11  * \brief Functions for ObsStore variable and attribute data storage
12  */
13 #pragma once
14 
15 #include <string>
16 #include <vector>
17 
18 #include "gsl/gsl-lite.hpp"
19 
20 #include "./Selection.hpp"
21 #include "./Types.hpp"
22 #include "ioda/Exception.h"
23 
24 namespace ioda {
25 namespace ObsStore {
26 /// \ingroup ioda_internals_engines_obsstore
28 private:
29 public:
31  virtual ~VarAttrStore_Base() {}
32 
33  /// \brief resizes memory allocated for data storage (vector)
34  /// \param newSize new size for allocated memory in number of vector elements
35  virtual void resize(std::size_t newSize) = 0;
36  /// \brief resizes memory allocated for data storage (vector)
37  /// \param newSize new size for allocated memory in number of vector elements
38  /// \param fillvalue new elements get initialized to fillValue
39  virtual void resize(std::size_t newSize, gsl::span<char> &fillValue) = 0;
40  /// \brief transfer data to data storage vector
41  /// \param data contiguous block of data to transfer
42  /// \param m_select Selection ojbect: how to select from data argument
43  /// \param f_select Selection ojbect: how to select to storage vector
44  virtual void write(gsl::span<char> data, Selection &m_select, Selection &f_select) = 0;
45  /// \brief transfer data from data storage vector
46  /// \param data contiguous block of data to transfer
47  /// \param m_select Selection ojbect: how to select to data argument
48  /// \param f_select Selection ojbect: how to select from storage vector
49  virtual void read(gsl::span<char> data, Selection &m_select, Selection &f_select) const = 0;
50 };
51 
52 // Templated versions for each data type
53 /// \ingroup ioda_internals_engines_obsstore
54 template <typename DataType>
56 private:
57  /// \brief data storage mechanism (vector)
58  std::vector<DataType> var_attr_data_;
59 
60 public:
63 
64  /// \brief resizes memory allocated for data storage (vector)
65  /// \param newSize new size for allocated memory in number of vector elements
66  void resize(std::size_t newSize) override { var_attr_data_.resize(newSize); }
67 
68  /// \brief resizes memory allocated for data storage (vector)
69  /// \param newSize new size for allocated memory in number of vector elements
70  /// \param fillvalue new elements get initialized to fillValue
71  void resize(std::size_t newSize, gsl::span<char> &fillValue) override {
72  gsl::span<DataType> fv_span(reinterpret_cast<DataType *>(fillValue.data()), 1);
73  var_attr_data_.resize(newSize, fv_span[0]);
74  }
75 
76  /// \brief transfer data to data storage vector
77  /// \param data contiguous block of data to transfer
78  /// \param m_select Selection ojbect: how to select from data argument
79  /// \param f_select Selection ojbect: how to select to storage vector
80  void write(gsl::span<char> data, Selection &m_select, Selection &f_select) override {
81  if (data.size() > 0) {
82  std::size_t numObjects = data.size() / sizeof(DataType);
83  gsl::span<DataType> d_span(reinterpret_cast<DataType *>(data.data()), numObjects);
84  // assumes m_select and f_select have same number of points
85  m_select.init_lin_indx();
86  f_select.init_lin_indx();
87  while (!m_select.end_lin_indx()) {
88  std::size_t m_indx = m_select.next_lin_indx();
89  std::size_t f_indx = f_select.next_lin_indx();
90  var_attr_data_[f_indx] = d_span[m_indx];
91  }
92  }
93  }
94 
95  /// \brief transfer data from data storage vector
96  /// \param data contiguous block of data to transfer
97  /// \param m_select Selection ojbect: how to select to data argument
98  /// \param f_select Selection ojbect: how to select from storage vector
99  void read(gsl::span<char> data, Selection &m_select, Selection &f_select) const override {
100  std::size_t datumLen = sizeof(DataType);
101  std::size_t numChars = var_attr_data_.size() * datumLen;
102  if (data.size() > 0) {
103  gsl::span<char> c_span(
104  const_cast<char *>(reinterpret_cast<const char *>(var_attr_data_.data())), numChars);
105  // assumes m_select and f_select have same number of points
106  m_select.init_lin_indx();
107  f_select.init_lin_indx();
108  while (!m_select.end_lin_indx()) {
109  std::size_t m_indx = m_select.next_lin_indx() * datumLen;
110  std::size_t f_indx = f_select.next_lin_indx() * datumLen;
111  for (std::size_t i = 0; i < datumLen; ++i) {
112  data[m_indx + i] = c_span[f_indx + i];
113  }
114  }
115  }
116  }
117 };
118 
119 // Specialization for std::string data type
120 /// \ingroup ioda_internals_engines_obsstore
121 template <>
122 class VarAttrStore<std::string> : public VarAttrStore_Base {
123 private:
124  /// \brief data storage mechanism (vector)
125  std::vector<std::string> var_attr_data_;
126 
127 public:
130 
131  /// \brief resizes memory allocated for data storage (vector)
132  /// \param newSize new size for allocated memory in number of vector elements
133  void resize(std::size_t newSize) override { var_attr_data_.resize(newSize); }
134 
135  /// \brief resizes memory allocated for data storage (vector)
136  /// \param newSize new size for allocated memory in number of vector elements
137  /// \param fillvalue new elements get initialized to fillValue
138  void resize(std::size_t newSize, gsl::span<char> &fillValue) override {
139  // At this point, fillValue[0] is a char * pointing to the string
140  // to be used for a fill value.
141  gsl::span<char *> fv_span(reinterpret_cast<char **>(fillValue.data()), 1);
142  var_attr_data_.resize(newSize, fv_span[0]);
143  }
144 
145  /// \brief transfer data to data storage vector
146  /// \param data contiguous block of data to transfer
147  /// \param m_select Selection object: how to select from data argument
148  /// \param f_select Selection object: how to select to storage vector
149  void write(gsl::span<char> data, Selection &m_select, Selection &f_select) override {
150  // data is a series of char * pointing to null terminated strings
151  // first place the char * values in a vector
152  if (data.size() > 0) {
153  std::size_t numObjects = data.size() / sizeof(char *);
154  gsl::span<char *> d_span(reinterpret_cast<char **>(data.data()), numObjects);
155  std::vector<char *> inStrings(numObjects);
156  for (std::size_t i = 0; i < numObjects; ++i) {
157  inStrings[i] = d_span[i];
158  }
159 
160  // assumes m_select and f_select have same number of points
161  m_select.init_lin_indx();
162  f_select.init_lin_indx();
163  while (!m_select.end_lin_indx()) {
164  std::size_t m_indx = m_select.next_lin_indx();
165  std::size_t f_indx = f_select.next_lin_indx();
166  var_attr_data_[f_indx] = inStrings[m_indx];
167  }
168  }
169  }
170 
171  /// \brief transfer data from data storage vector
172  /// \param data contiguous block of data to transfer
173  /// \param m_select Selection ojbect: how to select to data argument
174  /// \param f_select Selection ojbect: how to select from storage vector
175  void read(gsl::span<char> data, Selection &m_select, Selection &f_select) const override {
176  // First create a vector of char * pointers to each item in var_attr_data_.
177  std::size_t numObjects = var_attr_data_.size();
178  if (data.size() > 0) {
179  std::vector<const char *> outStrings(numObjects);
180  for (std::size_t i = 0; i < numObjects; ++i) {
181  outStrings[i] = var_attr_data_[i].data();
182  }
183 
184  std::size_t datumLen = sizeof(char *);
185  std::size_t numChars = outStrings.size() * datumLen;
186  gsl::span<char> c_span(reinterpret_cast<char *>(outStrings.data()), numChars);
187  // assumes m_select and f_select have same number of points
188  m_select.init_lin_indx();
189  f_select.init_lin_indx();
190  while (!m_select.end_lin_indx()) {
191  std::size_t m_indx = m_select.next_lin_indx() * datumLen;
192  std::size_t f_indx = f_select.next_lin_indx() * datumLen;
193  for (std::size_t i = 0; i < datumLen; ++i) {
194  data[m_indx + i] = c_span[f_indx + i];
195  }
196  }
197  }
198  }
199 };
200 
201 /// \brief factory style function to create a new templated object
202 /// \ingroup ioda_internals_engines_obsstore
204  VarAttrStore_Base *newStore = nullptr;
205 
206  // Use the dtype value to determine which templated version of the data store
207  // to instantiate.
208  if (dtype == ObsTypes::FLOAT) {
209  newStore = new VarAttrStore<float>;
210  } else if (dtype == ObsTypes::DOUBLE) {
211  newStore = new VarAttrStore<double>;
212  } else if (dtype == ObsTypes::LDOUBLE) {
213  newStore = new VarAttrStore<long double>;
214  } else if (dtype == ObsTypes::SCHAR) {
215  newStore = new VarAttrStore<signed char>;
216  } else if (dtype == ObsTypes::SHORT) {
217  newStore = new VarAttrStore<short>;
218  } else if (dtype == ObsTypes::INT) {
219  newStore = new VarAttrStore<int>;
220  } else if (dtype == ObsTypes::LONG) {
221  newStore = new VarAttrStore<long>;
222  } else if (dtype == ObsTypes::LLONG) {
223  newStore = new VarAttrStore<long long>;
224  } else if (dtype == ObsTypes::UCHAR) {
225  newStore = new VarAttrStore<unsigned char>;
226  } else if (dtype == ObsTypes::USHORT) {
227  newStore = new VarAttrStore<unsigned short>;
228  } else if (dtype == ObsTypes::UINT) {
229  newStore = new VarAttrStore<unsigned int>;
230  } else if (dtype == ObsTypes::ULONG) {
231  newStore = new VarAttrStore<unsigned long>;
232  } else if (dtype == ObsTypes::ULLONG) {
233  newStore = new VarAttrStore<unsigned long long>;
234  } else if (dtype == ObsTypes::CHAR) {
235  newStore = new VarAttrStore<char>;
236  } else if (dtype == ObsTypes::WCHAR) {
237  newStore = new VarAttrStore<wchar_t>;
238  } else if (dtype == ObsTypes::CHAR16) {
239  newStore = new VarAttrStore<char16_t>;
240  } else if (dtype == ObsTypes::CHAR32) {
241  newStore = new VarAttrStore<char32_t>;
242  } else if (dtype == ObsTypes::STRING) {
243  newStore = new VarAttrStore<std::string>;
244  } else
245  throw Exception("Unrecognized data type encountered during "
246  "Attribute object construnction", ioda_Here());
247 
248  return newStore;
249 }
250 
251 } // namespace ObsStore
252 } // namespace ioda
253 
254 /// @}
IODA's error system.
Functions for ObsStore Selection.
Functions for ObsStore type markers.
The ioda exception class.
Definition: Exception.h:54
bool end_lin_indx() const
returns true when at the end of the linear memory indices
void init_lin_indx()
initializes iterator for walking through linear memory indices
std::size_t next_lin_indx()
returns next linear memory index
void read(gsl::span< char > data, Selection &m_select, Selection &f_select) const override
transfer data from data storage vector
void resize(std::size_t newSize, gsl::span< char > &fillValue) override
resizes memory allocated for data storage (vector)
void resize(std::size_t newSize) override
resizes memory allocated for data storage (vector)
std::vector< std::string > var_attr_data_
data storage mechanism (vector)
void write(gsl::span< char > data, Selection &m_select, Selection &f_select) override
transfer data to data storage vector
virtual void write(gsl::span< char > data, Selection &m_select, Selection &f_select)=0
transfer data to data storage vector
virtual void resize(std::size_t newSize, gsl::span< char > &fillValue)=0
resizes memory allocated for data storage (vector)
virtual void resize(std::size_t newSize)=0
resizes memory allocated for data storage (vector)
virtual void read(gsl::span< char > data, Selection &m_select, Selection &f_select) const =0
transfer data from data storage vector
std::vector< DataType > var_attr_data_
data storage mechanism (vector)
void resize(std::size_t newSize) override
resizes memory allocated for data storage (vector)
void resize(std::size_t newSize, gsl::span< char > &fillValue) override
resizes memory allocated for data storage (vector)
void read(gsl::span< char > data, Selection &m_select, Selection &f_select) const override
transfer data from data storage vector
void write(gsl::span< char > data, Selection &m_select, Selection &f_select) override
transfer data to data storage vector
@ ObsStore
ObsStore in-memory.
VarAttrStore_Base * createVarAttrStore(ObsTypes dtype)
factory style function to create a new templated object
ObsTypes
ObsStore data type markers.
Definition: Types.hpp:30
#define ioda_Here()