cxgb4: notify uP to route ctrlq compl to rdma rspq

During the module initialisation there is a possible race
(basically race between uld and lld) where neither the uld
nor lld notifies the uP about where to route the ctrl queue
completions. LLD skips notifying uP as the rdma queues were
not created by then (will leave it to ULD to notify the uP).
As the ULD comes up, it also skips notifying the uP as the
flag FULL_INIT_DONE is not set yet (ULD assumes that the
interface is not up yet).

Consequently, this race between uld and lld leaves uP
unnotified about where to send the ctrl queue completions
to, leading to iwarp RI_RES WR failure.

Here is the race:

CPU 0                                   CPU1

- allocates nic rx queus
- t4_sge_alloc_ctrl_txq()
(if rdma rsp queues exists,
tell uP to route ctrl queue
compl to rdma rspq)
                                - acquires the mutex_lock
                                - allocates rdma response queues
                                - if FULL_INIT_DONE set,
                                  tell uP to route ctrl queue compl
                                  to rdma rspq
                                - relinquishes mutex_lock
- acquires the mutex_lock
- enable_rx()
- set FULL_INIT_DONE
- relinquishes mutex_lock

This patch fixes the above issue.

Fixes: e7519f9926f1('cxgb4: avoid enabling napi twice to the same queue')
Signed-off-by: Raju Rangoju <rajur@chelsio.com>
Acked-by: Steve Wise <swise@opengridcomputing.com>
CC: Stable <stable@vger.kernel.org> # 4.9+
Signed-off-by: Ganesh Goudar <ganeshgr@chelsio.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
Raju Rangoju 2017-06-19 17:40:48 +05:30 committed by David S. Miller
parent 89ff67718c
commit 910603818c
1 changed files with 6 additions and 4 deletions

View File

@ -2190,9 +2190,10 @@ static int cxgb_up(struct adapter *adap)
{ {
int err; int err;
mutex_lock(&uld_mutex);
err = setup_sge_queues(adap); err = setup_sge_queues(adap);
if (err) if (err)
goto out; goto rel_lock;
err = setup_rss(adap); err = setup_rss(adap);
if (err) if (err)
goto freeq; goto freeq;
@ -2216,7 +2217,6 @@ static int cxgb_up(struct adapter *adap)
goto irq_err; goto irq_err;
} }
mutex_lock(&uld_mutex);
enable_rx(adap); enable_rx(adap);
t4_sge_start(adap); t4_sge_start(adap);
t4_intr_enable(adap); t4_intr_enable(adap);
@ -2229,13 +2229,15 @@ static int cxgb_up(struct adapter *adap)
#endif #endif
/* Initialize hash mac addr list*/ /* Initialize hash mac addr list*/
INIT_LIST_HEAD(&adap->mac_hlist); INIT_LIST_HEAD(&adap->mac_hlist);
out:
return err; return err;
irq_err: irq_err:
dev_err(adap->pdev_dev, "request_irq failed, err %d\n", err); dev_err(adap->pdev_dev, "request_irq failed, err %d\n", err);
freeq: freeq:
t4_free_sge_resources(adap); t4_free_sge_resources(adap);
goto out; rel_lock:
mutex_unlock(&uld_mutex);
return err;
} }
static void cxgb_down(struct adapter *adapter) static void cxgb_down(struct adapter *adapter)