diff options
Diffstat (limited to 'drivers/infiniband/sw/rdmavt/qp.c')
| -rw-r--r-- | drivers/infiniband/sw/rdmavt/qp.c | 119 | 
1 files changed, 69 insertions, 50 deletions
| diff --git a/drivers/infiniband/sw/rdmavt/qp.c b/drivers/infiniband/sw/rdmavt/qp.c index 870b4f212fbc..6500c3b5a89c 100644 --- a/drivers/infiniband/sw/rdmavt/qp.c +++ b/drivers/infiniband/sw/rdmavt/qp.c @@ -488,60 +488,23 @@ static void rvt_remove_qp(struct rvt_dev_info *rdi, struct rvt_qp *qp)  	spin_unlock_irqrestore(&rdi->qp_dev->qpt_lock, flags);  	if (removed) {  		synchronize_rcu(); -		if (atomic_dec_and_test(&qp->refcount)) -			wake_up(&qp->wait); +		rvt_put_qp(qp);  	}  }  /** - * reset_qp - initialize the QP state to the reset state - * @qp: the QP to reset + * rvt_init_qp - initialize the QP state to the reset state + * @qp: the QP to init or reinit   * @type: the QP type - * r and s lock are required to be held by the caller + * + * This function is called from both rvt_create_qp() and + * rvt_reset_qp().   The difference is that the reset + * patch the necessary locks to protect against concurent + * access.   */ -static void rvt_reset_qp(struct rvt_dev_info *rdi, struct rvt_qp *qp, -		  enum ib_qp_type type) -	__releases(&qp->s_lock) -	__releases(&qp->s_hlock) -	__releases(&qp->r_lock) -	__acquires(&qp->r_lock) -	__acquires(&qp->s_hlock) -	__acquires(&qp->s_lock) +static void rvt_init_qp(struct rvt_dev_info *rdi, struct rvt_qp *qp, +			enum ib_qp_type type)  { -	if (qp->state != IB_QPS_RESET) { -		qp->state = IB_QPS_RESET; - -		/* Let drivers flush their waitlist */ -		rdi->driver_f.flush_qp_waiters(qp); -		qp->s_flags &= ~(RVT_S_TIMER | RVT_S_ANY_WAIT); -		spin_unlock(&qp->s_lock); -		spin_unlock(&qp->s_hlock); -		spin_unlock_irq(&qp->r_lock); - -		/* Stop the send queue and the retry timer */ -		rdi->driver_f.stop_send_queue(qp); - -		/* Wait for things to stop */ -		rdi->driver_f.quiesce_qp(qp); - -		/* take qp out the hash and wait for it to be unused */ -		rvt_remove_qp(rdi, qp); -		wait_event(qp->wait, !atomic_read(&qp->refcount)); - -		/* grab the lock b/c it was locked at call time */ -		spin_lock_irq(&qp->r_lock); -		spin_lock(&qp->s_hlock); -		spin_lock(&qp->s_lock); - -		rvt_clear_mr_refs(qp, 1); -	} - -	/* -	 * Let the driver do any tear down it needs to for a qp -	 * that has been reset -	 */ -	rdi->driver_f.notify_qp_reset(qp); -  	qp->remote_qpn = 0;  	qp->qkey = 0;  	qp->qp_access_flags = 0; @@ -587,6 +550,60 @@ static void rvt_reset_qp(struct rvt_dev_info *rdi, struct rvt_qp *qp,  }  /** + * rvt_reset_qp - initialize the QP state to the reset state + * @qp: the QP to reset + * @type: the QP type + * + * r_lock, s_hlock, and s_lock are required to be held by the caller + */ +static void rvt_reset_qp(struct rvt_dev_info *rdi, struct rvt_qp *qp, +			 enum ib_qp_type type) +	__must_hold(&qp->s_lock) +	__must_hold(&qp->s_hlock) +	__must_hold(&qp->r_lock) +{ +	lockdep_assert_held(&qp->r_lock); +	lockdep_assert_held(&qp->s_hlock); +	lockdep_assert_held(&qp->s_lock); +	if (qp->state != IB_QPS_RESET) { +		qp->state = IB_QPS_RESET; + +		/* Let drivers flush their waitlist */ +		rdi->driver_f.flush_qp_waiters(qp); +		qp->s_flags &= ~(RVT_S_TIMER | RVT_S_ANY_WAIT); +		spin_unlock(&qp->s_lock); +		spin_unlock(&qp->s_hlock); +		spin_unlock_irq(&qp->r_lock); + +		/* Stop the send queue and the retry timer */ +		rdi->driver_f.stop_send_queue(qp); + +		/* Wait for things to stop */ +		rdi->driver_f.quiesce_qp(qp); + +		/* take qp out the hash and wait for it to be unused */ +		rvt_remove_qp(rdi, qp); +		wait_event(qp->wait, !atomic_read(&qp->refcount)); + +		/* grab the lock b/c it was locked at call time */ +		spin_lock_irq(&qp->r_lock); +		spin_lock(&qp->s_hlock); +		spin_lock(&qp->s_lock); + +		rvt_clear_mr_refs(qp, 1); +		/* +		 * Let the driver do any tear down or re-init it needs to for +		 * a qp that has been reset +		 */ +		rdi->driver_f.notify_qp_reset(qp); +	} +	rvt_init_qp(rdi, qp, type); +	lockdep_assert_held(&qp->r_lock); +	lockdep_assert_held(&qp->s_hlock); +	lockdep_assert_held(&qp->s_lock); +} + +/**   * rvt_create_qp - create a queue pair for a device   * @ibpd: the protection domain who's device we create the queue pair for   * @init_attr: the attributes of the queue pair @@ -766,7 +783,7 @@ struct ib_qp *rvt_create_qp(struct ib_pd *ibpd,  		}  		qp->ibqp.qp_num = err;  		qp->port_num = init_attr->port_num; -		rvt_reset_qp(rdi, qp, init_attr->qp_type); +		rvt_init_qp(rdi, qp, init_attr->qp_type);  		break;  	default: @@ -906,6 +923,8 @@ int rvt_error_qp(struct rvt_qp *qp, enum ib_wc_status err)  	int ret = 0;  	struct rvt_dev_info *rdi = ib_to_rvt(qp->ibqp.device); +	lockdep_assert_held(&qp->r_lock); +	lockdep_assert_held(&qp->s_lock);  	if (qp->state == IB_QPS_ERR || qp->state == IB_QPS_RESET)  		goto bail; @@ -980,7 +999,7 @@ static void rvt_insert_qp(struct rvt_dev_info *rdi, struct rvt_qp *qp)  	struct rvt_ibport *rvp = rdi->ports[qp->port_num - 1];  	unsigned long flags; -	atomic_inc(&qp->refcount); +	rvt_get_qp(qp);  	spin_lock_irqsave(&rdi->qp_dev->qpt_lock, flags);  	if (qp->ibqp.qp_num <= 1) { @@ -997,7 +1016,7 @@ static void rvt_insert_qp(struct rvt_dev_info *rdi, struct rvt_qp *qp)  }  /** - * qib_modify_qp - modify the attributes of a queue pair + * rvt_modify_qp - modify the attributes of a queue pair   * @ibqp: the queue pair who's attributes we're modifying   * @attr: the new attributes   * @attr_mask: the mask of attributes to modify |