MOAB: Mesh Oriented datABase  (version 5.4.1)
CrystalRouterExample.cpp
Go to the documentation of this file.
00001 /*
00002  * This example will show one of the building blocks of parallel infrastructure in MOAB
00003  * More exactly, if we have some homogeneous data to communicate from each processor to a list of
00004  * other processors, how do we do it?
00005  *
00006  * introduce the TupleList and crystal router to MOAB users.
00007  *
00008  * This technology is used in resolving shared vertices / sets between partitions
00009  * It is used in the mbcoupler for sending data (target points) to the proper processor, and
00010  * communicate back the results. Also, it is used to communicate departure mesh for intersection in
00011  * parallel
00012  *
00013  *  It is a way of doing  MPI_gatheralltoallv(), when the communication matrix is sparse
00014  *
00015  *  It is assumed that every proc needs to communicate only with a few of the other processors.
00016  *  If every processor needs to communicate with all other, then we will have to use paired isend
00017  * and irecv, the communication matrix is full
00018  *
00019  *  the example needs to be launched in parallel.
00020  *  Every proc will build a list of tuples, that will be send to a few procs;
00021  *  In general, we will send to num_comms tasks, and about num_tuples to each task
00022  *  We vary num_comms and num_tuples for processor
00023  *
00024  *  we will send long ints of the form
00025  *    100000 * send + 1000* rank +j, where j is the index of tuple
00026  *
00027  *  after routing, we verify we received
00028  *    100000 * rank + 1000 * from
00029  *
00030  *    For some reportrank we also print the tuples.
00031  *
00032  *  after routing, we will see if we received, as expected. Should run on at least 2 processors.
00033  *
00034  * Note: We do not need a moab instance for this example
00035  *
00036  */
00037 
00038 /** @example CrystalRouterExample.cpp \n
00039  * \brief generalized gather scatter using tuples \n
00040  * <b>To run</b>: mpiexec -np <n> CrystalRouterExample -r [reportrank] -t [num_tuples] -n
00041  * [num_comms] \n
00042  *
00043  */
00044 //
00045 #include "moab/MOABConfig.h"
00046 #ifdef MOAB_HAVE_MPI
00047 #include "moab/ProcConfig.hpp"
00048 #endif
00049 #include "moab/TupleList.hpp"
00050 #include "moab/ProgOptions.hpp"
00051 #include "moab/ErrorHandler.hpp"
00052 #include <ctime>
00053 #include <iostream>
00054 #include <sstream>
00055 
00056 const char BRIEF_DESC[] = "Example of gather scatter with tuple lists \n";
00057 std::ostringstream LONG_DESC;
00058 
00059 using namespace moab;
00060 using namespace std;
00061 
00062 int main( int argc, char** argv )
00063 {
00064 #ifdef MOAB_HAVE_MPI
00065     MPI_Init( &argc, &argv );
00066 
00067     // Initialize error handler, required for this example (not using a moab instance)
00068     MBErrorHandler_Init();
00069 
00070     ProcConfig pc( MPI_COMM_WORLD );
00071     int size = pc.proc_size();
00072     int rank = pc.proc_rank();
00073 
00074     // Start copy
00075     LONG_DESC << "This program does a gather scatter with a list of tuples. \n"
00076                  " It tries to see how much communication costs in terms of time and memory. \n"
00077               << "It starts with creating a list of tuples to be sent from each processor, \n to a "
00078                  "list of other processors.\n"
00079               << "The number of tuples and how many tasks to communicate to are controlled by "
00080                  "input parameters.\n"
00081               << "After communication, we verify locally if we received what we expected. \n";
00082     ProgOptions opts( LONG_DESC.str(), BRIEF_DESC );
00083 
00084     // How many procs communicate to current proc, on average (we will vary that too)
00085     int num_comms = 2;
00086     opts.addOpt< int >( "num_comms,n", "each task will send to about num_comms other tasks some tuples (default 2)",
00087                         &num_comms );
00088 
00089     int num_tuples = 4;
00090     opts.addOpt< int >( "num_tuples,t", "each task will send to some task about num_tuples tuples (default 4)",
00091                         &num_tuples );
00092 
00093     int reportrank = size + 1;
00094     opts.addOpt< int >( "reporting_rank,r",
00095                         "this rank will report the tuples sent and the tuples received; it could "
00096                         "be higher than num_procs, then no reporting",
00097                         &reportrank );
00098 
00099     opts.parseCommandLine( argc, argv );
00100 
00101     if( rank == reportrank || ( reportrank >= size && 0 == rank ) )
00102     {
00103         cout << " There are " << size << " tasks in example.\n";
00104         cout << " We will send groups of " << num_tuples << " from each task towards " << num_comms
00105              << " other tasks.\n";
00106     }
00107 
00108     // Send some data from proc i to i + n/2, also to i + n/2 + 1 modulo n, where n is num procs
00109 
00110     gs_data::crystal_data* cd = pc.crystal_router();
00111 
00112     long total_n_tuples = num_comms * num_tuples;
00113 
00114     // Vary the number of tasks to send to, and the number of tuples to send
00115     if( rank < size / 2 )
00116         num_comms--;
00117     else
00118         num_comms++;
00119 
00120     if( rank < size / 3 )
00121         num_tuples *= 2;
00122     else if( rank > size - size / 3 )
00123         num_tuples /= 2;
00124 
00125     TupleList tl;
00126     // At most num_tuples* num_comms to send
00127     // We do a preallocate with this; some tuples on some processors might need more memory, to be
00128     // able to grow locally; Some tasks might receive more tuples though, and in the process, some
00129     // might grow more than others. By doing these logP sends/receives, we do not grow local memory
00130     // too much.
00131     tl.initialize( 1, 1, 0, 1, num_tuples * num_comms );
00132     tl.enableWriteAccess();
00133     // Form num_tuples*num_comms tuples, send to various ranks
00134     unsigned int n = tl.get_n();
00135     for( int i = 0; i < num_comms; i++ )
00136     {
00137         int sendTo     = rank + i * size / 2 + 1;  // Spread out the send to, for a stress-like test
00138         sendTo         = sendTo % size;            //
00139         long intToSend = 1000 * rank + 100000 * sendTo;
00140         for( int j = 0; j < num_tuples; j++ )
00141         {
00142             n           = tl.get_n();
00143             tl.vi_wr[n] = sendTo;
00144             tl.vl_wr[n] = intToSend + j;
00145             tl.vr_wr[n] = 10000. * rank + j;
00146             tl.inc_n();
00147         }
00148     }
00149 
00150     if( rank == reportrank )
00151     {
00152         cout << "rank " << rank << "\n";
00153         tl.print( " before sending" );
00154     }
00155 
00156     clock_t tt = clock();
00157     // All communication happens here; no mpi calls for the user
00158     ErrorCode rval = cd->gs_transfer( 1, tl, 0 );MB_CHK_SET_ERR( rval, "Error in tuple transfer" );
00159 
00160     double secs = 0;
00161     if( rank == reportrank || ( reportrank >= size && 0 == rank ) )
00162     {
00163         secs = ( clock() - tt ) / (double)CLOCKS_PER_SEC;
00164     }
00165     if( rank == reportrank )
00166     {
00167         cout << "rank " << rank << "\n";
00168         tl.print( " after transfer" );
00169     }
00170 
00171     // Check that all tuples received have the form 10000*rank + 100*from
00172     unsigned int received = tl.get_n();
00173     for( int i = 0; i < (int)received; i++ )
00174     {
00175         int from      = tl.vi_rd[i];
00176         long valrec   = tl.vl_rd[i];
00177         int remainder = valrec - 100000 * rank - 1000 * from;
00178         if( remainder < 0 || remainder >= num_tuples * 4 )
00179             cout << " error: tuple " << i << " received at proc rank " << rank << " from proc " << from << " has value "
00180                  << valrec << " remainder " << remainder << "\n";
00181     }
00182 
00183     if( rank == reportrank || ( reportrank >= size && 0 == rank ) )
00184     {
00185         cout << "communication of about " << total_n_tuples << " tuples/per proc took " << secs << " seconds"
00186              << "\n";
00187         tt = clock();
00188     }
00189 
00190     // Finalize error handler, required for this example (not using a moab instance)
00191     MBErrorHandler_Finalize();
00192 
00193     MPI_Finalize();
00194 #else
00195     std::cout << " Build with MPI for this example to work\n";
00196 #endif
00197 
00198     return 0;
00199 }
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Defines