8#define CheckMPIStatus(A,B) {}
14#include <dune/common/fvector.hh>
15#include <dune/common/hybridutilities.hh>
17#include <dune/geometry/type.hh>
26 struct MPITypeInfo {};
29 struct MPITypeInfo< int >
31 static const unsigned int size = 1;
32 static inline MPI_Datatype getType()
38 template<
typename K,
int N>
39 struct MPITypeInfo< Dune::FieldVector<K,N> >
41 static const unsigned int size = N;
42 static inline MPI_Datatype getType()
44 return Dune::MPITraits<K>::getType();
49 struct MPITypeInfo< unsigned int >
51 static const unsigned int size = 1;
52 static inline MPI_Datatype getType()
59 struct MPITypeInfo< Dune::GeometryType >
61 static const unsigned int size = 1;
62 static inline MPI_Datatype getType()
64 return Dune::MPITraits< Dune::GeometryType >::getType();
69 void MPI_SetVectorSize(
70 std::vector<T> & data,
73 typedef MPITypeInfo<T> Info;
75 MPI_Get_count(&status, Info::getType(), &sz);
76 assert(sz%Info::size == 0);
77 data.resize(sz/Info::size);
90 void MPI_SendVectorInRing(
91 std::vector<T> & data,
92 std::vector<T> & next,
102 [[maybe_unused]]
int result = 0;
103 typedef MPITypeInfo<T> Info;
105 next.resize(next.capacity());
109 &(data[0]), Info::size*data.size(), Info::getType(), rightrank, tag,
114 &(next[0]), Info::size*next.size(), Info::getType(), leftrank, tag,
126 template<
typename... Args>
127 struct call_MPI_SendVectorInRing
129 std::tuple<Args...> & remotedata;
130 std::tuple<Args...> & nextdata;
135 std::array<MPI_Request,
sizeof...(Args)> & requests_send;
136 std::array<MPI_Request,
sizeof...(Args)> & requests_recv;
141 MPI_SendVectorInRing(
142 std::get<i>(remotedata),
143 std::get<i>(nextdata),
145 rightrank, leftrank, mpicomm,
150 template<
typename... Args>
151 struct call_MPI_SetVectorSize
153 std::tuple<Args...> & nextdata;
154 std::array<MPI_Status,
sizeof...(Args)> & status_recv;
159 MPI_SetVectorSize(std::get<i>(nextdata),status_recv[i]);
163 template<
typename OP, std::size_t... Indices,
typename... Args>
164 void MPI_AllApply_impl(MPI_Comm mpicomm,
166 std::index_sequence<Indices...> indices,
169 constexpr std::size_t N =
sizeof...(Args);
173 MPI_Comm_rank(mpicomm, &myrank);
174 MPI_Comm_size(mpicomm, &commsize);
179#ifdef DEBUG_GRIDGLUE_PARALLELMERGE
180 std::cout << myrank <<
" Start Communication, size " << commsize << std::endl;
184 std::array<unsigned int, N> size({ ((
unsigned int)data.size())... });
187 std::array<unsigned int, N> maxSize;
188 MPI_Allreduce(&size, &maxSize,
189 size.size(), MPI_UNSIGNED, MPI_MAX, mpicomm);
190#ifdef DEBUG_GRIDGLUE_PARALLELMERGE
191 std::cout << myrank <<
" maxSize " <<
"done... " << std::endl;
195 std::tuple<Args...> remotedata { Args(maxSize[Indices])... };
198 remotedata = std::tie(data...);
201 std::tuple<Args...> nextdata { Args(maxSize[Indices])... };
204 int rightrank = (myrank + 1 + commsize) % commsize;
205 int leftrank = (myrank - 1 + commsize) % commsize;
207 std::cout << myrank <<
": size = " << commsize << std::endl;
208 std::cout << myrank <<
": left = " << leftrank
209 <<
" right = " << rightrank << std::endl;
212 int remoterank = myrank;
214 for (
int i=1; i<commsize; i++)
217 int nextrank = (myrank - i + commsize) % commsize;
219 std::cout << myrank <<
": next = " << nextrank << std::endl;
222 std::array<MPI_Request,N> requests_send;
223 std::array<MPI_Request,N> requests_recv;
226 Dune::Hybrid::forEach(indices,
236 call_MPI_SendVectorInRing<Args...>({
240 rightrank, leftrank, mpicomm,
246 op(remoterank,std::get<Indices>(remotedata)...);
249 std::array<MPI_Status,N> status_send;
250 std::array<MPI_Status,N> status_recv;
251 MPI_Waitall(N,&requests_recv[0],&status_recv[0]);
254 remoterank = nextrank;
258 Dune::Hybrid::forEach(indices,
262 call_MPI_SetVectorSize<Args...>({
263 nextdata, status_recv
266 MPI_Waitall(N,&requests_send[0],&status_send[0]);
269 std::swap(remotedata,nextdata);
273 op(remoterank,std::get<Indices>(remotedata)...);
296template<
typename OP,
typename... Args>
297void MPI_AllApply(MPI_Comm mpicomm,
299 const Args& ... data)
301 Impl::MPI_AllApply_impl(
303 std::forward<OP>(op),
304 std::make_index_sequence<
sizeof...(Args)>(),