IODA
ObsStore-selection.cpp
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 ObsStore-selection.cpp
11  * \brief Functions for transfering ioda::Selection to ObsStore
12  */
13 #include "./ObsStore-selection.h"
14 
15 #include <numeric>
16 #include <set>
17 
18 #include "ioda/defs.h"
19 #include "ioda/Exception.h"
20 
21 namespace ioda {
22 namespace Engines {
23 namespace ObsStore {
25  const std::vector<Dimensions_t>& dim_sizes) {
27  std::vector<ioda::ObsStore::SelectSpecs> dim_selects;
28  std::size_t start = 0;
29  std::size_t npoints = 0;
30 
31  // If incoming mode is set to select all, then pass that information
32  // along to the ObsStore selection object instead of filling up
33  // the selection indices.
34  //
35  // Assumptions:
36  // 1. Only one action is specified
37  // 2. No offset specs
38  if ((selection.getDefault() == SelectionState::ALL) && (selection.getActions().empty())) {
39  // Select all points
41  start = 0;
42 
43  // Count up the number of points
44  npoints = 1;
45  for (std::size_t idim = 0; idim < dim_sizes.size(); ++idim) {
46  npoints *= dim_sizes[idim];
47  }
48  } else {
49  auto first_action = selection.getActions().begin();
50  if (!first_action->start_.empty()) {
51  // Selection is specified as hyperslab
53  genDimSelects(first_action->start_, first_action->count_, first_action->stride_,
54  first_action->block_, dim_selects);
55  } else if (!first_action->points_.empty()) {
56  // Selection is specified as list of points
58  genDimSelects(first_action->points_, dim_selects);
59  } else if (!first_action->dimension_indices_starts_.empty()) {
60  // Selection is specified as dimension indices
62  genDimSelects(selection.getActions(), dim_sizes, dim_selects);
63  } else {
64  throw Exception("Unrecongnized selection mode", ioda_Here());
65  }
66  }
67 
68  if (mode == ioda::ObsStore::SelectionModes::ALL) return ioda::ObsStore::Selection{start, npoints};
69  return ioda::ObsStore::Selection{mode, dim_selects, dim_sizes};
70 }
71 
73  const Selection::VecDimensions_t& stride,
74  const Selection::VecDimensions_t& block,
75  std::vector<ioda::ObsStore::SelectSpecs>& selects) {
76  // Walk through the start, count, stride, block specs and generate
77  // the indices for each dimension.
78  std::size_t numDims = start.size();
79  selects.resize(numDims);
80  for (std::size_t idim = 0; idim < numDims; ++idim) {
81  // Get the start, count, stride, and block parameters for this dimension
82  std::size_t dim_start = start[idim];
83  std::size_t dim_count = count[idim];
84  std::size_t dim_stride = 1;
85  if (!stride.empty()) {
86  dim_stride = stride[idim];
87  }
88  std::size_t dim_block = 1;
89  if (!block.empty()) {
90  dim_block = block[idim];
91  }
92 
93  // Generate the dimension selects
94  for (std::size_t i = 0; i < dim_count; ++i) {
95  std::size_t block_start = dim_start + (i * dim_stride);
96  for (std::size_t j = 0; j < dim_block; ++j) {
97  selects[idim].push_back(block_start + j);
98  }
99  }
100  }
101 }
102 
103 void genDimSelects(const std::vector<Selection::VecDimensions_t>& points,
104  std::vector<ioda::ObsStore::SelectSpecs>& selects) {
105  // points[0] holds indices of first point
106  // points[1] holds indices of second point, etc.
107  // Copy indexes in points directly to select object
108  std::size_t numPoints = points.size();
109  std::size_t numDims = points[0].size();
110  selects.resize(numDims);
111  for (std::size_t idim = 0; idim < numDims; ++idim) {
112  selects[idim].resize(numPoints);
113  }
114 
115  for (std::size_t ipnt = 0; ipnt < numPoints; ++ipnt) {
116  for (std::size_t idim = 0; idim < numDims; ++idim) {
117  selects[idim][ipnt] = points[ipnt][idim];
118  }
119  }
120 }
121 
122 void genDimSelects(const std::vector<Selection::SingleSelection>& actions,
123  const std::vector<Dimensions_t>& dim_sizes,
124  std::vector<ioda::ObsStore::SelectSpecs>& selects) {
125  // Initialize selects to the same rank as dim_sizes. Fill in the dimension
126  // indices according to the selections actions. Then if any dimensions have
127  // not been filled, set their selections to all of the indices in
128  // that dimension.
129  selects.resize(dim_sizes.size());
130 
131  // Each element in actions is a list of indices for a particular dimension
132  for (std::size_t iact = 0; iact < actions.size(); ++iact) {
133  // Create an ordered set of indices (in case the starts and counts overlap)
134  // Then copy the set into the selects structure.
135  std::set<std::size_t> dim_indices;
136  bool haveCounts = (!actions[iact].dimension_indices_counts_.empty());
137  for (std::size_t i = 0; i < actions[iact].dimension_indices_starts_.size(); ++i) {
138  std::size_t idx = actions[iact].dimension_indices_starts_[i];
139  if (haveCounts) {
140  for (std::size_t j = 0; j < actions[iact].dimension_indices_counts_.size(); ++j) {
141  dim_indices.insert(idx + j);
142  }
143  } else {
144  dim_indices.insert(idx);
145  }
146  }
147 
148  // copy the set of indices into the selects structure
149  std::size_t idim = actions[iact].dimension_;
150  selects[idim].resize(dim_indices.size());
151  std::size_t iidx = 0;
152  for (auto iset = dim_indices.begin(); iset != dim_indices.end(); ++iset) {
153  selects[idim][iidx] = *iset;
154  iidx++;
155  }
156  }
157 
158  // For any unfilled dimensions, insert all indices.
159  for (std::size_t idim = 0; idim < selects.size(); ++idim) {
160  if (selects[idim].empty()) {
161  selects[idim].resize(dim_sizes[idim]);
162  std::iota(selects[idim].begin(), selects[idim].end(), 0);
163  }
164  }
165 }
166 } // namespace ObsStore
167 } // namespace Engines
168 } // namespace ioda
169 
170 /// @}
IODA's error system.
Functions for transfering ioda::Selection to ObsStore.
The ioda exception class.
Definition: Exception.h:54
A Selection represents the bounds of the data, in ioda or in userspace, that you are reading or writi...
Definition: Selection.h:48
Common preprocessor definitions used throughout IODA.
@ ObsStore
ObsStore in-memory.
const std::vector< SingleSelection > & getActions() const
Definition: Selection.h:108
std::vector< Dimensions_t > VecDimensions_t
Definition: Selection.h:50
SelectionState getDefault() const
Definition: Selection.h:109
void genDimSelects(const Selection::VecDimensions_t &start, const Selection::VecDimensions_t &count, const Selection::VecDimensions_t &stride, const Selection::VecDimensions_t &block, std::vector< ioda::ObsStore::SelectSpecs > &selects)
generate the dimension selection structure from hyperslab specs
SelectionModes
ObsStore selection modes.
Definition: Selection.hpp:56
ioda::ObsStore::Selection createObsStoreSelection(const ioda::Selection &selection, const std::vector< Dimensions_t > &dim_sizes)
translate a ioda::Selection to and ObsStore Selection
#define ioda_Here()