IODA
GeneralDistributionAccumulator.h
Go to the documentation of this file.
1 /*
2  * (C) Crown copyright 2021, Met Office
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 #ifndef DISTRIBUTION_GENERALDISTRIBUTIONACCUMULATOR_H_
9 #define DISTRIBUTION_GENERALDISTRIBUTIONACCUMULATOR_H_
10 
11 #include <cassert>
12 #include <vector>
13 
14 #include "eckit/mpi/Comm.h"
15 #include "ioda/distribution/Accumulator.h"
16 
17 namespace ioda {
18 
19 /// \brief Implementation of the Accumulator interface suitable for any (possibly overlapping)
20 /// distribution, but potentially less efficient than specialized implementations.
21 template <typename T>
23  public:
24  GeneralDistributionAccumulator(const T &, const eckit::mpi::Comm &comm,
25  const std::vector<bool> &patchObs)
26  : localResult_(0), comm_(comm), patchObs_(patchObs)
27  {}
28 
29  void addTerm(std::size_t loc, const T &term) override {
30  if (patchObs_[loc])
31  localResult_ += term;
32  }
33 
34  T computeResult() const override {
35  T result = localResult_;
36  comm_.allReduceInPlace(result, eckit::mpi::sum());
37  return result;
38  }
39 
40  private:
42  const eckit::mpi::Comm &comm_;
43  const std::vector<bool> &patchObs_;
44 };
45 
46 template <typename T>
47 class GeneralDistributionAccumulator<std::vector<T>> : public Accumulator<std::vector<T>> {
48  public:
49  /// Note: only the length of the `init` vector matters -- the values of its elements are ignored.
50  GeneralDistributionAccumulator(const std::vector<T> &init,
51  const eckit::mpi::Comm &comm,
52  const std::vector<bool> &patchObs)
53  : localResult_(init.size(), 0), comm_(comm), patchObs_(patchObs)
54  {}
55 
56  void addTerm(std::size_t loc, const std::vector<T> &term) override {
57  if (patchObs_[loc]) {
58  // Using assert() rather than ASSERT() since this can be called from a tight loop
59  // and I want this extra check to disappear in optimised builds.
60  assert(term.size() == localResult_.size());
61  for (std::size_t i = 0, n = localResult_.size(); i < n; ++i)
62  localResult_[i] += term[i];
63  }
64  }
65 
66  void addTerm(std::size_t loc, std::size_t item, const T &term) override {
67  if (patchObs_[loc])
68  localResult_[item] += term;
69  }
70 
71  std::vector<T> computeResult() const override {
72  std::vector<T> result = localResult_;
73  comm_.allReduceInPlace(result.begin(), result.end(), eckit::mpi::sum());
74  return result;
75  }
76 
77  private:
78  std::vector<T> localResult_;
79  const eckit::mpi::Comm &comm_;
80  const std::vector<bool> &patchObs_;
81 };
82 
83 } // namespace ioda
84 
85 #endif // DISTRIBUTION_GENERALDISTRIBUTIONACCUMULATOR_H_
Calculates the sum of a location-dependent quantity of type T over locations held on all PEs,...
Definition: Accumulator.h:27
void addTerm(std::size_t loc, std::size_t item, const T &term) override
Increment the ith sum with the contribution term of location loc held on the current PE.
std::vector< T > computeResult() const override
Return the sums of contributions associated with locations held on all PEs (each taken into account o...
GeneralDistributionAccumulator(const std::vector< T > &init, const eckit::mpi::Comm &comm, const std::vector< bool > &patchObs)
Note: only the length of the init vector matters – the values of its elements are ignored.
void addTerm(std::size_t loc, const std::vector< T > &term) override
Increment each sum with the contribution of location loc (held on the current PE) taken from the corr...
Implementation of the Accumulator interface suitable for any (possibly overlapping) distribution,...
T computeResult() const override
Return the sum of contributions associated with locations held on all PEs (each taken into account on...
GeneralDistributionAccumulator(const T &, const eckit::mpi::Comm &comm, const std::vector< bool > &patchObs)
void addTerm(std::size_t loc, const T &term) override
Increment the sum with the contribution term of location loc held on the current PE.