diff options
| author | Takashi Iwai <tiwai@suse.de> | 2018-03-12 17:30:38 +0100 | 
|---|---|---|
| committer | Takashi Iwai <tiwai@suse.de> | 2018-03-12 17:30:38 +0100 | 
| commit | db45dc9540ea4864bb9d22eed30d3e70ed5d56d4 (patch) | |
| tree | c611d75097036ff464e9af31aa729e34c863ec93 /drivers/infiniband/hw/bnxt_re/main.c | |
| parent | 40088dc4e1ead7df31728c73f5b51d71da18831d (diff) | |
| parent | a3e39ed1f40e14f24bec0f91c4fa0408fb6c4d02 (diff) | |
Merge tag 'asoc-fix-v4.16-rc5' of https://git.kernel.org/pub/scm/linux/kernel/git/broonie/sound into for-linus
ASoC: Fixes for v4.16
This is a fairly standard collection of fixes, there's no changes to the
core here just a bunch of small device specific changes for single
drivers plus an update to the MAINTAINERS file for the sgl5000.
Diffstat (limited to 'drivers/infiniband/hw/bnxt_re/main.c')
| -rw-r--r-- | drivers/infiniband/hw/bnxt_re/main.c | 24 | 
1 files changed, 19 insertions, 5 deletions
| diff --git a/drivers/infiniband/hw/bnxt_re/main.c b/drivers/infiniband/hw/bnxt_re/main.c index 508d00a5a106..f6e361750466 100644 --- a/drivers/infiniband/hw/bnxt_re/main.c +++ b/drivers/infiniband/hw/bnxt_re/main.c @@ -656,7 +656,6 @@ static void bnxt_re_dev_remove(struct bnxt_re_dev *rdev)  	mutex_unlock(&bnxt_re_dev_lock);  	synchronize_rcu(); -	flush_workqueue(bnxt_re_wq);  	ib_dealloc_device(&rdev->ibdev);  	/* rdev is gone */ @@ -731,6 +730,13 @@ static int bnxt_re_handle_qp_async_event(struct creq_qp_event *qp_event,  					 struct bnxt_re_qp *qp)  {  	struct ib_event event; +	unsigned int flags; + +	if (qp->qplib_qp.state == CMDQ_MODIFY_QP_NEW_STATE_ERR) { +		flags = bnxt_re_lock_cqs(qp); +		bnxt_qplib_add_flush_qp(&qp->qplib_qp); +		bnxt_re_unlock_cqs(qp, flags); +	}  	memset(&event, 0, sizeof(event));  	if (qp->qplib_qp.srq) { @@ -1417,9 +1423,12 @@ static void bnxt_re_task(struct work_struct *work)  	switch (re_work->event) {  	case NETDEV_REGISTER:  		rc = bnxt_re_ib_reg(rdev); -		if (rc) +		if (rc) {  			dev_err(rdev_to_dev(rdev),  				"Failed to register with IB: %#x", rc); +			bnxt_re_remove_one(rdev); +			bnxt_re_dev_unreg(rdev); +		}  		break;  	case NETDEV_UP:  		bnxt_re_dispatch_event(&rdev->ibdev, NULL, 1, @@ -1441,7 +1450,7 @@ static void bnxt_re_task(struct work_struct *work)  		break;  	}  	smp_mb__before_atomic(); -	clear_bit(BNXT_RE_FLAG_TASK_IN_PROG, &rdev->flags); +	atomic_dec(&rdev->sched_count);  	kfree(re_work);  } @@ -1503,7 +1512,7 @@ static int bnxt_re_netdev_event(struct notifier_block *notifier,  		/* netdev notifier will call NETDEV_UNREGISTER again later since  		 * we are still holding the reference to the netdev  		 */ -		if (test_bit(BNXT_RE_FLAG_TASK_IN_PROG, &rdev->flags)) +		if (atomic_read(&rdev->sched_count) > 0)  			goto exit;  		bnxt_re_ib_unreg(rdev, false);  		bnxt_re_remove_one(rdev); @@ -1523,7 +1532,7 @@ static int bnxt_re_netdev_event(struct notifier_block *notifier,  			re_work->vlan_dev = (real_dev == netdev ?  					     NULL : netdev);  			INIT_WORK(&re_work->work, bnxt_re_task); -			set_bit(BNXT_RE_FLAG_TASK_IN_PROG, &rdev->flags); +			atomic_inc(&rdev->sched_count);  			queue_work(bnxt_re_wq, &re_work->work);  		}  	} @@ -1578,6 +1587,11 @@ static void __exit bnxt_re_mod_exit(void)  	*/  	list_for_each_entry_safe_reverse(rdev, next, &to_be_deleted, list) {  		dev_info(rdev_to_dev(rdev), "Unregistering Device"); +		/* +		 * Flush out any scheduled tasks before destroying the +		 * resources +		 */ +		flush_workqueue(bnxt_re_wq);  		bnxt_re_dev_stop(rdev);  		bnxt_re_ib_unreg(rdev, true);  		bnxt_re_remove_one(rdev); | 
