2004-07-14 14:11:03 +00:00
|
|
|
// -*- c++ -*-
|
|
|
|
//
|
2005-11-05 19:57:48 +00:00
|
|
|
// Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
|
|
|
|
// University Research and Technology
|
|
|
|
// Corporation. All rights reserved.
|
|
|
|
// Copyright (c) 2004-2005 The University of Tennessee and The University
|
|
|
|
// of Tennessee Research Foundation. All rights
|
|
|
|
// reserved.
|
2004-11-28 20:09:25 +00:00
|
|
|
// Copyright (c) 2004-2005 High Performance Computing Center Stuttgart,
|
|
|
|
// University of Stuttgart. All rights reserved.
|
2005-03-24 12:43:37 +00:00
|
|
|
// Copyright (c) 2004-2005 The Regents of the University of California.
|
|
|
|
// All rights reserved.
|
2012-02-21 21:52:23 +00:00
|
|
|
// Copyright (c) 2007-2012 Cisco Systems, Inc. All rights reserved.
|
2012-02-17 11:42:51 +00:00
|
|
|
// Copyright (c) 2011 FUJITSU LIMITED. All rights reserved.
|
2004-11-22 01:38:40 +00:00
|
|
|
// $COPYRIGHT$
|
|
|
|
//
|
|
|
|
// Additional copyrights may follow
|
|
|
|
//
|
2004-07-14 14:11:03 +00:00
|
|
|
// $HEADER$
|
|
|
|
//
|
|
|
|
|
|
|
|
//
|
|
|
|
// Point-to-Point
|
|
|
|
//
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Send(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Send(const_cast<void *>(buf), count, datatype, dest, tag, mpi_comm);
|
2004-07-14 14:11:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Recv(void *buf, int count, const MPI::Datatype & datatype,
|
|
|
|
int source, int tag, MPI::Status & status) const
|
|
|
|
{
|
|
|
|
(void)MPI_Recv(buf, count, datatype, source, tag, mpi_comm, &status.mpi_status);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Recv(void *buf, int count, const MPI::Datatype & datatype,
|
|
|
|
int source, int tag) const
|
|
|
|
{
|
|
|
|
(void)MPI_Recv(buf, count, datatype, source,
|
|
|
|
tag, mpi_comm, MPI_STATUS_IGNORE);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Bsend(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Bsend(const_cast<void *>(buf), count, datatype,
|
2004-07-14 14:11:03 +00:00
|
|
|
dest, tag, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Ssend(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Ssend(const_cast<void *>(buf), count, datatype, dest,
|
2004-07-14 14:11:03 +00:00
|
|
|
tag, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Rsend(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Rsend(const_cast<void *>(buf), count, datatype,
|
2004-07-14 14:11:03 +00:00
|
|
|
dest, tag, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline MPI::Request
|
|
|
|
MPI::Comm::Isend(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
|
|
|
MPI_Request request;
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Isend(const_cast<void *>(buf), count, datatype,
|
2004-07-14 14:11:03 +00:00
|
|
|
dest, tag, mpi_comm, &request);
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline MPI::Request
|
|
|
|
MPI::Comm::Ibsend(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
|
|
|
MPI_Request request;
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Ibsend(const_cast<void *>(buf), count, datatype,
|
2004-07-14 14:11:03 +00:00
|
|
|
dest, tag, mpi_comm, &request);
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline MPI::Request
|
|
|
|
MPI::Comm::Issend(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
|
|
|
MPI_Request request;
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Issend(const_cast<void *>(buf), count, datatype,
|
2004-07-14 14:11:03 +00:00
|
|
|
dest, tag, mpi_comm, &request);
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline MPI::Request
|
|
|
|
MPI::Comm::Irsend(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
|
|
|
MPI_Request request;
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Irsend(const_cast<void *>(buf), count, datatype,
|
2004-07-14 14:11:03 +00:00
|
|
|
dest, tag, mpi_comm, &request);
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline MPI::Request
|
|
|
|
MPI::Comm::Irecv(void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int source, int tag) const
|
|
|
|
{
|
|
|
|
MPI_Request request;
|
|
|
|
(void)MPI_Irecv(buf, count, datatype, source,
|
|
|
|
tag, mpi_comm, &request);
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
MPI::Comm::Iprobe(int source, int tag, MPI::Status & status) const
|
|
|
|
{
|
|
|
|
int t;
|
|
|
|
(void)MPI_Iprobe(source, tag, mpi_comm, &t, &status.mpi_status);
|
2006-08-28 18:51:09 +00:00
|
|
|
return OPAL_INT_TO_BOOL(t);
|
2004-07-14 14:11:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
MPI::Comm::Iprobe(int source, int tag) const
|
|
|
|
{
|
|
|
|
int t;
|
|
|
|
(void)MPI_Iprobe(source, tag, mpi_comm, &t, MPI_STATUS_IGNORE);
|
2006-08-28 18:51:09 +00:00
|
|
|
return OPAL_INT_TO_BOOL(t);
|
2004-07-14 14:11:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Probe(int source, int tag, MPI::Status & status) const
|
|
|
|
{
|
|
|
|
(void)MPI_Probe(source, tag, mpi_comm, &status.mpi_status);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Probe(int source, int tag) const
|
|
|
|
{
|
|
|
|
(void)MPI_Probe(source, tag, mpi_comm, MPI_STATUS_IGNORE);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline MPI::Prequest
|
|
|
|
MPI::Comm::Send_init(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
|
|
|
MPI_Request request;
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Send_init(const_cast<void *>(buf), count, datatype,
|
2004-07-14 14:11:03 +00:00
|
|
|
dest, tag, mpi_comm, &request);
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline MPI::Prequest
|
|
|
|
MPI::Comm::Bsend_init(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
|
|
|
MPI_Request request;
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Bsend_init(const_cast<void *>(buf), count, datatype,
|
2004-07-14 14:11:03 +00:00
|
|
|
dest, tag, mpi_comm, &request);
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline MPI::Prequest
|
|
|
|
MPI::Comm::Ssend_init(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
|
|
|
MPI_Request request;
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Ssend_init(const_cast<void *>(buf), count, datatype,
|
2004-07-14 14:11:03 +00:00
|
|
|
dest, tag, mpi_comm, &request);
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline MPI::Prequest
|
|
|
|
MPI::Comm::Rsend_init(const void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest, int tag) const
|
|
|
|
{
|
|
|
|
MPI_Request request;
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Rsend_init(const_cast<void *>(buf), count, datatype,
|
2004-07-14 14:11:03 +00:00
|
|
|
dest, tag, mpi_comm, &request);
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline MPI::Prequest
|
|
|
|
MPI::Comm::Recv_init(void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int source, int tag) const
|
|
|
|
{
|
|
|
|
MPI_Request request;
|
|
|
|
(void)MPI_Recv_init(buf, count, datatype, source,
|
|
|
|
tag, mpi_comm, &request);
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Sendrecv(const void *sendbuf, int sendcount,
|
|
|
|
const MPI::Datatype & sendtype, int dest, int sendtag,
|
|
|
|
void *recvbuf, int recvcount,
|
|
|
|
const MPI::Datatype & recvtype, int source,
|
|
|
|
int recvtag, MPI::Status & status) const
|
|
|
|
{
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Sendrecv(const_cast<void *>(sendbuf), sendcount,
|
2004-07-14 14:11:03 +00:00
|
|
|
sendtype,
|
|
|
|
dest, sendtag, recvbuf, recvcount,
|
|
|
|
recvtype,
|
|
|
|
source, recvtag, mpi_comm, &status.mpi_status);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Sendrecv(const void *sendbuf, int sendcount,
|
|
|
|
const MPI::Datatype & sendtype, int dest, int sendtag,
|
|
|
|
void *recvbuf, int recvcount,
|
|
|
|
const MPI::Datatype & recvtype, int source,
|
|
|
|
int recvtag) const
|
|
|
|
{
|
2005-11-30 03:22:27 +00:00
|
|
|
(void)MPI_Sendrecv(const_cast<void *>(sendbuf), sendcount,
|
2004-07-14 14:11:03 +00:00
|
|
|
sendtype,
|
|
|
|
dest, sendtag, recvbuf, recvcount,
|
|
|
|
recvtype,
|
|
|
|
source, recvtag, mpi_comm, MPI_STATUS_IGNORE);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Sendrecv_replace(void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest,
|
|
|
|
int sendtag, int source,
|
|
|
|
int recvtag, MPI::Status & status) const
|
|
|
|
{
|
|
|
|
(void)MPI_Sendrecv_replace(buf, count, datatype, dest,
|
|
|
|
sendtag, source, recvtag, mpi_comm,
|
|
|
|
&status.mpi_status);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Sendrecv_replace(void *buf, int count,
|
|
|
|
const MPI::Datatype & datatype, int dest,
|
|
|
|
int sendtag, int source,
|
|
|
|
int recvtag) const
|
|
|
|
{
|
|
|
|
(void)MPI_Sendrecv_replace(buf, count, datatype, dest,
|
|
|
|
sendtag, source, recvtag, mpi_comm,
|
|
|
|
MPI_STATUS_IGNORE);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Groups, Contexts, and Communicators
|
|
|
|
//
|
|
|
|
|
|
|
|
inline MPI::Group
|
|
|
|
MPI::Comm::Get_group() const
|
|
|
|
{
|
|
|
|
MPI_Group group;
|
|
|
|
(void)MPI_Comm_group(mpi_comm, &group);
|
|
|
|
return group;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int
|
|
|
|
MPI::Comm::Get_size() const
|
|
|
|
{
|
|
|
|
int size;
|
|
|
|
(void)MPI_Comm_size (mpi_comm, &size);
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int
|
|
|
|
MPI::Comm::Get_rank() const
|
|
|
|
{
|
|
|
|
int rank;
|
|
|
|
(void)MPI_Comm_rank (mpi_comm, &rank);
|
|
|
|
return rank;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int
|
|
|
|
MPI::Comm::Compare(const MPI::Comm & comm1,
|
|
|
|
const MPI::Comm & comm2)
|
|
|
|
{
|
|
|
|
int result;
|
|
|
|
(void)MPI_Comm_compare(comm1, comm2, &result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
Fixes trac:817
The C++ bindings were not tracking keyvals properly -- they were
freeing some internal meta data when Free_keyval() was called, not
when the keyval was actually destroyed (keyvals are refcounted in the
C layer, just like all other MPI objects, because they can live for
long after their corresponding Free call is invoked). This commit
fixes this problem and several other things:
* Add infrastructure on the ompi_attribute_keyval_t for an "extra"
destructor pointer that will be invoked during the "real"
constructor (i.e., when OBJ_RELEASE puts the refcount to 0). This
allows calling back into the C++ layer to release meta data
associated with the keyval.
* Adjust all cases where keyvals are created to pass in relevant
destructors (NULL or the C++ destructor).
* Do essentially the same for MPI::Comm, MPI::Win, and MPI:Datatype:
* Move several functions out of the .cc file into the _inln.h file
since they no longer require locks
* Make the 4 Create_keyval() functions call a common back-end
keyval creation function that does the Right Thing depending on
whether C or C++ function pointers were used for the keyval
functions. The back-end function does not call the corresponding
C MPI_*_create_keyval function, but rather does the work itself
so that it can associate a "destructor" callback for the C++
bindings for when the keyval is actually destroyed.
* Change a few type names to be more indicative of what they are
(mostly dealing with keyvals [not "keys"]).
* Add the 3 missing bindings for MPI::Comm::Create_keyval().
* Remove MPI::Comm::comm_map (and associated types) because it's no
longer necessary in the intercepts -- it was a by-product of being
a portable C++ bindings layer. Now we can just query the C layer
directly to figure out what type a communicator is. This solves
some logistics / callback issues, too.
* Rename several types, variables, and fix many comments in the
back-end C attribute implementation to make the names really
reflect what they are (keyvals vs. attributes). The previous names
heavily overloaded the name "key" and were ''extremely''
confusing.
This commit was SVN r13565.
The following Trac tickets were found above:
Ticket 817 --> https://svn.open-mpi.org/trac/ompi/ticket/817
2007-02-08 23:50:04 +00:00
|
|
|
inline void
|
|
|
|
MPI::Comm::Free(void)
|
|
|
|
{
|
|
|
|
(void)MPI_Comm_free(&mpi_comm);
|
|
|
|
}
|
|
|
|
|
2004-07-14 14:11:03 +00:00
|
|
|
inline bool
|
|
|
|
MPI::Comm::Is_inter() const
|
|
|
|
{
|
|
|
|
int t;
|
|
|
|
(void)MPI_Comm_test_inter(mpi_comm, &t);
|
2006-08-28 18:51:09 +00:00
|
|
|
return OPAL_INT_TO_BOOL(t);
|
2004-07-14 14:11:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-21 21:12:52 +00:00
|
|
|
//
|
|
|
|
// Collective Communication
|
|
|
|
//
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Barrier() const
|
|
|
|
{
|
|
|
|
(void)MPI_Barrier(mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Bcast(void *buffer, int count,
|
|
|
|
const MPI::Datatype& datatype, int root) const
|
|
|
|
{
|
|
|
|
(void)MPI_Bcast(buffer, count, datatype, root, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Gather(const void *sendbuf, int sendcount,
|
|
|
|
const MPI::Datatype & sendtype,
|
|
|
|
void *recvbuf, int recvcount,
|
|
|
|
const MPI::Datatype & recvtype, int root) const
|
|
|
|
{
|
|
|
|
(void)MPI_Gather(const_cast<void *>(sendbuf), sendcount, sendtype,
|
|
|
|
recvbuf, recvcount, recvtype, root, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Gatherv(const void *sendbuf, int sendcount,
|
|
|
|
const MPI::Datatype & sendtype, void *recvbuf,
|
|
|
|
const int recvcounts[], const int displs[],
|
|
|
|
const MPI::Datatype & recvtype, int root) const
|
|
|
|
{
|
|
|
|
(void)MPI_Gatherv(const_cast<void *>(sendbuf), sendcount, sendtype,
|
|
|
|
recvbuf, const_cast<int *>(recvcounts),
|
|
|
|
const_cast<int *>(displs),
|
|
|
|
recvtype, root, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Scatter(const void *sendbuf, int sendcount,
|
|
|
|
const MPI::Datatype & sendtype,
|
|
|
|
void *recvbuf, int recvcount,
|
|
|
|
const MPI::Datatype & recvtype, int root) const
|
|
|
|
{
|
|
|
|
(void)MPI_Scatter(const_cast<void *>(sendbuf), sendcount, sendtype,
|
|
|
|
recvbuf, recvcount, recvtype, root, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Scatterv(const void *sendbuf, const int sendcounts[],
|
|
|
|
const int displs[], const MPI::Datatype & sendtype,
|
|
|
|
void *recvbuf, int recvcount,
|
|
|
|
const MPI::Datatype & recvtype, int root) const
|
|
|
|
{
|
|
|
|
(void)MPI_Scatterv(const_cast<void *>(sendbuf),
|
|
|
|
const_cast<int *>(sendcounts),
|
|
|
|
const_cast<int *>(displs), sendtype,
|
|
|
|
recvbuf, recvcount, recvtype,
|
|
|
|
root, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Allgather(const void *sendbuf, int sendcount,
|
|
|
|
const MPI::Datatype & sendtype, void *recvbuf,
|
|
|
|
int recvcount, const MPI::Datatype & recvtype) const
|
|
|
|
{
|
|
|
|
(void)MPI_Allgather(const_cast<void *>(sendbuf), sendcount,
|
|
|
|
sendtype, recvbuf, recvcount,
|
|
|
|
recvtype, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Allgatherv(const void *sendbuf, int sendcount,
|
|
|
|
const MPI::Datatype & sendtype, void *recvbuf,
|
|
|
|
const int recvcounts[], const int displs[],
|
|
|
|
const MPI::Datatype & recvtype) const
|
|
|
|
{
|
|
|
|
(void)MPI_Allgatherv(const_cast<void *>(sendbuf), sendcount,
|
|
|
|
sendtype, recvbuf,
|
|
|
|
const_cast<int *>(recvcounts),
|
|
|
|
const_cast<int *>(displs),
|
|
|
|
recvtype, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Alltoall(const void *sendbuf, int sendcount,
|
|
|
|
const MPI::Datatype & sendtype, void *recvbuf,
|
|
|
|
int recvcount, const MPI::Datatype & recvtype) const
|
|
|
|
{
|
|
|
|
(void)MPI_Alltoall(const_cast<void *>(sendbuf), sendcount,
|
|
|
|
sendtype, recvbuf, recvcount,
|
|
|
|
recvtype, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Alltoallv(const void *sendbuf, const int sendcounts[],
|
|
|
|
const int sdispls[], const MPI::Datatype & sendtype,
|
|
|
|
void *recvbuf, const int recvcounts[],
|
|
|
|
const int rdispls[],
|
|
|
|
const MPI::Datatype & recvtype) const
|
|
|
|
{
|
|
|
|
(void)MPI_Alltoallv(const_cast<void *>(sendbuf),
|
|
|
|
const_cast<int *>(sendcounts),
|
|
|
|
const_cast<int *>(sdispls), sendtype, recvbuf,
|
|
|
|
const_cast<int *>(recvcounts),
|
|
|
|
const_cast<int *>(rdispls),
|
|
|
|
recvtype,mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Alltoallw(const void *sendbuf, const int sendcounts[],
|
|
|
|
const int sdispls[], const MPI::Datatype sendtypes[],
|
|
|
|
void *recvbuf, const int recvcounts[],
|
|
|
|
const int rdispls[],
|
|
|
|
const MPI::Datatype recvtypes[]) const
|
|
|
|
{
|
|
|
|
const int comm_size = Get_size();
|
|
|
|
MPI_Datatype *const data_type_tbl = new MPI_Datatype [2*comm_size];
|
|
|
|
|
|
|
|
// This must be done because MPI::Datatype arrays cannot be
|
|
|
|
// converted directly into MPI_Datatype arrays.
|
|
|
|
for (int i_rank=0; i_rank < comm_size; i_rank++) {
|
|
|
|
data_type_tbl[i_rank] = sendtypes[i_rank];
|
|
|
|
data_type_tbl[i_rank + comm_size] = recvtypes[i_rank];
|
|
|
|
}
|
|
|
|
|
|
|
|
(void)MPI_Alltoallw(const_cast<void *>(sendbuf),
|
|
|
|
const_cast<int *>(sendcounts),
|
|
|
|
const_cast<int *>(sdispls),
|
|
|
|
data_type_tbl, recvbuf,
|
|
|
|
const_cast<int *>(recvcounts),
|
|
|
|
const_cast<int *>(rdispls),
|
|
|
|
&data_type_tbl[comm_size], mpi_comm);
|
|
|
|
|
|
|
|
delete[] data_type_tbl;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Reduce(const void *sendbuf, void *recvbuf, int count,
|
|
|
|
const MPI::Datatype & datatype, const MPI::Op& op,
|
|
|
|
int root) const
|
|
|
|
{
|
|
|
|
(void)MPI_Reduce(const_cast<void *>(sendbuf), recvbuf, count, datatype, op, root, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Allreduce(const void *sendbuf, void *recvbuf, int count,
|
|
|
|
const MPI::Datatype & datatype, const MPI::Op& op) const
|
|
|
|
{
|
|
|
|
(void)MPI_Allreduce (const_cast<void *>(sendbuf), recvbuf, count, datatype, op, mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Reduce_scatter(const void *sendbuf, void *recvbuf,
|
|
|
|
int recvcounts[],
|
|
|
|
const MPI::Datatype & datatype,
|
|
|
|
const MPI::Op& op) const
|
|
|
|
{
|
|
|
|
(void)MPI_Reduce_scatter(const_cast<void *>(sendbuf), recvbuf, recvcounts,
|
|
|
|
datatype, op, mpi_comm);
|
|
|
|
}
|
|
|
|
|
2004-07-14 14:11:03 +00:00
|
|
|
//
|
|
|
|
// Process Creation and Managemnt
|
|
|
|
//
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Disconnect()
|
|
|
|
{
|
|
|
|
(void) MPI_Comm_disconnect(&mpi_comm);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline MPI::Intercomm
|
|
|
|
MPI::Comm::Get_parent()
|
|
|
|
{
|
|
|
|
MPI_Comm parent;
|
|
|
|
MPI_Comm_get_parent(&parent);
|
|
|
|
return parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline MPI::Intercomm
|
|
|
|
MPI::Comm::Join(const int fd)
|
|
|
|
{
|
|
|
|
MPI_Comm newcomm;
|
|
|
|
(void) MPI_Comm_join((int) fd, &newcomm);
|
|
|
|
return newcomm;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// External Interfaces
|
|
|
|
//
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Get_name(char* comm_name, int& resultlen) const
|
|
|
|
{
|
|
|
|
(void) MPI_Comm_get_name(mpi_comm, comm_name, &resultlen);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Set_name(const char* comm_name)
|
|
|
|
{
|
2005-11-30 03:22:27 +00:00
|
|
|
(void) MPI_Comm_set_name(mpi_comm, const_cast<char *>(comm_name));
|
2004-07-14 14:11:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
//Process Topologies
|
|
|
|
//
|
|
|
|
|
|
|
|
inline int
|
|
|
|
MPI::Comm::Get_topology() const
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
(void)MPI_Topo_test(mpi_comm, &status);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Environmental Inquiry
|
|
|
|
//
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Abort(int errorcode)
|
|
|
|
{
|
|
|
|
(void)MPI_Abort(mpi_comm, errorcode);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// These C++ bindings are for MPI-2.
|
|
|
|
// The MPI-1.2 functions called below are all
|
|
|
|
// going to be deprecated and replaced in MPI-2.
|
|
|
|
//
|
|
|
|
|
|
|
|
inline MPI::Errhandler
|
|
|
|
MPI::Comm::Get_errhandler() const
|
|
|
|
{
|
2008-02-10 19:29:25 +00:00
|
|
|
MPI_Errhandler errhandler;
|
|
|
|
MPI_Comm_get_errhandler(mpi_comm, &errhandler);
|
|
|
|
return errhandler;
|
2004-07-14 14:11:03 +00:00
|
|
|
}
|
|
|
|
|
2008-02-10 19:29:25 +00:00
|
|
|
inline void
|
2012-02-17 11:42:51 +00:00
|
|
|
MPI::Comm::Set_errhandler(const MPI::Errhandler& errhandler)
|
2008-02-10 19:29:25 +00:00
|
|
|
{
|
|
|
|
(void)MPI_Comm_set_errhandler(mpi_comm, errhandler);
|
2004-07-14 14:11:03 +00:00
|
|
|
}
|
|
|
|
|
2008-09-05 12:26:06 +00:00
|
|
|
inline void
|
|
|
|
MPI::Comm::Call_errhandler(int errorcode) const
|
|
|
|
{
|
|
|
|
(void) MPI_Comm_call_errhandler(mpi_comm, errorcode);
|
|
|
|
}
|
|
|
|
|
Fixes trac:817
The C++ bindings were not tracking keyvals properly -- they were
freeing some internal meta data when Free_keyval() was called, not
when the keyval was actually destroyed (keyvals are refcounted in the
C layer, just like all other MPI objects, because they can live for
long after their corresponding Free call is invoked). This commit
fixes this problem and several other things:
* Add infrastructure on the ompi_attribute_keyval_t for an "extra"
destructor pointer that will be invoked during the "real"
constructor (i.e., when OBJ_RELEASE puts the refcount to 0). This
allows calling back into the C++ layer to release meta data
associated with the keyval.
* Adjust all cases where keyvals are created to pass in relevant
destructors (NULL or the C++ destructor).
* Do essentially the same for MPI::Comm, MPI::Win, and MPI:Datatype:
* Move several functions out of the .cc file into the _inln.h file
since they no longer require locks
* Make the 4 Create_keyval() functions call a common back-end
keyval creation function that does the Right Thing depending on
whether C or C++ function pointers were used for the keyval
functions. The back-end function does not call the corresponding
C MPI_*_create_keyval function, but rather does the work itself
so that it can associate a "destructor" callback for the C++
bindings for when the keyval is actually destroyed.
* Change a few type names to be more indicative of what they are
(mostly dealing with keyvals [not "keys"]).
* Add the 3 missing bindings for MPI::Comm::Create_keyval().
* Remove MPI::Comm::comm_map (and associated types) because it's no
longer necessary in the intercepts -- it was a by-product of being
a portable C++ bindings layer. Now we can just query the C layer
directly to figure out what type a communicator is. This solves
some logistics / callback issues, too.
* Rename several types, variables, and fix many comments in the
back-end C attribute implementation to make the names really
reflect what they are (keyvals vs. attributes). The previous names
heavily overloaded the name "key" and were ''extremely''
confusing.
This commit was SVN r13565.
The following Trac tickets were found above:
Ticket 817 --> https://svn.open-mpi.org/trac/ompi/ticket/817
2007-02-08 23:50:04 +00:00
|
|
|
// 1) original Create_keyval that takes the first 2 arguments as C++
|
|
|
|
// functions
|
|
|
|
inline int
|
|
|
|
MPI::Comm::Create_keyval(MPI::Comm::Copy_attr_function* comm_copy_attr_fn,
|
|
|
|
MPI::Comm::Delete_attr_function* comm_delete_attr_fn,
|
|
|
|
void* extra_state)
|
|
|
|
{
|
|
|
|
// Back-end function does the heavy lifting
|
2008-01-12 13:20:38 +00:00
|
|
|
int ret, keyval;
|
|
|
|
ret = do_create_keyval(NULL, NULL,
|
|
|
|
comm_copy_attr_fn, comm_delete_attr_fn,
|
|
|
|
extra_state, keyval);
|
|
|
|
return (MPI_SUCCESS == ret) ? keyval : ret;
|
Fixes trac:817
The C++ bindings were not tracking keyvals properly -- they were
freeing some internal meta data when Free_keyval() was called, not
when the keyval was actually destroyed (keyvals are refcounted in the
C layer, just like all other MPI objects, because they can live for
long after their corresponding Free call is invoked). This commit
fixes this problem and several other things:
* Add infrastructure on the ompi_attribute_keyval_t for an "extra"
destructor pointer that will be invoked during the "real"
constructor (i.e., when OBJ_RELEASE puts the refcount to 0). This
allows calling back into the C++ layer to release meta data
associated with the keyval.
* Adjust all cases where keyvals are created to pass in relevant
destructors (NULL or the C++ destructor).
* Do essentially the same for MPI::Comm, MPI::Win, and MPI:Datatype:
* Move several functions out of the .cc file into the _inln.h file
since they no longer require locks
* Make the 4 Create_keyval() functions call a common back-end
keyval creation function that does the Right Thing depending on
whether C or C++ function pointers were used for the keyval
functions. The back-end function does not call the corresponding
C MPI_*_create_keyval function, but rather does the work itself
so that it can associate a "destructor" callback for the C++
bindings for when the keyval is actually destroyed.
* Change a few type names to be more indicative of what they are
(mostly dealing with keyvals [not "keys"]).
* Add the 3 missing bindings for MPI::Comm::Create_keyval().
* Remove MPI::Comm::comm_map (and associated types) because it's no
longer necessary in the intercepts -- it was a by-product of being
a portable C++ bindings layer. Now we can just query the C layer
directly to figure out what type a communicator is. This solves
some logistics / callback issues, too.
* Rename several types, variables, and fix many comments in the
back-end C attribute implementation to make the names really
reflect what they are (keyvals vs. attributes). The previous names
heavily overloaded the name "key" and were ''extremely''
confusing.
This commit was SVN r13565.
The following Trac tickets were found above:
Ticket 817 --> https://svn.open-mpi.org/trac/ompi/ticket/817
2007-02-08 23:50:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 2) overload Create_keyval to take the first 2 arguments as C
|
|
|
|
// functions
|
|
|
|
inline int
|
|
|
|
MPI::Comm::Create_keyval(MPI_Comm_copy_attr_function* comm_copy_attr_fn,
|
|
|
|
MPI_Comm_delete_attr_function* comm_delete_attr_fn,
|
|
|
|
void* extra_state)
|
|
|
|
{
|
|
|
|
// Back-end function does the heavy lifting
|
2008-01-12 13:20:38 +00:00
|
|
|
int ret, keyval;
|
2008-01-21 14:34:25 +00:00
|
|
|
ret = do_create_keyval(comm_copy_attr_fn, comm_delete_attr_fn,
|
|
|
|
NULL, NULL,
|
|
|
|
extra_state, keyval);
|
2008-01-12 13:20:38 +00:00
|
|
|
return (MPI_SUCCESS == ret) ? keyval : ret;
|
Fixes trac:817
The C++ bindings were not tracking keyvals properly -- they were
freeing some internal meta data when Free_keyval() was called, not
when the keyval was actually destroyed (keyvals are refcounted in the
C layer, just like all other MPI objects, because they can live for
long after their corresponding Free call is invoked). This commit
fixes this problem and several other things:
* Add infrastructure on the ompi_attribute_keyval_t for an "extra"
destructor pointer that will be invoked during the "real"
constructor (i.e., when OBJ_RELEASE puts the refcount to 0). This
allows calling back into the C++ layer to release meta data
associated with the keyval.
* Adjust all cases where keyvals are created to pass in relevant
destructors (NULL or the C++ destructor).
* Do essentially the same for MPI::Comm, MPI::Win, and MPI:Datatype:
* Move several functions out of the .cc file into the _inln.h file
since they no longer require locks
* Make the 4 Create_keyval() functions call a common back-end
keyval creation function that does the Right Thing depending on
whether C or C++ function pointers were used for the keyval
functions. The back-end function does not call the corresponding
C MPI_*_create_keyval function, but rather does the work itself
so that it can associate a "destructor" callback for the C++
bindings for when the keyval is actually destroyed.
* Change a few type names to be more indicative of what they are
(mostly dealing with keyvals [not "keys"]).
* Add the 3 missing bindings for MPI::Comm::Create_keyval().
* Remove MPI::Comm::comm_map (and associated types) because it's no
longer necessary in the intercepts -- it was a by-product of being
a portable C++ bindings layer. Now we can just query the C layer
directly to figure out what type a communicator is. This solves
some logistics / callback issues, too.
* Rename several types, variables, and fix many comments in the
back-end C attribute implementation to make the names really
reflect what they are (keyvals vs. attributes). The previous names
heavily overloaded the name "key" and were ''extremely''
confusing.
This commit was SVN r13565.
The following Trac tickets were found above:
Ticket 817 --> https://svn.open-mpi.org/trac/ompi/ticket/817
2007-02-08 23:50:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 3) overload Create_keyval to take the first 2 arguments as C++ & C
|
|
|
|
// functions
|
|
|
|
inline int
|
|
|
|
MPI::Comm::Create_keyval(MPI::Comm::Copy_attr_function* comm_copy_attr_fn,
|
|
|
|
MPI_Comm_delete_attr_function* comm_delete_attr_fn,
|
|
|
|
void* extra_state)
|
|
|
|
{
|
|
|
|
// Back-end function does the heavy lifting
|
2008-01-12 13:20:38 +00:00
|
|
|
int ret, keyval;
|
2008-01-21 14:34:25 +00:00
|
|
|
ret = do_create_keyval(NULL, comm_delete_attr_fn,
|
|
|
|
comm_copy_attr_fn, NULL,
|
|
|
|
extra_state, keyval);
|
2008-01-12 13:20:38 +00:00
|
|
|
return (MPI_SUCCESS == ret) ? keyval : ret;
|
Fixes trac:817
The C++ bindings were not tracking keyvals properly -- they were
freeing some internal meta data when Free_keyval() was called, not
when the keyval was actually destroyed (keyvals are refcounted in the
C layer, just like all other MPI objects, because they can live for
long after their corresponding Free call is invoked). This commit
fixes this problem and several other things:
* Add infrastructure on the ompi_attribute_keyval_t for an "extra"
destructor pointer that will be invoked during the "real"
constructor (i.e., when OBJ_RELEASE puts the refcount to 0). This
allows calling back into the C++ layer to release meta data
associated with the keyval.
* Adjust all cases where keyvals are created to pass in relevant
destructors (NULL or the C++ destructor).
* Do essentially the same for MPI::Comm, MPI::Win, and MPI:Datatype:
* Move several functions out of the .cc file into the _inln.h file
since they no longer require locks
* Make the 4 Create_keyval() functions call a common back-end
keyval creation function that does the Right Thing depending on
whether C or C++ function pointers were used for the keyval
functions. The back-end function does not call the corresponding
C MPI_*_create_keyval function, but rather does the work itself
so that it can associate a "destructor" callback for the C++
bindings for when the keyval is actually destroyed.
* Change a few type names to be more indicative of what they are
(mostly dealing with keyvals [not "keys"]).
* Add the 3 missing bindings for MPI::Comm::Create_keyval().
* Remove MPI::Comm::comm_map (and associated types) because it's no
longer necessary in the intercepts -- it was a by-product of being
a portable C++ bindings layer. Now we can just query the C layer
directly to figure out what type a communicator is. This solves
some logistics / callback issues, too.
* Rename several types, variables, and fix many comments in the
back-end C attribute implementation to make the names really
reflect what they are (keyvals vs. attributes). The previous names
heavily overloaded the name "key" and were ''extremely''
confusing.
This commit was SVN r13565.
The following Trac tickets were found above:
Ticket 817 --> https://svn.open-mpi.org/trac/ompi/ticket/817
2007-02-08 23:50:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 4) overload Create_keyval to take the first 2 arguments as C & C++
|
|
|
|
// functions
|
|
|
|
inline int
|
|
|
|
MPI::Comm::Create_keyval(MPI_Comm_copy_attr_function* comm_copy_attr_fn,
|
|
|
|
MPI::Comm::Delete_attr_function* comm_delete_attr_fn,
|
|
|
|
void* extra_state)
|
|
|
|
{
|
|
|
|
// Back-end function does the heavy lifting
|
2008-01-12 13:20:38 +00:00
|
|
|
int ret, keyval;
|
2008-01-21 14:34:25 +00:00
|
|
|
ret = do_create_keyval(comm_copy_attr_fn, NULL,
|
|
|
|
NULL, comm_delete_attr_fn,
|
|
|
|
extra_state, keyval);
|
2008-01-12 13:20:38 +00:00
|
|
|
return (MPI_SUCCESS == ret) ? keyval : ret;
|
Fixes trac:817
The C++ bindings were not tracking keyvals properly -- they were
freeing some internal meta data when Free_keyval() was called, not
when the keyval was actually destroyed (keyvals are refcounted in the
C layer, just like all other MPI objects, because they can live for
long after their corresponding Free call is invoked). This commit
fixes this problem and several other things:
* Add infrastructure on the ompi_attribute_keyval_t for an "extra"
destructor pointer that will be invoked during the "real"
constructor (i.e., when OBJ_RELEASE puts the refcount to 0). This
allows calling back into the C++ layer to release meta data
associated with the keyval.
* Adjust all cases where keyvals are created to pass in relevant
destructors (NULL or the C++ destructor).
* Do essentially the same for MPI::Comm, MPI::Win, and MPI:Datatype:
* Move several functions out of the .cc file into the _inln.h file
since they no longer require locks
* Make the 4 Create_keyval() functions call a common back-end
keyval creation function that does the Right Thing depending on
whether C or C++ function pointers were used for the keyval
functions. The back-end function does not call the corresponding
C MPI_*_create_keyval function, but rather does the work itself
so that it can associate a "destructor" callback for the C++
bindings for when the keyval is actually destroyed.
* Change a few type names to be more indicative of what they are
(mostly dealing with keyvals [not "keys"]).
* Add the 3 missing bindings for MPI::Comm::Create_keyval().
* Remove MPI::Comm::comm_map (and associated types) because it's no
longer necessary in the intercepts -- it was a by-product of being
a portable C++ bindings layer. Now we can just query the C layer
directly to figure out what type a communicator is. This solves
some logistics / callback issues, too.
* Rename several types, variables, and fix many comments in the
back-end C attribute implementation to make the names really
reflect what they are (keyvals vs. attributes). The previous names
heavily overloaded the name "key" and were ''extremely''
confusing.
This commit was SVN r13565.
The following Trac tickets were found above:
Ticket 817 --> https://svn.open-mpi.org/trac/ompi/ticket/817
2007-02-08 23:50:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Free_keyval(int& comm_keyval)
|
|
|
|
{
|
2012-02-21 21:52:23 +00:00
|
|
|
(void) MPI_Comm_free_keyval(&comm_keyval);
|
Fixes trac:817
The C++ bindings were not tracking keyvals properly -- they were
freeing some internal meta data when Free_keyval() was called, not
when the keyval was actually destroyed (keyvals are refcounted in the
C layer, just like all other MPI objects, because they can live for
long after their corresponding Free call is invoked). This commit
fixes this problem and several other things:
* Add infrastructure on the ompi_attribute_keyval_t for an "extra"
destructor pointer that will be invoked during the "real"
constructor (i.e., when OBJ_RELEASE puts the refcount to 0). This
allows calling back into the C++ layer to release meta data
associated with the keyval.
* Adjust all cases where keyvals are created to pass in relevant
destructors (NULL or the C++ destructor).
* Do essentially the same for MPI::Comm, MPI::Win, and MPI:Datatype:
* Move several functions out of the .cc file into the _inln.h file
since they no longer require locks
* Make the 4 Create_keyval() functions call a common back-end
keyval creation function that does the Right Thing depending on
whether C or C++ function pointers were used for the keyval
functions. The back-end function does not call the corresponding
C MPI_*_create_keyval function, but rather does the work itself
so that it can associate a "destructor" callback for the C++
bindings for when the keyval is actually destroyed.
* Change a few type names to be more indicative of what they are
(mostly dealing with keyvals [not "keys"]).
* Add the 3 missing bindings for MPI::Comm::Create_keyval().
* Remove MPI::Comm::comm_map (and associated types) because it's no
longer necessary in the intercepts -- it was a by-product of being
a portable C++ bindings layer. Now we can just query the C layer
directly to figure out what type a communicator is. This solves
some logistics / callback issues, too.
* Rename several types, variables, and fix many comments in the
back-end C attribute implementation to make the names really
reflect what they are (keyvals vs. attributes). The previous names
heavily overloaded the name "key" and were ''extremely''
confusing.
This commit was SVN r13565.
The following Trac tickets were found above:
Ticket 817 --> https://svn.open-mpi.org/trac/ompi/ticket/817
2007-02-08 23:50:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Set_attr(int comm_keyval, const void* attribute_val) const
|
|
|
|
{
|
2012-02-21 21:52:23 +00:00
|
|
|
(void)MPI_Comm_set_attr(mpi_comm, comm_keyval, const_cast<void*>(attribute_val));
|
Fixes trac:817
The C++ bindings were not tracking keyvals properly -- they were
freeing some internal meta data when Free_keyval() was called, not
when the keyval was actually destroyed (keyvals are refcounted in the
C layer, just like all other MPI objects, because they can live for
long after their corresponding Free call is invoked). This commit
fixes this problem and several other things:
* Add infrastructure on the ompi_attribute_keyval_t for an "extra"
destructor pointer that will be invoked during the "real"
constructor (i.e., when OBJ_RELEASE puts the refcount to 0). This
allows calling back into the C++ layer to release meta data
associated with the keyval.
* Adjust all cases where keyvals are created to pass in relevant
destructors (NULL or the C++ destructor).
* Do essentially the same for MPI::Comm, MPI::Win, and MPI:Datatype:
* Move several functions out of the .cc file into the _inln.h file
since they no longer require locks
* Make the 4 Create_keyval() functions call a common back-end
keyval creation function that does the Right Thing depending on
whether C or C++ function pointers were used for the keyval
functions. The back-end function does not call the corresponding
C MPI_*_create_keyval function, but rather does the work itself
so that it can associate a "destructor" callback for the C++
bindings for when the keyval is actually destroyed.
* Change a few type names to be more indicative of what they are
(mostly dealing with keyvals [not "keys"]).
* Add the 3 missing bindings for MPI::Comm::Create_keyval().
* Remove MPI::Comm::comm_map (and associated types) because it's no
longer necessary in the intercepts -- it was a by-product of being
a portable C++ bindings layer. Now we can just query the C layer
directly to figure out what type a communicator is. This solves
some logistics / callback issues, too.
* Rename several types, variables, and fix many comments in the
back-end C attribute implementation to make the names really
reflect what they are (keyvals vs. attributes). The previous names
heavily overloaded the name "key" and were ''extremely''
confusing.
This commit was SVN r13565.
The following Trac tickets were found above:
Ticket 817 --> https://svn.open-mpi.org/trac/ompi/ticket/817
2007-02-08 23:50:04 +00:00
|
|
|
}
|
|
|
|
|
2004-07-14 14:11:03 +00:00
|
|
|
inline bool
|
|
|
|
MPI::Comm::Get_attr(int comm_keyval, void* attribute_val) const
|
|
|
|
{
|
|
|
|
int flag;
|
2012-02-21 21:52:23 +00:00
|
|
|
(void)MPI_Comm_get_attr(mpi_comm, comm_keyval, attribute_val, &flag);
|
2006-08-28 18:51:09 +00:00
|
|
|
return OPAL_INT_TO_BOOL(flag);
|
2004-07-14 14:11:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
MPI::Comm::Delete_attr(int comm_keyval)
|
|
|
|
{
|
2012-02-21 21:52:23 +00:00
|
|
|
(void)MPI_Comm_delete_attr(mpi_comm, comm_keyval);
|
2004-07-14 14:11:03 +00:00
|
|
|
}
|
|
|
|
|
2011-08-04 15:39:12 +00:00
|
|
|
// Comment out the unused parameters so that compilers don't warn
|
|
|
|
// about them. Use comments instead of just deleting the param names
|
|
|
|
// outright so that we know/remember what they are.
|
2004-07-14 14:11:03 +00:00
|
|
|
inline int
|
2011-08-04 15:39:12 +00:00
|
|
|
MPI::Comm::NULL_COPY_FN(const MPI::Comm& /* oldcomm */,
|
|
|
|
int /* comm_keyval */,
|
|
|
|
void* /* extra_state */,
|
|
|
|
void* /* attribute_val_in */,
|
|
|
|
void* /* attribute_val_out */,
|
|
|
|
bool& flag)
|
2004-07-14 14:11:03 +00:00
|
|
|
{
|
2007-12-21 16:23:13 +00:00
|
|
|
flag = false;
|
2004-07-14 14:11:03 +00:00
|
|
|
return MPI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int
|
|
|
|
MPI::Comm::DUP_FN(const MPI::Comm& oldcomm, int comm_keyval,
|
|
|
|
void* extra_state, void* attribute_val_in,
|
|
|
|
void* attribute_val_out, bool& flag)
|
|
|
|
{
|
2009-08-04 18:45:11 +00:00
|
|
|
if (sizeof(bool) != sizeof(int)) {
|
|
|
|
int f = (int)flag;
|
|
|
|
int ret;
|
2012-02-21 21:52:23 +00:00
|
|
|
ret = MPI_COMM_DUP_FN(oldcomm, comm_keyval, extra_state,
|
|
|
|
attribute_val_in, attribute_val_out, &f);
|
2009-08-04 18:45:11 +00:00
|
|
|
flag = OPAL_INT_TO_BOOL(f);
|
|
|
|
return ret;
|
|
|
|
} else {
|
2012-02-21 21:52:23 +00:00
|
|
|
return MPI_COMM_DUP_FN(oldcomm, comm_keyval, extra_state,
|
|
|
|
attribute_val_in, attribute_val_out,
|
|
|
|
(int*)&flag);
|
2009-08-04 18:45:11 +00:00
|
|
|
}
|
2004-07-14 14:11:03 +00:00
|
|
|
}
|
|
|
|
|
2011-08-04 15:39:12 +00:00
|
|
|
// Comment out the unused parameters so that compilers don't warn
|
|
|
|
// about them. Use comments instead of just deleting the param names
|
|
|
|
// outright so that we know/remember what they are.
|
2004-07-14 14:11:03 +00:00
|
|
|
inline int
|
2011-08-04 15:39:12 +00:00
|
|
|
MPI::Comm::NULL_DELETE_FN(MPI::Comm& /* comm */,
|
|
|
|
int /* comm_keyval */,
|
|
|
|
void* /* attribute_val */,
|
|
|
|
void* /* extra_state */)
|
2004-07-14 14:11:03 +00:00
|
|
|
{
|
|
|
|
return MPI_SUCCESS;
|
|
|
|
}
|
|
|
|
|