Teuchos Package Browser (Single Doxygen Collection)  Version of the Day
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Groups Pages
scatterv.cpp
Go to the documentation of this file.
1 // @HEADER
2 // *****************************************************************************
3 // Teuchos: Common Tools Package
4 //
5 // Copyright 2004 NTESS and the Teuchos contributors.
6 // SPDX-License-Identifier: BSD-3-Clause
7 // *****************************************************************************
8 // @HEADER
9 
10 #include "Teuchos_ConfigDefs.hpp"
11 #include "Teuchos_DefaultComm.hpp"
12 #include "Teuchos_CommHelpers.hpp"
13 #ifdef HAVE_TEUCHOS_MPI
15 #endif // HAVE_TEUCHOS_MPI
17 
18 
19 template<class PacketType>
20 bool
21 testScatterv (bool& success, std::ostream& out,
22  const int root, const Teuchos::Comm<int>& comm)
23 {
24  using Teuchos::scatterv;
26  using std::endl;
27  typedef PacketType packet_type;
28 
29  // Teuchos constructs the output stream such that it only prints on
30  // Process 0 anyway.
31  Teuchos::OSTab tab0 (out);
32  out << "Testing Teuchos::scatterv<int, "
33  << TypeNameTraits<packet_type>::name ()
34  << "> with root = " << root << endl;
35  Teuchos::OSTab tab1 (out);
36 
37  int lclSuccess = 1;
38  int gblSuccess = lclSuccess;
39 
40 #ifdef HAVE_TEUCHOS_MPI
41  using Teuchos::MpiComm;
42  int errCode = MPI_SUCCESS;
43  const MpiComm<int>& mpiComm = Teuchos::dyn_cast<const MpiComm<int>> (comm);
44  MPI_Comm rawMpiComm = * (mpiComm.getRawMpiComm());
45 #endif // HAVE_TEUCHOS_MPI
46 
47  const packet_type ZERO = Teuchos::ScalarTraits<packet_type>::zero ();
48  const packet_type ONE = Teuchos::ScalarTraits<packet_type>::one ();
49  const int myRank = comm.getRank ();
50  const int numProcs = comm.getSize ();
51  const int sendCount = 10; // fixed count
52  const int recvCount = 10; // fixed count
53 
54  out << "Initializing receive buffer (on all processes)" << endl;
55 
56  // Set up the receive buffer on each process.
57  Teuchos::Array<packet_type> recvBuf (recvCount);
58  for (int i = 0; i < recvCount; ++i) {
59  recvBuf[i] = ZERO;
60  }
61 
62  out << "Filling send buffer (on root process only)" << endl;
63 
64  // The send buffer is only needed on the root process. It exists
65  // elsewhere for syntax's sake, but is only allocated with nonzero
66  // size on the root process.
68  if (myRank == root) {
69  sendBuf.resize (sendCount * numProcs);
70  try {
71  // Fill the send buffer. The root process will send (p+1)*1, (p+1)*2,
72  // ..., (p+1)*sendCount to Process p. This helps us test whether the
73  // scatterv worked correctly (with fixed count).
74  for (int p = 0; p < numProcs; ++p) {
75  for (int k = 0; k < sendCount; ++k) {
76  const packet_type val = static_cast<packet_type> (p+1) *
77  (ONE + static_cast<packet_type> (k));
78  sendBuf[p*sendCount + k] = val;
79  }
80  }
81  } catch (std::exception& e) {
82  std::cerr << "Root process " << root << " threw an exception: "
83  << e.what () << endl;
84  lclSuccess = 0;
85  }
86  }
87 
88 #ifdef HAVE_TEUCHOS_MPI
89  // Make sure that the root process didn't throw an exception when
90  // filling the send buffer, by broadcasting from it to all the other
91  // processes.
92  gblSuccess = lclSuccess;
93  errCode = MPI_Bcast (&gblSuccess, 1, MPI_INT, root, rawMpiComm);
95  (errCode != MPI_SUCCESS, std::logic_error, "MPI_Bcast failed!");
97  (gblSuccess != 1, std::logic_error, "Filling the send buffer failed on "
98  "the root (" << root << ") process! This probably indicates a bug in "
99  "the test.");
100 #endif // HAVE_TEUCHOS_MPI
101 
102  // Fill send count/displ arrays (fixed count)
103  Teuchos::Array<int> sendCounts(numProcs), displs(1+numProcs);
104  displs[0] = 0;
105  for( int i = 0; i < numProcs; ++i ) {
106  displs[i+1] = displs[i] + sendCount;
107  sendCounts[i] = sendCount;
108  }
109 
110  // Invoke the function to test.
111  out << "About to invoke scatterv" << endl;
112  Teuchos::scatterv <int, packet_type> (sendBuf.getRawPtr (), sendCounts.data(), displs.data(),
113  recvBuf.getRawPtr (), recvCount, root, comm);
114  out << "Done with scatterv" << endl;
115 
116  // Test whether each process got the expected results.
117  for (int k = 0; k < recvCount; ++k) {
118  const packet_type expectedVal = static_cast<packet_type> (myRank+1) *
119  (ONE + static_cast<packet_type> (k));
120  TEST_EQUALITY( recvBuf[k], expectedVal );
121  }
122  lclSuccess = success ? 1 : 0;
123  gblSuccess = lclSuccess;
124 
125  //
126  // Use an all-reduce to check whether all processes got what they
127  // should have gotten.
128  //
129  // Don't trust that any other Teuchos communicator wrapper functions
130  // work here. Instead, if building with MPI, use raw MPI. If not
131  // building with MPI, first test that comm has only one process,
132  // then assume this in the test.
133 
134 #ifdef HAVE_TEUCHOS_MPI
135  errCode = MPI_Allreduce (&lclSuccess, &gblSuccess, 1,
136  MPI_INT, MPI_MIN, rawMpiComm);
138  (errCode != MPI_SUCCESS, std::logic_error, "MPI_Allreduce failed!");
139 #else // HAVE_TEUCHOS_MPI
141  (comm.getSize () != 1, std::logic_error, "Not building with MPI, but "
142  "communicator has size = " << comm.getSize () << " != 1. We don't know "
143  "how to test this case.");
145  (comm.getRank () != root, std::logic_error, "Not building with MPI, but "
146  "the one process of rank " << comm.getRank () << " is not the root "
147  "process of rank " << root << ". We don't know how to test this case.");
148 #endif // HAVE_TEUCHOS_MPI
149 
150  return gblSuccess == 1;
151 }
152 
153 
154 TEUCHOS_UNIT_TEST_TEMPLATE_1_DECL( Comm, Scatterv, PacketType )
155 {
156  using Teuchos::Comm;
157  using Teuchos::RCP;
158  using std::endl;
159  typedef PacketType packet_type;
160 
161  Teuchos::OSTab tab0 (out);
162  out << "Testing Teuchos::scatterv" << endl;
163  Teuchos::OSTab tab1 (out);
164 
165  RCP<const Comm<int> > comm = Teuchos::DefaultComm<int>::getComm ();
166  const int numProcs = comm->getSize ();
167 
168  // Make sure that it works for all possible root processes in the
169  // communicator, not just Process 0.
170  for (int root = 0; root < numProcs; ++root) {
171  out << "Test for root = " << root << endl;
172  const bool curSuccess = testScatterv<packet_type> (success, out, root, *comm);
173  TEST_EQUALITY_CONST( curSuccess, true );
174  success = success && curSuccess;
175  }
176 
177  comm->barrier (); // make sure that everybody finished
178  out << "Done with test!" << endl;
179 }
180 
181 //
182 TEUCHOS_UNIT_TEST_TEMPLATE_1_INSTANT( Comm, Scatterv, double )
183 TEUCHOS_UNIT_TEST_TEMPLATE_1_INSTANT( Comm, Scatterv, float )
virtual int getSize() const =0
Returns the number of processes that make up this communicator.
virtual int getRank() const =0
Returns the rank of this process.
#define TEUCHOS_TEST_FOR_EXCEPTION(throw_exception_test, Exception, msg)
Macro for throwing an exception with breakpointing to ease debugging.
#define TEUCHOS_UNIT_TEST_TEMPLATE_1_DECL(TEST_GROUP, TEST_NAME, TYPE)
Macro for defining a templated unit test with one template parameter.
Teuchos header file which uses auto-configuration information to include necessary C++ headers...
#define TEST_EQUALITY(v1, v2)
Assert the equality of v1 and v2.
T * getRawPtr()
Return a raw pointer to beginning of array or NULL if unsized.
static Teuchos::RCP< const Comm< OrdinalType > > getComm()
Return the default global communicator.
Tabbing class for helping to create formated, indented output for a basic_FancyOStream object...
T_To & dyn_cast(T_From &from)
Dynamic casting utility function meant to replace dynamic_cast&lt;T&amp;&gt; by throwing a better documented er...
Unit testing support.
Implementation of Teuchos wrappers for MPI.
#define TEST_EQUALITY_CONST(v1, v2)
Assert the equality of v1 and constant v2.
T * data()
Return a raw pointer to beginning of array.
void resize(size_type new_size, const value_type &x=value_type())
Default traits class that just returns typeid(T).name().
static T zero()
Returns representation of zero for this scalar type.
#define TEUCHOS_UNIT_TEST_TEMPLATE_1_INSTANT(TEST_GROUP, TEST_NAME, TYPE)
Instantiate a templated unit test with one template parameter.
Smart reference counting pointer class for automatic garbage collection.
void scatterv(const Packet sendBuf[], const Ordinal sendCounts[], const Ordinal displs[], Packet recvBuf[], const Ordinal recvCount, const Ordinal root, const Comm< Ordinal > &comm)
static T one()
Returns representation of one for this scalar type.
bool testScatterv(bool &success, std::ostream &out, const int root, const Teuchos::Comm< int > &comm)
Definition: scatterv.cpp:21
Replacement for std::vector that is compatible with the Teuchos Memory Management classes...