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( 
unsigned 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...