13 #include <Eigen/Dense>
19 #include "eckit/exception/Exceptions.h"
20 #include "eckit/mpi/Comm.h"
22 #include "oops/util/Timer.h"
34 const eckit::mpi::Comm &
world();
37 const eckit::mpi::Comm &
myself();
43 template <
typename SERIALIZABLE>
44 void send(
const eckit::mpi::Comm & comm,
const SERIALIZABLE & sendobj,
45 const int dest,
const int tag) {
46 util::Timer timer(
"oops::mpi",
"send");
47 std::vector<double> sendbuf;
48 sendobj.serialize(sendbuf);
49 comm.send(sendbuf.data(), sendbuf.size(),
dest, tag);
54 template <
typename SERIALIZABLE>
55 void receive(
const eckit::mpi::Comm & comm, SERIALIZABLE & recvobj,
56 const int source,
const int tag) {
57 util::Timer timer(
"oops::mpi",
"receive");
58 size_t sz = recvobj.serialSize();
59 std::vector<double> recvbuf(sz);
60 eckit::mpi::Status
status = comm.receive(recvbuf.data(), sz, source, tag);
62 recvobj.deserialize(recvbuf, ii);
68 void gather(
const eckit::mpi::Comm & comm,
const std::vector<double> &
send,
69 std::vector<double> & recv,
const size_t root);
73 template <
typename SERIALIZABLE>
74 void gather(
const eckit::mpi::Comm & comm,
const std::vector<SERIALIZABLE> &
send,
75 std::vector<SERIALIZABLE> & recv,
const size_t root) {
76 if (comm.size() > 1) {
77 std::vector<double> sendbuf;
78 std::vector<double> recvbuf;
80 for (
const SERIALIZABLE & jsend :
send) jsend.serialize(sendbuf);
82 gather(comm, sendbuf, recvbuf, root);
84 if (comm.rank() == root) {
86 for (SERIALIZABLE & jrecv : recv) jrecv.deserialize(recvbuf, indx);
96 void allGather(
const eckit::mpi::Comm & comm,
97 const Eigen::VectorXd &, Eigen::MatrixXd &);
118 template <
typename CIter,
typename Iter>
121 std::vector<double> serializedLocalData;
122 for (CIter it = first; it != last; ++it)
123 it->serialize(serializedLocalData);
125 eckit::mpi::Buffer<double> buffer(comm.size());
126 comm.allGatherv(serializedLocalData.begin(), serializedLocalData.end(), buffer);
128 size_t numDeserializedDoubles = 0;
129 for (Iter it = recvbuf; numDeserializedDoubles != buffer.buffer.size(); ++it)
130 it->deserialize(buffer.buffer, numDeserializedDoubles);
151 template <
typename T>
152 void allGatherv(
const eckit::mpi::Comm & comm, std::vector<T> &x) {
153 eckit::mpi::Buffer<T> buffer(comm.size());
154 comm.allGatherv(x.begin(), x.end(), buffer);
155 x = std::move(buffer.buffer);
169 void allGatherv(
const eckit::mpi::Comm & comm, std::vector<util::DateTime> &x);
182 void allGatherv(
const eckit::mpi::Comm & comm, std::vector<std::string> &x);
void allGather(const eckit::mpi::Comm &comm, const Eigen::VectorXd &sendbuf, Eigen::MatrixXd &recvbuf)
const eckit::mpi::Comm & myself()
Default communicator with each MPI task by itself.
void allGathervUsingSerialize(const eckit::mpi::Comm &comm, CIter first, CIter last, Iter recvbuf)
A wrapper around the MPI all gather operation for serializable types.
void gather(const eckit::mpi::Comm &comm, const std::vector< double > &send, std::vector< double > &recv, const size_t root)
void exclusiveScan(const eckit::mpi::Comm &comm, size_t &x)
Perform the exclusive scan operation.
void send(const eckit::mpi::Comm &comm, const SERIALIZABLE &sendobj, const int dest, const int tag)
Extend eckit Comm for Serializable oops objects.
const eckit::mpi::Comm & world()
Default communicator with all MPI tasks (ie MPI_COMM_WORLD)
void allGatherv(const eckit::mpi::Comm &comm, std::vector< util::DateTime > &x)
Perform the MPI all gather operation on a vector of DateTime objects.
void receive(const eckit::mpi::Comm &comm, SERIALIZABLE &recvobj, const int source, const int tag)
The namespace for the main oops code.