2004-03-06 22:40:26 +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.
|
|
|
|
* 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.
|
2008-03-05 12:22:34 +00:00
|
|
|
* Copyright (c) 2008 Cisco Systems, Inc. All rights reserved.
|
2004-11-22 01:38:40 +00:00
|
|
|
* $COPYRIGHT$
|
|
|
|
*
|
|
|
|
* Additional copyrights may follow
|
|
|
|
*
|
2004-03-06 22:40:26 +00:00
|
|
|
* $HEADER$
|
|
|
|
*/
|
|
|
|
|
2004-10-20 01:03:09 +00:00
|
|
|
#include "ompi_config.h"
|
2006-02-12 01:33:29 +00:00
|
|
|
#include "ompi/mca/topo/base/base.h"
|
|
|
|
#include "ompi/communicator/communicator.h"
|
2004-03-06 22:40:26 +00:00
|
|
|
|
2004-03-08 06:48:24 +00:00
|
|
|
/*
|
|
|
|
* function - partitions a communicator into subgroups which
|
|
|
|
* form lower-dimensional cartesian subgrids
|
|
|
|
*
|
|
|
|
* @param comm communicator with cartesian structure (handle)
|
|
|
|
* @param remain_dims the 'i'th entry of 'remain_dims' specifies whether
|
|
|
|
* the 'i'th dimension is kept in the subgrid (true)
|
|
|
|
* or is dropped (false) (logical vector)
|
|
|
|
* @param new_comm communicator containing the subgrid that includes the
|
|
|
|
* calling process (handle)
|
|
|
|
*
|
|
|
|
* @retval MPI_SUCCESS
|
|
|
|
* @retval MPI_ERR_TOPOLOGY
|
|
|
|
* @retval MPI_ERR_COMM
|
|
|
|
*/
|
2009-03-12 22:38:14 +00:00
|
|
|
int mca_topo_base_cart_sub (ompi_communicator_t* comm,
|
2004-03-08 06:48:24 +00:00
|
|
|
int *remain_dims,
|
2009-03-12 22:38:14 +00:00
|
|
|
ompi_communicator_t** new_comm){
|
2004-07-20 22:32:45 +00:00
|
|
|
|
|
|
|
struct ompi_communicator_t *temp_comm;
|
2004-03-08 06:48:24 +00:00
|
|
|
int errcode;
|
|
|
|
int colour;
|
|
|
|
int key;
|
|
|
|
int colfactor;
|
|
|
|
int keyfactor;
|
|
|
|
int rank;
|
|
|
|
int ndim;
|
|
|
|
int dim;
|
|
|
|
int i;
|
2012-09-06 14:16:29 +00:00
|
|
|
int *d, *dold;
|
2004-03-08 06:48:24 +00:00
|
|
|
int *c;
|
|
|
|
int *r;
|
2012-09-06 14:16:29 +00:00
|
|
|
int *p, *pold;
|
2004-03-08 06:48:24 +00:00
|
|
|
|
2004-07-20 22:32:45 +00:00
|
|
|
*new_comm = MPI_COMM_NULL;
|
|
|
|
|
2004-03-08 06:48:24 +00:00
|
|
|
/*
|
|
|
|
* Compute colour and key used in splitting the communicator.
|
|
|
|
*/
|
|
|
|
colour = key = 0;
|
|
|
|
colfactor = keyfactor = 1;
|
|
|
|
ndim = 0;
|
|
|
|
|
2004-07-20 22:32:45 +00:00
|
|
|
i = comm->c_topo_comm->mtc_ndims_or_nnodes - 1;
|
|
|
|
d = comm->c_topo_comm->mtc_dims_or_index + i;
|
2004-04-16 20:54:48 +00:00
|
|
|
c = comm->c_topo_comm->mtc_coords + i;
|
2004-03-08 06:48:24 +00:00
|
|
|
r = remain_dims + i;
|
|
|
|
|
|
|
|
for (; i >= 0; --i, --d, --c, --r) {
|
2008-03-05 12:22:34 +00:00
|
|
|
dim = *d;
|
2004-03-08 06:48:24 +00:00
|
|
|
if (*r == 0) {
|
|
|
|
colour += colfactor * (*c);
|
|
|
|
colfactor *= dim;
|
|
|
|
} else {
|
|
|
|
++ndim;
|
|
|
|
key += keyfactor * (*c);
|
|
|
|
keyfactor *= dim;
|
|
|
|
}
|
|
|
|
}
|
2008-08-31 19:31:10 +00:00
|
|
|
/* Special case: if all of remain_dims were false, we need to make
|
|
|
|
a 0-dimension cartesian communicator with just ourselves in it
|
|
|
|
(you can't have a communicator unless you're in it). */
|
|
|
|
if (0 == ndim) {
|
|
|
|
colour = ompi_comm_rank (comm);
|
2004-03-08 06:48:24 +00:00
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Split the communicator.
|
|
|
|
*/
|
2004-10-07 19:14:27 +00:00
|
|
|
errcode = ompi_comm_split (comm, colour, key, &temp_comm, true);
|
2004-03-08 06:48:24 +00:00
|
|
|
if (errcode != MPI_SUCCESS) {
|
|
|
|
return errcode;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Fill the communicator with topology information.
|
|
|
|
*/
|
2004-07-20 22:32:45 +00:00
|
|
|
if (temp_comm != MPI_COMM_NULL) {
|
|
|
|
|
|
|
|
temp_comm->c_topo_comm->mtc_ndims_or_nnodes = ndim;
|
2012-09-06 14:16:29 +00:00
|
|
|
|
2008-08-31 19:31:10 +00:00
|
|
|
if (ndim >= 1) {
|
2012-09-06 14:16:29 +00:00
|
|
|
/* Copy the dimensions */
|
|
|
|
d = temp_comm->c_topo_comm->mtc_dims_or_index;
|
|
|
|
dold = comm->c_topo_comm->mtc_dims_or_index;
|
|
|
|
r = remain_dims;
|
|
|
|
for (i = 0; i < comm->c_topo_comm->mtc_ndims_or_nnodes;
|
|
|
|
++i, ++dold, ++r) {
|
|
|
|
if (*r) {
|
|
|
|
*d++ = *dold;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy the periods */
|
|
|
|
p = temp_comm->c_topo_comm->mtc_periods_or_edges;
|
|
|
|
pold = comm->c_topo_comm->mtc_periods_or_edges;
|
2008-08-31 19:31:10 +00:00
|
|
|
r = remain_dims;
|
|
|
|
for (i = 0; i < comm->c_topo_comm->mtc_ndims_or_nnodes;
|
2012-09-06 14:16:29 +00:00
|
|
|
++i, ++pold, ++r) {
|
2008-08-31 19:31:10 +00:00
|
|
|
if (*r) {
|
2012-09-06 14:16:29 +00:00
|
|
|
*p++ = *pold;
|
2008-08-31 19:31:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Compute the caller's coordinates, if ndims>0
|
|
|
|
*/
|
|
|
|
rank = ompi_comm_rank (temp_comm);
|
|
|
|
if (MPI_SUCCESS != errcode) {
|
|
|
|
OBJ_RELEASE(temp_comm);
|
|
|
|
return errcode;
|
|
|
|
}
|
|
|
|
errcode = temp_comm->c_topo->topo_cart_coords (temp_comm, rank,
|
|
|
|
ndim, temp_comm->c_topo_comm->mtc_coords);
|
|
|
|
if (MPI_SUCCESS != errcode) {
|
|
|
|
OBJ_RELEASE(temp_comm);
|
|
|
|
return errcode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-03-08 06:48:24 +00:00
|
|
|
|
2004-07-20 22:32:45 +00:00
|
|
|
*new_comm = temp_comm;
|
|
|
|
return MPI_SUCCESS;
|
2004-03-08 06:48:24 +00:00
|
|
|
}
|