27 std::vector<int> remain_dims;
33inline std::map<MPI_Comm, MPICartInfo> mpi_cart_registry;
35inline void mpi_cart_registry_display( std::ostream& out = std::cout)
37 out <<
"MPI Cart registry\n";
38 for (
auto pair : detail::mpi_cart_registry)
40 auto& re = pair.second.remain_dims;
41 out <<
"Comm "<<pair.first<<
" Root "<<pair.second.root<<
" Remains size "<<re.size();
43 for( uint i=0; i<re.size(); i++)
49inline void mpi_cart_registry_clear( )
51 mpi_cart_registry.clear();
54inline void mpi_cart_register_cart( MPI_Comm comm)
57 auto it = detail::mpi_cart_registry.find( comm);
58 if( it == detail::mpi_cart_registry.end())
61 MPI_Cartdim_get( comm, &ndims);
62 std::vector<int> remain_dims( ndims,
true);
63 detail::MPICartInfo info{ comm, remain_dims};
64 detail::mpi_cart_registry[comm] = info;
80inline void register_mpi_cart_sub( MPI_Comm comm,
const int remain_dims[], MPI_Comm newcomm)
82 detail::mpi_cart_register_cart( comm);
83 detail::MPICartInfo info = detail::mpi_cart_registry.at(comm);
84 for(
unsigned u=0; u<info.remain_dims.size(); u++)
85 info.remain_dims[u] = remain_dims[u];
86 detail::mpi_cart_registry[newcomm] = info;
111inline MPI_Comm
mpi_cart_sub( MPI_Comm comm, std::vector<int> remain_dims,
bool
115 MPI_Cartdim_get( comm, &ndims);
116 assert( (
unsigned) ndims == remain_dims.size());
118 detail::mpi_cart_register_cart( comm);
119 detail::MPICartInfo info = detail::mpi_cart_registry.at(comm);
123 for(
unsigned u=0; u<info.remain_dims.size(); u++)
124 if( info.remain_dims[u])
126 info.remain_dims[u] = remain_dims[counter];
129 assert( counter == (
int)remain_dims.size());
132 for (
auto it = detail::mpi_cart_registry.begin(); it !=
133 detail::mpi_cart_registry.end(); ++it)
139 if( it->second.root == info.root && it->second.remain_dims ==
147 int err = MPI_Cart_sub( comm, &remain_dims[0], &newcomm);
148 if( err != MPI_SUCCESS)
150 "Cannot create Cartesian sub comm from given communicator");
151 register_mpi_cart_sub( comm, &remain_dims[0], newcomm);
183 return MPI_COMM_NULL;
184 if( comms.size() == 1)
187 std::vector<detail::MPICartInfo> infos(comms.size());
189 for(
unsigned u=0; u<comms.size(); u++)
192 infos [u] = detail::mpi_cart_registry.at(comms[u]);
193 }
catch( std::exception& e)
195 std::cerr <<
"Did not find "<<comms[u]<<
" in Registry!\n";
196 detail::mpi_cart_registry_display();
201 MPI_Comm root = infos[0].root;
202 for(
unsigned u=0; u<comms.size(); u++)
203 if( infos[u].root != root)
205 "In mpi_cart_kron all comms must have same root comm "
206 <<root<<
" Offending comm number "<<u<<
" with root "
208 auto root_info = detail::mpi_cart_registry.at(root);
209 size_t ndims = root_info.remain_dims.size();
210 std::vector<int> remain_dims( ndims,
false) ;
211 unsigned current_free_k=0;
212 for(
unsigned u=0; u<comms.size(); u++)
214 for(
unsigned k=0; k<ndims; k++)
215 if( infos[u].remain_dims[k])
219 "Cannot form kronecker product with given communicators as remaining dims must be orthogonal ");
220 if( k < current_free_k)
222 "Cannot form kronecker product with communicators in reverse order compared to original ");
223 remain_dims[k] = infos[u].remain_dims[k];
224 current_free_k = k+1;
234template<
class Vector>
237 return mpi_cart_kron( std::vector<MPI_Comm>(comms.begin(), comms.end()));
252 MPI_Cartdim_get( comm, &ndims);
254 std::vector<MPI_Comm> comms(ndims);
255 for(
int u=0; u<ndims; u++)
257 std::vector<int> remain_dims(ndims, 0);
275 std::array<MPI_Comm, Nd> arr;
276 std::copy_n(
split.begin(), Nd, arr.begin());
class intended for the use in throw statements
Definition exceptions.h:83
small class holding a stringstream
Definition exceptions.h:29
Error classes or the dg library.
#define _ping_
Definition exceptions.h:12
std::array< MPI_Comm, Nd > mpi_cart_split_as(MPI_Comm comm)
Same as mpi_cart_split but differen return type.
Definition mpi_kron.h:272
std::vector< MPI_Comm > mpi_cart_split(MPI_Comm comm)
Split a Cartesian communicator along each dimensions.
Definition mpi_kron.h:248
MPI_Comm mpi_cart_sub(MPI_Comm comm, std::vector< int > remain_dims, bool duplicate=false)
Call and register a call to MPI_Cart_sub with the dg library.
Definition mpi_kron.h:111
MPI_Comm mpi_cart_kron(std::vector< MPI_Comm > comms)
Form a Kronecker product among Cartesian communicators.
Definition mpi_kron.h:178
void split(SharedContainer &in, std::vector< View< SharedContainer > > &out, const aRealTopology3d< real_type > &grid)
Split a vector into planes along the last dimension (fast version)
Definition split_and_join.h:32
This is the namespace for all functions and classes defined and used by the discontinuous Galerkin li...