Discontinuous Galerkin Library
#include "dg/algorithm.h"
Loading...
Searching...
No Matches
mpi_kron.h
Go to the documentation of this file.
1#pragma once
2#include <cassert>
3#include <vector>
4#include <algorithm> // std::copy_n
5#include <map>
6#include "exceptions.h"
7
8
9namespace dg
10{
19namespace detail{
20// I think the mpi registry should not be exposed to the User
21// It may be an idea to hint at it in the grid documentation
22// inline declared functions generate no warning about "defined but not used"...
23// and can be defined in multiple translation units
24struct MPICartInfo
25{
26 MPI_Comm root;
27 std::vector<int> remain_dims;
28 // std::vector<int> periods; // no need to track because the root and all the children have the same periods
29};
30//we keep track of communicators that were created in the past
31// inline variables have the same address in multiple translation units
32// and have external linkage by default
33inline std::map<MPI_Comm, MPICartInfo> mpi_cart_registry;
34
35inline void mpi_cart_registry_display( std::ostream& out = std::cout)
36{
37 out << "MPI Cart registry\n";
38 for ( auto pair : detail::mpi_cart_registry)
39 {
40 auto& re = pair.second.remain_dims;
41 out << "Comm "<<pair.first<<" Root "<<pair.second.root<<" Remains size "<<re.size();
42 out << " r:";
43 for( uint i=0; i<re.size(); i++)
44 out << " "<<re[i];
45 out << std::endl;
46 }
47}
48
49inline void mpi_cart_registry_clear( )
50{
51 mpi_cart_registry.clear();
52}
53
54inline void mpi_cart_register_cart( MPI_Comm comm)
55{
56 // Register Cartesian comm if not already there
57 auto it = detail::mpi_cart_registry.find( comm);
58 if( it == detail::mpi_cart_registry.end())
59 {
60 int ndims;
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;
65 }
66}
67}
68
80inline void register_mpi_cart_sub( MPI_Comm comm, const int remain_dims[], MPI_Comm newcomm)
81{
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;
87}
88// Can't decide whether to make register_mpi_cart_sub public ...
92
111inline MPI_Comm mpi_cart_sub( MPI_Comm comm, std::vector<int> remain_dims, bool
112 duplicate = false)
113{
114 int ndims;
115 MPI_Cartdim_get( comm, &ndims);
116 assert( (unsigned) ndims == remain_dims.size());
117
118 detail::mpi_cart_register_cart( comm);
119 detail::MPICartInfo info = detail::mpi_cart_registry.at(comm);
120 // info.remain_dims may be larger than remain_dims because comm may have a parent
121 // but exactly remain_dims.size() entries are true
122 int counter =0;
123 for( unsigned u=0; u<info.remain_dims.size(); u++)
124 if( info.remain_dims[u])
125 {
126 info.remain_dims[u] = remain_dims[counter];
127 counter ++;
128 }
129 assert( counter == (int)remain_dims.size());
130 if( not duplicate)
131 {
132 for (auto it = detail::mpi_cart_registry.begin(); it !=
133 detail::mpi_cart_registry.end(); ++it)
134 {
135// int comp_root;
136// MPI_Comm_compare( it->second.root, info.root, &comp_root);
137// if( (comp_root == MPI_IDENT or comp_root == MPI_CONGRUENT) and
138// it->second.remain_dims == info.remain_dims)
139 if( it->second.root == info.root && it->second.remain_dims ==
140 info.remain_dims)
141 {
142 return it->first;
143 }
144 }
145 }
146 MPI_Comm newcomm;
147 int err = MPI_Cart_sub( comm, &remain_dims[0], &newcomm);
148 if( err != MPI_SUCCESS)
149 throw Error(Message(_ping_)<<
150 "Cannot create Cartesian sub comm from given communicator");
151 register_mpi_cart_sub( comm, &remain_dims[0], newcomm);
152 return newcomm;
153}
154
178inline MPI_Comm mpi_cart_kron( std::vector<MPI_Comm> comms)
179{
180 // This non-template interface must exist so compiler can deduce call
181 // mpi_cart_kron( {comm0, comm1});
182 if ( comms.empty())
183 return MPI_COMM_NULL;
184 if( comms.size() == 1)
185 return comms[0];
186
187 std::vector<detail::MPICartInfo> infos(comms.size());
188
189 for( unsigned u=0; u<comms.size(); u++)
190 {
191 try{
192 infos [u] = detail::mpi_cart_registry.at(comms[u]);
193 }catch( std::exception& e)
194 {
195 std::cerr << "Did not find "<<comms[u]<<" in Registry!\n";
196 detail::mpi_cart_registry_display();
197 throw e;
198
199 }
200 }
201 MPI_Comm root = infos[0].root;
202 for( unsigned u=0; u<comms.size(); u++)
203 if( infos[u].root != root)
204 throw Error(Message(_ping_)<<
205 "In mpi_cart_kron all comms must have same root comm "
206 <<root<<" Offending comm number "<<u<<" with root "
207 <<infos[u].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++)
213 {
214 for( unsigned k=0; k<ndims; k++)
215 if( infos[u].remain_dims[k])
216 {
217 if( remain_dims[k])
218 throw Error(Message(_ping_)<<
219 "Cannot form kronecker product with given communicators as remaining dims must be orthogonal ");
220 if( k < current_free_k)
221 throw Error(Message(_ping_)<<
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;
225 }
226 }
227 return dg::mpi_cart_sub( root_info.root, remain_dims, false);
228}
229
234template<class Vector>
235MPI_Comm mpi_cart_kron( Vector comms)
236{
237 return mpi_cart_kron( std::vector<MPI_Comm>(comms.begin(), comms.end()));
238}
239
240
248inline std::vector<MPI_Comm> mpi_cart_split( MPI_Comm comm)
249{
250 // Check that there is a Comm that was already split
251 int ndims;
252 MPI_Cartdim_get( comm, &ndims);
253
254 std::vector<MPI_Comm> comms(ndims);
255 for( int u=0; u<ndims; u++)
256 {
257 std::vector<int> remain_dims(ndims, 0);
258 remain_dims[u] = 1;
259 comms[u] = mpi_cart_sub( comm, remain_dims, false);
260 }
261 return comms;
262}
271template<size_t Nd>
272std::array<MPI_Comm, Nd> mpi_cart_split_as( MPI_Comm comm)
273{
274 auto split = mpi_cart_split( comm);
275 std::array<MPI_Comm, Nd> arr;
276 std::copy_n( split.begin(), Nd, arr.begin());
277 return arr;
278}
279
280
281
282// Need to think about those again
283// /*! @brief unregister a communicator
284// * For example if a communicator was freed
285// * @param comm delete associated registry entry
286// */
287//void unregister_mpi_comm( MPI_Comm comm)
288//{
289// detail::mpi_cart_registry.erase( comm);
290//}
291//
292// /*! @brief call \c MPI_Comm_free(comm) followed by \c dg::unregister_mpi_comm(comm)
293// * @param comm free communicator and delete associated registry entry
294// */
295//void mpi_comm_free( MPI_Comm * comm)
296//{
297// MPI_Comm_free(comm);
298// unregister_mpi_comm( *comm);
299//}
300
302
303
304}
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...