IODA
InefficientDistribution.cc
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2017-2019 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/InefficientDistribution.h"
9 
10 #include <algorithm>
11 #include <iostream>
12 #include <numeric>
13 
14 #include <boost/make_unique.hpp>
15 
16 #include "ioda/distribution/DistributionFactory.h"
17 #include "ioda/distribution/InefficientDistributionAccumulator.h"
18 #include "oops/util/DateTime.h"
19 #include "oops/util/Logger.h"
20 
21 // -----------------------------------------------------------------------------
22 
23 namespace ioda {
24 
25 // -----------------------------------------------------------------------------
26 static DistributionMaker<InefficientDistribution> maker("InefficientDistribution");
27 
28 // -----------------------------------------------------------------------------
30  const eckit::Configuration & config)
31  : Distribution(Comm) {
32  oops::Log::trace() << "InefficientDistribution constructed" << std::endl;
33 }
34 
35 // -----------------------------------------------------------------------------
37  oops::Log::trace() << "InefficientDistribution destructed" << std::endl;
38 }
39 
40 // -----------------------------------------------------------------------------
41 void InefficientDistribution::patchObs(std::vector<bool> & patchObsVec) const {
42  // a copy of obs is present on all PEs but only rank=0 "owns" this obs. as patch
43  size_t MyRank = comm_.rank();
44  if (MyRank == 0) {
45  std::fill(patchObsVec.begin(), patchObsVec.end(), true);
46  } else {
47  std::fill(patchObsVec.begin(), patchObsVec.end(), false);
48  }
49 }
50 
51 // -----------------------------------------------------------------------------
52 std::unique_ptr<Accumulator<int>>
54  return createAccumulatorImplT(init);
55 }
56 
57 std::unique_ptr<Accumulator<std::size_t>>
59  return createAccumulatorImplT(init);
60 }
61 
62 std::unique_ptr<Accumulator<float>>
64  return createAccumulatorImplT(init);
65 }
66 
67 std::unique_ptr<Accumulator<double>>
69  return createAccumulatorImplT(init);
70 }
71 
72 std::unique_ptr<Accumulator<std::vector<int>>>
73 InefficientDistribution::createAccumulatorImpl(const std::vector<int> &init) const {
74  return createAccumulatorImplT(init);
75 }
76 
77 std::unique_ptr<Accumulator<std::vector<std::size_t>>>
78 InefficientDistribution::createAccumulatorImpl(const std::vector<std::size_t> &init) const {
79  return createAccumulatorImplT(init);
80 }
81 
82 std::unique_ptr<Accumulator<std::vector<float>>>
83 InefficientDistribution::createAccumulatorImpl(const std::vector<float> &init) const {
84  return createAccumulatorImplT(init);
85 }
86 
87 std::unique_ptr<Accumulator<std::vector<double>>>
88 InefficientDistribution::createAccumulatorImpl(const std::vector<double> &init) const {
89  return createAccumulatorImplT(init);
90 }
91 
92 template <typename T>
93 std::unique_ptr<Accumulator<T>>
95  return boost::make_unique<InefficientDistributionAccumulator<T>>(init);
96 }
97 
98 // -----------------------------------------------------------------------------
100  return loc;
101 }
102 
103 // -----------------------------------------------------------------------------
104 } // namespace ioda
class for distributing obs across multiple process elements
const eckit::mpi::Comm & comm_
Local MPI communicator.
A class able to instantiate objects of type T, which should be a subclass of Distribution.
void patchObs(std::vector< bool > &) const override
Sets each element of the provided vector to true if the corresponding location is a "patch obs",...
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...
InefficientDistribution(const eckit::mpi::Comm &Comm, const eckit::Configuration &config)
std::unique_ptr< Accumulator< T > > createAccumulatorImplT(const T &init) const
static DistributionMaker< AtlasDistribution > maker(DIST_NAME)