IODA
NonoverlappingDistribution.cc
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2017-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 
8 #include "ioda/distribution/NonoverlappingDistribution.h"
9 
10 #include <algorithm>
11 #include <iostream>
12 #include <numeric>
13 #include <set>
14 
15 #include <boost/make_unique.hpp>
16 
17 #include "ioda/distribution/DistributionFactory.h"
18 #include "ioda/distribution/NonoverlappingDistributionAccumulator.h"
19 #include "oops/mpi/mpi.h"
20 #include "oops/util/DateTime.h"
21 #include "oops/util/Logger.h"
22 
23 namespace ioda {
24 
25 // -----------------------------------------------------------------------------
27  : Distribution(Comm) {
28  oops::Log::trace() << "NonoverlappingDistribution constructed" << std::endl;
29 }
30 
31 // -----------------------------------------------------------------------------
33  oops::Log::trace() << "NonoverlappingDistribution destructed" << std::endl;
34 }
35 
36 // -----------------------------------------------------------------------------
37 void NonoverlappingDistribution::assignRecord(const std::size_t RecNum,
38  const std::size_t /*LocNum*/,
39  const eckit::geometry::Point2 & /*point*/) {
40  if (isMyRecord(RecNum))
42 }
43 
44 // -----------------------------------------------------------------------------
47  oops::mpi::exclusiveScan(comm_, numLocationsOnLowerRanks_);
48 }
49 
50 // -----------------------------------------------------------------------------
51 void NonoverlappingDistribution::patchObs(std::vector<bool> & patchObsVec) const {
52  std::fill(patchObsVec.begin(), patchObsVec.end(), true);
53 }
54 
55 // -----------------------------------------------------------------------------
56 void NonoverlappingDistribution::min(int & x) const {
57  minImpl(x);
58 }
59 
60 void NonoverlappingDistribution::min(std::size_t & x) const {
61  minImpl(x);
62 }
63 
64 void NonoverlappingDistribution::min(float & x) const {
65  minImpl(x);
66 }
67 
68 void NonoverlappingDistribution::min(double & x) const {
69  minImpl(x);
70 }
71 
72 void NonoverlappingDistribution::min(std::vector<int> & x) const {
73  minImpl(x);
74 }
75 
76 void NonoverlappingDistribution::min(std::vector<std::size_t> & x) const {
77  minImpl(x);
78 }
79 
80 void NonoverlappingDistribution::min(std::vector<float> & x) const {
81  minImpl(x);
82 }
83 
84 void NonoverlappingDistribution::min(std::vector<double> & x) const {
85  minImpl(x);
86 }
87 
88 template <typename T>
90  reductionImpl(x, eckit::mpi::min());
91 }
92 
93 // -----------------------------------------------------------------------------
94 void NonoverlappingDistribution::max(int & x) const {
95  maxImpl(x);
96 }
97 
98 void NonoverlappingDistribution::max(std::size_t & x) const {
99  maxImpl(x);
100 }
101 
102 void NonoverlappingDistribution::max(float & x) const {
103  maxImpl(x);
104 }
105 
106 void NonoverlappingDistribution::max(double & x) const {
107  maxImpl(x);
108 }
109 
110 void NonoverlappingDistribution::max(std::vector<int> & x) const {
111  maxImpl(x);
112 }
113 
114 void NonoverlappingDistribution::max(std::vector<std::size_t> & x) const {
115  maxImpl(x);
116 }
117 
118 void NonoverlappingDistribution::max(std::vector<float> & x) const {
119  maxImpl(x);
120 }
121 
122 void NonoverlappingDistribution::max(std::vector<double> & x) const {
123  maxImpl(x);
124 }
125 
126 template <typename T>
128  reductionImpl(x, eckit::mpi::max());
129 }
130 
131 // -----------------------------------------------------------------------------
132 template <typename T>
133 void NonoverlappingDistribution::reductionImpl(T & x, eckit::mpi::Operation::Code op) const {
134  comm_.allReduceInPlace(x, op);
135 }
136 
137 template <typename T>
139  eckit::mpi::Operation::Code op) const {
140  comm_.allReduceInPlace(x.begin(), x.end(), op);
141 }
142 
143 // -----------------------------------------------------------------------------
144 std::unique_ptr<Accumulator<int>>
146  return createAccumulatorImplT(init);
147 }
148 
149 std::unique_ptr<Accumulator<std::size_t>>
151  return createAccumulatorImplT(init);
152 }
153 
154 std::unique_ptr<Accumulator<float>>
156  return createAccumulatorImplT(init);
157 }
158 
159 std::unique_ptr<Accumulator<double>>
161  return createAccumulatorImplT(init);
162 }
163 
164 std::unique_ptr<Accumulator<std::vector<int>>>
165 NonoverlappingDistribution::createAccumulatorImpl(const std::vector<int> &init) const {
166  return createAccumulatorImplT(init);
167 }
168 
169 std::unique_ptr<Accumulator<std::vector<std::size_t>>>
170 NonoverlappingDistribution::createAccumulatorImpl(const std::vector<std::size_t> &init) const {
171  return createAccumulatorImplT(init);
172 }
173 
174 std::unique_ptr<Accumulator<std::vector<float>>>
175 NonoverlappingDistribution::createAccumulatorImpl(const std::vector<float> &init) const {
176  return createAccumulatorImplT(init);
177 }
178 
179 std::unique_ptr<Accumulator<std::vector<double>>>
180 NonoverlappingDistribution::createAccumulatorImpl(const std::vector<double> &init) const {
181  return createAccumulatorImplT(init);
182 }
183 
184 template <typename T>
185 std::unique_ptr<Accumulator<T>>
187  return boost::make_unique<NonoverlappingDistributionAccumulator<T>>(init, comm_);
188 }
189 
190 // -----------------------------------------------------------------------------
191 void NonoverlappingDistribution::allGatherv(std::vector<size_t> &x) const {
192  ASSERT(x.size() == numLocationsOnThisRank_);
193  oops::mpi::allGatherv(comm_, x);
194 }
195 
196 void NonoverlappingDistribution::allGatherv(std::vector<int> &x) const {
197  ASSERT(x.size() == numLocationsOnThisRank_);
198  oops::mpi::allGatherv(comm_, x);
199 }
200 
201 void NonoverlappingDistribution::allGatherv(std::vector<float> &x) const {
202  ASSERT(x.size() == numLocationsOnThisRank_);
203  oops::mpi::allGatherv(comm_, x);
204 }
205 
206 void NonoverlappingDistribution::allGatherv(std::vector<double> &x) const {
207  ASSERT(x.size() == numLocationsOnThisRank_);
208  oops::mpi::allGatherv(comm_, x);
209 }
210 
211 void NonoverlappingDistribution::allGatherv(std::vector<util::DateTime> &x) const {
212  ASSERT(x.size() == numLocationsOnThisRank_);
213  oops::mpi::allGatherv(comm_, x);
214 }
215 
216 void NonoverlappingDistribution::allGatherv(std::vector<std::string> &x) const {
217  ASSERT(x.size() == numLocationsOnThisRank_);
218  oops::mpi::allGatherv(comm_, x);
219 }
220 
222  return numLocationsOnLowerRanks_ + loc;
223 }
224 
225 // -----------------------------------------------------------------------------
226 
227 } // namespace ioda
class for distributing obs across multiple process elements
const eckit::mpi::Comm & comm_
Local MPI communicator.
virtual bool isMyRecord(std::size_t RecNum) const =0
Returns true if record RecNum has been assigned to the calling PE during a previous call to assignRec...
void patchObs(std::vector< bool > &patchObsVec) const override
Sets each element of the provided vector to true if the corresponding location is a "patch obs",...
NonoverlappingDistribution(const eckit::mpi::Comm &Comm)
void computePatchLocs() override
If necessary, identifies locations of "patch obs", i.e.
void max(int &x) const override
Calculates the global maximum (over all locations on all PEs) of a location-dependent quantity.
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 ...
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...
void allGatherv(std::vector< size_t > &x) const override
Gather observation data from all processes and deliver the combined data to all processes.
std::unique_ptr< Accumulator< T > > createAccumulatorImplT(const T &init) 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
If the record RecNum has not yet been assigned to a PE, assigns it to the appropriate PE.
void reductionImpl(T &x, eckit::mpi::Operation::Code op) const