/* * 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. * 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 #include "include/constants.h" #include "opal/threads/condition.h" #include "opal/util/output.h" #include "util/proc_info.h" #include "orte/dps/dps.h" #include "mca/oob/oob.h" #include "mca/oob/base/base.h" #include "mca/ns/ns.h" #include "mca/gpr/gpr.h" #include "mca/errmgr/errmgr.h" #include "mca/soh/base/base.h" #include "runtime/runtime.h" /** * A "broadcast-like" function over the specified set of peers. * @param root The process acting as the root of the broadcast. * @param peers The list of processes receiving the broadcast (excluding root). * @param buffer The data to broadcast - only significant at root. * @param cbfunc Callback function on receipt of data - not significant at root. * * Note that the callback function is provided so that the data can be * received and interpreted by the application prior to the broadcast * continuing to forward data along the distribution tree. */ struct mca_oob_xcast_t { opal_object_t super; opal_mutex_t mutex; opal_condition_t cond; size_t counter; }; typedef struct mca_oob_xcast_t mca_oob_xcast_t; static void mca_oob_xcast_construct(mca_oob_xcast_t* xcast) { OBJ_CONSTRUCT(&xcast->mutex, opal_mutex_t); OBJ_CONSTRUCT(&xcast->cond, opal_condition_t); } static void mca_oob_xcast_destruct(mca_oob_xcast_t* xcast) { OBJ_DESTRUCT(&xcast->mutex); OBJ_DESTRUCT(&xcast->cond); } static OBJ_CLASS_INSTANCE( mca_oob_xcast_t, opal_object_t, mca_oob_xcast_construct, mca_oob_xcast_destruct); static void mca_oob_xcast_cb(int status, orte_process_name_t* peer, orte_buffer_t* buffer, int tag, void* cbdata) { mca_oob_xcast_t* xcast = (mca_oob_xcast_t*)cbdata; OPAL_THREAD_LOCK(&xcast->mutex); if(--xcast->counter == 0) { opal_condition_signal(&xcast->cond); } OPAL_THREAD_UNLOCK(&xcast->mutex); } int mca_oob_xcast( orte_process_name_t* root, orte_process_name_t* peers, size_t num_peers, orte_buffer_t* buffer, orte_gpr_trigger_cb_fn_t cbfunc) { size_t i; int rc; int tag = MCA_OOB_TAG_XCAST; int cmpval; int status; orte_proc_state_t state; /* check to see if I am the root process name */ cmpval = orte_ns.compare(ORTE_NS_CMP_ALL, root, orte_process_info.my_name); if(NULL != root && 0 == cmpval) { mca_oob_xcast_t *xcast = OBJ_NEW(mca_oob_xcast_t); xcast->counter = num_peers; for(i=0; imutex); while(xcast->counter > 0) { opal_condition_wait(&xcast->cond, &xcast->mutex); } OPAL_THREAD_UNLOCK(&xcast->mutex); OBJ_RELEASE(xcast); } else { orte_buffer_t rbuf; orte_gpr_notify_message_t *msg; OBJ_CONSTRUCT(&rbuf, orte_buffer_t); rc = mca_oob_recv_packed(MCA_OOB_NAME_ANY, &rbuf, tag); if(rc < 0) { OBJ_DESTRUCT(&rbuf); return rc; } if (cbfunc != NULL) { msg = OBJ_NEW(orte_gpr_notify_message_t); if (NULL == msg) { ORTE_ERROR_LOG(ORTE_ERR_OUT_OF_RESOURCE); return ORTE_ERR_OUT_OF_RESOURCE; } i=1; if (ORTE_SUCCESS != (rc = orte_dps.unpack(&rbuf, &msg, &i, ORTE_GPR_NOTIFY_MSG))) { ORTE_ERROR_LOG(rc); OBJ_RELEASE(msg); return rc; } cbfunc(msg); OBJ_RELEASE(msg); } OBJ_DESTRUCT(&rbuf); } return ORTE_SUCCESS; }