2015-03-29 20:35:37 +01:00
|
|
|
#include "Grid.h"
|
|
|
|
#include <mpi.h>
|
|
|
|
|
2015-04-03 05:29:54 +01:00
|
|
|
namespace Grid {
|
2015-03-29 20:35:37 +01:00
|
|
|
|
|
|
|
// Should error check all MPI calls.
|
|
|
|
|
|
|
|
CartesianCommunicator::CartesianCommunicator(std::vector<int> &processors)
|
|
|
|
{
|
2015-04-03 04:52:53 +01:00
|
|
|
_ndimension = processors.size();
|
2015-03-29 20:35:37 +01:00
|
|
|
std::vector<int> periodic(_ndimension,1);
|
|
|
|
|
2015-04-03 04:52:53 +01:00
|
|
|
_Nprocessors=1;
|
2015-03-29 20:35:37 +01:00
|
|
|
_processors = processors;
|
2015-04-03 04:52:53 +01:00
|
|
|
_processor_coor.resize(_ndimension);
|
|
|
|
|
|
|
|
MPI_Cart_create(MPI_COMM_WORLD, _ndimension,&_processors[0],&periodic[0],1,&communicator);
|
2015-03-29 20:35:37 +01:00
|
|
|
MPI_Comm_rank(communicator,&_processor);
|
2015-04-03 04:52:53 +01:00
|
|
|
MPI_Cart_coords(communicator,_processor,_ndimension,&_processor_coor[0]);
|
2015-04-03 22:54:13 +01:00
|
|
|
|
2015-04-03 04:52:53 +01:00
|
|
|
for(int i=0;i<_ndimension;i++){
|
|
|
|
_Nprocessors*=_processors[i];
|
|
|
|
}
|
2015-04-03 22:54:13 +01:00
|
|
|
|
|
|
|
int Size;
|
|
|
|
MPI_Comm_size(communicator,&Size);
|
|
|
|
|
|
|
|
assert(Size==_Nprocessors);
|
2015-03-29 20:35:37 +01:00
|
|
|
}
|
|
|
|
|
2015-04-06 06:30:48 +01:00
|
|
|
void CartesianCommunicator::GlobalSum(float &f){
|
2015-03-29 20:35:37 +01:00
|
|
|
MPI_Allreduce(&f,&f,1,MPI_FLOAT,MPI_SUM,communicator);
|
|
|
|
}
|
2015-04-06 06:30:48 +01:00
|
|
|
void CartesianCommunicator::GlobalSumVector(float *f,int N)
|
2015-03-29 20:35:37 +01:00
|
|
|
{
|
|
|
|
MPI_Allreduce(f,f,N,MPI_FLOAT,MPI_SUM,communicator);
|
|
|
|
}
|
2015-04-06 06:30:48 +01:00
|
|
|
void CartesianCommunicator::GlobalSum(double &d)
|
2015-03-29 20:35:37 +01:00
|
|
|
{
|
|
|
|
MPI_Allreduce(&d,&d,1,MPI_DOUBLE,MPI_SUM,communicator);
|
|
|
|
}
|
2015-04-06 06:30:48 +01:00
|
|
|
void CartesianCommunicator::GlobalSumVector(double *d,int N)
|
2015-03-29 20:35:37 +01:00
|
|
|
{
|
|
|
|
MPI_Allreduce(d,d,N,MPI_DOUBLE,MPI_SUM,communicator);
|
|
|
|
}
|
|
|
|
|
2015-04-03 04:52:53 +01:00
|
|
|
void CartesianCommunicator::ShiftedRanks(int dim,int shift,int &source,int &dest)
|
2015-03-29 20:35:37 +01:00
|
|
|
{
|
2015-04-03 04:52:53 +01:00
|
|
|
MPI_Cart_shift(communicator,dim,shift,&source,&dest);
|
2015-03-29 20:35:37 +01:00
|
|
|
}
|
2015-04-06 06:30:48 +01:00
|
|
|
int CartesianCommunicator::RankFromProcessorCoor(std::vector<int> &coor)
|
2015-03-29 20:35:37 +01:00
|
|
|
{
|
|
|
|
int rank;
|
|
|
|
MPI_Cart_rank (communicator, &coor[0], &rank);
|
|
|
|
return rank;
|
|
|
|
}
|
2015-04-06 06:30:48 +01:00
|
|
|
void CartesianCommunicator::ProcessorCoorFromRank(int rank, std::vector<int> &coor)
|
|
|
|
{
|
|
|
|
coor.resize(_ndimension);
|
|
|
|
MPI_Cart_coords (communicator, rank, _ndimension,&coor[0]);
|
|
|
|
}
|
2015-03-29 20:35:37 +01:00
|
|
|
|
|
|
|
// Basic Halo comms primitive
|
|
|
|
void CartesianCommunicator::SendToRecvFrom(void *xmit,
|
2015-04-03 04:52:53 +01:00
|
|
|
int dest,
|
2015-03-29 20:35:37 +01:00
|
|
|
void *recv,
|
2015-04-03 04:52:53 +01:00
|
|
|
int from,
|
2015-03-29 20:35:37 +01:00
|
|
|
int bytes)
|
|
|
|
{
|
2015-04-03 04:52:53 +01:00
|
|
|
MPI_Request reqs[2];
|
|
|
|
MPI_Status OkeyDokey[2];
|
|
|
|
int rank = _processor;
|
|
|
|
MPI_Isend(xmit, bytes, MPI_CHAR,dest,_processor,communicator,&reqs[0]);
|
|
|
|
MPI_Irecv(recv, bytes, MPI_CHAR,from,from,communicator,&reqs[1]);
|
|
|
|
MPI_Waitall(2,reqs,OkeyDokey);
|
2015-03-29 20:35:37 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-04-06 06:30:48 +01:00
|
|
|
void CartesianCommunicator::Barrier(void)
|
|
|
|
{
|
|
|
|
MPI_Barrier(communicator);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CartesianCommunicator::Broadcast(int root,void* data, int bytes)
|
|
|
|
{
|
|
|
|
MPI_Bcast(data,
|
|
|
|
bytes,
|
|
|
|
MPI_BYTE,
|
|
|
|
root,
|
|
|
|
communicator);
|
|
|
|
}
|
|
|
|
|
2015-03-29 20:35:37 +01:00
|
|
|
}
|
|
|
|
|