opm-simulators/opm/models/parallel/mpibuffer.hh
2021-06-18 13:26:57 +02:00

240 lines
6.0 KiB
C++

// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
// vi: set et ts=4 sw=4 sts=4:
/*
This file is part of the Open Porous Media project (OPM).
OPM is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 2 of the License, or
(at your option) any later version.
OPM is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with OPM. If not, see <http://www.gnu.org/licenses/>.
Consult the COPYING file in the top-level source directory of this
module for the precise wording of the license and the list of
copyright holders.
*/
/*!
* \file
* \copydoc Opm::MpiBuffer
*/
#ifndef EWOMS_MPI_BUFFER_HH
#define EWOMS_MPI_BUFFER_HH
#if HAVE_MPI
#include <mpi.h>
#endif
#include <stddef.h>
#include <type_traits>
#include <cassert>
namespace Opm {
/*!
* \brief Simplifies handling of buffers to be used in conjunction with MPI
*/
template <class DataType>
class MpiBuffer
{
public:
MpiBuffer()
{
data_ = NULL;
dataSize_ = 0;
setMpiDataType_();
updateMpiDataSize_();
}
MpiBuffer(size_t size)
{
data_ = new DataType[size];
dataSize_ = size;
setMpiDataType_();
updateMpiDataSize_();
}
MpiBuffer(const MpiBuffer&) = default;
~MpiBuffer()
{ delete[] data_; }
/*!
* \brief Set the size of the buffer
*/
void resize(size_t newSize)
{
delete[] data_;
data_ = new DataType[newSize];
dataSize_ = newSize;
updateMpiDataSize_();
}
/*!
* \brief Send the buffer asyncronously to a peer process.
*/
void send([[maybe_unused]] unsigned peerRank)
{
#if HAVE_MPI
MPI_Isend(data_,
static_cast<int>(mpiDataSize_),
mpiDataType_,
static_cast<int>(peerRank),
0, // tag
MPI_COMM_WORLD,
&mpiRequest_);
#endif
}
/*!
* \brief Wait until the buffer was send to the peer completely.
*/
void wait()
{
#if HAVE_MPI
MPI_Wait(&mpiRequest_, &mpiStatus_);
#endif // HAVE_MPI
}
/*!
* \brief Receive the buffer syncronously from a peer rank
*/
void receive([[maybe_unused]] unsigned peerRank)
{
#if HAVE_MPI
MPI_Recv(data_,
static_cast<int>(mpiDataSize_),
mpiDataType_,
static_cast<int>(peerRank),
0, // tag
MPI_COMM_WORLD,
&mpiStatus_);
assert(!mpiStatus_.MPI_ERROR);
#endif // HAVE_MPI
}
#if HAVE_MPI
/*!
* \brief Returns the current MPI_Request object.
*
* This object is only well defined after the send() method.
*/
MPI_Request& request()
{ return mpiRequest_; }
/*!
* \brief Returns the current MPI_Request object.
*
* This object is only well defined after the send() method.
*/
const MPI_Request& request() const
{ return mpiRequest_; }
/*!
* \brief Returns the current MPI_Status object.
*
* This object is only well defined after the receive() and wait() methods.
*/
MPI_Status& status()
{ return mpiStatus_; }
/*!
* \brief Returns the current MPI_Status object.
*
* This object is only well defined after the receive() and wait() methods.
*/
const MPI_Status& status() const
{ return mpiStatus_; }
#endif // HAVE_MPI
/*!
* \brief Returns the number of data objects in the buffer
*/
size_t size() const
{ return dataSize_; }
/*!
* \brief Provide access to the buffer data.
*/
DataType& operator[](size_t i)
{
assert(i < dataSize_);
return data_[i];
}
/*!
* \brief Provide access to the buffer data.
*/
const DataType& operator[](size_t i) const
{
assert(i < dataSize_);
return data_[i];
}
private:
void setMpiDataType_()
{
#if HAVE_MPI
// set the MPI data type
if (std::is_same<DataType, char>::value)
mpiDataType_ = MPI_CHAR;
else if (std::is_same<DataType, unsigned char>::value)
mpiDataType_ = MPI_UNSIGNED_CHAR;
else if (std::is_same<DataType, short>::value)
mpiDataType_ = MPI_SHORT;
else if (std::is_same<DataType, unsigned short>::value)
mpiDataType_ = MPI_UNSIGNED_SHORT;
else if (std::is_same<DataType, int>::value)
mpiDataType_ = MPI_INT;
else if (std::is_same<DataType, unsigned>::value)
mpiDataType_ = MPI_UNSIGNED;
else if (std::is_same<DataType, long>::value)
mpiDataType_ = MPI_LONG;
else if (std::is_same<DataType, unsigned long>::value)
mpiDataType_ = MPI_UNSIGNED_LONG;
else if (std::is_same<DataType, long long>::value)
mpiDataType_ = MPI_LONG_LONG;
else if (std::is_same<DataType, unsigned long long>::value)
mpiDataType_ = MPI_UNSIGNED_LONG_LONG;
else if (std::is_same<DataType, float>::value)
mpiDataType_ = MPI_FLOAT;
else if (std::is_same<DataType, double>::value)
mpiDataType_ = MPI_DOUBLE;
else if (std::is_same<DataType, long double>::value)
mpiDataType_ = MPI_LONG_DOUBLE;
else {
mpiDataType_ = MPI_BYTE;
}
#endif // HAVE_MPI
}
void updateMpiDataSize_()
{
#if HAVE_MPI
mpiDataSize_ = dataSize_;
if (mpiDataType_ == MPI_BYTE)
mpiDataSize_ *= sizeof(DataType);
#endif // HAVE_MPI
}
DataType *data_;
size_t dataSize_;
#if HAVE_MPI
size_t mpiDataSize_;
MPI_Datatype mpiDataType_;
MPI_Request mpiRequest_;
MPI_Status mpiStatus_;
#endif // HAVE_MPI
};
} // namespace Opm
#endif