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

📄 cm.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 5 页
字号:
	cm_rep_set_target_ack_delay(rep_msg, param->target_ack_delay);	cm_rep_set_failover(rep_msg, param->failover_accepted);	cm_rep_set_flow_ctrl(rep_msg, param->flow_control);	cm_rep_set_rnr_retry_count(rep_msg, param->rnr_retry_count);	cm_rep_set_srq(rep_msg, param->srq);	rep_msg->local_ca_guid = cm_id_priv->av.port->cm_dev->ca_guid;	if (param->private_data && param->private_data_len)		memcpy(rep_msg->private_data, param->private_data,		       param->private_data_len);}int ib_send_cm_rep(struct ib_cm_id *cm_id,		   struct ib_cm_rep_param *param){	struct cm_id_private *cm_id_priv;	struct ib_mad_send_buf *msg;	struct cm_rep_msg *rep_msg;	unsigned long flags;	int ret;	if (param->private_data &&	    param->private_data_len > IB_CM_REP_PRIVATE_DATA_SIZE)		return -EINVAL;	cm_id_priv = container_of(cm_id, struct cm_id_private, id);	spin_lock_irqsave(&cm_id_priv->lock, flags);	if (cm_id->state != IB_CM_REQ_RCVD &&	    cm_id->state != IB_CM_MRA_REQ_SENT) {		ret = -EINVAL;		goto out;	}	ret = cm_alloc_msg(cm_id_priv, &msg);	if (ret)		goto out;	rep_msg = (struct cm_rep_msg *) msg->mad;	cm_format_rep(rep_msg, cm_id_priv, param);	msg->timeout_ms = cm_id_priv->timeout_ms;	msg->context[1] = (void *) (unsigned long) IB_CM_REP_SENT;	ret = ib_post_send_mad(msg, NULL);	if (ret) {		spin_unlock_irqrestore(&cm_id_priv->lock, flags);		cm_free_msg(msg);		return ret;	}	cm_id->state = IB_CM_REP_SENT;	cm_id_priv->msg = msg;	cm_id_priv->initiator_depth = param->initiator_depth;	cm_id_priv->responder_resources = param->responder_resources;	cm_id_priv->rq_psn = cm_rep_get_starting_psn(rep_msg);	cm_id_priv->local_qpn = cm_rep_get_local_qpn(rep_msg);out:	spin_unlock_irqrestore(&cm_id_priv->lock, flags);	return ret;}EXPORT_SYMBOL(ib_send_cm_rep);static void cm_format_rtu(struct cm_rtu_msg *rtu_msg,			  struct cm_id_private *cm_id_priv,			  const void *private_data,			  u8 private_data_len){	cm_format_mad_hdr(&rtu_msg->hdr, CM_RTU_ATTR_ID, cm_id_priv->tid);	rtu_msg->local_comm_id = cm_id_priv->id.local_id;	rtu_msg->remote_comm_id = cm_id_priv->id.remote_id;	if (private_data && private_data_len)		memcpy(rtu_msg->private_data, private_data, private_data_len);}int ib_send_cm_rtu(struct ib_cm_id *cm_id,		   const void *private_data,		   u8 private_data_len){	struct cm_id_private *cm_id_priv;	struct ib_mad_send_buf *msg;	unsigned long flags;	void *data;	int ret;	if (private_data && private_data_len > IB_CM_RTU_PRIVATE_DATA_SIZE)		return -EINVAL;	data = cm_copy_private_data(private_data, private_data_len);	if (IS_ERR(data))		return PTR_ERR(data);	cm_id_priv = container_of(cm_id, struct cm_id_private, id);	spin_lock_irqsave(&cm_id_priv->lock, flags);	if (cm_id->state != IB_CM_REP_RCVD &&	    cm_id->state != IB_CM_MRA_REP_SENT) {		ret = -EINVAL;		goto error;	}	ret = cm_alloc_msg(cm_id_priv, &msg);	if (ret)		goto error;	cm_format_rtu((struct cm_rtu_msg *) msg->mad, cm_id_priv,		      private_data, private_data_len);	ret = ib_post_send_mad(msg, NULL);	if (ret) {		spin_unlock_irqrestore(&cm_id_priv->lock, flags);		cm_free_msg(msg);		kfree(data);		return ret;	}	cm_id->state = IB_CM_ESTABLISHED;	cm_set_private_data(cm_id_priv, data, private_data_len);	spin_unlock_irqrestore(&cm_id_priv->lock, flags);	return 0;error:	spin_unlock_irqrestore(&cm_id_priv->lock, flags);	kfree(data);	return ret;}EXPORT_SYMBOL(ib_send_cm_rtu);static void cm_format_rep_event(struct cm_work *work){	struct cm_rep_msg *rep_msg;	struct ib_cm_rep_event_param *param;	rep_msg = (struct cm_rep_msg *)work->mad_recv_wc->recv_buf.mad;	param = &work->cm_event.param.rep_rcvd;	param->remote_ca_guid = rep_msg->local_ca_guid;	param->remote_qkey = be32_to_cpu(rep_msg->local_qkey);	param->remote_qpn = be32_to_cpu(cm_rep_get_local_qpn(rep_msg));	param->starting_psn = be32_to_cpu(cm_rep_get_starting_psn(rep_msg));	param->responder_resources = rep_msg->initiator_depth;	param->initiator_depth = rep_msg->resp_resources;	param->target_ack_delay = cm_rep_get_target_ack_delay(rep_msg);	param->failover_accepted = cm_rep_get_failover(rep_msg);	param->flow_control = cm_rep_get_flow_ctrl(rep_msg);	param->rnr_retry_count = cm_rep_get_rnr_retry_count(rep_msg);	param->srq = cm_rep_get_srq(rep_msg);	work->cm_event.private_data = &rep_msg->private_data;}static void cm_dup_rep_handler(struct cm_work *work){	struct cm_id_private *cm_id_priv;	struct cm_rep_msg *rep_msg;	struct ib_mad_send_buf *msg = NULL;	unsigned long flags;	int ret;	rep_msg = (struct cm_rep_msg *) work->mad_recv_wc->recv_buf.mad;	cm_id_priv = cm_acquire_id(rep_msg->remote_comm_id,				   rep_msg->local_comm_id);	if (!cm_id_priv)		return;	ret = cm_alloc_response_msg(work->port, work->mad_recv_wc, &msg);	if (ret)		goto deref;	spin_lock_irqsave(&cm_id_priv->lock, flags);	if (cm_id_priv->id.state == IB_CM_ESTABLISHED)		cm_format_rtu((struct cm_rtu_msg *) msg->mad, cm_id_priv,			      cm_id_priv->private_data,			      cm_id_priv->private_data_len);	else if (cm_id_priv->id.state == IB_CM_MRA_REP_SENT)		cm_format_mra((struct cm_mra_msg *) msg->mad, cm_id_priv,			      CM_MSG_RESPONSE_REP, cm_id_priv->service_timeout,			      cm_id_priv->private_data,			      cm_id_priv->private_data_len);	else		goto unlock;	spin_unlock_irqrestore(&cm_id_priv->lock, flags);	ret = ib_post_send_mad(msg, NULL);	if (ret)		goto free;	goto deref;unlock:	spin_unlock_irqrestore(&cm_id_priv->lock, flags);free:	cm_free_msg(msg);deref:	cm_deref_id(cm_id_priv);}static int cm_rep_handler(struct cm_work *work){	struct cm_id_private *cm_id_priv;	struct cm_rep_msg *rep_msg;	unsigned long flags;	int ret;	rep_msg = (struct cm_rep_msg *)work->mad_recv_wc->recv_buf.mad;	cm_id_priv = cm_acquire_id(rep_msg->remote_comm_id, 0);	if (!cm_id_priv) {		cm_dup_rep_handler(work);		return -EINVAL;	}	cm_id_priv->timewait_info->work.remote_id = rep_msg->local_comm_id;	cm_id_priv->timewait_info->remote_ca_guid = rep_msg->local_ca_guid;	cm_id_priv->timewait_info->remote_qpn = cm_rep_get_local_qpn(rep_msg);	spin_lock_irqsave(&cm.lock, flags);	/* Check for duplicate REP. */	if (cm_insert_remote_id(cm_id_priv->timewait_info)) {		spin_unlock_irqrestore(&cm.lock, flags);		ret = -EINVAL;		goto error;	}	/* Check for a stale connection. */	if (cm_insert_remote_qpn(cm_id_priv->timewait_info)) {		spin_unlock_irqrestore(&cm.lock, flags);		cm_issue_rej(work->port, work->mad_recv_wc,			     IB_CM_REJ_STALE_CONN, CM_MSG_RESPONSE_REP,			     NULL, 0);		ret = -EINVAL;		goto error;	}	spin_unlock_irqrestore(&cm.lock, flags);	cm_format_rep_event(work);	spin_lock_irqsave(&cm_id_priv->lock, flags);	switch (cm_id_priv->id.state) {	case IB_CM_REQ_SENT:	case IB_CM_MRA_REQ_RCVD:		break;	default:		spin_unlock_irqrestore(&cm_id_priv->lock, flags);		ret = -EINVAL;		goto error;	}	cm_id_priv->id.state = IB_CM_REP_RCVD;	cm_id_priv->id.remote_id = rep_msg->local_comm_id;	cm_id_priv->remote_qpn = cm_rep_get_local_qpn(rep_msg);	cm_id_priv->initiator_depth = rep_msg->resp_resources;	cm_id_priv->responder_resources = rep_msg->initiator_depth;	cm_id_priv->sq_psn = cm_rep_get_starting_psn(rep_msg);	cm_id_priv->rnr_retry_count = cm_rep_get_rnr_retry_count(rep_msg);	/* todo: handle peer_to_peer */	ib_cancel_mad(cm_id_priv->av.port->mad_agent, cm_id_priv->msg);	ret = atomic_inc_and_test(&cm_id_priv->work_count);	if (!ret)		list_add_tail(&work->list, &cm_id_priv->work_list);	spin_unlock_irqrestore(&cm_id_priv->lock, flags);	if (ret)		cm_process_work(cm_id_priv, work);	else		cm_deref_id(cm_id_priv);	return 0;error:	cm_cleanup_timewait(cm_id_priv->timewait_info);	cm_deref_id(cm_id_priv);	return ret;}static int cm_establish_handler(struct cm_work *work){	struct cm_id_private *cm_id_priv;	unsigned long flags;	int ret;	/* See comment in ib_cm_establish about lookup. */	cm_id_priv = cm_acquire_id(work->local_id, work->remote_id);	if (!cm_id_priv)		return -EINVAL;	spin_lock_irqsave(&cm_id_priv->lock, flags);	if (cm_id_priv->id.state != IB_CM_ESTABLISHED) {		spin_unlock_irqrestore(&cm_id_priv->lock, flags);		goto out;	}	ib_cancel_mad(cm_id_priv->av.port->mad_agent, cm_id_priv->msg);	ret = atomic_inc_and_test(&cm_id_priv->work_count);	if (!ret)		list_add_tail(&work->list, &cm_id_priv->work_list);	spin_unlock_irqrestore(&cm_id_priv->lock, flags);	if (ret)		cm_process_work(cm_id_priv, work);	else		cm_deref_id(cm_id_priv);	return 0;out:	cm_deref_id(cm_id_priv);	return -EINVAL;}static int cm_rtu_handler(struct cm_work *work){	struct cm_id_private *cm_id_priv;	struct cm_rtu_msg *rtu_msg;	unsigned long flags;	int ret;	rtu_msg = (struct cm_rtu_msg *)work->mad_recv_wc->recv_buf.mad;	cm_id_priv = cm_acquire_id(rtu_msg->remote_comm_id,				   rtu_msg->local_comm_id);	if (!cm_id_priv)		return -EINVAL;	work->cm_event.private_data = &rtu_msg->private_data;	spin_lock_irqsave(&cm_id_priv->lock, flags);	if (cm_id_priv->id.state != IB_CM_REP_SENT &&	    cm_id_priv->id.state != IB_CM_MRA_REP_RCVD) {		spin_unlock_irqrestore(&cm_id_priv->lock, flags);		goto out;	}	cm_id_priv->id.state = IB_CM_ESTABLISHED;	ib_cancel_mad(cm_id_priv->av.port->mad_agent, cm_id_priv->msg);	ret = atomic_inc_and_test(&cm_id_priv->work_count);	if (!ret)		list_add_tail(&work->list, &cm_id_priv->work_list);	spin_unlock_irqrestore(&cm_id_priv->lock, flags);	if (ret)		cm_process_work(cm_id_priv, work);	else		cm_deref_id(cm_id_priv);	return 0;out:	cm_deref_id(cm_id_priv);	return -EINVAL;}static void cm_format_dreq(struct cm_dreq_msg *dreq_msg,			  struct cm_id_private *cm_id_priv,			  const void *private_data,			  u8 private_data_len){	cm_format_mad_hdr(&dreq_msg->hdr, CM_DREQ_ATTR_ID,			  cm_form_tid(cm_id_priv, CM_MSG_SEQUENCE_DREQ));	dreq_msg->local_comm_id = cm_id_priv->id.local_id;	dreq_msg->remote_comm_id = cm_id_priv->id.remote_id;	cm_dreq_set_remote_qpn(dreq_msg, cm_id_priv->remote_qpn);	if (private_data && private_data_len)		memcpy(dreq_msg->private_data, private_data, private_data_len);}int ib_send_cm_dreq(struct ib_cm_id *cm_id,		    const void *private_data,		    u8 private_data_len){	struct cm_id_private *cm_id_priv;	struct ib_mad_send_buf *msg;	unsigned long flags;	int ret;	if (private_data && private_data_len > IB_CM_DREQ_PRIVATE_DATA_SIZE)		return -EINVAL;	cm_id_priv = container_of(cm_id, struct cm_id_private, id);	spin_lock_irqsave(&cm_id_priv->lock, flags);	if (cm_id->state != IB_CM_ESTABLISHED) {		ret = -EINVAL;		goto out;	}	ret = cm_alloc_msg(cm_id_priv, &msg);	if (ret) {		cm_enter_timewait(cm_id_priv);		goto out;	}	cm_format_dreq((struct cm_dreq_msg *) msg->mad, cm_id_priv,		       private_data, private_data_len);	msg->timeout_ms = cm_id_priv->timeout_ms;	msg->context[1] = (void *) (unsigned long) IB_CM_DREQ_SENT;	ret = ib_post_send_mad(msg, NULL);	if (ret) {		cm_enter_timewait(cm_id_priv);		spin_unlock_irqrestore(&cm_id_priv->lock, flags);		cm_free_msg(msg);		return ret;	}	cm_id->state = IB_CM_DREQ_SENT;	cm_id_priv->msg = msg;out:	spin_unlock_irqrestore(&cm_id_priv->lock, flags);	return ret;}EXPORT_SYMBOL(ib_send_cm_dreq);static void cm_format_drep(struct cm_drep_msg *drep_msg,			  struct cm_id_private *cm_id_priv,			  const void *private_data,			  u8 private_data_len){	cm_format_mad_hdr(&drep_msg->hdr, CM_DREP_ATTR_ID, cm_id_priv->tid);	drep_msg->local_comm_id = cm_id_priv->id.local_id;	drep_msg->remote_comm_id = cm_id_priv->id.remote_id;	if (private_data && private_data_len)		memcpy(drep_msg->private_data, private_data, private_data_len);}int ib_send_cm_drep(struct ib_cm_id *cm_id,		    const void *private_data,		    u8 private_data_len){	struct cm_id_private *cm_id_priv;	struct ib_mad_send_buf *msg;	unsigned long flags;	void *data;	int ret;	if (private_data && private_data_len > IB_CM_DREP_PRIVATE_DATA_SIZE)		return -EINVAL;	data = cm_copy_private_data(private_data, private_data_len);	if (IS_ERR(data))		return PTR_ERR(data);	cm_id_priv = container_of(cm_id, struct cm_id_private, id);	spin_lock_irqsave(&cm_id_priv->lock, flags);	if (cm_id->state != IB_CM_DREQ_RCVD) {		spin_unlock_irqrestore(&cm_id_priv->lock, flags);		kfree(data);		return -EINVAL;	}	cm_set_private_data(cm_id_priv, data, private_data_len);	cm_enter_timewait(cm_id_priv);	ret = cm_alloc_msg(cm_id_priv, &msg);	if (ret)		goto out;	cm_format_drep((struct cm_drep_msg *) msg->mad, cm_id_priv,		       private_data, private_data_len);	ret = ib_post_send_mad(msg, NULL);	if (ret) {		spin_unlock_irqrestore(&cm_id_priv->lock, flags);		cm_free_msg(msg);

⌨️ 快捷键说明

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