2005-05-24 02:06:50 +04:00
|
|
|
/*
|
2005-11-05 22:57:48 +03:00
|
|
|
* Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
|
|
|
|
* University Research and Technology
|
|
|
|
* Corporation. All rights reserved.
|
2006-03-16 01:53:41 +03:00
|
|
|
* Copyright (c) 2004-2006 The University of Tennessee and The University
|
2005-11-05 22:57:48 +03:00
|
|
|
* of Tennessee Research Foundation. All rights
|
|
|
|
* reserved.
|
2005-05-24 02:06:50 +04: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"
|
2006-02-12 04:33:29 +03:00
|
|
|
#include "ompi/constants.h"
|
|
|
|
#include "ompi/mca/pml/pml.h"
|
|
|
|
#include "ompi/mca/btl/btl.h"
|
|
|
|
#include "orte/mca/errmgr/errmgr.h"
|
|
|
|
#include "ompi/mca/mpool/mpool.h"
|
2005-05-24 02:06:50 +04:00
|
|
|
#include "pml_ob1.h"
|
|
|
|
#include "pml_ob1_hdr.h"
|
|
|
|
#include "pml_ob1_sendreq.h"
|
2005-06-10 00:16:33 +04:00
|
|
|
#include "pml_ob1_rdmafrag.h"
|
2005-05-24 02:06:50 +04:00
|
|
|
#include "pml_ob1_recvreq.h"
|
2006-02-12 04:33:29 +03:00
|
|
|
#include "ompi/mca/bml/base/base.h"
|
2006-02-26 03:45:54 +03:00
|
|
|
#include "ompi/datatype/dt_arch.h"
|
2005-05-24 02:06:50 +04:00
|
|
|
|
2006-10-26 17:21:47 +04:00
|
|
|
|
|
|
|
void mca_pml_ob1_send_request_process_pending(mca_bml_base_btl_t *bml_btl)
|
2006-07-20 18:44:35 +04:00
|
|
|
{
|
|
|
|
int i, s = opal_list_get_size(&mca_pml_ob1.send_pending);
|
|
|
|
|
|
|
|
/* advance pending requests */
|
|
|
|
for(i = 0; i < s; i++) {
|
|
|
|
mca_pml_ob1_send_pending_t pending_type;
|
|
|
|
mca_pml_ob1_send_request_t* sendreq;
|
2006-10-26 17:21:47 +04:00
|
|
|
mca_bml_base_btl_t *send_dst;
|
2006-07-20 18:44:35 +04:00
|
|
|
OPAL_THREAD_LOCK(&mca_pml_ob1.lock);
|
|
|
|
sendreq = (mca_pml_ob1_send_request_t*)
|
|
|
|
opal_list_remove_first(&mca_pml_ob1.send_pending);
|
|
|
|
OPAL_THREAD_UNLOCK(&mca_pml_ob1.lock);
|
|
|
|
if(NULL == sendreq)
|
|
|
|
break;
|
|
|
|
pending_type = sendreq->req_pending;
|
|
|
|
sendreq->req_pending = MCA_PML_OB1_SEND_PENDING_NONE;
|
|
|
|
switch(pending_type) {
|
|
|
|
case MCA_PML_OB1_SEND_PENDING_SCHEDULE:
|
|
|
|
if(mca_pml_ob1_send_request_schedule_exclusive(sendreq) ==
|
|
|
|
OMPI_ERR_OUT_OF_RESOURCE) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MCA_PML_OB1_SEND_PENDING_START:
|
2006-10-26 17:21:47 +04:00
|
|
|
send_dst = mca_bml_base_btl_array_find(
|
|
|
|
&sendreq->req_endpoint->btl_eager, bml_btl->btl);
|
|
|
|
if(NULL == send_dst ||
|
|
|
|
mca_pml_ob1_send_request_start_btl(sendreq, send_dst) ==
|
2006-07-20 18:44:35 +04:00
|
|
|
OMPI_ERR_OUT_OF_RESOURCE) {
|
2006-10-26 17:21:47 +04:00
|
|
|
/* if dst of this sendreq cannot be reached through the
|
|
|
|
* endpoint or no resources put request back on the list */
|
|
|
|
OPAL_THREAD_LOCK(&mca_pml_ob1.lock);
|
|
|
|
sendreq->req_pending = MCA_PML_OB1_SEND_PENDING_START;
|
|
|
|
if(NULL == send_dst) {
|
|
|
|
opal_list_append(&mca_pml_ob1.send_pending,
|
|
|
|
(opal_list_item_t*)sendreq);
|
|
|
|
} else {
|
|
|
|
/* prepend to the pending list to minimaze reordering */
|
|
|
|
opal_list_prepend(&mca_pml_ob1.send_pending,
|
|
|
|
(opal_list_item_t*)sendreq);
|
|
|
|
}
|
|
|
|
OPAL_THREAD_UNLOCK(&mca_pml_ob1.lock);
|
|
|
|
/* if no destination try next request otherwise give up,
|
|
|
|
* no more resources on this btl */
|
|
|
|
if(send_dst != NULL)
|
|
|
|
return;
|
2006-07-20 18:44:35 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
opal_output(0, "[%s:%d] wrong send request type\n",
|
|
|
|
__FILE__, __LINE__);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-03-16 01:53:41 +03:00
|
|
|
/*
|
|
|
|
* The free call mark the final stage in a request life-cycle. Starting from this
|
|
|
|
* point the request is completed at both PML and user level, and can be used
|
|
|
|
* for others p2p communications. Therefore, in the case of the OB1 PML it should
|
|
|
|
* be added to the free request list.
|
|
|
|
*/
|
2006-03-24 07:21:30 +03:00
|
|
|
static int mca_pml_ob1_send_request_free(struct ompi_request_t** request)
|
2005-05-24 02:06:50 +04:00
|
|
|
{
|
2006-03-16 01:53:41 +03:00
|
|
|
mca_pml_ob1_send_request_t* sendreq = *(mca_pml_ob1_send_request_t**)request;
|
|
|
|
|
|
|
|
assert( false == sendreq->req_send.req_base.req_free_called );
|
|
|
|
|
2006-03-02 03:39:07 +03:00
|
|
|
OPAL_THREAD_LOCK(&ompi_request_lock);
|
2006-03-16 01:53:41 +03:00
|
|
|
sendreq->req_send.req_base.req_free_called = true;
|
2006-03-31 21:09:09 +04:00
|
|
|
|
|
|
|
PERUSE_TRACE_COMM_EVENT( PERUSE_COMM_REQ_NOTIFY,
|
|
|
|
&(sendreq->req_send.req_base), PERUSE_SEND );
|
|
|
|
|
2006-09-29 03:54:38 +04:00
|
|
|
if( true == sendreq->req_send.req_base.req_pml_complete ) {
|
|
|
|
MCA_PML_OB1_SEND_REQUEST_RETURN( sendreq );
|
|
|
|
}
|
|
|
|
|
2006-03-02 03:39:07 +03:00
|
|
|
OPAL_THREAD_UNLOCK(&ompi_request_lock);
|
2005-05-24 02:06:50 +04:00
|
|
|
|
2006-02-14 12:09:05 +03:00
|
|
|
*request = MPI_REQUEST_NULL;
|
2005-05-24 02:06:50 +04:00
|
|
|
return OMPI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mca_pml_ob1_send_request_cancel(struct ompi_request_t* request, int complete)
|
|
|
|
{
|
|
|
|
/* we dont cancel send requests by now */
|
|
|
|
return OMPI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mca_pml_ob1_send_request_construct(mca_pml_ob1_send_request_t* req)
|
|
|
|
{
|
|
|
|
req->req_send.req_base.req_type = MCA_PML_REQUEST_SEND;
|
|
|
|
req->req_send.req_base.req_ompi.req_free = mca_pml_ob1_send_request_free;
|
|
|
|
req->req_send.req_base.req_ompi.req_cancel = mca_pml_ob1_send_request_cancel;
|
2006-03-16 01:53:41 +03:00
|
|
|
req->req_rdma_cnt = 0;
|
2005-05-24 02:06:50 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
OBJ_CLASS_INSTANCE(
|
|
|
|
mca_pml_ob1_send_request_t,
|
|
|
|
mca_pml_base_send_request_t,
|
|
|
|
mca_pml_ob1_send_request_construct,
|
2006-09-29 03:57:49 +04:00
|
|
|
NULL);
|
2005-05-24 02:06:50 +04:00
|
|
|
|
|
|
|
/**
|
2006-03-16 01:53:41 +03:00
|
|
|
* Completion of a short message - nothing left to schedule. Note that this
|
|
|
|
* function is only called for 0 sized messages.
|
2005-05-24 02:06:50 +04:00
|
|
|
*/
|
|
|
|
|
2005-09-14 21:08:08 +04:00
|
|
|
void mca_pml_ob1_match_completion_cache(
|
2005-08-12 06:41:14 +04:00
|
|
|
struct mca_btl_base_module_t* btl,
|
2005-06-30 09:50:55 +04:00
|
|
|
struct mca_btl_base_endpoint_t* ep,
|
|
|
|
struct mca_btl_base_descriptor_t* descriptor,
|
2005-05-24 02:06:50 +04:00
|
|
|
int status)
|
|
|
|
{
|
|
|
|
mca_pml_ob1_send_request_t* sendreq = (mca_pml_ob1_send_request_t*)descriptor->des_cbdata;
|
2005-08-12 06:41:14 +04:00
|
|
|
mca_bml_base_btl_t* bml_btl = (mca_bml_base_btl_t*) descriptor->des_context;
|
2005-05-24 02:06:50 +04:00
|
|
|
|
2006-07-06 03:39:13 +04:00
|
|
|
if( sendreq->req_send.req_bytes_packed > 0 ) {
|
|
|
|
PERUSE_TRACE_COMM_EVENT( PERUSE_COMM_REQ_XFER_BEGIN,
|
|
|
|
&(sendreq->req_send.req_base), PERUSE_SEND );
|
|
|
|
}
|
2006-03-31 21:09:09 +04:00
|
|
|
|
2005-06-10 00:16:33 +04:00
|
|
|
/* check completion status */
|
|
|
|
if(OMPI_SUCCESS != status) {
|
|
|
|
/* TSW - FIX */
|
2005-07-04 03:31:27 +04:00
|
|
|
opal_output(0, "%s:%d FATAL", __FILE__, __LINE__);
|
2005-06-10 00:16:33 +04:00
|
|
|
orte_errmgr.abort();
|
|
|
|
}
|
|
|
|
|
2005-06-09 00:37:19 +04:00
|
|
|
/* attempt to cache the descriptor */
|
2005-08-12 23:29:05 +04:00
|
|
|
MCA_BML_BASE_BTL_DES_RETURN( bml_btl, descriptor );
|
2005-06-02 23:51:51 +04:00
|
|
|
|
2005-06-09 00:37:19 +04:00
|
|
|
/* signal request completion */
|
2005-09-15 22:47:59 +04:00
|
|
|
MCA_PML_OB1_SEND_REQUEST_PML_COMPLETE(sendreq);
|
2006-07-20 18:44:35 +04:00
|
|
|
|
|
|
|
/* check for pending requests */
|
|
|
|
MCA_PML_OB1_PROGRESS_PENDING(bml_btl);
|
2005-06-09 00:37:19 +04:00
|
|
|
}
|
|
|
|
|
2005-09-14 21:08:08 +04:00
|
|
|
/**
|
|
|
|
* Completion of a short message - nothing left to schedule.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void mca_pml_ob1_match_completion_free(
|
|
|
|
struct mca_btl_base_module_t* btl,
|
|
|
|
struct mca_btl_base_endpoint_t* ep,
|
|
|
|
struct mca_btl_base_descriptor_t* descriptor,
|
|
|
|
int status)
|
|
|
|
{
|
|
|
|
mca_pml_ob1_send_request_t* sendreq = (mca_pml_ob1_send_request_t*)descriptor->des_cbdata;
|
|
|
|
mca_bml_base_btl_t* bml_btl = (mca_bml_base_btl_t*) descriptor->des_context;
|
|
|
|
|
2006-07-06 03:39:13 +04:00
|
|
|
if( sendreq->req_send.req_bytes_packed > 0 ) {
|
|
|
|
PERUSE_TRACE_COMM_EVENT( PERUSE_COMM_REQ_XFER_BEGIN,
|
|
|
|
&(sendreq->req_send.req_base), PERUSE_SEND );
|
|
|
|
}
|
2006-03-31 21:09:09 +04:00
|
|
|
|
2005-09-14 21:08:08 +04:00
|
|
|
/* check completion status */
|
|
|
|
if(OMPI_SUCCESS != status) {
|
|
|
|
/* TSW - FIX */
|
|
|
|
opal_output(0, "%s:%d FATAL", __FILE__, __LINE__);
|
|
|
|
orte_errmgr.abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* free the descriptor */
|
|
|
|
mca_bml_base_free( bml_btl, descriptor );
|
|
|
|
|
|
|
|
/* signal request completion */
|
2005-09-15 22:47:59 +04:00
|
|
|
MCA_PML_OB1_SEND_REQUEST_PML_COMPLETE(sendreq);
|
2006-07-20 18:44:35 +04:00
|
|
|
|
|
|
|
/* check for pending requests */
|
|
|
|
MCA_PML_OB1_PROGRESS_PENDING(bml_btl);
|
2005-09-14 21:08:08 +04:00
|
|
|
}
|
|
|
|
|
2005-07-18 22:54:25 +04:00
|
|
|
/*
|
|
|
|
* Completion of the first fragment of a long message that
|
|
|
|
* requires an acknowledgement
|
2005-06-09 00:37:19 +04:00
|
|
|
*/
|
2005-07-18 22:54:25 +04:00
|
|
|
static void mca_pml_ob1_rndv_completion(
|
2005-06-30 09:50:55 +04:00
|
|
|
mca_btl_base_module_t* btl,
|
|
|
|
struct mca_btl_base_endpoint_t* ep,
|
|
|
|
struct mca_btl_base_descriptor_t* descriptor,
|
2005-06-09 00:37:19 +04:00
|
|
|
int status)
|
|
|
|
{
|
|
|
|
mca_pml_ob1_send_request_t* sendreq = (mca_pml_ob1_send_request_t*)descriptor->des_cbdata;
|
2005-08-12 06:41:14 +04:00
|
|
|
mca_bml_base_btl_t* bml_btl = (mca_bml_base_btl_t*) descriptor->des_context;
|
2006-02-08 09:03:54 +03:00
|
|
|
|
2006-07-06 03:39:13 +04:00
|
|
|
if( sendreq->req_send.req_bytes_packed > 0 ) {
|
|
|
|
PERUSE_TRACE_COMM_EVENT( PERUSE_COMM_REQ_XFER_BEGIN,
|
|
|
|
&(sendreq->req_send.req_base), PERUSE_SEND );
|
|
|
|
}
|
2006-03-31 21:09:09 +04:00
|
|
|
|
2005-06-10 00:16:33 +04:00
|
|
|
/* check completion status */
|
|
|
|
if(OMPI_SUCCESS != status) {
|
|
|
|
/* TSW - FIX */
|
2005-07-04 03:31:27 +04:00
|
|
|
opal_output(0, "%s:%d FATAL", __FILE__, __LINE__);
|
2005-06-10 00:16:33 +04:00
|
|
|
orte_errmgr.abort();
|
|
|
|
}
|
2005-06-09 00:37:19 +04:00
|
|
|
|
2006-02-08 09:03:54 +03:00
|
|
|
/* count bytes of user data actually delivered. As the rndv completion only
|
|
|
|
* happens in one thread, the increase of the req_bytes_delivered does not
|
|
|
|
* have to be atomic.
|
|
|
|
*/
|
|
|
|
MCA_PML_OB1_COMPUTE_SEGMENT_LENGTH( descriptor->des_src,
|
|
|
|
descriptor->des_src_cnt,
|
|
|
|
sizeof(mca_pml_ob1_rendezvous_hdr_t),
|
|
|
|
sendreq->req_bytes_delivered );
|
2005-06-09 00:37:19 +04:00
|
|
|
|
2005-07-18 22:54:25 +04:00
|
|
|
/* return the descriptor */
|
2005-08-12 06:41:14 +04:00
|
|
|
mca_bml_base_free(bml_btl, descriptor);
|
2005-07-18 22:54:25 +04:00
|
|
|
|
|
|
|
/* advance the request */
|
|
|
|
MCA_PML_OB1_SEND_REQUEST_ADVANCE(sendreq);
|
2005-07-19 01:22:55 +04:00
|
|
|
|
|
|
|
/* check for pending requests */
|
2006-07-20 18:44:35 +04:00
|
|
|
MCA_PML_OB1_PROGRESS_PENDING(bml_btl);
|
2005-07-18 22:54:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-08-17 22:23:38 +04:00
|
|
|
/**
|
|
|
|
* Completion of a get request.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void mca_pml_ob1_rget_completion(
|
|
|
|
mca_btl_base_module_t* btl,
|
|
|
|
struct mca_btl_base_endpoint_t* ep,
|
|
|
|
struct mca_btl_base_descriptor_t* des,
|
|
|
|
int status)
|
|
|
|
{
|
|
|
|
mca_pml_ob1_send_request_t* sendreq = (mca_pml_ob1_send_request_t*)des->des_cbdata;
|
2006-07-20 18:44:35 +04:00
|
|
|
mca_bml_base_btl_t* bml_btl = (mca_bml_base_btl_t*)des->des_context;
|
2006-02-08 09:03:54 +03:00
|
|
|
size_t req_bytes_delivered = 0;
|
2005-08-17 22:23:38 +04:00
|
|
|
|
|
|
|
/* count bytes of user data actually delivered and check for request completion */
|
2006-02-08 09:03:54 +03:00
|
|
|
MCA_PML_OB1_COMPUTE_SEGMENT_LENGTH( des->des_src, des->des_src_cnt,
|
|
|
|
0, req_bytes_delivered );
|
|
|
|
if( OPAL_THREAD_ADD_SIZE_T( &sendreq->req_bytes_delivered, req_bytes_delivered )
|
|
|
|
== sendreq->req_send.req_bytes_packed ) {
|
2005-09-15 22:47:59 +04:00
|
|
|
MCA_PML_OB1_SEND_REQUEST_PML_COMPLETE(sendreq);
|
2005-08-17 22:23:38 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* release resources */
|
|
|
|
btl->btl_free(btl,des);
|
2006-07-20 18:44:35 +04:00
|
|
|
MCA_PML_OB1_PROGRESS_PENDING(bml_btl);
|
2005-08-17 22:23:38 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Completion of a control message - return resources.
|
|
|
|
*/
|
|
|
|
|
2006-03-31 21:09:09 +04:00
|
|
|
static void mca_pml_ob1_send_ctl_completion(
|
2005-08-17 22:23:38 +04:00
|
|
|
mca_btl_base_module_t* btl,
|
|
|
|
struct mca_btl_base_endpoint_t* ep,
|
|
|
|
struct mca_btl_base_descriptor_t* descriptor,
|
|
|
|
int status)
|
|
|
|
{
|
|
|
|
/* return the descriptor */
|
|
|
|
mca_bml_base_btl_t* bml_btl = (mca_bml_base_btl_t*) descriptor->des_context;
|
|
|
|
mca_bml_base_free(bml_btl, descriptor);
|
2006-07-20 18:44:35 +04:00
|
|
|
|
|
|
|
/* check for pending requests */
|
|
|
|
MCA_PML_OB1_PROGRESS_PENDING(bml_btl);
|
2005-08-17 22:23:38 +04:00
|
|
|
}
|
|
|
|
|
2005-07-18 22:54:25 +04:00
|
|
|
/**
|
|
|
|
* Completion of additional fragments of a large message - may need
|
|
|
|
* to schedule additional fragments.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void mca_pml_ob1_frag_completion(
|
|
|
|
mca_btl_base_module_t* btl,
|
|
|
|
struct mca_btl_base_endpoint_t* ep,
|
|
|
|
struct mca_btl_base_descriptor_t* descriptor,
|
|
|
|
int status)
|
|
|
|
{
|
|
|
|
mca_pml_ob1_send_request_t* sendreq = (mca_pml_ob1_send_request_t*)descriptor->des_cbdata;
|
2005-08-12 06:41:14 +04:00
|
|
|
mca_bml_base_btl_t* bml_btl = (mca_bml_base_btl_t*) descriptor->des_context;
|
2006-02-08 09:03:54 +03:00
|
|
|
size_t req_bytes_delivered = 0;
|
2005-07-18 22:54:25 +04:00
|
|
|
|
|
|
|
/* check completion status */
|
|
|
|
if(OMPI_SUCCESS != status) {
|
|
|
|
/* TSW - FIX */
|
|
|
|
opal_output(0, "%s:%d FATAL", __FILE__, __LINE__);
|
|
|
|
orte_errmgr.abort();
|
2005-06-09 00:37:19 +04:00
|
|
|
}
|
|
|
|
|
2005-08-17 22:23:38 +04:00
|
|
|
/* count bytes of user data actually delivered */
|
2006-02-08 09:03:54 +03:00
|
|
|
MCA_PML_OB1_COMPUTE_SEGMENT_LENGTH( descriptor->des_src,
|
|
|
|
descriptor->des_src_cnt,
|
|
|
|
sizeof(mca_pml_ob1_frag_hdr_t),
|
|
|
|
req_bytes_delivered );
|
2006-07-20 18:44:35 +04:00
|
|
|
|
|
|
|
/* return the descriptor */
|
|
|
|
mca_bml_base_free(bml_btl, descriptor);
|
|
|
|
|
2006-02-08 09:03:54 +03:00
|
|
|
req_bytes_delivered = OPAL_THREAD_ADD_SIZE_T( &sendreq->req_bytes_delivered,
|
|
|
|
req_bytes_delivered );
|
2005-08-02 21:36:01 +04:00
|
|
|
if (OPAL_THREAD_ADD_SIZE_T(&sendreq->req_pipeline_depth,-1) == 0 &&
|
2006-02-08 09:03:54 +03:00
|
|
|
req_bytes_delivered == sendreq->req_send.req_bytes_packed) {
|
2006-03-16 01:53:41 +03:00
|
|
|
MCA_PML_OB1_SEND_REQUEST_PML_COMPLETE(sendreq);
|
2005-07-18 22:54:25 +04:00
|
|
|
} else {
|
|
|
|
mca_pml_ob1_send_request_schedule(sendreq);
|
|
|
|
}
|
2005-06-10 00:16:33 +04:00
|
|
|
|
2005-07-19 01:22:55 +04:00
|
|
|
/* check for pending requests */
|
2006-07-20 18:44:35 +04:00
|
|
|
MCA_PML_OB1_PROGRESS_PENDING(bml_btl);
|
2005-05-24 02:06:50 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-09-13 02:28:23 +04:00
|
|
|
|
2005-09-15 22:47:59 +04:00
|
|
|
/**
|
|
|
|
* Buffer the entire message and mark as complete.
|
|
|
|
*/
|
|
|
|
|
|
|
|
int mca_pml_ob1_send_request_start_buffered(
|
|
|
|
mca_pml_ob1_send_request_t* sendreq,
|
|
|
|
mca_bml_base_btl_t* bml_btl,
|
|
|
|
size_t size)
|
|
|
|
{
|
|
|
|
mca_btl_base_descriptor_t* descriptor;
|
|
|
|
mca_btl_base_segment_t* segment;
|
|
|
|
mca_pml_ob1_hdr_t* hdr;
|
|
|
|
struct iovec iov;
|
|
|
|
unsigned int iov_count;
|
|
|
|
size_t max_data;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* allocate descriptor */
|
|
|
|
mca_bml_base_alloc(bml_btl, &descriptor, sizeof(mca_pml_ob1_rendezvous_hdr_t) + size);
|
|
|
|
if(NULL == descriptor) {
|
|
|
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
|
|
|
}
|
|
|
|
segment = descriptor->des_src;
|
|
|
|
|
|
|
|
/* pack the data into the BTL supplied buffer */
|
2006-08-24 20:38:08 +04:00
|
|
|
iov.iov_base = (IOVBASE_TYPE*)((unsigned char*)segment->seg_addr.pval +
|
|
|
|
sizeof(mca_pml_ob1_rendezvous_hdr_t));
|
2005-09-15 22:47:59 +04:00
|
|
|
iov.iov_len = size;
|
|
|
|
iov_count = 1;
|
|
|
|
max_data = size;
|
|
|
|
if((rc = ompi_convertor_pack(
|
|
|
|
&sendreq->req_send.req_convertor,
|
|
|
|
&iov,
|
|
|
|
&iov_count,
|
2006-10-27 03:11:26 +04:00
|
|
|
&max_data)) < 0) {
|
2005-09-15 22:47:59 +04:00
|
|
|
mca_bml_base_free(bml_btl, descriptor);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* build rendezvous header */
|
|
|
|
hdr = (mca_pml_ob1_hdr_t*)segment->seg_addr.pval;
|
|
|
|
hdr->hdr_common.hdr_flags = 0;
|
|
|
|
hdr->hdr_common.hdr_type = MCA_PML_OB1_HDR_TYPE_RNDV;
|
|
|
|
hdr->hdr_match.hdr_ctx = sendreq->req_send.req_base.req_comm->c_contextid;
|
|
|
|
hdr->hdr_match.hdr_src = sendreq->req_send.req_base.req_comm->c_my_rank;
|
|
|
|
hdr->hdr_match.hdr_tag = sendreq->req_send.req_base.req_tag;
|
2006-08-24 20:38:08 +04:00
|
|
|
hdr->hdr_match.hdr_seq = (uint16_t)sendreq->req_send.req_base.req_sequence;
|
2005-09-15 22:47:59 +04:00
|
|
|
hdr->hdr_rndv.hdr_msg_length = sendreq->req_send.req_bytes_packed;
|
|
|
|
hdr->hdr_rndv.hdr_src_req.pval = sendreq;
|
|
|
|
|
2006-02-28 22:54:46 +03:00
|
|
|
#if OMPI_ENABLE_HETEROGENEOUS_SUPPORT
|
2006-02-26 03:45:54 +03:00
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
2006-02-28 22:54:46 +03:00
|
|
|
#else
|
2006-02-26 03:45:54 +03:00
|
|
|
/* if we are little endian and the remote side is big endian,
|
|
|
|
we're responsible for making sure the data is in network byte
|
|
|
|
order */
|
|
|
|
if (sendreq->req_send.req_base.req_proc->proc_arch & OMPI_ARCH_ISBIGENDIAN) {
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
|
|
|
MCA_PML_OB1_RNDV_HDR_HTON(hdr->hdr_rndv);
|
|
|
|
}
|
2006-02-28 22:54:46 +03:00
|
|
|
#endif
|
2006-02-26 03:45:54 +03:00
|
|
|
#endif
|
|
|
|
|
2005-09-15 22:47:59 +04:00
|
|
|
/* update lengths */
|
|
|
|
segment->seg_len = sizeof(mca_pml_ob1_rendezvous_hdr_t) + max_data;
|
|
|
|
sendreq->req_send_offset = max_data;
|
|
|
|
|
|
|
|
descriptor->des_cbfunc = mca_pml_ob1_rndv_completion;
|
|
|
|
descriptor->des_flags |= MCA_BTL_DES_FLAGS_PRIORITY;
|
|
|
|
descriptor->des_cbdata = sendreq;
|
|
|
|
|
|
|
|
/* buffer the remainder of the message */
|
|
|
|
rc = mca_pml_base_bsend_request_alloc((ompi_request_t*)sendreq);
|
|
|
|
if(OMPI_SUCCESS != rc) {
|
|
|
|
mca_bml_base_free(bml_btl, descriptor);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2006-08-24 20:38:08 +04:00
|
|
|
iov.iov_base = (IOVBASE_TYPE*)(((unsigned char*)sendreq->req_send.req_addr) + sendreq->req_send_offset);
|
2005-09-15 22:47:59 +04:00
|
|
|
iov.iov_len = max_data = sendreq->req_send.req_bytes_packed - sendreq->req_send_offset;
|
|
|
|
|
2006-04-01 11:42:43 +04:00
|
|
|
if((rc = ompi_convertor_pack( &sendreq->req_send.req_convertor,
|
|
|
|
&iov,
|
|
|
|
&iov_count,
|
2006-10-27 03:11:26 +04:00
|
|
|
&max_data)) < 0) {
|
2006-04-01 11:42:43 +04:00
|
|
|
mca_bml_base_free(bml_btl, descriptor);
|
|
|
|
return rc;
|
2005-09-15 22:47:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* re-init convertor for packed data */
|
2006-04-01 11:42:43 +04:00
|
|
|
ompi_convertor_prepare_for_send( &sendreq->req_send.req_convertor,
|
2006-07-06 21:58:25 +04:00
|
|
|
MPI_BYTE,
|
|
|
|
sendreq->req_send.req_bytes_packed,
|
2006-04-01 11:42:43 +04:00
|
|
|
sendreq->req_send.req_addr );
|
2005-09-15 22:47:59 +04:00
|
|
|
/* request is complete at mpi level */
|
|
|
|
OPAL_THREAD_LOCK(&ompi_request_lock);
|
|
|
|
MCA_PML_OB1_SEND_REQUEST_MPI_COMPLETE(sendreq);
|
|
|
|
OPAL_THREAD_UNLOCK(&ompi_request_lock);
|
|
|
|
|
|
|
|
/* send */
|
|
|
|
rc = mca_bml_base_send(bml_btl, descriptor, MCA_BTL_TAG_PML);
|
|
|
|
if(OMPI_SUCCESS != rc) {
|
|
|
|
mca_bml_base_free(bml_btl, descriptor );
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* BTL requires "specially" allocated memory. Request a segment that
|
2006-03-16 01:53:41 +03:00
|
|
|
* is used for initial hdr and any eager data. This is used only from
|
|
|
|
* the _START macro.
|
2005-09-15 22:47:59 +04:00
|
|
|
*/
|
|
|
|
|
|
|
|
int mca_pml_ob1_send_request_start_copy(
|
|
|
|
mca_pml_ob1_send_request_t* sendreq,
|
|
|
|
mca_bml_base_btl_t* bml_btl,
|
|
|
|
size_t size)
|
|
|
|
{
|
|
|
|
mca_btl_base_descriptor_t* descriptor;
|
|
|
|
mca_btl_base_segment_t* segment;
|
|
|
|
mca_pml_ob1_hdr_t* hdr;
|
|
|
|
struct iovec iov;
|
|
|
|
unsigned int iov_count;
|
|
|
|
size_t max_data;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* allocate descriptor */
|
2006-07-20 18:44:35 +04:00
|
|
|
if(0==size)
|
|
|
|
MCA_PML_OB1_DES_ALLOC(bml_btl, descriptor,
|
|
|
|
sizeof(mca_pml_ob1_match_hdr_t));
|
|
|
|
else
|
|
|
|
mca_bml_base_alloc(bml_btl, &descriptor,
|
|
|
|
sizeof(mca_pml_ob1_match_hdr_t) + size);
|
2005-09-15 22:47:59 +04:00
|
|
|
if(NULL == descriptor) {
|
|
|
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
|
|
|
}
|
|
|
|
segment = descriptor->des_src;
|
|
|
|
|
|
|
|
max_data = size;
|
2006-02-08 20:39:33 +03:00
|
|
|
if(size > 0) {
|
|
|
|
/* pack the data into the supplied buffer */
|
2006-08-24 20:38:08 +04:00
|
|
|
iov.iov_base = (IOVBASE_TYPE*)((unsigned char*)segment->seg_addr.pval + sizeof(mca_pml_ob1_match_hdr_t));
|
2006-02-08 20:39:33 +03:00
|
|
|
iov.iov_len = size;
|
|
|
|
iov_count = 1;
|
|
|
|
if((rc = ompi_convertor_pack(
|
|
|
|
&sendreq->req_send.req_convertor,
|
|
|
|
&iov,
|
|
|
|
&iov_count,
|
2006-10-27 03:11:26 +04:00
|
|
|
&max_data)) < 0) {
|
2005-09-15 22:47:59 +04:00
|
|
|
mca_bml_base_free(bml_btl, descriptor);
|
|
|
|
return rc;
|
2006-02-08 20:39:33 +03:00
|
|
|
}
|
2005-09-15 22:47:59 +04:00
|
|
|
}
|
2006-02-08 20:39:33 +03:00
|
|
|
|
2005-09-15 22:47:59 +04:00
|
|
|
/* build match header */
|
|
|
|
hdr = (mca_pml_ob1_hdr_t*)segment->seg_addr.pval;
|
|
|
|
hdr->hdr_common.hdr_flags = 0;
|
|
|
|
hdr->hdr_common.hdr_type = MCA_PML_OB1_HDR_TYPE_MATCH;
|
|
|
|
hdr->hdr_match.hdr_ctx = sendreq->req_send.req_base.req_comm->c_contextid;
|
|
|
|
hdr->hdr_match.hdr_src = sendreq->req_send.req_base.req_comm->c_my_rank;
|
|
|
|
hdr->hdr_match.hdr_tag = sendreq->req_send.req_base.req_tag;
|
2006-08-24 20:38:08 +04:00
|
|
|
hdr->hdr_match.hdr_seq = (uint16_t)sendreq->req_send.req_base.req_sequence;
|
2005-09-15 22:47:59 +04:00
|
|
|
|
2006-02-28 22:54:46 +03:00
|
|
|
#if OMPI_ENABLE_HETEROGENEOUS_SUPPORT
|
2006-02-26 03:45:54 +03:00
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
2006-02-28 22:54:46 +03:00
|
|
|
#else
|
2006-02-26 03:45:54 +03:00
|
|
|
/* if we are little endian and the remote side is big endian,
|
|
|
|
we're responsible for making sure the data is in network byte
|
|
|
|
order */
|
|
|
|
if (sendreq->req_send.req_base.req_proc->proc_arch & OMPI_ARCH_ISBIGENDIAN) {
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
|
|
|
MCA_PML_OB1_MATCH_HDR_HTON(hdr->hdr_match);
|
|
|
|
}
|
2006-02-28 22:54:46 +03:00
|
|
|
#endif
|
2006-02-26 03:45:54 +03:00
|
|
|
#endif
|
|
|
|
|
2005-09-15 22:47:59 +04:00
|
|
|
/* update lengths */
|
|
|
|
segment->seg_len = sizeof(mca_pml_ob1_match_hdr_t) + max_data;
|
|
|
|
sendreq->req_send_offset = max_data;
|
|
|
|
sendreq->req_rdma_offset = max_data;
|
|
|
|
|
|
|
|
/* short message */
|
2006-07-20 18:44:35 +04:00
|
|
|
descriptor->des_cbfunc = size?mca_pml_ob1_match_completion_free:
|
|
|
|
mca_pml_ob1_match_completion_cache;
|
2005-09-15 22:47:59 +04:00
|
|
|
descriptor->des_flags |= MCA_BTL_DES_FLAGS_PRIORITY;
|
|
|
|
descriptor->des_cbdata = sendreq;
|
|
|
|
|
|
|
|
/* signal request completion */
|
|
|
|
OPAL_THREAD_LOCK(&ompi_request_lock);
|
|
|
|
MCA_PML_OB1_SEND_REQUEST_MPI_COMPLETE(sendreq);
|
|
|
|
OPAL_THREAD_UNLOCK(&ompi_request_lock);
|
|
|
|
|
|
|
|
/* send */
|
|
|
|
rc = mca_bml_base_send(bml_btl, descriptor, MCA_BTL_TAG_PML);
|
|
|
|
if(OMPI_SUCCESS != rc) {
|
2006-07-20 18:44:35 +04:00
|
|
|
if(0==size)
|
|
|
|
MCA_BML_BASE_BTL_DES_RETURN(bml_btl, descriptor);
|
|
|
|
else
|
|
|
|
mca_bml_base_free(bml_btl, descriptor );
|
2005-09-15 22:47:59 +04:00
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* BTL can send directly from user buffer so allow the BTL
|
2006-03-16 01:53:41 +03:00
|
|
|
* to prepare the segment list. Start sending a small message.
|
2005-09-15 22:47:59 +04:00
|
|
|
*/
|
|
|
|
|
|
|
|
int mca_pml_ob1_send_request_start_prepare(
|
|
|
|
mca_pml_ob1_send_request_t* sendreq,
|
|
|
|
mca_bml_base_btl_t* bml_btl,
|
|
|
|
size_t size)
|
|
|
|
{
|
|
|
|
mca_btl_base_descriptor_t* descriptor;
|
|
|
|
mca_btl_base_segment_t* segment;
|
|
|
|
mca_pml_ob1_hdr_t* hdr;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* prepare descriptor */
|
|
|
|
mca_bml_base_prepare_src(
|
2006-03-16 01:53:41 +03:00
|
|
|
bml_btl,
|
2005-09-15 22:47:59 +04:00
|
|
|
NULL,
|
|
|
|
&sendreq->req_send.req_convertor,
|
|
|
|
sizeof(mca_pml_ob1_match_hdr_t),
|
|
|
|
&size,
|
|
|
|
&descriptor);
|
|
|
|
if(NULL == descriptor) {
|
|
|
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
2006-03-16 01:53:41 +03:00
|
|
|
}
|
2005-09-15 22:47:59 +04:00
|
|
|
segment = descriptor->des_src;
|
|
|
|
|
|
|
|
/* build match header */
|
|
|
|
hdr = (mca_pml_ob1_hdr_t*)segment->seg_addr.pval;
|
|
|
|
hdr->hdr_common.hdr_flags = 0;
|
|
|
|
hdr->hdr_common.hdr_type = MCA_PML_OB1_HDR_TYPE_MATCH;
|
|
|
|
hdr->hdr_match.hdr_ctx = sendreq->req_send.req_base.req_comm->c_contextid;
|
|
|
|
hdr->hdr_match.hdr_src = sendreq->req_send.req_base.req_comm->c_my_rank;
|
|
|
|
hdr->hdr_match.hdr_tag = sendreq->req_send.req_base.req_tag;
|
2006-08-24 20:38:08 +04:00
|
|
|
hdr->hdr_match.hdr_seq = (uint16_t)sendreq->req_send.req_base.req_sequence;
|
2005-09-15 22:47:59 +04:00
|
|
|
|
2006-02-28 22:54:46 +03:00
|
|
|
#if OMPI_ENABLE_HETEROGENEOUS_SUPPORT
|
2006-02-26 03:45:54 +03:00
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
2006-02-28 22:54:46 +03:00
|
|
|
#else
|
2006-02-26 03:45:54 +03:00
|
|
|
/* if we are little endian and the remote side is big endian,
|
|
|
|
we're responsible for making sure the data is in network byte
|
|
|
|
order */
|
|
|
|
if (sendreq->req_send.req_base.req_proc->proc_arch & OMPI_ARCH_ISBIGENDIAN) {
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
|
|
|
MCA_PML_OB1_MATCH_HDR_HTON(hdr->hdr_match);
|
|
|
|
}
|
2006-02-28 22:54:46 +03:00
|
|
|
#endif
|
2006-02-26 03:45:54 +03:00
|
|
|
#endif
|
|
|
|
|
2005-09-15 22:47:59 +04:00
|
|
|
/* short message */
|
|
|
|
descriptor->des_cbfunc = mca_pml_ob1_match_completion_free;
|
2006-03-16 01:53:41 +03:00
|
|
|
|
2005-09-15 22:47:59 +04:00
|
|
|
/* update lengths */
|
|
|
|
sendreq->req_send_offset = size;
|
|
|
|
sendreq->req_rdma_offset = size;
|
|
|
|
|
|
|
|
descriptor->des_flags |= MCA_BTL_DES_FLAGS_PRIORITY;
|
|
|
|
descriptor->des_cbdata = sendreq;
|
|
|
|
|
|
|
|
/* send */
|
|
|
|
rc = mca_bml_base_send(bml_btl, descriptor, MCA_BTL_TAG_PML);
|
|
|
|
if(OMPI_SUCCESS != rc) {
|
|
|
|
mca_bml_base_free(bml_btl, descriptor );
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-09-13 02:28:23 +04:00
|
|
|
/**
|
|
|
|
* We have contigous data that is registered - schedule across
|
|
|
|
* available nics.
|
|
|
|
*/
|
|
|
|
|
2005-09-15 22:47:59 +04:00
|
|
|
int mca_pml_ob1_send_request_start_rdma(
|
|
|
|
mca_pml_ob1_send_request_t* sendreq,
|
|
|
|
mca_bml_base_btl_t* bml_btl,
|
|
|
|
size_t size)
|
2005-09-13 02:28:23 +04:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* FIX - to get the basics working - schedule on the
|
|
|
|
* first BTL only
|
|
|
|
*/
|
|
|
|
|
|
|
|
mca_mpool_base_registration_t* reg = sendreq->req_rdma[0].btl_reg;
|
|
|
|
mca_btl_base_descriptor_t* src;
|
|
|
|
mca_btl_base_descriptor_t* des;
|
|
|
|
mca_btl_base_segment_t* segment;
|
|
|
|
mca_pml_ob1_hdr_t* hdr;
|
|
|
|
size_t i;
|
|
|
|
int rc;
|
|
|
|
|
2005-11-03 23:52:56 +03:00
|
|
|
|
2005-09-15 22:47:59 +04:00
|
|
|
bml_btl = sendreq->req_rdma[0].bml_btl;
|
2005-11-03 23:52:56 +03:00
|
|
|
if(sendreq->req_rdma_cnt == 1 &&
|
|
|
|
bml_btl->btl_flags & MCA_BTL_FLAGS_GET) {
|
2006-07-20 18:44:35 +04:00
|
|
|
size_t old_position = sendreq->req_send.req_convertor.bConverted;
|
2005-09-13 02:28:23 +04:00
|
|
|
|
|
|
|
/* prepare source descriptor/segment(s) */
|
2006-02-13 20:53:30 +03:00
|
|
|
mca_bml_base_prepare_src(
|
2005-09-13 02:28:23 +04:00
|
|
|
bml_btl,
|
|
|
|
reg,
|
|
|
|
&sendreq->req_send.req_convertor,
|
|
|
|
0,
|
|
|
|
&size,
|
|
|
|
&src);
|
|
|
|
if(NULL == src) {
|
2006-07-20 18:44:35 +04:00
|
|
|
ompi_convertor_set_position(&sendreq->req_send.req_convertor,
|
|
|
|
&old_position);
|
2005-09-13 02:28:23 +04:00
|
|
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
|
|
|
}
|
|
|
|
src->des_cbfunc = mca_pml_ob1_rget_completion;
|
|
|
|
src->des_cbdata = sendreq;
|
|
|
|
|
|
|
|
/* allocate space for get hdr + segment list */
|
|
|
|
mca_bml_base_alloc(bml_btl, &des,
|
|
|
|
sizeof(mca_pml_ob1_rget_hdr_t) + (sizeof(mca_btl_base_segment_t)*(src->des_src_cnt-1)));
|
|
|
|
if(NULL == des) {
|
2006-07-20 18:44:35 +04:00
|
|
|
ompi_convertor_set_position(&sendreq->req_send.req_convertor,
|
|
|
|
&old_position);
|
2005-09-13 02:28:23 +04:00
|
|
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
|
|
|
}
|
|
|
|
segment = des->des_src;
|
|
|
|
|
|
|
|
/* build match header */
|
|
|
|
hdr = (mca_pml_ob1_hdr_t*)segment->seg_addr.pval;
|
2005-11-11 18:33:25 +03:00
|
|
|
hdr->hdr_common.hdr_flags = MCA_PML_OB1_HDR_FLAGS_CONTIG|MCA_PML_OB1_HDR_FLAGS_PIN;
|
2005-09-13 02:28:23 +04:00
|
|
|
hdr->hdr_common.hdr_type = MCA_PML_OB1_HDR_TYPE_RGET;
|
|
|
|
hdr->hdr_match.hdr_ctx = sendreq->req_send.req_base.req_comm->c_contextid;
|
|
|
|
hdr->hdr_match.hdr_src = sendreq->req_send.req_base.req_comm->c_my_rank;
|
|
|
|
hdr->hdr_match.hdr_tag = sendreq->req_send.req_base.req_tag;
|
2006-08-24 20:38:08 +04:00
|
|
|
hdr->hdr_match.hdr_seq = (uint16_t)sendreq->req_send.req_base.req_sequence;
|
2005-09-13 02:28:23 +04:00
|
|
|
hdr->hdr_rndv.hdr_msg_length = sendreq->req_send.req_bytes_packed;
|
|
|
|
hdr->hdr_rndv.hdr_src_req.pval = sendreq;
|
|
|
|
hdr->hdr_rget.hdr_des.pval = src;
|
|
|
|
hdr->hdr_rget.hdr_seg_cnt = src->des_src_cnt;
|
2006-02-26 03:45:54 +03:00
|
|
|
|
2006-02-28 22:54:46 +03:00
|
|
|
#if OMPI_ENABLE_HETEROGENEOUS_SUPPORT
|
2006-02-26 03:45:54 +03:00
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
2006-02-28 22:54:46 +03:00
|
|
|
#else
|
2006-02-26 03:45:54 +03:00
|
|
|
/* if we are little endian and the remote side is big endian,
|
|
|
|
we're responsible for making sure the data is in network byte
|
|
|
|
order */
|
2006-02-28 22:54:46 +03:00
|
|
|
/* RDMA is currently disabled by bml if arch doesn't
|
|
|
|
match, so this shouldn't be needed. here to make sure
|
|
|
|
we remember if we ever change the bml. */
|
|
|
|
assert(0 == (sendreq->req_send.req_base.req_proc->proc_arch &
|
|
|
|
OMPI_ARCH_ISBIGENDIAN));
|
|
|
|
#endif
|
2006-02-26 03:45:54 +03:00
|
|
|
#endif
|
|
|
|
|
2005-09-13 02:28:23 +04:00
|
|
|
for(i=0; i<src->des_src_cnt; i++)
|
|
|
|
hdr->hdr_rget.hdr_segs[i] = src->des_src[i];
|
2006-07-20 18:44:35 +04:00
|
|
|
|
2006-03-31 21:09:09 +04:00
|
|
|
des->des_cbfunc = mca_pml_ob1_send_ctl_completion;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Well, it's a get so we will not know when the peer get the data anyway.
|
|
|
|
* If we generate the PERUSE event here, at least we will know when do we
|
|
|
|
* sent the GET message ...
|
|
|
|
*/
|
2006-07-06 03:39:13 +04:00
|
|
|
if( sendreq->req_send.req_bytes_packed > 0 ) {
|
|
|
|
PERUSE_TRACE_COMM_EVENT( PERUSE_COMM_REQ_XFER_BEGIN,
|
|
|
|
&(sendreq->req_send.req_base), PERUSE_SEND );
|
|
|
|
}
|
2005-09-13 02:28:23 +04:00
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
/* allocate a rendezvous header - dont eager send any data
|
|
|
|
* receiver will schedule rdma put(s) of the entire message
|
|
|
|
*/
|
|
|
|
|
|
|
|
mca_bml_base_alloc(bml_btl, &des, sizeof(mca_pml_ob1_rendezvous_hdr_t));
|
|
|
|
if(NULL == des) {
|
|
|
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
|
|
|
}
|
|
|
|
segment = des->des_src;
|
|
|
|
|
|
|
|
/* build hdr */
|
|
|
|
hdr = (mca_pml_ob1_hdr_t*)segment->seg_addr.pval;
|
2005-11-11 18:33:25 +03:00
|
|
|
hdr->hdr_common.hdr_flags = MCA_PML_OB1_HDR_FLAGS_CONTIG|MCA_PML_OB1_HDR_FLAGS_PIN;
|
2005-09-13 02:28:23 +04:00
|
|
|
hdr->hdr_common.hdr_type = MCA_PML_OB1_HDR_TYPE_RNDV;
|
|
|
|
hdr->hdr_match.hdr_ctx = sendreq->req_send.req_base.req_comm->c_contextid;
|
|
|
|
hdr->hdr_match.hdr_src = sendreq->req_send.req_base.req_comm->c_my_rank;
|
|
|
|
hdr->hdr_match.hdr_tag = sendreq->req_send.req_base.req_tag;
|
2006-08-24 20:38:08 +04:00
|
|
|
hdr->hdr_match.hdr_seq = (uint16_t)sendreq->req_send.req_base.req_sequence;
|
2005-09-13 02:28:23 +04:00
|
|
|
hdr->hdr_rndv.hdr_msg_length = sendreq->req_send.req_bytes_packed;
|
|
|
|
hdr->hdr_rndv.hdr_src_req.pval = sendreq;
|
|
|
|
|
2006-02-28 22:54:46 +03:00
|
|
|
#if OMPI_ENABLE_HETEROGENEOUS_SUPPORT
|
2006-02-26 03:45:54 +03:00
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
2006-02-28 22:54:46 +03:00
|
|
|
#else
|
2006-02-26 03:45:54 +03:00
|
|
|
/* if we are little endian and the remote side is big endian,
|
|
|
|
we're responsible for making sure the data is in network byte
|
|
|
|
order */
|
|
|
|
if (sendreq->req_send.req_base.req_proc->proc_arch & OMPI_ARCH_ISBIGENDIAN) {
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
|
|
|
MCA_PML_OB1_RNDV_HDR_HTON(hdr->hdr_rndv);
|
|
|
|
}
|
2006-02-28 22:54:46 +03:00
|
|
|
#endif
|
2006-02-26 03:45:54 +03:00
|
|
|
#endif
|
|
|
|
|
2005-09-13 02:28:23 +04:00
|
|
|
/* update lengths with number of bytes actually packed */
|
|
|
|
segment->seg_len = sizeof(mca_pml_ob1_rendezvous_hdr_t);
|
2006-06-20 18:12:32 +04:00
|
|
|
sendreq->req_rdma_offset = 0;
|
2005-09-13 02:28:23 +04:00
|
|
|
|
|
|
|
/* first fragment of a long message */
|
|
|
|
des->des_cbfunc = mca_pml_ob1_rndv_completion;
|
|
|
|
}
|
|
|
|
|
|
|
|
des->des_flags |= MCA_BTL_DES_FLAGS_PRIORITY;
|
|
|
|
des->des_cbdata = sendreq;
|
|
|
|
|
|
|
|
/* send */
|
|
|
|
rc = mca_bml_base_send(bml_btl, des, MCA_BTL_TAG_PML);
|
|
|
|
if(OMPI_SUCCESS != rc) {
|
|
|
|
mca_bml_base_free(bml_btl, des);
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Rendezvous is required. Not doing rdma so eager send up to
|
|
|
|
* the btls eager limit.
|
|
|
|
*/
|
|
|
|
|
2005-09-15 22:47:59 +04:00
|
|
|
int mca_pml_ob1_send_request_start_rndv(
|
2005-09-13 02:28:23 +04:00
|
|
|
mca_pml_ob1_send_request_t* sendreq,
|
|
|
|
mca_bml_base_btl_t* bml_btl,
|
2005-11-11 18:33:25 +03:00
|
|
|
size_t size,
|
|
|
|
int flags)
|
2005-09-13 02:28:23 +04:00
|
|
|
{
|
|
|
|
mca_btl_base_descriptor_t* des;
|
|
|
|
mca_btl_base_segment_t* segment;
|
|
|
|
mca_pml_ob1_hdr_t* hdr;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* prepare descriptor */
|
2005-09-14 06:17:04 +04:00
|
|
|
if(size == 0) {
|
|
|
|
mca_bml_base_alloc(
|
|
|
|
bml_btl,
|
|
|
|
&des,
|
|
|
|
sizeof(mca_pml_ob1_rendezvous_hdr_t)
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
mca_bml_base_prepare_src(
|
|
|
|
bml_btl,
|
|
|
|
NULL,
|
|
|
|
&sendreq->req_send.req_convertor,
|
|
|
|
sizeof(mca_pml_ob1_rendezvous_hdr_t),
|
|
|
|
&size,
|
|
|
|
&des);
|
|
|
|
}
|
2005-09-13 02:28:23 +04:00
|
|
|
|
|
|
|
if(NULL == des) {
|
|
|
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
|
|
|
}
|
|
|
|
segment = des->des_src;
|
|
|
|
|
|
|
|
/* build hdr */
|
|
|
|
hdr = (mca_pml_ob1_hdr_t*)segment->seg_addr.pval;
|
2005-11-11 18:33:25 +03:00
|
|
|
hdr->hdr_common.hdr_flags = flags;
|
2005-09-13 02:28:23 +04:00
|
|
|
hdr->hdr_common.hdr_type = MCA_PML_OB1_HDR_TYPE_RNDV;
|
|
|
|
hdr->hdr_match.hdr_ctx = sendreq->req_send.req_base.req_comm->c_contextid;
|
|
|
|
hdr->hdr_match.hdr_src = sendreq->req_send.req_base.req_comm->c_my_rank;
|
|
|
|
hdr->hdr_match.hdr_tag = sendreq->req_send.req_base.req_tag;
|
2006-08-24 20:38:08 +04:00
|
|
|
hdr->hdr_match.hdr_seq = (uint16_t)sendreq->req_send.req_base.req_sequence;
|
2005-09-13 02:28:23 +04:00
|
|
|
hdr->hdr_rndv.hdr_msg_length = sendreq->req_send.req_bytes_packed;
|
|
|
|
hdr->hdr_rndv.hdr_src_req.pval = sendreq;
|
|
|
|
|
2006-02-28 22:54:46 +03:00
|
|
|
#if OMPI_ENABLE_HETEROGENEOUS_SUPPORT
|
2006-02-26 03:45:54 +03:00
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
2006-02-28 22:54:46 +03:00
|
|
|
#else
|
2006-02-26 03:45:54 +03:00
|
|
|
/* if we are little endian and the remote side is big endian,
|
|
|
|
we're responsible for making sure the data is in network byte
|
|
|
|
order */
|
|
|
|
if (sendreq->req_send.req_base.req_proc->proc_arch & OMPI_ARCH_ISBIGENDIAN) {
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
|
|
|
MCA_PML_OB1_RNDV_HDR_HTON(hdr->hdr_rndv);
|
|
|
|
}
|
2006-02-28 22:54:46 +03:00
|
|
|
#endif
|
2006-02-26 03:45:54 +03:00
|
|
|
#endif
|
|
|
|
|
2005-09-13 02:28:23 +04:00
|
|
|
/* first fragment of a long message */
|
|
|
|
des->des_flags |= MCA_BTL_DES_FLAGS_PRIORITY;
|
|
|
|
des->des_cbdata = sendreq;
|
|
|
|
des->des_cbfunc = mca_pml_ob1_rndv_completion;
|
|
|
|
sendreq->req_send_offset = size;
|
|
|
|
|
|
|
|
/* send */
|
|
|
|
rc = mca_bml_base_send(bml_btl, des, MCA_BTL_TAG_PML);
|
|
|
|
if(OMPI_SUCCESS != rc) {
|
|
|
|
mca_bml_base_free(bml_btl, des );
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-09-14 21:08:08 +04:00
|
|
|
/**
|
2005-09-15 22:47:59 +04:00
|
|
|
* Schedule pipeline of send descriptors for the given request.
|
|
|
|
* Up to the rdma threshold. If this is a send based protocol,
|
|
|
|
* the rdma threshold is the end of the message. Otherwise, schedule
|
|
|
|
* fragments up to the threshold to overlap initial registration/setup
|
2006-07-20 18:44:35 +04:00
|
|
|
* costs of the rdma. Only one thread can be inside this function.
|
2005-06-01 18:34:22 +04:00
|
|
|
*/
|
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
int mca_pml_ob1_send_request_schedule_exclusive(
|
|
|
|
mca_pml_ob1_send_request_t* sendreq)
|
2005-06-01 18:34:22 +04:00
|
|
|
{
|
2005-09-13 02:28:23 +04:00
|
|
|
mca_bml_base_endpoint_t* bml_endpoint = sendreq->req_endpoint;
|
2006-07-20 18:44:35 +04:00
|
|
|
size_t num_btl_avail =
|
|
|
|
mca_bml_base_btl_array_get_size(&bml_endpoint->btl_send);
|
|
|
|
|
|
|
|
do {
|
|
|
|
/* allocate remaining bytes to BTLs */
|
2006-09-10 10:17:33 +04:00
|
|
|
size_t bytes_remaining = sendreq->req_rdma_offset -
|
2006-07-20 18:44:35 +04:00
|
|
|
sendreq->req_send_offset;
|
|
|
|
size_t prev_bytes_remaining = 0, num_fail = 0;
|
|
|
|
|
2006-09-10 10:17:33 +04:00
|
|
|
while((int32_t)bytes_remaining > 0 &&
|
2006-07-20 18:44:35 +04:00
|
|
|
(sendreq->req_pipeline_depth < mca_pml_ob1.send_pipeline_depth
|
|
|
|
||
|
|
|
|
sendreq->req_rdma_offset < sendreq->req_send.req_bytes_packed))
|
|
|
|
{
|
|
|
|
mca_pml_ob1_frag_hdr_t* hdr;
|
|
|
|
mca_btl_base_descriptor_t* des;
|
|
|
|
int rc;
|
|
|
|
size_t size;
|
|
|
|
mca_bml_base_btl_t* bml_btl =
|
|
|
|
mca_bml_base_btl_array_get_next(&bml_endpoint->btl_send);
|
|
|
|
|
|
|
|
if(prev_bytes_remaining == bytes_remaining)
|
|
|
|
num_fail++;
|
|
|
|
else
|
|
|
|
num_fail = 0;
|
|
|
|
|
|
|
|
prev_bytes_remaining = bytes_remaining;
|
|
|
|
|
|
|
|
if (num_fail == num_btl_avail) {
|
|
|
|
assert(sendreq->req_pending == MCA_PML_OB1_SEND_PENDING_NONE);
|
|
|
|
OPAL_THREAD_LOCK(&mca_pml_ob1.lock);
|
|
|
|
sendreq->req_pending = MCA_PML_OB1_SEND_PENDING_SCHEDULE;
|
|
|
|
opal_list_append(&mca_pml_ob1.send_pending,
|
|
|
|
(opal_list_item_t*)sendreq);
|
|
|
|
OPAL_THREAD_UNLOCK(&mca_pml_ob1.lock);
|
|
|
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
|
|
|
}
|
2005-06-01 18:34:22 +04:00
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
if(num_btl_avail == 1 ||
|
|
|
|
bytes_remaining < bml_btl->btl_min_send_size) {
|
|
|
|
size = bytes_remaining;
|
|
|
|
} else {
|
|
|
|
/* otherwise attempt to give the BTL a percentage of the message
|
|
|
|
* based on a weighting factor. for simplicity calculate this as
|
|
|
|
* a percentage of the overall message length (regardless of
|
|
|
|
* amount previously assigned)
|
|
|
|
*/
|
|
|
|
size = (size_t)(bml_btl->btl_weight * bytes_remaining);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* makes sure that we don't exceed BTL max send size */
|
|
|
|
if (bml_btl->btl_max_send_size != 0 &&
|
|
|
|
size > (bml_btl->btl_max_send_size -
|
|
|
|
sizeof(mca_pml_ob1_frag_hdr_t))) {
|
|
|
|
size = bml_btl->btl_max_send_size -
|
|
|
|
sizeof(mca_pml_ob1_frag_hdr_t);
|
|
|
|
}
|
2005-08-12 06:41:14 +04:00
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
/* pack into a descriptor */
|
|
|
|
ompi_convertor_set_position(&sendreq->req_send.req_convertor,
|
|
|
|
&sendreq->req_send_offset);
|
|
|
|
|
|
|
|
mca_bml_base_prepare_src(bml_btl, NULL,
|
|
|
|
&sendreq->req_send.req_convertor,
|
|
|
|
sizeof(mca_pml_ob1_frag_hdr_t), &size, &des);
|
2005-08-12 06:41:14 +04:00
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
if(des == NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
des->des_cbfunc = mca_pml_ob1_frag_completion;
|
|
|
|
des->des_cbdata = sendreq;
|
|
|
|
|
|
|
|
/* setup header */
|
|
|
|
hdr = (mca_pml_ob1_frag_hdr_t*)des->des_src->seg_addr.pval;
|
|
|
|
hdr->hdr_common.hdr_flags = 0;
|
|
|
|
hdr->hdr_common.hdr_type = MCA_PML_OB1_HDR_TYPE_FRAG;
|
|
|
|
hdr->hdr_frag_offset = sendreq->req_send_offset;
|
|
|
|
hdr->hdr_src_req.pval = sendreq;
|
|
|
|
hdr->hdr_dst_req = sendreq->req_recv;
|
2005-06-01 18:34:22 +04:00
|
|
|
|
2006-02-28 22:54:46 +03:00
|
|
|
#if OMPI_ENABLE_HETEROGENEOUS_SUPPORT
|
2006-02-26 03:45:54 +03:00
|
|
|
#ifdef WORDS_BIGENDIAN
|
2006-07-20 18:44:35 +04:00
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
2006-02-28 22:54:46 +03:00
|
|
|
#else
|
2006-07-20 18:44:35 +04:00
|
|
|
/*
|
|
|
|
* if we are little endian and the remote side is big endian,
|
|
|
|
* we're responsible for making sure the data is in network byte
|
|
|
|
* order
|
|
|
|
*/
|
|
|
|
if(sendreq->req_send.req_base.req_proc->proc_arch &
|
|
|
|
OMPI_ARCH_ISBIGENDIAN) {
|
|
|
|
hdr->hdr_common.hdr_flags |= MCA_PML_OB1_HDR_FLAGS_NBO;
|
|
|
|
MCA_PML_OB1_FRAG_HDR_HTON(*hdr);
|
|
|
|
}
|
2006-02-28 22:54:46 +03:00
|
|
|
#endif
|
2006-02-26 03:45:54 +03:00
|
|
|
#endif
|
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
/*
|
|
|
|
* The if-clause should be optimized away, in case the macro
|
|
|
|
* extends to ;
|
|
|
|
*/
|
2006-03-31 21:09:09 +04:00
|
|
|
#if OMPI_WANT_PERUSE
|
2006-07-20 18:44:35 +04:00
|
|
|
if( 0 != sendreq->req_send_offset ) {
|
|
|
|
PERUSE_TRACE_COMM_OMPI_EVENT(PERUSE_COMM_REQ_XFER_CONTINUE,
|
|
|
|
&(sendreq->req_send.req_base),
|
|
|
|
size, PERUSE_SEND);
|
|
|
|
}
|
2006-03-31 21:09:09 +04:00
|
|
|
#endif /* OMPI_WANT_PERUSE */
|
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
/* initiate send - note that this may complete before the call returns */
|
2006-10-25 12:52:39 +04:00
|
|
|
OPAL_THREAD_ADD_SIZE_T(&sendreq->req_pipeline_depth,1);
|
2006-07-20 18:44:35 +04:00
|
|
|
rc = mca_bml_base_send(bml_btl, des, MCA_BTL_TAG_PML);
|
|
|
|
|
|
|
|
if(rc == OMPI_SUCCESS) {
|
|
|
|
bytes_remaining -= size;
|
2005-06-01 18:34:22 +04:00
|
|
|
/* update state */
|
2005-06-09 00:37:19 +04:00
|
|
|
sendreq->req_send_offset += size;
|
2006-10-25 12:52:39 +04:00
|
|
|
} else {
|
|
|
|
OPAL_THREAD_ADD_SIZE_T(&sendreq->req_pipeline_depth,-1);
|
2006-07-20 18:44:35 +04:00
|
|
|
mca_bml_base_free(bml_btl,des);
|
|
|
|
continue;
|
2005-06-01 18:34:22 +04:00
|
|
|
}
|
2006-07-20 18:44:35 +04:00
|
|
|
mca_bml.bml_progress();
|
|
|
|
}
|
|
|
|
} while (OPAL_THREAD_ADD32(&sendreq->req_lock,-1) > 0);
|
|
|
|
|
2005-06-01 18:34:22 +04:00
|
|
|
return OMPI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-06-10 00:16:33 +04:00
|
|
|
/**
|
|
|
|
* An RDMA put operation has completed:
|
|
|
|
* (1) Update request status and if required set completed
|
|
|
|
* (2) Send FIN control message to the destination
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void mca_pml_ob1_put_completion(
|
2005-06-30 09:50:55 +04:00
|
|
|
mca_btl_base_module_t* btl,
|
|
|
|
struct mca_btl_base_endpoint_t* ep,
|
|
|
|
struct mca_btl_base_descriptor_t* des,
|
2005-06-10 00:16:33 +04:00
|
|
|
int status)
|
|
|
|
{
|
|
|
|
mca_pml_ob1_rdma_frag_t* frag = (mca_pml_ob1_rdma_frag_t*)des->des_cbdata;
|
2006-08-24 20:38:08 +04:00
|
|
|
mca_pml_ob1_send_request_t* sendreq = (mca_pml_ob1_send_request_t*)frag->rdma_req;
|
2006-07-20 18:44:35 +04:00
|
|
|
mca_bml_base_btl_t* bml_btl = (mca_bml_base_btl_t*) des->des_context;
|
2005-06-10 00:16:33 +04:00
|
|
|
|
|
|
|
/* check completion status */
|
|
|
|
if(OMPI_SUCCESS != status) {
|
|
|
|
/* TSW - FIX */
|
|
|
|
ORTE_ERROR_LOG(status);
|
|
|
|
orte_errmgr.abort();
|
|
|
|
}
|
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
mca_pml_ob1_send_fin(sendreq->req_send.req_base.req_proc,
|
|
|
|
frag->rdma_hdr.hdr_rdma.hdr_des.pval);
|
2005-06-10 00:16:33 +04:00
|
|
|
|
2006-03-16 01:53:41 +03:00
|
|
|
/* check for request completion */
|
|
|
|
if( OPAL_THREAD_ADD_SIZE_T(&sendreq->req_bytes_delivered, frag->rdma_length)
|
|
|
|
>= sendreq->req_send.req_bytes_packed) {
|
2006-06-14 20:55:35 +04:00
|
|
|
/* bump up the req_state after the last fin was sent..
|
|
|
|
if rndv completion occurs after this (can happen!) then
|
|
|
|
the rndv completion will properly clean up after the request
|
|
|
|
we can't just do this on the first RDMA PUT + ACK ctl message in
|
|
|
|
mca_pml_ob1_send_request_put because then we might fall into sender
|
|
|
|
side scheduleing (pml pipeline protocol) */
|
|
|
|
if(true == sendreq->req_got_put_ack) {
|
|
|
|
MCA_PML_OB1_SEND_REQUEST_ADVANCE_NO_SCHEDULE(sendreq);
|
|
|
|
}
|
2006-06-12 18:00:43 +04:00
|
|
|
/* if we've got completion on rndv packet */
|
2006-06-13 20:57:41 +04:00
|
|
|
if (sendreq->req_state == 2) {
|
2006-06-12 18:00:43 +04:00
|
|
|
MCA_PML_OB1_SEND_REQUEST_PML_COMPLETE(sendreq);
|
|
|
|
}
|
2005-06-10 00:16:33 +04:00
|
|
|
}
|
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
MCA_PML_OB1_RDMA_FRAG_RETURN(frag);
|
2005-06-10 00:16:33 +04:00
|
|
|
/* return rdma descriptor - do this after queuing the fin message - as
|
|
|
|
* release rdma resources (unpin memory) can take some time.
|
|
|
|
*/
|
2006-07-20 18:44:35 +04:00
|
|
|
des->des_dst = NULL;
|
|
|
|
des->des_dst_cnt = 0;
|
|
|
|
mca_bml_base_free(bml_btl, des);
|
2005-06-10 00:16:33 +04:00
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
MCA_PML_OB1_PROGRESS_PENDING(bml_btl);
|
2005-06-10 00:16:33 +04:00
|
|
|
}
|
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
int mca_pml_ob1_send_request_put_frag(
|
|
|
|
mca_pml_ob1_rdma_frag_t* frag
|
|
|
|
)
|
|
|
|
{
|
2006-08-24 20:38:08 +04:00
|
|
|
mca_pml_ob1_send_request_t* sendreq = (mca_pml_ob1_send_request_t*)frag->rdma_req;
|
2005-06-25 01:12:38 +04:00
|
|
|
mca_mpool_base_registration_t* reg = NULL;
|
2005-09-13 02:28:23 +04:00
|
|
|
mca_bml_base_btl_t* bml_btl;
|
2005-06-30 09:50:55 +04:00
|
|
|
mca_btl_base_descriptor_t* des;
|
2006-08-24 20:38:08 +04:00
|
|
|
size_t offset = (size_t)frag->rdma_hdr.hdr_rdma.hdr_rdma_offset;
|
2006-07-20 18:44:35 +04:00
|
|
|
size_t i, save_size = frag->rdma_length;
|
2005-06-10 00:16:33 +04:00
|
|
|
int rc;
|
2006-02-09 18:49:51 +03:00
|
|
|
bool release = false;
|
2005-06-10 00:16:33 +04:00
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
bml_btl = mca_bml_base_btl_array_find(&frag->rdma_ep->btl_rdma,
|
|
|
|
frag->rdma_btl);
|
|
|
|
|
2005-09-13 02:28:23 +04:00
|
|
|
/* lookup the corresponding registration */
|
|
|
|
for(i=0; i<sendreq->req_rdma_cnt; i++) {
|
|
|
|
if(sendreq->req_rdma[i].bml_btl == bml_btl) {
|
|
|
|
reg = sendreq->req_rdma[i].btl_reg;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2006-02-09 18:49:51 +03:00
|
|
|
|
|
|
|
/* set convertor at current offset */
|
|
|
|
ompi_convertor_set_position(&sendreq->req_send.req_convertor, &offset);
|
|
|
|
|
|
|
|
/* if registration doesnt exist - create one */
|
|
|
|
if (mca_pml_ob1.leave_pinned_pipeline && reg == NULL) {
|
|
|
|
unsigned char* base;
|
2006-10-18 00:20:58 +04:00
|
|
|
ptrdiff_t lb;
|
2006-02-09 18:49:51 +03:00
|
|
|
ompi_ddt_type_lb(sendreq->req_send.req_convertor.pDesc, &lb);
|
|
|
|
base = (unsigned char*)sendreq->req_send.req_convertor.pBaseBuf + lb + offset;
|
2006-07-20 18:44:35 +04:00
|
|
|
reg = mca_pml_ob1_rdma_register(bml_btl, base, frag->rdma_length);
|
2006-02-09 18:49:51 +03:00
|
|
|
release = true;
|
|
|
|
}
|
2005-09-13 02:28:23 +04:00
|
|
|
|
2005-06-22 00:58:24 +04:00
|
|
|
/* setup descriptor */
|
2005-08-12 06:41:14 +04:00
|
|
|
mca_bml_base_prepare_src(
|
2005-09-13 02:28:23 +04:00
|
|
|
bml_btl,
|
|
|
|
reg,
|
|
|
|
&sendreq->req_send.req_convertor,
|
|
|
|
0,
|
2006-07-20 18:44:35 +04:00
|
|
|
&frag->rdma_length,
|
2005-09-13 02:28:23 +04:00
|
|
|
&des
|
|
|
|
);
|
2005-08-12 06:41:14 +04:00
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
if(reg && release == true && bml_btl->btl_mpool) {
|
|
|
|
bml_btl->btl_mpool->mpool_release(bml_btl->btl_mpool, reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(NULL == des) {
|
|
|
|
frag->rdma_length = save_size;
|
2005-07-04 02:45:48 +04:00
|
|
|
OPAL_THREAD_LOCK(&mca_pml_ob1.lock);
|
2005-07-03 20:22:16 +04:00
|
|
|
opal_list_append(&mca_pml_ob1.rdma_pending, (opal_list_item_t*)frag);
|
2005-07-04 02:45:48 +04:00
|
|
|
OPAL_THREAD_UNLOCK(&mca_pml_ob1.lock);
|
2006-07-20 18:44:35 +04:00
|
|
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
2005-06-10 00:16:33 +04:00
|
|
|
}
|
2006-02-09 18:49:51 +03:00
|
|
|
|
2005-06-10 00:16:33 +04:00
|
|
|
des->des_dst = frag->rdma_segs;
|
2006-07-20 18:44:35 +04:00
|
|
|
des->des_dst_cnt = frag->rdma_hdr.hdr_rdma.hdr_seg_cnt;
|
2005-06-10 00:16:33 +04:00
|
|
|
des->des_cbfunc = mca_pml_ob1_put_completion;
|
|
|
|
des->des_cbdata = frag;
|
|
|
|
|
2006-06-27 00:08:33 +04:00
|
|
|
PERUSE_TRACE_COMM_OMPI_EVENT( PERUSE_COMM_REQ_XFER_CONTINUE,
|
2006-09-18 11:41:09 +04:00
|
|
|
&(sendreq->req_send.req_base), save_size, PERUSE_SEND );
|
2006-06-26 23:01:22 +04:00
|
|
|
|
2005-08-12 06:41:14 +04:00
|
|
|
if(OMPI_SUCCESS != (rc = mca_bml_base_put(bml_btl, des))) {
|
2006-07-20 18:44:35 +04:00
|
|
|
mca_bml_base_free(bml_btl, des);
|
|
|
|
frag->rdma_length = save_size;
|
|
|
|
if(OMPI_ERR_OUT_OF_RESOURCE == rc) {
|
2005-07-04 02:45:48 +04:00
|
|
|
OPAL_THREAD_LOCK(&mca_pml_ob1.lock);
|
2006-07-20 18:44:35 +04:00
|
|
|
opal_list_append(&mca_pml_ob1.rdma_pending,
|
|
|
|
(opal_list_item_t*)frag);
|
2005-07-04 02:45:48 +04:00
|
|
|
OPAL_THREAD_UNLOCK(&mca_pml_ob1.lock);
|
2006-07-20 18:44:35 +04:00
|
|
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
2005-06-10 00:16:33 +04:00
|
|
|
} else {
|
|
|
|
/* TSW - FIX */
|
|
|
|
ORTE_ERROR_LOG(rc);
|
|
|
|
orte_errmgr.abort();
|
|
|
|
}
|
|
|
|
}
|
2006-07-20 18:44:35 +04:00
|
|
|
return OMPI_SUCCESS;
|
2005-06-09 07:34:33 +04:00
|
|
|
}
|
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
/**
|
|
|
|
* Receiver has scheduled an RDMA operation:
|
|
|
|
* (1) Allocate an RDMA fragment to maintain the state of the operation
|
|
|
|
* (2) Call BTL prepare_src to pin/prepare source buffers
|
|
|
|
* (3) Queue the RDMA put
|
|
|
|
*/
|
|
|
|
|
|
|
|
void mca_pml_ob1_send_request_put(
|
|
|
|
mca_pml_ob1_send_request_t* sendreq,
|
|
|
|
mca_btl_base_module_t* btl,
|
|
|
|
mca_pml_ob1_rdma_hdr_t* hdr)
|
|
|
|
{
|
|
|
|
mca_bml_base_endpoint_t *bml_endpoint = sendreq->req_endpoint;
|
|
|
|
mca_pml_ob1_rdma_frag_t* frag;
|
|
|
|
int rc;
|
|
|
|
size_t i, size = 0;
|
2005-06-09 07:34:33 +04:00
|
|
|
|
2006-07-20 18:44:35 +04:00
|
|
|
if(hdr->hdr_common.hdr_flags & MCA_PML_OB1_HDR_TYPE_ACK) {
|
|
|
|
sendreq->req_got_put_ack = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
MCA_PML_OB1_RDMA_FRAG_ALLOC(frag, rc);
|
|
|
|
|
|
|
|
if(NULL == frag) {
|
|
|
|
/* TSW - FIX */
|
|
|
|
ORTE_ERROR_LOG(rc);
|
|
|
|
orte_errmgr.abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* setup fragment */
|
|
|
|
for(i=0; i<hdr->hdr_seg_cnt; i++) {
|
|
|
|
size += hdr->hdr_segs[i].seg_len;
|
|
|
|
frag->rdma_segs[i] = hdr->hdr_segs[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
frag->rdma_hdr.hdr_rdma = *hdr;
|
|
|
|
frag->rdma_req = sendreq;
|
|
|
|
frag->rdma_ep = bml_endpoint;
|
|
|
|
frag->rdma_btl = btl;
|
|
|
|
frag->rdma_length = size;
|
|
|
|
frag->rdma_state = MCA_PML_OB1_RDMA_PUT;
|
|
|
|
|
|
|
|
mca_pml_ob1_send_request_put_frag(frag);
|
|
|
|
}
|