IODA
Engines/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 Selection.cpp
11  * \brief Functions for ObsStore Selection
12  */
13 
14 #include "gsl/gsl-lite.hpp"
15 
16 #include "./Selection.hpp"
17 #include "ioda/Exception.h"
18 
19 namespace ioda {
20 namespace ObsStore {
21 
22 //*************************************************************************
23 // SelectCounter functions
24 //*************************************************************************
25 SelectCounter::SelectCounter() : icount_(0), ndigits_(0) {}
26 
27 void SelectCounter::reset(const SelectionModes mode, const std::vector<std::size_t>& digit_sizes) {
28  mode_ = mode;
29  digit_sizes_ = digit_sizes;
30  ndigits_ = digit_sizes.size();
31  digits_.assign(digit_sizes.size(), 0);
32  counter_end_ = false;
33 }
34 
37  if (digits_[0] == ndigits_ - 1) {
38  // about to overflow the counter --> done
39  counter_end_ = true;
40  return;
41  }
42 
43  // increment all digits: 0,0 1,1 2,2 etc.
44  for (std::size_t i = 0; i < ndigits_; ++i) {
45  digits_[i]++;
46  }
47  } else {
48  // Increment the least significant digit
49  icount_ = ndigits_ - 1;
50  digits_[icount_]++;
51 
52  // Apply carry rules
53  while (digits_[icount_] == digit_sizes_[icount_]) {
54  if (icount_ == 0) {
55  // About to overflow the counter --> done
56  counter_end_ = true;
57  return;
58  }
59 
60  // Carry
61  digits_[icount_] = 0;
62  icount_--;
63  digits_[icount_]++;
64  }
65  }
66 }
67 
68 bool SelectCounter::finished() const { return counter_end_; }
69 
70 const std::vector<std::size_t>& SelectCounter::count() const { return digits_; }
71 
72 //*************************************************************************
73 // Selection functions
74 //*************************************************************************
75 Selection::Selection(const std::size_t start, const std::size_t npoints)
76  : mode_(SelectionModes::ALL),
77  end_(gsl::narrow<int>(start + npoints) - 1),
78  index_(0), npoints_(npoints) {
79  max_index_ = end_;
80 }
81 
82 Selection::Selection(const SelectionModes mode, const std::vector<SelectSpecs>& dim_selects,
83  const std::vector<Dimensions_t>& dim_sizes)
84  : mode_(mode),
85  end_(0),
86  index_(0),
87  npoints_(1),
88  max_index_(1),
89  dim_selects_(dim_selects),
90  dim_sizes_(dim_sizes),
91  // dim_select_sizes
92  counter_(std::unique_ptr<SelectCounter>(new SelectCounter)) {
93  // record the sizes of each dimension selection
94  dim_select_sizes_.clear();
95  for (std::size_t i = 0; i < dim_selects_.size(); ++i) {
96  std::size_t dim_select_size = dim_selects_[i].size();
97  dim_select_sizes_.push_back(dim_select_size);
98  if ((i == 0) || (mode == SelectionModes::INTERSECT)) {
99  npoints_ *= dim_select_size;
100  }
101 
102  max_index_ *= dim_sizes[i];
103  }
104  // At this point, max_index_ equals the total number of points that can
105  // be accessed by dimensions, so need to decrement by 1 for it to contain
106  // the maximum allowed index.
107  max_index_--;
108 }
109 
110 Selection::Selection() = default;
111 
113 
114 // Following are a set of functions that provide an iterator style capability
115 // that generates the linear memory indices correspoding to the selection
116 // settings. The linear memory indices are generated by creating a counter
117 // with the same number of digits as the size of the dim_selects_ vector, and
118 // making the maximum values of each digit place match the size of the corresponding
119 // subvector in dim_selects_. This yields the same sequence as running through
120 // nested for loops that walk through the dim_selects_ structure.
122  if (mode_ == SelectionModes::ALL) {
123  index_ = 0;
124  } else {
126  }
127 }
128 
130  std::size_t lin_index = 0;
131  if (mode_ == SelectionModes::ALL) {
132  lin_index = index_;
133  index_++;
134  } else {
135  // Calculate linear index from current count
136  const std::vector<std::size_t>& curCount = counter_->count();
137  lin_index = dim_selects_[0][curCount[0]];
138  for (std::size_t i = 1; i < dim_selects_.size(); ++i) {
139  lin_index *= dim_sizes_[i];
140  lin_index += dim_selects_[i][curCount[i]];
141  }
142 
143  // increment counter for next time around
144  counter_->inc();
145  }
146 
147  // Make sure lin_index_ is in bounds.
148  if (lin_index > max_index_)
149  throw Exception("Next linear index is out of bounds.", ioda_Here())
150  .add(" Next linear index: ", lin_index)
151  .add(" Maximum allowed index: ", max_index_);
152 
153  return lin_index;
154 }
155 
157  bool finished = false;
158  ;
159  if (mode_ == SelectionModes::ALL) {
160  finished = (index_ > end_);
161  } else {
162  finished = counter_->finished();
163  }
164  return finished;
165 }
166 
167 std::size_t Selection::npoints() const { return npoints_; }
168 } // namespace ObsStore
169 } // namespace ioda
170 
171 /// @}
IODA's error system.
Functions for ObsStore Selection.
The ioda exception class.
Definition: Exception.h:54
Exception & add(const std::string &key, const T value)
Add a key-value pair to the error message.
Definition: Exception.h:75
bool finished() const
returns true when inc attempts to go past final state of counter
const std::vector< std::size_t > & count() const
return current counter value
std::vector< std::size_t > digits_
counter digits
Definition: Selection.hpp:65
bool counter_end_
true when finished counting
Definition: Selection.hpp:73
std::size_t icount_
index to counter digit
Definition: Selection.hpp:67
SelectionModes mode_
mode of selection
Definition: Selection.hpp:62
void reset(const SelectionModes mode, const std::vector< std::size_t > &digit_sizes_)
allocate n digits for counter and set count to zero
std::size_t ndigits_
number of digits in counter
Definition: Selection.hpp:69
std::vector< std::size_t > digit_sizes_
size (range) for each digit place
Definition: Selection.hpp:71
std::size_t npoints() const
returns number of points in selection
int end_
end of selection for ALL mode
Definition: Selection.hpp:98
std::size_t npoints_
total number of points in selection
Definition: Selection.hpp:103
std::unique_ptr< SelectCounter > counter_
counter for generating linear memory indices
Definition: Selection.hpp:115
bool end_lin_indx() const
returns true when at the end of the linear memory indices
SelectionModes mode() const
returns selection mode
void init_lin_indx()
initializes iterator for walking through linear memory indices
std::vector< std::size_t > dim_select_sizes_
number of dimension selections per dimension
Definition: Selection.hpp:112
SelectionModes mode_
mode of selection (which impacts how linear memory is accessed)
Definition: Selection.hpp:95
std::vector< Dimensions_t > dim_sizes_
sizes of data dimensions (length is rank of dimensions)
Definition: Selection.hpp:110
int index_
index value for ALL mode
Definition: Selection.hpp:100
std::size_t max_index_
maximum allowed index value
Definition: Selection.hpp:105
std::size_t next_lin_indx()
returns next linear memory index
std::vector< SelectSpecs > dim_selects_
selection indices for each dimension
Definition: Selection.hpp:108
@ ObsStore
ObsStore in-memory.
SelectionModes
ObsStore selection modes.
Definition: Selection.hpp:56
#define ioda_Here()