OOPS
EnsembleApplication.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018-2020 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 #ifndef OOPS_RUNS_ENSEMBLEAPPLICATION_H_
9 #define OOPS_RUNS_ENSEMBLEAPPLICATION_H_
10 
11 #include <string>
12 #include <vector>
13 
14 #include "eckit/config/LocalConfiguration.h"
15 #include "eckit/config/YAMLConfiguration.h"
16 #include "eckit/exception/Exceptions.h"
17 #include "eckit/mpi/Comm.h"
18 #include "oops/mpi/mpi.h"
19 #include "oops/runs/Application.h"
20 #include "oops/util/abor1_cpp.h"
21 #include "oops/util/Logger.h"
22 
23 namespace oops {
24 
25 template <typename APP>
27  public:
28 // -----------------------------------------------------------------------------
29  explicit EnsembleApplication(const eckit::mpi::Comm & comm = oops::mpi::world()) :
30  Application(comm) {}
31 // -----------------------------------------------------------------------------
32  virtual ~EnsembleApplication() {}
33 // -----------------------------------------------------------------------------
34  int execute(const eckit::Configuration & fullConfig) const {
35  // Get the list of yaml files
36  std::vector<std::string> listConf;
37  fullConfig.get("files", listConf);
38  Log::info() << "EnsembleApplication yaml files:" << listConf << std::endl;
39 
40  // Get the MPI partition
41  const int nmembers = listConf.size();
42  const int ntasks = this->getComm().size();
43  const int mytask = this->getComm().rank();
44  const int tasks_per_member = ntasks / nmembers;
45  const int mymember = mytask / tasks_per_member + 1;
46 
47  Log::info() << "Running " << nmembers << " EnsembleApplication members handled by "
48  << ntasks << " total MPI tasks and "
49  << tasks_per_member << " MPI tasks per member." << std::endl;
50 
51  ASSERT(ntasks%nmembers == 0);
52 
53  // Create the communicator for each member, named comm_member_{i}:
54  std::string commNameStr = "comm_member_" + std::to_string(mymember);
55  char const *commName = commNameStr.c_str();
56  eckit::mpi::Comm & commMember = this->getComm().split(mymember, commName);
57 
58  // Each member uses a different configuration:
59  eckit::PathName confPath = listConf[mymember-1];
60  eckit::YAMLConfiguration memberConf(confPath);
61 
62  Log::debug() << "EnsembleApplication config for member " << mymember << ": "
63  << memberConf << std::endl;
64 
65  APP ensapp(commMember);
66  return ensapp.execute(memberConf);
67  }
68 // -----------------------------------------------------------------------------
69  private:
70  std::string appname() const {
71  return "oops::EnsembleApplication<>";
72  }
73 // -----------------------------------------------------------------------------
74 };
75 
76 } // namespace oops
77 #endif // OOPS_RUNS_ENSEMBLEAPPLICATION_H_
const eckit::mpi::Comm & getComm() const
Definition: Application.h:36
EnsembleApplication(const eckit::mpi::Comm &comm=oops::mpi::world())
int execute(const eckit::Configuration &fullConfig) const
std::string appname() const
const eckit::mpi::Comm & world()
Default communicator with all MPI tasks (ie MPI_COMM_WORLD)
Definition: oops/mpi/mpi.cc:84
The namespace for the main oops code.