Skip to content
Snippets Groups Projects
MpiBodyComm.cpp 2.54 KiB
Newer Older
#include <iostream>
#include "MpiBodyComm.hpp"

namespace nbody {
	using namespace std;

	MpiBodyComm::MpiBodyComm(MPI::Datatype* bodyDatatype) {
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		//keep pointer to datatype for send/recv
		this->datatype = bodyDatatype;
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		this->request = MPI::REQUEST_NULL;
	}

	MpiBodyComm::~MpiBodyComm() {
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		if (this->request != MPI::REQUEST_NULL) {
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
			//wait for termination of possibly open request
			this->request.Wait();
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
			//TODO: check why request cannot be freed
			//this->request.Free();
	void MpiBodyComm::cleanup() {
		if (this->request != MPI::REQUEST_NULL) {
			this->request.Wait();
			//this->request.Free();
		}
	}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
	void MpiBodyComm::sendBlocking(int target, vector<Body> bodies) {
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		MPI::COMM_WORLD.Send(&bodies[0], bodies.size(), *this->datatype, target, 0);
	}

	void MpiBodyComm::recvBlocking(int source, vector<Body>& bodies) {
		MPI::Status status;
		int currentSource;
		int currentSize;

Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		//check for message beforehand to get number of bodies transmitted
		//check for actual source because input parameter could be MPI::ANY_SOURCE
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		MPI::COMM_WORLD.Probe(source, 0, status);
		currentSource = status.Get_source();
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		currentSize = status.Get_count(*this->datatype);
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		//allocate correct vector size for incoming data
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		bodies.resize(currentSize);
		MPI::COMM_WORLD.Recv(&bodies[0], currentSize, *this->datatype, currentSource, 0);
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
	}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
	bool MpiBodyComm::sendUnblocking(int target, vector<Body> bodies) {
		if (this->request != MPI::REQUEST_NULL && !this->request.Test()) {
			//unblocking send still not finished
			//cout << "Test failed" << endl;
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
			return false;
		}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		if (this->request != MPI::REQUEST_NULL) {
			//TODO: check if works for reuse of a MPIBodyComm
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
			this->request.Free();
		}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		this->buffer.resize(bodies.size());
		std::copy(bodies.begin(), bodies.end(), this->buffer.begin());
		this->request = MPI::COMM_WORLD.Isend(&this->buffer[0], this->buffer.size(), *this->datatype, target, 0);
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		return true;
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
	}

	bool MpiBodyComm::recvUnblocking(int source, vector<Body>& bodies) {
		MPI::Status status;
		int currentSource;
		int currentSize;
		//check for message beforehand to get number of bodies transmitted
		//check for actual source because input parameter could be MPI::ANY_SOURCE
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		if (!MPI::COMM_WORLD.Iprobe(source, 0, status)) {
			return false;
		}
		currentSource = status.Get_source();
		currentSize = status.Get_count(*this->datatype);
		//allocate correct vector size for incoming data
		bodies.resize(currentSize);
		MPI::COMM_WORLD.Recv(&bodies[0], currentSize, *this->datatype, currentSource, 0);
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		return true;
	}