IODA
PairOfDistributions.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_MASTERANDREPLICADISTRIBUTION_H_
9 #define DISTRIBUTION_MASTERANDREPLICADISTRIBUTION_H_
10 
11 #include "ioda/distribution/Distribution.h"
12 
13 namespace ioda {
14 
15 /// \brief Represents a concatenation of locations and records from two distributions.
17  public:
18  /// \brief Create a PairOfDistributions object.
19  ///
20  /// \param comm
21  /// Communicator used by both `first` and `second`.
22  /// \param first
23  /// The first distribution.
24  /// \param second
25  /// The second distribution.
26  /// \param firstNumLocs
27  /// Number of locations in the first distribution held on the calling process. The
28  /// PairOfDistributions will map locations 0, 1, ..., (`firstNumLocs` - 1) to the
29  /// same locations from the first distribution, and locations `firstNumLocs`,
30  /// (`firstNumLocs` + 1), ... to locations 0, 1, ... from the second distribution.
31  /// \param secondRecordNumOffset
32  /// Offset to apply to record indices from the second distribution. The PairOfDistributions
33  /// will map each record `r` < `secondRecordNumOffset` to record `r` from the
34  /// first distribution, and each record `r` >= `secondRecordNumOffset` to record
35  /// `r` - `secondRecordNumOffset` from the second distribution.
36  explicit PairOfDistributions(const eckit::mpi::Comm & comm,
37  std::shared_ptr<const Distribution> first,
38  std::shared_ptr<const Distribution> second,
39  std::size_t firstNumLocs,
40  std::size_t secondRecordNumOffset);
41  ~PairOfDistributions() override;
42 
43  /// This function should not be called. Records are meant to be assigned to the two distributions
44  /// before wrapping them in a PairOfDistributions.
45  void assignRecord(const std::size_t RecNum, const std::size_t LocNum,
46  const eckit::geometry::Point2 & point) override;
47  bool isMyRecord(std::size_t RecNum) const override;
48  void patchObs(std::vector<bool> &) const override;
49 
50  void min(int & x) const override;
51  void min(std::size_t & x) const override;
52  void min(float & x) const override;
53  void min(double & x) const override;
54  void min(std::vector<int> & x) const override;
55  void min(std::vector<std::size_t> & x) const override;
56  void min(std::vector<float> & x) const override;
57  void min(std::vector<double> & x) const override;
58 
59  void max(int & x) const override;
60  void max(std::size_t & x) const override;
61  void max(float & x) const override;
62  void max(double & x) const override;
63  void max(std::vector<int> & x) const override;
64  void max(std::vector<std::size_t> & x) const override;
65  void max(std::vector<float> & x) const override;
66  void max(std::vector<double> & x) const override;
67 
68  void allGatherv(std::vector<size_t> &x) const override;
69  void allGatherv(std::vector<int> &x) const override;
70  void allGatherv(std::vector<float> &x) const override;
71  void allGatherv(std::vector<double> &x) const override;
72  void allGatherv(std::vector<util::DateTime> &x) const override;
73  void allGatherv(std::vector<std::string> &x) const override;
74 
75  size_t globalUniqueConsecutiveLocationIndex(size_t loc) const override;
76 
77  std::string name() const override { return "PairOfDistributions"; }
78 
79  private:
80  template <typename T>
81  void minImpl(T & x) const;
82 
83  template <typename T>
84  void maxImpl(T & x) const;
85 
86  std::unique_ptr<Accumulator<int>>
87  createAccumulatorImpl(int init) const override;
88  std::unique_ptr<Accumulator<std::size_t>>
89  createAccumulatorImpl(std::size_t init) const override;
90  std::unique_ptr<Accumulator<float>>
91  createAccumulatorImpl(float init) const override;
92  std::unique_ptr<Accumulator<double>>
93  createAccumulatorImpl(double init) const override;
94  std::unique_ptr<Accumulator<std::vector<int>>>
95  createAccumulatorImpl(const std::vector<int> &init) const override;
96  std::unique_ptr<Accumulator<std::vector<std::size_t>>>
97  createAccumulatorImpl(const std::vector<std::size_t> &init) const override;
98  std::unique_ptr<Accumulator<std::vector<float>>>
99  createAccumulatorImpl(const std::vector<float> &init) const override;
100  std::unique_ptr<Accumulator<std::vector<double>>>
101  createAccumulatorImpl(const std::vector<double> &init) const override;
102 
103  template <typename T>
104  std::unique_ptr<Accumulator<T>> createScalarAccumulator() const;
105 
106  template <typename T>
107  std::unique_ptr<Accumulator<std::vector<T>>> createVectorAccumulator(std::size_t n) const;
108 
109  template <typename T>
110  void allGathervImpl(std::vector<T> &x) const;
111 
112  std::shared_ptr<const Distribution> first_;
113  std::shared_ptr<const Distribution> second_;
114  std::size_t firstNumLocs_;
117 };
118 
119 } // namespace ioda
120 
121 #endif // DISTRIBUTION_MASTERANDREPLICADISTRIBUTION_H_
class for distributing obs across multiple process elements
Represents a concatenation of locations and records from two distributions.
bool isMyRecord(std::size_t RecNum) const override
Returns true if record RecNum has been assigned to the calling PE during a previous call to assignRec...
void allGatherv(std::vector< size_t > &x) const override
Gather observation data from all processes and deliver the combined data to all processes.
std::size_t secondGlobalUniqueConsecutiveLocationIndexOffset_
std::string name() const override
void allGathervImpl(std::vector< T > &x) const
void min(int &x) const override
Calculates the global minimum (over all locations on all PEs) of a location-dependent quantity.
void assignRecord(const std::size_t RecNum, const std::size_t LocNum, const eckit::geometry::Point2 &point) override
std::shared_ptr< const Distribution > second_
void patchObs(std::vector< bool > &) const override
Sets each element of the provided vector to true if the corresponding location is a "patch obs",...
std::unique_ptr< Accumulator< int > > createAccumulatorImpl(int init) const override
Create an object that can be used to calculate the sum of a location-dependent quantity over location...
std::shared_ptr< const Distribution > first_
std::unique_ptr< Accumulator< T > > createScalarAccumulator() const
void max(int &x) const override
Calculates the global maximum (over all locations on all PEs) of a location-dependent quantity.
std::unique_ptr< Accumulator< std::vector< T > > > createVectorAccumulator(std::size_t n) const
size_t globalUniqueConsecutiveLocationIndex(size_t loc) const override
Map the index of a location held on the calling process to the index of the corresponding element of ...
PairOfDistributions(const eckit::mpi::Comm &comm, std::shared_ptr< const Distribution > first, std::shared_ptr< const Distribution > second, std::size_t firstNumLocs, std::size_t secondRecordNumOffset)
Create a PairOfDistributions object.