Commit c768e490 authored by Jacob Keller's avatar Jacob Keller Committed by Jeff Kirsher

i40e: factor out queue control from i40e_vsi_control_(tx|rx)

A future patch will need to be able to handle controlling queues without
waiting until all VSIs are handled. Factor out the direct queue
modification so that we can easily re-use this code. The result is also
a bit easier to read since we don't embed multiple single-letter loop
counters.

Change-ID: Id923cbfa43127b1c24d8ed4f809b1012c736d9ac
Signed-off-by: default avatarJacob Keller <jacob.e.keller@intel.com>
Tested-by: default avatarAndrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: default avatarJeff Kirsher <jeffrey.t.kirsher@intel.com>
parent 024b05f4
...@@ -3919,6 +3919,8 @@ static void i40e_netpoll(struct net_device *netdev) ...@@ -3919,6 +3919,8 @@ static void i40e_netpoll(struct net_device *netdev)
} }
#endif #endif
#define I40E_QTX_ENA_WAIT_COUNT 50
/** /**
* i40e_pf_txq_wait - Wait for a PF's Tx queue to be enabled or disabled * i40e_pf_txq_wait - Wait for a PF's Tx queue to be enabled or disabled
* @pf: the PF being configured * @pf: the PF being configured
...@@ -3949,35 +3951,37 @@ static int i40e_pf_txq_wait(struct i40e_pf *pf, int pf_q, bool enable) ...@@ -3949,35 +3951,37 @@ static int i40e_pf_txq_wait(struct i40e_pf *pf, int pf_q, bool enable)
} }
/** /**
* i40e_vsi_control_tx - Start or stop a VSI's rings * i40e_control_tx_q - Start or stop a particular Tx queue
* @vsi: the VSI being configured * @pf: the PF structure
* @enable: start or stop the rings * @pf_q: the PF queue to configure
* @enable: start or stop the queue
*
* This function enables or disables a single queue. Note that any delay
* required after the operation is expected to be handled by the caller of
* this function.
**/ **/
static int i40e_vsi_control_tx(struct i40e_vsi *vsi, bool enable) static void i40e_control_tx_q(struct i40e_pf *pf, int pf_q, bool enable)
{ {
struct i40e_pf *pf = vsi->back;
struct i40e_hw *hw = &pf->hw; struct i40e_hw *hw = &pf->hw;
int i, j, pf_q, ret = 0;
u32 tx_reg; u32 tx_reg;
int i;
pf_q = vsi->base_queue;
for (i = 0; i < vsi->num_queue_pairs; i++, pf_q++) {
/* warn the TX unit of coming changes */ /* warn the TX unit of coming changes */
i40e_pre_tx_queue_cfg(&pf->hw, pf_q, enable); i40e_pre_tx_queue_cfg(&pf->hw, pf_q, enable);
if (!enable) if (!enable)
usleep_range(10, 20); usleep_range(10, 20);
for (j = 0; j < 50; j++) { for (i = 0; i < I40E_QTX_ENA_WAIT_COUNT; i++) {
tx_reg = rd32(hw, I40E_QTX_ENA(pf_q)); tx_reg = rd32(hw, I40E_QTX_ENA(pf_q));
if (((tx_reg >> I40E_QTX_ENA_QENA_REQ_SHIFT) & 1) == if (((tx_reg >> I40E_QTX_ENA_QENA_REQ_SHIFT) & 1) ==
((tx_reg >> I40E_QTX_ENA_QENA_STAT_SHIFT) & 1)) ((tx_reg >> I40E_QTX_ENA_QENA_STAT_SHIFT) & 1))
break; break;
usleep_range(1000, 2000); usleep_range(1000, 2000);
} }
/* Skip if the queue is already in the requested state */ /* Skip if the queue is already in the requested state */
if (enable == !!(tx_reg & I40E_QTX_ENA_QENA_STAT_MASK)) if (enable == !!(tx_reg & I40E_QTX_ENA_QENA_STAT_MASK))
continue; return;
/* turn on/off the queue */ /* turn on/off the queue */
if (enable) { if (enable) {
...@@ -3988,7 +3992,23 @@ static int i40e_vsi_control_tx(struct i40e_vsi *vsi, bool enable) ...@@ -3988,7 +3992,23 @@ static int i40e_vsi_control_tx(struct i40e_vsi *vsi, bool enable)
} }
wr32(hw, I40E_QTX_ENA(pf_q), tx_reg); wr32(hw, I40E_QTX_ENA(pf_q), tx_reg);
/* No waiting for the Tx queue to disable */ }
/**
* i40e_vsi_control_tx - Start or stop a VSI's rings
* @vsi: the VSI being configured
* @enable: start or stop the rings
**/
static int i40e_vsi_control_tx(struct i40e_vsi *vsi, bool enable)
{
struct i40e_pf *pf = vsi->back;
int i, pf_q, ret = 0;
pf_q = vsi->base_queue;
for (i = 0; i < vsi->num_queue_pairs; i++, pf_q++) {
i40e_control_tx_q(pf, pf_q, enable);
/* Don't wait to disable when port Tx is suspended */
if (!enable && test_bit(__I40E_PORT_TX_SUSPENDED, &pf->state)) if (!enable && test_bit(__I40E_PORT_TX_SUSPENDED, &pf->state))
continue; continue;
...@@ -4035,20 +4055,22 @@ static int i40e_pf_rxq_wait(struct i40e_pf *pf, int pf_q, bool enable) ...@@ -4035,20 +4055,22 @@ static int i40e_pf_rxq_wait(struct i40e_pf *pf, int pf_q, bool enable)
} }
/** /**
* i40e_vsi_control_rx - Start or stop a VSI's rings * i40e_control_rx_q - Start or stop a particular Rx queue
* @vsi: the VSI being configured * @pf: the PF structure
* @enable: start or stop the rings * @pf_q: the PF queue to configure
* @enable: start or stop the queue
*
* This function enables or disables a single queue. Note that any delay
* required after the operation is expected to be handled by the caller of
* this function.
**/ **/
static int i40e_vsi_control_rx(struct i40e_vsi *vsi, bool enable) static void i40e_control_rx_q(struct i40e_pf *pf, int pf_q, bool enable)
{ {
struct i40e_pf *pf = vsi->back;
struct i40e_hw *hw = &pf->hw; struct i40e_hw *hw = &pf->hw;
int i, j, pf_q, ret = 0;
u32 rx_reg; u32 rx_reg;
int i;
pf_q = vsi->base_queue; for (i = 0; i < I40E_QTX_ENA_WAIT_COUNT; i++) {
for (i = 0; i < vsi->num_queue_pairs; i++, pf_q++) {
for (j = 0; j < 50; j++) {
rx_reg = rd32(hw, I40E_QRX_ENA(pf_q)); rx_reg = rd32(hw, I40E_QRX_ENA(pf_q));
if (((rx_reg >> I40E_QRX_ENA_QENA_REQ_SHIFT) & 1) == if (((rx_reg >> I40E_QRX_ENA_QENA_REQ_SHIFT) & 1) ==
((rx_reg >> I40E_QRX_ENA_QENA_STAT_SHIFT) & 1)) ((rx_reg >> I40E_QRX_ENA_QENA_STAT_SHIFT) & 1))
...@@ -4058,15 +4080,32 @@ static int i40e_vsi_control_rx(struct i40e_vsi *vsi, bool enable) ...@@ -4058,15 +4080,32 @@ static int i40e_vsi_control_rx(struct i40e_vsi *vsi, bool enable)
/* Skip if the queue is already in the requested state */ /* Skip if the queue is already in the requested state */
if (enable == !!(rx_reg & I40E_QRX_ENA_QENA_STAT_MASK)) if (enable == !!(rx_reg & I40E_QRX_ENA_QENA_STAT_MASK))
continue; return;
/* turn on/off the queue */ /* turn on/off the queue */
if (enable) if (enable)
rx_reg |= I40E_QRX_ENA_QENA_REQ_MASK; rx_reg |= I40E_QRX_ENA_QENA_REQ_MASK;
else else
rx_reg &= ~I40E_QRX_ENA_QENA_REQ_MASK; rx_reg &= ~I40E_QRX_ENA_QENA_REQ_MASK;
wr32(hw, I40E_QRX_ENA(pf_q), rx_reg); wr32(hw, I40E_QRX_ENA(pf_q), rx_reg);
/* No waiting for the Tx queue to disable */ }
/**
* i40e_vsi_control_rx - Start or stop a VSI's rings
* @vsi: the VSI being configured
* @enable: start or stop the rings
**/
static int i40e_vsi_control_rx(struct i40e_vsi *vsi, bool enable)
{
struct i40e_pf *pf = vsi->back;
int i, pf_q, ret = 0;
pf_q = vsi->base_queue;
for (i = 0; i < vsi->num_queue_pairs; i++, pf_q++) {
i40e_control_rx_q(pf, pf_q, enable);
/* Don't wait to disable when port Tx is suspended */
if (!enable && test_bit(__I40E_PORT_TX_SUSPENDED, &pf->state)) if (!enable && test_bit(__I40E_PORT_TX_SUSPENDED, &pf->state))
continue; continue;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment