⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ch3_isendv.c

📁 刚才是说明 现在是安装程序在 LINUX环境下进行编程的MPICH安装文件
💻 C
字号:
/* -*- Mode: C; c-basic-offset:4 ; -*- *//* *  (C) 2001 by Argonne National Laboratory. *      See COPYRIGHT in top-level directory. */#include "mpidi_ch3_impl.h"static void update_request(MPID_Request * sreq, MPID_IOV * iov, int iov_count, int iov_offset, int nb){    int i;    MPIDI_STATE_DECL(MPID_STATE_UPDATE_REQUEST);    MPIDI_FUNC_ENTER(MPID_STATE_UPDATE_REQUEST);        /* memcpy(sreq->ch3.iov, iov, iov_count * sizeof(MPID_IOV)); */    for (i = 0; i < iov_count; i++)    {	sreq->ch3.iov[i] = iov[i];    }    if (iov_offset == 0)    {	/* memcpy(&sreq->tcp.pkt, iov[0].MPID_IOV_BUF, iov[0].MPID_IOV_LEN); */	assert(iov[0].MPID_IOV_LEN == sizeof(MPIDI_CH3_Pkt_t));	sreq->tcp.pkt = *(MPIDI_CH3_Pkt_t *) iov[0].MPID_IOV_BUF;	sreq->ch3.iov[0].MPID_IOV_BUF = &sreq->tcp.pkt;    }    sreq->ch3.iov[iov_offset].MPID_IOV_BUF = 	(char *) sreq->ch3.iov[iov_offset].MPID_IOV_BUF + nb;    sreq->ch3.iov[iov_offset].MPID_IOV_LEN -= nb;    sreq->tcp.iov_offset = iov_offset;    sreq->ch3.iov_count = iov_count;    MPIDI_FUNC_EXIT(MPID_STATE_UPDATE_REQUEST);}#undef FUNCNAME#define FUNCNAME MPIDI_CH3_iSendv#undef FCNAME#define FCNAME MPIDI_QUOTE(FUNCNAME)void MPIDI_CH3_iSendv(MPIDI_VC * vc, MPID_Request * sreq, MPID_IOV * iov, int n_iov){    MPIDI_STATE_DECL(MPID_STATE_MPIDI_CH3_ISENDV);    MPIDI_STATE_DECL(MPID_STATE_WRITEV);    MPIDI_FUNC_ENTER(MPID_STATE_MPIDI_CH3_ISENDV);    MPIDI_DBG_PRINTF((50, FCNAME, "entering"));    assert(n_iov <= MPID_IOV_LIMIT);    assert(iov[0].MPID_IOV_LEN <= sizeof(MPIDI_CH3_Pkt_t));    /* The TCP implementation uses a fixed length header, the size of which is the maximum of all possible packet headers */    iov[0].MPID_IOV_LEN = sizeof(MPIDI_CH3_Pkt_t);        if (vc->tcp.state == MPIDI_CH3I_VC_STATE_CONNECTED) /* MT */    {	/* Connection already formed.  If send queue is empty attempt to send data, queuing any unsent data. */	if (MPIDI_CH3I_SendQ_empty(vc)) /* MT */	{	    int nb;	    MPIDI_DBG_PRINTF((55, FCNAME, "send queue empty, attempting to write"));	    	    /* MT - need some signalling to lock down our right to use the channel, thus insuring that the progress engine does               also try to write */	    /* FIXME: the current code only agressively writes the first IOV.  Eventually it should be changed to agressively write               as much as possible.  Ideally, the code would be shared between the send routines and the progress engine. */	    do	    {		MPIDI_FUNC_ENTER(MPID_STATE_WRITEV);		nb = writev(vc->tcp.fd, iov, n_iov);		MPIDI_FUNC_EXIT(MPID_STATE_WRITEV);	    }	    while (nb == -1 && errno == EINTR);	    	    if (nb > 0)	    {		int offset = 0;		MPIDI_DBG_PRINTF((55, FCNAME, "wrote %d bytes", nb));				while (offset < n_iov)		{		    if (iov[offset].MPID_IOV_LEN <= nb)		    {			nb -= iov[offset].MPID_IOV_LEN;			offset++;		    }		    else		    {			MPIDI_DBG_PRINTF((55, FCNAME, "partial write, enqueuing at head"));			update_request(sreq, iov, n_iov, offset, nb);			MPIDI_CH3I_SendQ_enqueue_head(vc, sreq);			MPIDI_CH3I_TCP_post_write(vc, sreq);			break;		    }		}		if (offset == n_iov)		{		    MPIDI_DBG_PRINTF((55, FCNAME, "write complete, calling MPIDI_CH3U_Handle_send_req()"));		    MPIDI_CH3U_Handle_send_req(vc, sreq);		    if (sreq->ch3.iov_count != 0)		    {			/* NOTE: ch3.iov_count is used to detect completion instead of cc because the transfer may be complete, but                           request may still be active (see MPI_Ssend()) */			MPIDI_CH3I_SendQ_enqueue_head(vc, sreq);			MPIDI_CH3I_TCP_post_write(vc, sreq);		    }		}	    }	    else if (nb == 0 || errno == EAGAIN ||		     errno == EWOULDBLOCK || errno == ENOMEM)	    {		MPIDI_DBG_PRINTF((55, FCNAME, "unable to write, enqueuing"));		update_request(sreq, iov, n_iov, 0, 0);		MPIDI_CH3I_SendQ_enqueue(vc, sreq);		MPIDI_CH3I_TCP_post_write(vc, sreq);	    }	    else	    {		/* Connection just failed.  Mark the request complete and return an error. */		vc->tcp.state = MPIDI_CH3I_VC_STATE_FAILED;		/* TODO: Create an appropriate error message based on the value of errno */		sreq->status.MPI_ERROR = MPI_ERR_INTERN;		 /* MT - CH3U_Request_complete performs write barrier */		MPIDI_CH3U_Request_complete(sreq);	    }	}	else	{	    MPIDI_DBG_PRINTF((55, FCNAME, "send queue not empty, enqueuing"));	    update_request(sreq, iov, n_iov, 0, 0);	    MPIDI_CH3I_SendQ_enqueue(vc, sreq);	}    }    else if (vc->tcp.state == MPIDI_CH3I_VC_STATE_UNCONNECTED)    {	/* Form a new connection, queuing the data so it can be sent later. */	MPIDI_DBG_PRINTF((55, FCNAME, "unconnected.  enqueuing request"));	MPIDI_CH3I_TCP_post_connect(vc);	update_request(sreq, iov, n_iov, 0, 0);	MPIDI_CH3I_SendQ_enqueue(vc, sreq);    }    else if (vc->tcp.state != MPIDI_CH3I_VC_STATE_FAILED)    {	/* Unable to send data at the moment, so queue it for later */	MPIDI_DBG_PRINTF((55, FCNAME, "still connecting.  enqueuing request"));	update_request(sreq, iov, n_iov, 0, 0);	MPIDI_CH3I_SendQ_enqueue(vc, sreq);    }    else    {	/* Connection failed.  Mark the request complete and return an           error. */	/* TODO: Create an appropriate error message */	sreq->status.MPI_ERROR = MPI_ERR_INTERN;	/* MT - CH3U_Request_complete performs write barrier */	MPIDI_CH3U_Request_complete(sreq);    }        MPIDI_DBG_PRINTF((50, FCNAME, "exiting"));    MPIDI_FUNC_EXIT(MPID_STATE_MPIDI_CH3_ISENDV);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -