i40e: Set XPS bit mask to zero in DCB mode
authorNeerav Parikh <neerav.parikh@intel.com>
Wed, 12 Nov 2014 00:19:02 +0000 (00:19 +0000)
committerJeff Kirsher <jeffrey.t.kirsher@intel.com>
Tue, 18 Nov 2014 09:09:35 +0000 (01:09 -0800)
Due to DCBX configuration change if the VSI needs to use more than 1 TC;
it needs to disable the XPS maps that were set when operating in 1 TC mode.
Without disabling XPS the netdev layer will select queues based on those
settings and not use the TC queue mapping to make the queue selection.

This patch allows the driver to enable/disable the XPS based on the number
of TCs being enabled for the given VSI.

Change-ID: Idc4dec47a672d2a509f6d7fe11ed1ee65b4f0e08
Signed-off-by: Neerav Parikh <neerav.parikh@intel.com>
Tested-By: Jack Morgan <jack.morgan@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
drivers/net/ethernet/intel/i40e/i40e_main.c

index c3107dca24da38eb2c4ad1dc80a40d3fa75f6aef..c998d82da0fc0986a01635332ffa2d5561dce7b4 100644 (file)
@@ -2381,6 +2381,35 @@ static void i40e_vsi_free_rx_resources(struct i40e_vsi *vsi)
 #endif
 }
 
+/**
+ * i40e_config_xps_tx_ring - Configure XPS for a Tx ring
+ * @ring: The Tx ring to configure
+ *
+ * This enables/disables XPS for a given Tx descriptor ring
+ * based on the TCs enabled for the VSI that ring belongs to.
+ **/
+static void i40e_config_xps_tx_ring(struct i40e_ring *ring)
+{
+       struct i40e_vsi *vsi = ring->vsi;
+       cpumask_var_t mask;
+
+       if (ring->q_vector && ring->netdev) {
+               /* Single TC mode enable XPS */
+               if (vsi->tc_config.numtc <= 1 &&
+                   !test_and_set_bit(__I40E_TX_XPS_INIT_DONE, &ring->state)) {
+                       netif_set_xps_queue(ring->netdev,
+                                           &ring->q_vector->affinity_mask,
+                                           ring->queue_index);
+               } else if (alloc_cpumask_var(&mask, GFP_KERNEL)) {
+                       /* Disable XPS to allow selection based on TC */
+                       bitmap_zero(cpumask_bits(mask), nr_cpumask_bits);
+                       netif_set_xps_queue(ring->netdev, mask,
+                                           ring->queue_index);
+                       free_cpumask_var(mask);
+               }
+       }
+}
+
 /**
  * i40e_configure_tx_ring - Configure a transmit ring context and rest
  * @ring: The Tx ring to configure
@@ -2404,13 +2433,8 @@ static int i40e_configure_tx_ring(struct i40e_ring *ring)
                ring->atr_sample_rate = 0;
        }
 
-       /* initialize XPS */
-       if (ring->q_vector && ring->netdev &&
-           vsi->tc_config.numtc <= 1 &&
-           !test_and_set_bit(__I40E_TX_XPS_INIT_DONE, &ring->state))
-               netif_set_xps_queue(ring->netdev,
-                                   &ring->q_vector->affinity_mask,
-                                   ring->queue_index);
+       /* configure XPS */
+       i40e_config_xps_tx_ring(ring);
 
        /* clear the context structure first */
        memset(&tx_ctx, 0, sizeof(tx_ctx));