2005-09-13 04:28:18 +00:00
/*
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 .
2006-04-19 23:42:06 +00:00
* Copyright ( c ) 2004 - 2006 The University of Tennessee and The University
2005-11-05 19:57:48 +00:00
* of Tennessee Research Foundation . All rights
* reserved .
2005-09-13 04:28:18 +00:00
* Copyright ( c ) 2004 - 2005 High Performance Computing Center Stuttgart ,
* University of Stuttgart . All rights reserved .
* Copyright ( c ) 2004 - 2005 The Regents of the University of California .
* All rights reserved .
* $ COPYRIGHT $
*
* Additional copyrights may follow
*
* $ HEADER $
*/
# include "ompi_config.h"
# include "mpi.h"
2006-02-12 01:33:29 +00:00
# include "ompi/constants.h"
# include "ompi/datatype/datatype.h"
# include "ompi/communicator/communicator.h"
# include "ompi/mca/coll/coll.h"
# include "ompi/mca/coll/base/coll_tags.h"
# include "ompi/mca/pml/pml.h"
# include "ompi/op/op.h"
2005-09-13 04:28:18 +00:00
# include "coll_tuned.h"
# include "coll_tuned_topo.h"
# include "coll_tuned_util.h"
2005-12-22 13:49:33 +00:00
int ompi_coll_tuned_alltoall_intra_pairwise ( void * sbuf , int scount ,
2006-10-18 02:00:46 +00:00
struct ompi_datatype_t * sdtype ,
void * rbuf , int rcount ,
struct ompi_datatype_t * rdtype ,
2007-08-19 03:37:49 +00:00
struct ompi_communicator_t * comm ,
struct mca_coll_base_module_1_1_0_t * module )
2005-09-13 04:28:18 +00:00
{
int line = - 1 , err = 0 ;
int rank , size , step ;
int sendto , recvfrom ;
void * tmpsend , * tmprecv ;
2006-10-17 20:20:58 +00:00
ptrdiff_t lb , sext , rext ;
2005-09-13 04:28:18 +00:00
size = ompi_comm_size ( comm ) ;
rank = ompi_comm_rank ( comm ) ;
2007-07-07 01:06:19 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream ,
" coll:tuned:alltoall_intra_pairwise rank %d " , rank ) ) ;
2005-09-13 04:28:18 +00:00
err = ompi_ddt_get_extent ( sdtype , & lb , & sext ) ;
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
err = ompi_ddt_get_extent ( rdtype , & lb , & rext ) ;
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
2007-07-07 01:06:19 +00:00
2006-10-18 02:00:46 +00:00
/* Perform pairwise exchange - starting from 1 so the local copy is last */
2007-07-07 01:06:19 +00:00
for ( step = 1 ; step < size + 1 ; step + + ) {
2005-09-13 04:28:18 +00:00
2007-07-07 01:06:19 +00:00
/* Determine sender and receiver for this step. */
sendto = ( rank + step ) % size ;
recvfrom = ( rank + size - step ) % size ;
2005-09-13 04:28:18 +00:00
2007-07-07 01:06:19 +00:00
/* Determine sending and receiving locations */
tmpsend = ( char * ) sbuf + sendto * sext * scount ;
tmprecv = ( char * ) rbuf + recvfrom * rext * rcount ;
2005-09-13 04:28:18 +00:00
2006-10-18 02:00:46 +00:00
/* send and receive */
2007-07-07 01:06:19 +00:00
err = ompi_coll_tuned_sendrecv ( tmpsend , scount , sdtype , sendto ,
MCA_COLL_BASE_TAG_ALLTOALL ,
tmprecv , rcount , rdtype , recvfrom ,
MCA_COLL_BASE_TAG_ALLTOALL ,
2006-10-18 02:00:46 +00:00
comm , MPI_STATUS_IGNORE , rank ) ;
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
}
2005-09-13 04:28:18 +00:00
2006-10-18 02:00:46 +00:00
return MPI_SUCCESS ;
2005-09-13 04:28:18 +00:00
err_hndl :
2007-07-07 01:06:19 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream ,
" %s:%4d \t Error occurred %d, rank %2d " , __FILE__ , line ,
err , rank ) ) ;
2006-10-18 02:00:46 +00:00
return err ;
2005-09-13 04:28:18 +00:00
}
2005-12-22 13:49:33 +00:00
int ompi_coll_tuned_alltoall_intra_bruck ( void * sbuf , int scount ,
2006-10-18 02:00:46 +00:00
struct ompi_datatype_t * sdtype ,
void * rbuf , int rcount ,
struct ompi_datatype_t * rdtype ,
2007-08-19 03:37:49 +00:00
struct ompi_communicator_t * comm ,
struct mca_coll_base_module_1_1_0_t * module )
2005-09-13 04:28:18 +00:00
{
int i , k , line = - 1 ;
int rank , size ;
2007-07-07 01:06:19 +00:00
int sendto , recvfrom , distance , * displs = NULL , * blen = NULL ;
char * tmpbuf = NULL , * tmpbuf_free = NULL ;
ptrdiff_t rlb , slb , tlb , sext , rext , tsext ;
2005-09-13 04:28:18 +00:00
int err = 0 ;
int weallocated = 0 ;
2007-07-07 01:06:19 +00:00
struct ompi_datatype_t * new_ddt ;
2007-08-19 03:37:49 +00:00
# ifdef blahblah
mca_coll_tuned_module_t * tuned_module = ( mca_coll_tuned_module_t * ) module ;
mca_coll_tuned_comm_t * data = tuned_module - > tuned_data ;
# endif
2005-09-13 04:28:18 +00:00
size = ompi_comm_size ( comm ) ;
rank = ompi_comm_rank ( comm ) ;
2007-07-07 01:06:19 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream ,
" coll:tuned:alltoall_intra_bruck rank %d " , rank ) ) ;
2005-09-13 04:28:18 +00:00
2007-07-07 01:06:19 +00:00
err = ompi_ddt_get_extent ( sdtype , & slb , & sext ) ;
2005-09-13 04:28:18 +00:00
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
2007-07-07 01:06:19 +00:00
err = ompi_ddt_get_true_extent ( sdtype , & tlb , & tsext ) ;
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
err = ompi_ddt_get_extent ( rdtype , & rlb , & rext ) ;
2005-09-13 04:28:18 +00:00
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
# ifdef blahblah
2007-07-07 01:06:19 +00:00
/* try and SAVE memory by using the data segment hung off
the communicator if possible */
2007-08-19 03:37:49 +00:00
if ( data - > mcct_num_reqs > = size ) {
2005-09-13 04:28:18 +00:00
/* we have enought preallocated for displments and lengths */
2007-08-19 03:37:49 +00:00
displs = ( int * ) data - > mcct_reqs ;
2005-09-13 04:28:18 +00:00
blen = ( int * ) ( displs + size ) ;
weallocated = 0 ;
}
else { /* allocate the buffers ourself */
# endif
2007-07-07 01:06:19 +00:00
displs = ( int * ) malloc ( size * sizeof ( int ) ) ;
2005-09-13 04:28:18 +00:00
if ( displs = = NULL ) { line = __LINE__ ; err = - 1 ; goto err_hndl ; }
2007-07-07 01:06:19 +00:00
blen = ( int * ) malloc ( size * sizeof ( int ) ) ;
2005-09-13 04:28:18 +00:00
if ( blen = = NULL ) { line = __LINE__ ; err = - 1 ; goto err_hndl ; }
weallocated = 1 ;
# ifdef blahblah
}
# endif
/* tmp buffer allocation for message data */
2007-07-07 01:06:19 +00:00
tmpbuf_free = ( char * ) malloc ( tsext + ( scount * size - 1 ) * sext ) ;
if ( tmpbuf_free = = NULL ) { line = __LINE__ ; err = - 1 ; goto err_hndl ; }
tmpbuf = tmpbuf_free - slb ;
2005-09-13 04:28:18 +00:00
/* Step 1 - local rotation - shift up by rank */
2007-07-07 01:06:19 +00:00
err = ompi_ddt_copy_content_same_ddt ( sdtype ,
( int32_t ) ( ( size - rank ) * scount ) ,
tmpbuf ,
( ( char * ) sbuf ) + rank * scount * sext ) ;
2005-09-13 04:28:18 +00:00
if ( err < 0 ) {
2006-10-18 02:00:46 +00:00
line = __LINE__ ; err = - 1 ; goto err_hndl ;
2005-09-13 04:28:18 +00:00
}
if ( rank ! = 0 ) {
2007-07-07 01:06:19 +00:00
err = ompi_ddt_copy_content_same_ddt ( sdtype , ( int32_t ) ( rank * scount ) ,
tmpbuf + ( size - rank ) * scount * sext ,
( char * ) sbuf ) ;
2005-09-13 04:28:18 +00:00
if ( err < 0 ) {
2006-10-18 02:00:46 +00:00
line = __LINE__ ; err = - 1 ; goto err_hndl ;
2005-09-13 04:28:18 +00:00
}
}
2006-10-18 02:00:46 +00:00
/* perform communication step */
for ( distance = 1 ; distance < size ; distance < < = 1 ) {
2007-07-07 01:06:19 +00:00
sendto = ( rank + distance ) % size ;
recvfrom = ( rank - distance + size ) % size ;
2006-10-18 02:00:46 +00:00
k = 0 ;
/* create indexed datatype */
for ( i = 1 ; i < size ; i + + ) {
2007-07-07 01:06:19 +00:00
if ( ( i & distance ) = = distance ) {
displs [ k ] = i * scount ;
blen [ k ] = scount ;
2006-10-18 02:00:46 +00:00
k + + ;
}
}
/* Set indexes and displacements */
2007-07-07 01:06:19 +00:00
err = ompi_ddt_create_indexed ( k , blen , displs , sdtype , & new_ddt ) ;
2006-10-18 02:00:46 +00:00
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
/* Commit the new datatype */
2007-07-07 01:06:19 +00:00
err = ompi_ddt_commit ( & new_ddt ) ;
2006-10-18 02:00:46 +00:00
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
/* Sendreceive */
2007-07-07 01:06:19 +00:00
err = ompi_coll_tuned_sendrecv ( tmpbuf , 1 , new_ddt , sendto ,
2006-10-18 02:00:46 +00:00
MCA_COLL_BASE_TAG_ALLTOALL ,
2007-07-07 01:06:19 +00:00
rbuf , 1 , new_ddt , recvfrom ,
2006-10-18 02:00:46 +00:00
MCA_COLL_BASE_TAG_ALLTOALL ,
2007-07-07 01:06:19 +00:00
comm , MPI_STATUS_IGNORE , rank ) ;
2006-10-18 02:00:46 +00:00
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
2007-07-07 01:06:19 +00:00
/* Copy back new data from recvbuf to tmpbuf */
2007-09-12 15:29:58 +00:00
err = ompi_ddt_copy_content_same_ddt ( new_ddt , 1 , tmpbuf , ( char * ) rbuf ) ;
2007-07-07 01:06:19 +00:00
if ( err < 0 ) { line = __LINE__ ; err = - 1 ; goto err_hndl ; }
2005-09-13 04:28:18 +00:00
2006-10-18 02:00:46 +00:00
/* free ddt */
2007-07-07 01:06:19 +00:00
err = ompi_ddt_destroy ( & new_ddt ) ;
2006-10-18 02:00:46 +00:00
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
} /* end of for (distance = 1... */
2005-09-13 04:28:18 +00:00
2006-10-18 02:00:46 +00:00
/* Step 3 - local rotation - */
for ( i = 0 ; i < size ; i + + ) {
2005-09-13 18:35:45 +00:00
2005-09-13 04:28:18 +00:00
err = ompi_ddt_copy_content_same_ddt ( rdtype , ( int32_t ) rcount ,
2007-07-07 01:06:19 +00:00
( ( char * ) rbuf ) + ( ( ( rank - i + size ) % size ) * rcount * rext ) ,
tmpbuf + i * rcount * rext ) ;
if ( err < 0 ) { line = __LINE__ ; err = - 1 ; goto err_hndl ; }
2006-10-18 02:00:46 +00:00
}
2005-09-13 04:28:18 +00:00
2006-10-18 02:00:46 +00:00
/* Step 4 - clean up */
2007-07-07 01:06:19 +00:00
if ( tmpbuf ! = NULL ) free ( tmpbuf_free ) ;
2006-10-18 02:00:46 +00:00
if ( weallocated ) {
if ( displs ! = NULL ) free ( displs ) ;
if ( blen ! = NULL ) free ( blen ) ;
}
return OMPI_SUCCESS ;
2005-09-13 04:28:18 +00:00
err_hndl :
2007-07-07 01:06:19 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream ,
" %s:%4d \t Error occurred %d, rank %2d " , __FILE__ , line , err ,
rank ) ) ;
if ( tmpbuf ! = NULL ) free ( tmpbuf_free ) ;
2006-10-18 02:00:46 +00:00
if ( weallocated ) {
if ( displs ! = NULL ) free ( displs ) ;
if ( blen ! = NULL ) free ( blen ) ;
}
return err ;
2005-09-13 04:28:18 +00:00
}
2007-02-20 04:25:00 +00:00
/*
* alltoall_intra_linear_sync
*
* Function : Linear implementation of alltoall with limited number
* of outstanding requests .
* Accepts : Same as MPI_Alltoall ( ) , and the maximum number of
* outstanding requests ( actual number is 2 * max , since
* we count receive and send requests separately ) .
* Returns : MPI_SUCCESS or error code
*
* Description : Algorithm is the following :
* 1 ) post K irecvs , K < = N
* 2 ) post K isends , K < = N
* 3 ) while not done
* - wait for any request to complete
* - replace that request by the new one of the same type .
*/
int ompi_coll_tuned_alltoall_intra_linear_sync ( void * sbuf , int scount ,
struct ompi_datatype_t * sdtype ,
void * rbuf , int rcount ,
struct ompi_datatype_t * rdtype ,
struct ompi_communicator_t * comm ,
2007-08-19 03:37:49 +00:00
struct mca_coll_base_module_1_1_0_t * module ,
2007-02-20 04:25:00 +00:00
int max_outstanding_reqs )
{
int line , error ;
int ri , si ;
int rank ;
int size ;
int nreqs , nrreqs , nsreqs , total_reqs ;
char * psnd ;
char * prcv ;
ptrdiff_t slb , sext ;
ptrdiff_t rlb , rext ;
ompi_request_t * * reqs = NULL ;
/* Initialize. */
size = ompi_comm_size ( comm ) ;
rank = ompi_comm_rank ( comm ) ;
2007-07-07 01:06:19 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream ,
" ompi_coll_tuned_alltoall_intra_linear_sync rank %d " , rank ) ) ;
2007-02-20 04:25:00 +00:00
error = ompi_ddt_get_extent ( sdtype , & slb , & sext ) ;
if ( OMPI_SUCCESS ! = error ) {
return error ;
}
sext * = scount ;
error = ompi_ddt_get_extent ( rdtype , & rlb , & rext ) ;
if ( OMPI_SUCCESS ! = error ) {
return error ;
}
rext * = rcount ;
/* simple optimization */
psnd = ( ( char * ) sbuf ) + ( rank * sext ) ;
prcv = ( ( char * ) rbuf ) + ( rank * rext ) ;
error = ompi_ddt_sndrcv ( psnd , scount , sdtype , prcv , rcount , rdtype ) ;
if ( MPI_SUCCESS ! = error ) {
return error ;
}
/* If only one process, we're done. */
if ( 1 = = size ) {
return MPI_SUCCESS ;
}
/* Initiate send/recv to/from others. */
total_reqs = ( ( ( max_outstanding_reqs > ( size - 1 ) ) | |
( max_outstanding_reqs < = 0 ) ) ?
( size - 1 ) : ( max_outstanding_reqs ) ) ;
reqs = ( ompi_request_t * * ) malloc ( 2 * total_reqs *
sizeof ( ompi_request_t * ) ) ;
if ( NULL = = reqs ) { error = - 1 ; goto error_hndl ; }
prcv = ( char * ) rbuf ;
psnd = ( char * ) sbuf ;
/* Post first batch or ireceive and isend requests */
for ( nreqs = 0 , nrreqs = 0 , ri = ( rank + 1 ) % size ; nreqs < total_reqs ;
ri = ( ri + 1 ) % size , + + nreqs , + + nrreqs ) {
error =
MCA_PML_CALL ( irecv
( prcv + ( ri * rext ) , rcount , rdtype , ri ,
MCA_COLL_BASE_TAG_ALLTOALL , comm , & reqs [ nreqs ] ) ) ;
if ( MPI_SUCCESS ! = error ) { line = __LINE__ ; goto error_hndl ; }
}
for ( nsreqs = 0 , si = ( rank + size - 1 ) % size ; nreqs < 2 * total_reqs ;
si = ( si + size - 1 ) % size , + + nreqs , + + nsreqs ) {
error =
MCA_PML_CALL ( isend
( psnd + ( si * sext ) , scount , sdtype , si ,
MCA_COLL_BASE_TAG_ALLTOALL ,
MCA_PML_BASE_SEND_STANDARD , comm , & reqs [ nreqs ] ) ) ;
if ( MPI_SUCCESS ! = error ) { line = __LINE__ ; goto error_hndl ; }
}
/* Wait for requests to complete */
if ( nreqs = = 2 * ( size - 1 ) ) {
/* Optimization for the case when all requests have been posted */
error = ompi_request_wait_all ( nreqs , reqs , MPI_STATUSES_IGNORE ) ;
if ( MPI_SUCCESS ! = error ) { line = __LINE__ ; goto error_hndl ; }
} else {
/* As requests complete, replace them with corresponding requests:
- wait for any request to complete , mark the request as
MPI_REQUEST_NULL
- If it was a receive request , replace it with new irecv request
( if any )
- if it was a send request , replace it with new isend request ( if any )
*/
int ncreqs = 0 ;
while ( ncreqs < 2 * ( size - 1 ) ) {
int completed ;
error = ompi_request_wait_any ( 2 * total_reqs , reqs , & completed ,
MPI_STATUS_IGNORE ) ;
if ( MPI_SUCCESS ! = error ) { line = __LINE__ ; goto error_hndl ; }
reqs [ completed ] = MPI_REQUEST_NULL ;
ncreqs + + ;
if ( completed < total_reqs ) {
if ( nrreqs < ( size - 1 ) ) {
error =
MCA_PML_CALL ( irecv
( prcv + ( ri * rext ) , rcount , rdtype , ri ,
MCA_COLL_BASE_TAG_ALLTOALL , comm ,
& reqs [ completed ] ) ) ;
if ( MPI_SUCCESS ! = error ) { line = __LINE__ ; goto error_hndl ; }
+ + nrreqs ;
ri = ( ri + 1 ) % size ;
}
} else {
if ( nsreqs < ( size - 1 ) ) {
error = MCA_PML_CALL ( isend
( psnd + ( si * sext ) , scount , sdtype , si ,
MCA_COLL_BASE_TAG_ALLTOALL ,
MCA_PML_BASE_SEND_STANDARD , comm ,
& reqs [ completed ] ) ) ;
+ + nsreqs ;
si = ( si + size - 1 ) % size ;
}
}
}
}
/* Free the reqs */
free ( reqs ) ;
/* All done */
return MPI_SUCCESS ;
error_hndl :
2007-07-07 01:06:19 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream ,
" %s:%4d \t Error occurred %d, rank %2d " , __FILE__ , line , error ,
rank ) ) ;
2007-02-20 04:25:00 +00:00
if ( NULL ! = reqs ) free ( reqs ) ;
return error ;
}
2005-09-13 04:28:18 +00:00
2005-12-22 13:49:33 +00:00
int ompi_coll_tuned_alltoall_intra_two_procs ( void * sbuf , int scount ,
2006-10-18 02:00:46 +00:00
struct ompi_datatype_t * sdtype ,
void * rbuf , int rcount ,
struct ompi_datatype_t * rdtype ,
2007-08-19 03:37:49 +00:00
struct ompi_communicator_t * comm ,
struct mca_coll_base_module_1_1_0_t * module )
2005-09-13 04:28:18 +00:00
{
int line = - 1 , err = 0 ;
int rank ;
2007-07-07 01:06:19 +00:00
int remote ;
2005-09-13 04:28:18 +00:00
void * tmpsend , * tmprecv ;
2006-10-17 20:20:58 +00:00
ptrdiff_t sext , rext , lb ;
2005-09-13 04:28:18 +00:00
rank = ompi_comm_rank ( comm ) ;
2007-07-07 01:06:19 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream ,
" ompi_coll_tuned_alltoall_intra_two_procs rank %d " , rank ) ) ;
2005-09-13 04:28:18 +00:00
err = ompi_ddt_get_extent ( sdtype , & lb , & sext ) ;
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
err = ompi_ddt_get_extent ( rdtype , & lb , & rext ) ;
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
/* exchange data */
2007-07-07 01:06:19 +00:00
remote = rank ^ 1 ;
2005-09-13 04:28:18 +00:00
2007-07-07 01:06:19 +00:00
tmpsend = ( char * ) sbuf + remote * sext * scount ;
tmprecv = ( char * ) rbuf + remote * rext * rcount ;
2005-09-13 04:28:18 +00:00
/* send and receive */
2007-07-07 01:06:19 +00:00
err = ompi_coll_tuned_sendrecv ( tmpsend , scount , sdtype , remote ,
MCA_COLL_BASE_TAG_ALLTOALL ,
tmprecv , rcount , rdtype , remote ,
MCA_COLL_BASE_TAG_ALLTOALL ,
2006-10-18 02:00:46 +00:00
comm , MPI_STATUS_IGNORE , rank ) ;
2005-09-13 04:28:18 +00:00
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
/* ddt sendrecv your own data */
2007-07-07 01:06:19 +00:00
err = ompi_ddt_sndrcv ( ( char * ) sbuf + rank * sext * scount ,
( int32_t ) scount , sdtype ,
( char * ) rbuf + rank * rext * rcount ,
( int32_t ) rcount , rdtype ) ;
2005-09-13 04:28:18 +00:00
if ( err ! = MPI_SUCCESS ) { line = __LINE__ ; goto err_hndl ; }
/* done */
return MPI_SUCCESS ;
err_hndl :
2007-07-07 01:06:19 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream ,
" %s:%4d \t Error occurred %d, rank %2d " , __FILE__ , line , err ,
rank ) ) ;
2006-10-18 02:00:46 +00:00
return err ;
2005-09-13 04:28:18 +00:00
}
2005-10-26 23:51:56 +00:00
/*
* Linear functions are copied from the BASIC coll module
* they do not segment the message and are simple implementations
* but for some small number of nodes and / or small data sizes they
* are just as fast as tuned / tree based segmenting operations
* and as such may be selected by the decision functions
* These are copied into this module due to the way we select modules
* in V1 . i . e . in V2 we will handle this differently and so will not
* have to duplicate code .
* GEF Oct05 after asking Jeff .
*/
/* copied function (with appropriate renaming) starts here */
2005-12-22 13:49:33 +00:00
int ompi_coll_tuned_alltoall_intra_basic_linear ( void * sbuf , int scount ,
2006-10-18 02:00:46 +00:00
struct ompi_datatype_t * sdtype ,
void * rbuf , int rcount ,
struct ompi_datatype_t * rdtype ,
2007-08-19 03:37:49 +00:00
struct ompi_communicator_t * comm ,
struct mca_coll_base_module_1_1_0_t * module )
2005-09-13 04:28:18 +00:00
{
2005-10-26 23:51:56 +00:00
int i ;
int rank ;
int size ;
int err ;
int nreqs ;
char * psnd ;
char * prcv ;
MPI_Aint lb ;
MPI_Aint sndinc ;
MPI_Aint rcvinc ;
2005-09-13 04:28:18 +00:00
2005-10-26 23:51:56 +00:00
ompi_request_t * * req ;
ompi_request_t * * sreq ;
ompi_request_t * * rreq ;
2007-08-19 03:37:49 +00:00
mca_coll_tuned_module_t * tuned_module = ( mca_coll_tuned_module_t * ) module ;
mca_coll_tuned_comm_t * data = tuned_module - > tuned_data ;
2005-10-26 23:51:56 +00:00
/* Initialize. */
2005-09-13 04:28:18 +00:00
size = ompi_comm_size ( comm ) ;
rank = ompi_comm_rank ( comm ) ;
2007-07-07 01:06:19 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream ,
" ompi_coll_tuned_alltoall_intra_basic_linear rank %d " , rank ) ) ;
2005-09-13 04:28:18 +00:00
2005-10-26 23:51:56 +00:00
err = ompi_ddt_get_extent ( sdtype , & lb , & sndinc ) ;
if ( OMPI_SUCCESS ! = err ) {
return err ;
}
sndinc * = scount ;
2005-09-13 04:28:18 +00:00
2005-10-26 23:51:56 +00:00
err = ompi_ddt_get_extent ( rdtype , & lb , & rcvinc ) ;
if ( OMPI_SUCCESS ! = err ) {
return err ;
}
rcvinc * = rcount ;
2005-09-13 04:28:18 +00:00
2005-10-26 23:51:56 +00:00
/* simple optimization */
psnd = ( ( char * ) sbuf ) + ( rank * sndinc ) ;
prcv = ( ( char * ) rbuf ) + ( rank * rcvinc ) ;
err = ompi_ddt_sndrcv ( psnd , scount , sdtype , prcv , rcount , rdtype ) ;
if ( MPI_SUCCESS ! = err ) {
return err ;
}
/* If only one process, we're done. */
if ( 1 = = size ) {
return MPI_SUCCESS ;
}
/* Initiate all send/recv to/from others. */
2007-08-19 03:37:49 +00:00
req = rreq = data - > mcct_reqs ;
2005-10-26 23:51:56 +00:00
sreq = rreq + size - 1 ;
prcv = ( char * ) rbuf ;
psnd = ( char * ) sbuf ;
/* Post all receives first -- a simple optimization */
2007-01-26 21:59:31 +00:00
for ( nreqs = 0 , i = ( rank + 1 ) % size ; i ! = rank ;
i = ( i + 1 ) % size , + + rreq , + + nreqs ) {
2005-10-26 23:51:56 +00:00
err =
MCA_PML_CALL ( irecv_init
( prcv + ( i * rcvinc ) , rcount , rdtype , i ,
MCA_COLL_BASE_TAG_ALLTOALL , comm , rreq ) ) ;
if ( MPI_SUCCESS ! = err ) {
2005-12-22 13:49:33 +00:00
ompi_coll_tuned_free_reqs ( req , rreq - req ) ;
2005-10-26 23:51:56 +00:00
return err ;
}
}
2007-01-26 21:59:31 +00:00
/* Now post all sends in reverse order
- We would like to minimize the search time through message queue
when messages actually arrive in the order in which they were posted .
*/
for ( nreqs = 0 , i = ( rank + size - 1 ) % size ; i ! = rank ;
i = ( i + size - 1 ) % size , + + sreq , + + nreqs ) {
2005-10-26 23:51:56 +00:00
err =
MCA_PML_CALL ( isend_init
( psnd + ( i * sndinc ) , scount , sdtype , i ,
MCA_COLL_BASE_TAG_ALLTOALL ,
MCA_PML_BASE_SEND_STANDARD , comm , sreq ) ) ;
if ( MPI_SUCCESS ! = err ) {
2005-12-22 13:49:33 +00:00
ompi_coll_tuned_free_reqs ( req , sreq - req ) ;
2005-10-26 23:51:56 +00:00
return err ;
}
}
2006-11-08 16:54:03 +00:00
nreqs = ( size - 1 ) * 2 ;
2005-10-26 23:51:56 +00:00
/* Start your engines. This will never return an error. */
MCA_PML_CALL ( start ( nreqs , req ) ) ;
/* Wait for them all. If there's an error, note that we don't
* care what the error was - - just that there * was * an error . The
* PML will finish all requests , even if one or more of them fail .
* i . e . , by the end of this call , all the requests are free - able .
* So free them anyway - - even if there was an error , and return
* the error after we free everything . */
err = ompi_request_wait_all ( nreqs , req , MPI_STATUSES_IGNORE ) ;
/* Free the reqs */
2005-12-22 13:49:33 +00:00
ompi_coll_tuned_free_reqs ( req , nreqs ) ;
2005-10-26 23:51:56 +00:00
/* All done */
return err ;
2005-09-13 04:28:18 +00:00
}
2005-10-26 23:51:56 +00:00
/* copied function (with appropriate renaming) ends here */
2005-10-25 03:55:58 +00:00
/* The following are used by dynamic and forced rules */
2005-09-13 04:28:18 +00:00
2005-10-25 03:55:58 +00:00
/* publish details of each algorithm and if its forced/fixed/locked in */
/* as you add methods/algorithms you must update this and the query/map routines */
2005-09-13 04:28:18 +00:00
2006-04-19 23:42:06 +00:00
/* this routine is called by the component only */
/* this makes sure that the mca parameters are set to their initial values and perms */
/* module does not call this they call the forced_getvalues routine instead */
int ompi_coll_tuned_alltoall_intra_check_forced_init ( coll_tuned_force_algorithm_mca_param_indices_t * mca_param_indices )
2005-10-25 03:55:58 +00:00
{
2007-09-12 20:13:26 +00:00
int max_alg = 5 , requested_alg , max_requests ;
2006-04-19 23:42:06 +00:00
ompi_coll_tuned_forced_max_algorithms [ ALLTOALL ] = max_alg ;
2007-09-12 20:13:26 +00:00
mca_base_param_reg_int ( & mca_coll_tuned_component . super . collm_version ,
" alltoall_algorithm_count " ,
" Number of alltoall algorithms available " ,
false , true , max_alg , NULL ) ;
2006-10-18 02:00:46 +00:00
2006-11-10 19:54:09 +00:00
mca_param_indices - > algorithm_param_index
= mca_base_param_reg_int ( & mca_coll_tuned_component . super . collm_version ,
" alltoall_algorithm " ,
2007-02-20 04:25:00 +00:00
" Which alltoall algorithm is used. Can be locked down to choice of: 0 ignore, 1 basic linear, 2 pairwise, 3: modified bruck, 4: linear with sync, 5:two proc only. " ,
2006-11-10 19:54:09 +00:00
false , false , 0 , NULL ) ;
2006-11-10 21:47:07 +00:00
mca_base_param_lookup_int ( mca_param_indices - > algorithm_param_index , & ( requested_alg ) ) ;
if ( requested_alg > max_alg ) {
2006-11-10 19:54:09 +00:00
if ( 0 = = ompi_comm_rank ( MPI_COMM_WORLD ) ) {
opal_output ( 0 , " Alltoall algorithm #%d is not available (range [0..%d]). Switching back to ignore(0) \n " ,
2006-11-10 21:47:07 +00:00
requested_alg , max_alg ) ;
2006-11-10 19:54:09 +00:00
}
2006-11-10 21:47:07 +00:00
mca_base_param_set_int ( mca_param_indices - > algorithm_param_index , 0 ) ;
2006-11-10 19:54:09 +00:00
}
mca_param_indices - > segsize_param_index
= mca_base_param_reg_int ( & mca_coll_tuned_component . super . collm_version ,
" alltoall_algorithm_segmentsize " ,
" Segment size in bytes used by default for alltoall algorithms. Only has meaning if algorithm is forced and supports segmenting. 0 bytes means no segmentation. " ,
false , false , 0 , NULL ) ;
mca_param_indices - > tree_fanout_param_index
= mca_base_param_reg_int ( & mca_coll_tuned_component . super . collm_version ,
" alltoall_algorithm_tree_fanout " ,
" Fanout for n-tree used for alltoall algorithms. Only has meaning if algorithm is forced and supports n-tree topo based operation. " ,
false , false ,
ompi_coll_tuned_init_tree_fanout , /* get system wide default */
NULL ) ;
mca_param_indices - > chain_fanout_param_index
= mca_base_param_reg_int ( & mca_coll_tuned_component . super . collm_version ,
" alltoall_algorithm_chain_fanout " ,
" Fanout for chains used for alltoall algorithms. Only has meaning if algorithm is forced and supports chain topo based operation. " ,
false , false ,
ompi_coll_tuned_init_chain_fanout , /* get system wide default */
NULL ) ;
2006-10-18 02:00:46 +00:00
2007-02-20 04:25:00 +00:00
mca_param_indices - > max_requests_param_index
= mca_base_param_reg_int ( & mca_coll_tuned_component . super . collm_version ,
" alltoall_algorithm_max_requests " ,
" Maximum number of outstanding send or recv requests. Only has meaning for synchronized algorithms. " ,
false , false ,
ompi_coll_tuned_init_max_requests , /* get system wide default */
NULL ) ;
mca_base_param_lookup_int ( mca_param_indices - > max_requests_param_index , & ( max_requests ) ) ;
if ( max_requests < = 1 ) {
if ( 0 = = ompi_comm_rank ( MPI_COMM_WORLD ) ) {
opal_output ( 0 , " Maximum outstanding requests must be positive number greater than 1. Switching to system level default %d \n " ,
ompi_coll_tuned_init_max_requests ) ;
}
mca_base_param_set_int ( mca_param_indices - > max_requests_param_index ,
ompi_coll_tuned_init_max_requests ) ;
}
2006-10-18 02:00:46 +00:00
return ( MPI_SUCCESS ) ;
2005-10-25 03:55:58 +00:00
}
2005-12-22 13:49:33 +00:00
int ompi_coll_tuned_alltoall_intra_do_forced ( void * sbuf , int scount ,
2006-10-18 02:00:46 +00:00
struct ompi_datatype_t * sdtype ,
void * rbuf , int rcount ,
struct ompi_datatype_t * rdtype ,
2007-08-19 03:37:49 +00:00
struct ompi_communicator_t * comm ,
struct mca_coll_base_module_1_1_0_t * module )
2005-10-25 03:55:58 +00:00
{
2007-08-19 03:37:49 +00:00
mca_coll_tuned_module_t * tuned_module = ( mca_coll_tuned_module_t * ) module ;
mca_coll_tuned_comm_t * data = tuned_module - > tuned_data ;
2006-04-19 23:42:06 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream , " coll:tuned:alltoall_intra_do_forced selected algorithm %d " ,
2007-08-19 03:37:49 +00:00
data - > user_forced [ ALLTOALL ] . algorithm ) ) ;
switch ( data - > user_forced [ ALLTOALL ] . algorithm ) {
case ( 0 ) : return ompi_coll_tuned_alltoall_intra_dec_fixed ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module ) ;
case ( 1 ) : return ompi_coll_tuned_alltoall_intra_basic_linear ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module ) ;
case ( 2 ) : return ompi_coll_tuned_alltoall_intra_pairwise ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module ) ;
case ( 3 ) : return ompi_coll_tuned_alltoall_intra_bruck ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module ) ;
case ( 4 ) : return ompi_coll_tuned_alltoall_intra_linear_sync ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module , data - > user_forced [ ALLTOALL ] . max_requests ) ;
case ( 5 ) : return ompi_coll_tuned_alltoall_intra_two_procs ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module ) ;
2005-10-25 03:55:58 +00:00
default :
2005-12-22 13:49:33 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream , " coll:tuned:alltoall_intra_do_forced attempt to select algorithm %d when only 0-%d is valid? " ,
2007-08-19 03:37:49 +00:00
data - > user_forced [ ALLTOALL ] . algorithm , ompi_coll_tuned_forced_max_algorithms [ ALLTOALL ] ) ) ;
2005-10-25 03:55:58 +00:00
return ( MPI_ERR_ARG ) ;
} /* switch */
}
2005-09-13 04:28:18 +00:00
2005-11-11 04:49:29 +00:00
2005-12-22 13:49:33 +00:00
int ompi_coll_tuned_alltoall_intra_do_this ( void * sbuf , int scount ,
2006-10-18 02:00:46 +00:00
struct ompi_datatype_t * sdtype ,
void * rbuf , int rcount ,
struct ompi_datatype_t * rdtype ,
struct ompi_communicator_t * comm ,
2007-08-19 03:37:49 +00:00
struct mca_coll_base_module_1_1_0_t * module ,
2007-02-20 04:25:00 +00:00
int algorithm , int faninout , int segsize ,
int max_requests )
2005-11-11 04:49:29 +00:00
{
2005-12-22 13:49:33 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream , " coll:tuned:alltoall_intra_do_this selected algorithm %d topo faninout %d segsize %d " ,
2006-10-18 02:00:46 +00:00
algorithm , faninout , segsize ) ) ;
2005-11-11 04:49:29 +00:00
2006-10-18 02:00:46 +00:00
switch ( algorithm ) {
2007-08-19 03:37:49 +00:00
case ( 0 ) : return ompi_coll_tuned_alltoall_intra_dec_fixed ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module ) ;
case ( 1 ) : return ompi_coll_tuned_alltoall_intra_basic_linear ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module ) ;
case ( 2 ) : return ompi_coll_tuned_alltoall_intra_pairwise ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module ) ;
case ( 3 ) : return ompi_coll_tuned_alltoall_intra_bruck ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module ) ;
case ( 4 ) : return ompi_coll_tuned_alltoall_intra_linear_sync ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module , max_requests ) ;
case ( 5 ) : return ompi_coll_tuned_alltoall_intra_two_procs ( sbuf , scount , sdtype , rbuf , rcount , rdtype , comm , module ) ;
2005-11-11 04:49:29 +00:00
default :
2005-12-22 13:49:33 +00:00
OPAL_OUTPUT ( ( ompi_coll_tuned_stream , " coll:tuned:alltoall_intra_do_this attempt to select algorithm %d when only 0-%d is valid? " ,
2006-10-18 02:00:46 +00:00
algorithm , ompi_coll_tuned_forced_max_algorithms [ ALLTOALL ] ) ) ;
2005-11-11 04:49:29 +00:00
return ( MPI_ERR_ARG ) ;
} /* switch */
}