8 #ifndef TEST_MPI_MPI_H_
9 #define TEST_MPI_MPI_H_
11 #include <Eigen/Dense>
16 #include "eckit/config/LocalConfiguration.h"
17 #include "eckit/mpi/Comm.h"
18 #include "eckit/testing/Test.h"
23 #include "oops/util/DateTime.h"
24 #include "oops/util/Expect.h"
25 #include "oops/util/parameters/Parameters.h"
26 #include "oops/util/parameters/RequiredParameter.h"
32 template <>
struct VectorPrintSelector<
util::DateTime> {
typedef VectorPrintSimple selector; };
41 oops::RequiredParameter<std::vector<util::DateTime>>
datetime{
"datetime",
this};
42 oops::RequiredParameter<std::vector<int>>
int_{
"int",
this};
43 oops::RequiredParameter<std::vector<std::string>>
string{
"string",
this};
48 const std::vector<T> &
getTestData(
const TestParameters ¶ms);
56 const std::vector<int> &
getTestData(
const TestParameters ¶ms) {
61 const std::vector<std::string> &
getTestData(
const TestParameters ¶ms) {
72 const size_t rank = comm.rank();
73 localParams.deserialize(conf.getSubConfiguration(
"local" + std::to_string(rank)));
74 std::vector<T> values = getTestData<T>(localParams);
77 globalParams.deserialize(conf.getSubConfiguration(
"global"));
78 const std::vector<T> &expectedResult = getTestData<T>(globalParams);
81 EXPECT_EQUAL(values, expectedResult);
85 CASE(
"mpi/mpi/defaultCommunicators") {
87 size_t worldsize =
world.size();
88 EXPECT_EQUAL(worldsize, 4);
91 size_t myownsize = talk_to_myself.size();
92 EXPECT_EQUAL(myownsize, 1);
95 CASE(
"mpi/mpi/allGathervUsingSerialize") {
100 const size_t rank = comm.rank();
101 localParams.deserialize(conf.getSubConfiguration(
"local" + std::to_string(rank)));
102 const std::vector<util::DateTime> &localValues = localParams.
datetime;
105 globalParams.deserialize(conf.getSubConfiguration(
"global"));
106 const std::vector<util::DateTime> &expectedGlobalValues = globalParams.
datetime;
108 size_t numGlobalValues;
109 comm.allReduce(localValues.size(), numGlobalValues, eckit::mpi::Operation::SUM);
111 std::vector<util::DateTime> globalValues(numGlobalValues);
113 globalValues.begin());
114 EXPECT_EQUAL(globalValues, expectedGlobalValues);
117 CASE(
"mpi/mpi/allGathervInt") {
118 testAllGatherv<int>();
121 CASE(
"mpi/mpi/allGathervDateTime") {
122 testAllGatherv<util::DateTime>();
125 CASE(
"mpi/mpi/allGathervInt") {
126 testAllGatherv<std::string>();
132 const size_t rank = comm.rank();
133 int source = (rank + 3) % comm.size();
134 int destination = (rank + 1) % comm.size();
135 int tag_send = destination;
138 util::DateTime sendValue(conf.getString(
"send"+ std::to_string(rank)));
139 util::DateTime expectedValue(conf.getString(
"expected"+ std::to_string(rank)));
140 util::DateTime receivedValue;
149 EXPECT_EQUAL(receivedValue, expectedValue);
152 CASE(
"mpi/mpi/gatherSerializable") {
157 const size_t rank = comm.rank();
158 localParams.deserialize(conf.getSubConfiguration(
"local" + std::to_string(rank)));
159 const std::vector<util::DateTime> &localValues = localParams.
datetime;
162 globalParams.deserialize(conf.getSubConfiguration(
"global"));
163 const std::vector<util::DateTime> &expectedGlobalValues = globalParams.
datetime;
165 size_t numGlobalValues;
166 comm.allReduce(localValues.size(), numGlobalValues, eckit::mpi::Operation::SUM);
168 std::vector<util::DateTime> globalValues(numGlobalValues);
170 util::DateTime zeroDate(
"0001-01-01T00:00:00Z");
171 for (
size_t ii = 0; ii < numGlobalValues; ++ii) {
172 globalValues[ii] = zeroDate;
175 std::vector<util::DateTime> zeroValues = globalValues;
177 size_t root_gather = conf.getInt(
"root for gathering", 0);
180 if (rank == root_gather) {
181 EXPECT_EQUAL(globalValues, expectedGlobalValues);
183 EXPECT_EQUAL(globalValues, zeroValues);
190 const size_t rank = comm.rank();
192 std::vector<double> localDouble;
193 conf.get(
"localDouble" + std::to_string(rank), localDouble);
195 std::vector<double> globalDoubleExpected;
196 conf.get(
"globalDouble", globalDoubleExpected);
198 size_t numGlobalDouble;
199 comm.allReduce(localDouble.size(), numGlobalDouble, eckit::mpi::Operation::SUM);
201 std::vector<double> globalDouble(numGlobalDouble, 0.0);
202 std::vector<double> zerosDouble = globalDouble;
204 size_t root_gather = conf.getInt(
"root for gathering", 0);
208 if (rank == root_gather) {
209 EXPECT_EQUAL(globalDouble, globalDoubleExpected);
211 EXPECT_EQUAL(globalDouble, zerosDouble);
215 CASE(
"mpi/mpi/allGatherEigen") {
217 const size_t rank = comm.rank();
219 Eigen::VectorXd localEigen = rank * Eigen::VectorXd::Ones(4);
221 Eigen::MatrixXd globalEigen(4, 4);
222 globalEigen << Eigen::VectorXd::Zero(4),
223 Eigen::VectorXd::Zero(4),
224 Eigen::VectorXd::Zero(4),
225 Eigen::VectorXd::Zero(4);
227 Eigen::MatrixXd expectedEigen(4, 4);
228 expectedEigen << 0*Eigen::VectorXd::Ones(4),
229 1*Eigen::VectorXd::Ones(4),
230 2*Eigen::VectorXd::Ones(4),
231 3*Eigen::VectorXd::Ones(4);
234 EXPECT_EQUAL(expectedEigen, globalEigen);
237 CASE(
"mpi/mpi/exclusiveScan") {
239 const size_t rank = comm.rank();
241 size_t expectedResult = 0;
242 for (
size_t lowerRank = 0; lowerRank < rank; ++lowerRank)
243 expectedResult += lowerRank;
245 size_t result = rank;
247 EXPECT_EQUAL(result, expectedResult);
253 std::string
testid()
const override {
return "test::mpi::mpi";}
void register_tests() const override
std::string testid() const override
void clear() const override
static const eckit::Configuration & config()
oops::RequiredParameter< std::vector< util::DateTime > > datetime
oops::RequiredParameter< std::vector< int > > int_
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)
const std::vector< T > & getTestData(const TestParameters ¶ms)
CASE("test_linearmodelparameterswrapper_valid_name")