IODA
ObsFrame.cc
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2017-2019 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 
8 #include "oops/util/abor1_cpp.h"
9 
10 #include "ioda/core/IodaUtils.h"
11 #include "ioda/Engines/HH.h"
12 #include "ioda/io/ObsFrame.h"
13 #include "ioda/Layout.h"
15 
16 namespace ioda {
17 
18 //--------------------------- public functions ---------------------------------------
19 //------------------------------------------------------------------------------------
21  params_(params), gnlocs_(0), gnlocs_outside_timewindow_(0) {
22  oops::Log::trace() << "Constructing ObsFrame" << std::endl;
23 }
24 
25 //------------------------------------------------------------------------------------
26 Selection ObsFrame::createMemSelection(const std::vector<Dimensions_t> & varShape,
27  const Dimensions_t frameCount) {
28  // Use hyperslab selection on the memory side to make sure a slab selected
29  // from a backend with a start greater than zero ends up in the memory
30  // starting at zero. If the memory is left with a default selection ("ALL")
31  // ioda-engines will allocate the memory to match the entire size of the backend variable
32  // and place the selected slab from the backend in the exact same position in the memory.
33  std::vector<Dimensions_t> memShape = varShape;
34  memShape[0] = frameCount;
35 
36  // Treat the memory side as a buffer (vector) so make sure any extra dimensions
37  // are accounted for in the size.
38  Dimensions_t numElements = std::accumulate(
39  memShape.begin(), memShape.end(), 1, std::multiplies<Dimensions_t>());
40 
41  std::vector<Dimensions_t> memStarts(1, 0);
42  std::vector<Dimensions_t> memCounts(1, numElements);
43  Selection memSelect;
44  memSelect.extent(memCounts).select({ SelectionOperator::SET, memStarts, memCounts });
45  return memSelect;
46 }
47 
48 //------------------------------------------------------------------------------------
49 Selection ObsFrame::createEntireFrameSelection(const std::vector<Dimensions_t> & varShape,
50  const Dimensions_t frameCount) {
51  // In this case we want the memory select to go from zero to frameCount in the
52  // first dimension of the variable.
53  std::vector<Dimensions_t> entireFrameCounts = varShape;
54  entireFrameCounts[0] = frameCount;
55 
56  // The frame extent will go from zero to frameCount in the first dimension
57  // of the varShape. varShape is the shape from the storage containing the
58  // entire variable which might be larger than one frame.
59  std::vector<Dimensions_t> frameVarShape = varShape;
60  frameVarShape[0] = frameCount;
61 
62  // Treat the frame side as multi-dimensioned storage. Take the entire range for
63  // the second dimension, third dimension, etc.
64  std::vector<Dimensions_t> entireFrameStarts(entireFrameCounts.size(), 0);
65  Selection entireFrameSelect;
66  entireFrameSelect.extent(frameVarShape)
67  .select({ SelectionOperator::SET, entireFrameStarts, entireFrameCounts });
68  return entireFrameSelect;
69 }
70 
71 //------------------------------------------------------------------------------------
72 Selection ObsFrame::createVarSelection(const std::vector<Dimensions_t> & varShape,
73  const Dimensions_t frameStart,
74  const Dimensions_t frameCount) {
75  // In this case we want the memory select to go from frameStart and be of size frameCount
76  // in the first dimension of the variable.
77  std::vector<Dimensions_t> varCounts = varShape;
78  varCounts[0] = frameCount;
79 
80  // Treat the frame size as multi-dimensioned storage. Take the entire range for
81  // the second dimension, third dimension, etc.
82  std::vector<Dimensions_t> varStarts(varCounts.size(), 0);
83  varStarts[0] = frameStart;
84  Selection varSelect;
85  varSelect.extent(varShape).select({ SelectionOperator::SET, varStarts, varCounts });
86  return varSelect;
87 }
88 
89 //--------------------------- protected functions ---------------------------------------
90 //------------------------------------------------------------------------------------
91 Selection ObsFrame::createObsIoSelection(const std::vector<Dimensions_t> & varShape,
92  const Dimensions_t frameStart,
93  const Dimensions_t frameCount) {
94  // We want ObsIo selection to go from frameStart to frameCount in the first
95  // dimension of the variable shape.
96  std::vector<Dimensions_t> obsIoCounts = varShape;
97  obsIoCounts[0] = frameCount;
98 
99  // Treat the ObsIo side as multi-dimensioned storage. Take the entire range for
100  // the second dimension, third dimension, etc.
101  std::vector<Dimensions_t> obsIoStarts(obsIoCounts.size(), 0);
102  obsIoStarts[0] = frameStart;
103  Selection obsIoSelect;
104  obsIoSelect.extent(varShape).select({ SelectionOperator::SET, obsIoStarts, obsIoCounts });
105  return obsIoSelect;
106 }
107 
108 //------------------------------------------------------------------------------------
110  const VarNameObjectList & dimVarList,
111  const VarDimMap & varDimMap) {
112  // create an ObsGroup with an in-memory backend
113  Engines::BackendNames backendName;
117  backendParams.fileName = ioda::Engines::HH::genUniqueName();
118  backendParams.allocBytes = 1024*1024*50;
119  backendParams.flush = false;
120 
121  backendName = Engines::BackendNames::ObsStore; // Hdf5Mem; ObsStore;
122  Group backend = constructBackend(backendName, backendParams);
123 
124  // create dimensions for frame
126  for (auto & dimNameObject : dimVarList) {
127  std::string dimName = dimNameObject.first;
128  Variable srcDimVar = dimNameObject.second;
129  Dimensions_t dimSize = dimNameObject.second.getDimensions().dimsCur[0];
130  // Don't allow nchans to be limited by the frame size since nchans is
131  // the second dimension (and we are only limiting the frame size on
132  // the first dimension, typically nlocs).
133  if (dimName != "nchans") {
134  if (dimSize > max_frame_size_) {
135  dimSize = max_frame_size_;
136  }
137  }
138  Dimensions_t maxDimSize = dimSize;
139  Dimensions_t chunkSize = dimSize;
140 
141  if (srcDimVar.isA<int>()) {
142  newDims.push_back(
143  ioda::NewDimensionScale<int>(
144  dimName, dimSize, maxDimSize, chunkSize));
145  } else if (srcDimVar.isA<float>()) {
146  newDims.push_back(
147  ioda::NewDimensionScale<float>(
148  dimName, dimSize, maxDimSize, chunkSize));
149  }
150  }
152 
153  // fill in dimension coordinate values
154  for (auto & dimVarNameObject : dimVarList) {
155  std::string dimVarName = dimVarNameObject.first;
156  Variable srcDimVar = dimVarNameObject.second;
157  Variable destDimVar = obs_frame_.vars.open(dimVarName);
158 
159  // Set up the dimension selection objects. The prior loop declared the
160  // sizes of all the dimensions in the frame so use that as a guide, and
161  // transfer the first frame's worth of coordinate values accordingly.
162  std::vector<Dimensions_t> srcDimShape = srcDimVar.getDimensions().dimsCur;
163  std::vector<Dimensions_t> destDimShape = destDimVar.getDimensions().dimsCur;
164  Dimensions_t frameCount = destDimShape[0];
165  // Transfer the coordinate values
166  if (frameCount > 0) {
167  Selection srcSelect = createObsIoSelection(srcDimShape, 0, frameCount);
168  Selection memSelect = createMemSelection(destDimShape, frameCount);
169  Selection destSelect = createEntireFrameSelection(destDimShape, frameCount);
170 
171  if (srcDimVar.isA<int>()) {
172  std::vector<int> dimCoords;
173  srcDimVar.read<int>(dimCoords, memSelect, srcSelect);
174  destDimVar.write<int>(dimCoords, memSelect, destSelect);
175  } else if (srcDimVar.isA<float>()) {
176  std::vector<float> dimCoords;
177  srcDimVar.read<float>(dimCoords, memSelect, srcSelect);
178  destDimVar.write<float>(dimCoords, memSelect, destSelect);
179  }
180  }
181  }
182 
183  // create variables for frame
184  for (auto & varNameObject : varList) {
185  std::string varName = varNameObject.first;
186 
187  // get the dimensions attached to this variable
188  std::vector<std::string> dimVarNames = varDimMap.at(varName);
189  std::vector<Variable> dimVars;
190  for (auto & dimVarName : dimVarNames) {
191  dimVars.push_back(obs_frame_.vars.open(dimVarName));
192  }
193 
194  Variable sourceVar = varNameObject.second;
195  if (sourceVar.isA<int>()) {
197  if (sourceVar.hasFillValue()) {
198  auto varFillValue = sourceVar.getFillValue();
199  params.setFillValue<int>(ioda::detail::getFillValue<int>(varFillValue));
200  }
201  obs_frame_.vars.createWithScales<int>(varName, dimVars, params);
202  } else if (sourceVar.isA<float>()) {
204  if (sourceVar.hasFillValue()) {
205  auto varFillValue = sourceVar.getFillValue();
206  params.setFillValue<float>(ioda::detail::getFillValue<float>(varFillValue));
207  }
208  obs_frame_.vars.createWithScales<float>(varName, dimVars, params);
209  } else if (sourceVar.isA<std::string>()) {
211  if (sourceVar.hasFillValue()) {
212  auto varFillValue = sourceVar.getFillValue();
213  params.setFillValue<std::string>(
214  ioda::detail::getFillValue<std::string>(varFillValue));
215  }
216  obs_frame_.vars.createWithScales<std::string>(varName, dimVars, params);
217  }
218  }
219 }
220 
221 } // namespace ioda
HDF5 engine.
Contains definitions for how data are arranged in ioda internally.
Interfaces for ioda::Variable and related classes.
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
Selection createMemSelection(const std::vector< Dimensions_t > &varShape, const Dimensions_t frameCount)
create selection object for accessing a memory buffer
Definition: ObsFrame.cc:26
void createFrameFromObsGroup(const VarNameObjectList &varList, const VarNameObjectList &dimVarList, const VarDimMap &varDimMap)
create a frame object based on dimensions and variables from a source ObsGroup
Definition: ObsFrame.cc:109
Dimensions_t max_frame_size_
maximum frame size
Definition: ObsFrame.h:183
virtual Dimensions_t frameCount(const std::string &varName)=0
return current frame count for variable
Selection createEntireFrameSelection(const std::vector< Dimensions_t > &varShape, const Dimensions_t frameCount)
create selection object for accessing the entire frame variable
Definition: ObsFrame.cc:49
Selection createVarSelection(const std::vector< Dimensions_t > &varShape, const Dimensions_t frameStart, const Dimensions_t frameCount)
create selection object for accessing a frame from a whole variable
Definition: ObsFrame.cc:72
Selection createObsIoSelection(const std::vector< Dimensions_t > &varShape, const Dimensions_t frameStart, const Dimensions_t frameCount)
create selection object for accessing an ObsIo variable
Definition: ObsFrame.cc:91
ObsFrame(const ObsSpaceParameters &params)
Definition: ObsFrame.cc:20
virtual Dimensions_t frameStart()=0
return current frame starting index
ObsGroup obs_frame_
ObsGroup object (temporary storage for a single frame)
Definition: ObsFrame.h:164
static ObsGroup generate(Group &emptyGroup, const NewDimensionScales_t &fundamentalDims, std::shared_ptr< const detail::DataLayoutPolicy > layout=nullptr)
Create an empty ObsGroup and populate it with the fundamental dimensions.
Definition: ObsGroup.cpp:72
A Selection represents the bounds of the data, in ioda or in userspace, that you are reading or writi...
Definition: Selection.h:48
Variables store data!
Definition: Variable.h:680
Has_Variables vars
Use this to access variables.
Definition: Group.h:123
Variable createWithScales(const std::string &name, const std::vector< Variable > &dimension_scales, const VariableCreationParameters &params=VariableCreationParameters::defaulted< DataType >())
Convenience function to create a Variable from certain dimension scales.
virtual Variable open(const std::string &name) const
Open a Variable by name.
virtual FillValueData_t getFillValue() const
Retrieve the fill value.
Definition: Variable.cpp:111
bool isA() const
Convenience function to check a Variable's storage type.
Definition: Variable.h:99
virtual bool hasFillValue() const
Check if a variable has a fill value set.
Definition: Variable.cpp:98
virtual Dimensions getDimensions() const
Definition: Variable.cpp:160
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
IODA_DL std::string genUniqueName()
Convenience function to generate a random file name.
Definition: HH.cpp:50
BackendNames
Backend names.
Definition: Factory.h:28
IODA_DL Group constructBackend(BackendNames name, BackendCreationParameters &params)
This is a simple factory style function that will instantiate a different backend based on a given na...
Definition: Factory.cpp:124
@ ObsStore
ObsStore in-memory.
@ Truncate_If_Exists
If the file already exists, overwrite it.
Selection & extent(const VecDimensions_t &sz)
Provide the dimensions of the object that you are selecting from.
Definition: Selection.h:111
Selection & select(const SingleSelection &s)
Append a new selection.
Definition: Selection.h:103
list newDims
Definition: 05-ObsGroup.py:95
std::vector< std::pair< std::string, Variable > > VarNameObjectList
typedef for holding list of variable names with associated variable object
Definition: IodaUtils.h:30
std::vector< std::shared_ptr< NewDimensionScale_Base > > NewDimensionScales_t
std::map< std::string, std::vector< std::string > > VarDimMap
typedef for holding dim names attached to variables
Definition: IodaUtils.h:36
std::vector< Dimensions_t > dimsCur
The dimensions of the data.
Definition: Dimensions.h:23
Used to specify backend creation-time properties.
Definition: Factory.h:59
Used to specify Variable creation-time properties.
Definition: Has_Variables.h:57