i40e/i40evf: check for stopped admin queue
It's possible that while we are waiting for the spinlock, another entity (that owns the spinlock) has shut down the admin queue. If we then attempt to use the queue, we will panic. Add a check for this condition on the receive side. This matches an existing check on the send queue side. Signed-off-by: Mitch Williams <mitch.a.williams@intel.com> Acked-by: Jesse Brandeburg <jesse.brandeburg@intel.com> Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
c4bbac3913
commit
43ae93a93e
|
@ -946,6 +946,13 @@ i40e_status i40e_clean_arq_element(struct i40e_hw *hw,
|
||||||
/* take the lock before we start messing with the ring */
|
/* take the lock before we start messing with the ring */
|
||||||
mutex_lock(&hw->aq.arq_mutex);
|
mutex_lock(&hw->aq.arq_mutex);
|
||||||
|
|
||||||
|
if (hw->aq.arq.count == 0) {
|
||||||
|
i40e_debug(hw, I40E_DEBUG_AQ_MESSAGE,
|
||||||
|
"AQRX: Admin queue not initialized.\n");
|
||||||
|
ret_code = I40E_ERR_QUEUE_EMPTY;
|
||||||
|
goto clean_arq_element_err;
|
||||||
|
}
|
||||||
|
|
||||||
/* set next_to_use to head */
|
/* set next_to_use to head */
|
||||||
ntu = (rd32(hw, hw->aq.arq.head) & I40E_PF_ARQH_ARQH_MASK);
|
ntu = (rd32(hw, hw->aq.arq.head) & I40E_PF_ARQH_ARQH_MASK);
|
||||||
if (ntu == ntc) {
|
if (ntu == ntc) {
|
||||||
|
@ -1007,6 +1014,8 @@ i40e_status i40e_clean_arq_element(struct i40e_hw *hw,
|
||||||
/* Set pending if needed, unlock and return */
|
/* Set pending if needed, unlock and return */
|
||||||
if (pending != NULL)
|
if (pending != NULL)
|
||||||
*pending = (ntc > ntu ? hw->aq.arq.count : 0) + (ntu - ntc);
|
*pending = (ntc > ntu ? hw->aq.arq.count : 0) + (ntu - ntc);
|
||||||
|
|
||||||
|
clean_arq_element_err:
|
||||||
mutex_unlock(&hw->aq.arq_mutex);
|
mutex_unlock(&hw->aq.arq_mutex);
|
||||||
|
|
||||||
if (i40e_is_nvm_update_op(&e->desc)) {
|
if (i40e_is_nvm_update_op(&e->desc)) {
|
||||||
|
|
|
@ -887,6 +887,13 @@ i40e_status i40evf_clean_arq_element(struct i40e_hw *hw,
|
||||||
/* take the lock before we start messing with the ring */
|
/* take the lock before we start messing with the ring */
|
||||||
mutex_lock(&hw->aq.arq_mutex);
|
mutex_lock(&hw->aq.arq_mutex);
|
||||||
|
|
||||||
|
if (hw->aq.arq.count == 0) {
|
||||||
|
i40e_debug(hw, I40E_DEBUG_AQ_MESSAGE,
|
||||||
|
"AQRX: Admin queue not initialized.\n");
|
||||||
|
ret_code = I40E_ERR_QUEUE_EMPTY;
|
||||||
|
goto clean_arq_element_err;
|
||||||
|
}
|
||||||
|
|
||||||
/* set next_to_use to head */
|
/* set next_to_use to head */
|
||||||
ntu = (rd32(hw, hw->aq.arq.head) & I40E_VF_ARQH1_ARQH_MASK);
|
ntu = (rd32(hw, hw->aq.arq.head) & I40E_VF_ARQH1_ARQH_MASK);
|
||||||
if (ntu == ntc) {
|
if (ntu == ntc) {
|
||||||
|
@ -948,6 +955,8 @@ i40e_status i40evf_clean_arq_element(struct i40e_hw *hw,
|
||||||
/* Set pending if needed, unlock and return */
|
/* Set pending if needed, unlock and return */
|
||||||
if (pending != NULL)
|
if (pending != NULL)
|
||||||
*pending = (ntc > ntu ? hw->aq.arq.count : 0) + (ntu - ntc);
|
*pending = (ntc > ntu ? hw->aq.arq.count : 0) + (ntu - ntc);
|
||||||
|
|
||||||
|
clean_arq_element_err:
|
||||||
mutex_unlock(&hw->aq.arq_mutex);
|
mutex_unlock(&hw->aq.arq_mutex);
|
||||||
|
|
||||||
return ret_code;
|
return ret_code;
|
||||||
|
|
Loading…
Reference in New Issue