Commit 5536d210 authored by Alexander Duyck's avatar Alexander Duyck Committed by Jeff Kirsher

igb: Combine q_vector and ring allocation into a single function

This change combines the the allocation of q_vectors and rings into a single
function.  The advantage of this is that we are guaranteed we will avoid
overlap in the L1 cache sets.
Signed-off-by: default avatarAlexander Duyck <alexander.h.duyck@intel.com>
Tested-by: default avatarAaron Brown <aaron.f.brown@intel.com>
Signed-off-by: default avatarJeff Kirsher <jeffrey.t.kirsher@intel.com>
parent de78d1f9
...@@ -204,22 +204,6 @@ struct igb_ring_container { ...@@ -204,22 +204,6 @@ struct igb_ring_container {
u8 itr; /* current ITR setting for ring */ u8 itr; /* current ITR setting for ring */
}; };
struct igb_q_vector {
struct igb_adapter *adapter; /* backlink */
int cpu; /* CPU for DCA */
u32 eims_value; /* EIMS mask value */
struct igb_ring_container rx, tx;
struct napi_struct napi;
u16 itr_val;
u8 set_itr;
void __iomem *itr_register;
char name[IFNAMSIZ + 9];
};
struct igb_ring { struct igb_ring {
struct igb_q_vector *q_vector; /* backlink to q_vector */ struct igb_q_vector *q_vector; /* backlink to q_vector */
struct net_device *netdev; /* back pointer to net_device */ struct net_device *netdev; /* back pointer to net_device */
...@@ -231,14 +215,15 @@ struct igb_ring { ...@@ -231,14 +215,15 @@ struct igb_ring {
void *desc; /* descriptor ring memory */ void *desc; /* descriptor ring memory */
unsigned long flags; /* ring specific flags */ unsigned long flags; /* ring specific flags */
void __iomem *tail; /* pointer to ring tail register */ void __iomem *tail; /* pointer to ring tail register */
dma_addr_t dma; /* phys address of the ring */
unsigned int size; /* length of desc. ring in bytes */
u16 count; /* number of desc. in the ring */ u16 count; /* number of desc. in the ring */
u8 queue_index; /* logical index of the ring*/ u8 queue_index; /* logical index of the ring*/
u8 reg_idx; /* physical index of the ring */ u8 reg_idx; /* physical index of the ring */
u32 size; /* length of desc. ring in bytes */
/* everything past this point are written often */ /* everything past this point are written often */
u16 next_to_clean ____cacheline_aligned_in_smp; u16 next_to_clean;
u16 next_to_use; u16 next_to_use;
u16 next_to_alloc; u16 next_to_alloc;
...@@ -256,8 +241,25 @@ struct igb_ring { ...@@ -256,8 +241,25 @@ struct igb_ring {
struct u64_stats_sync rx_syncp; struct u64_stats_sync rx_syncp;
}; };
}; };
/* Items past this point are only used during ring alloc / free */ } ____cacheline_internodealigned_in_smp;
dma_addr_t dma; /* phys address of the ring */
struct igb_q_vector {
struct igb_adapter *adapter; /* backlink */
int cpu; /* CPU for DCA */
u32 eims_value; /* EIMS mask value */
u16 itr_val;
u8 set_itr;
void __iomem *itr_register;
struct igb_ring_container rx, tx;
struct napi_struct napi;
struct rcu_head rcu; /* to avoid race with update stats on free */
char name[IFNAMSIZ + 9];
/* for dynamic allocation of rings associated with this q_vector */
struct igb_ring ring[0] ____cacheline_internodealigned_in_smp;
}; };
enum e1000_ring_flags_t { enum e1000_ring_flags_t {
......
...@@ -652,80 +652,6 @@ static void igb_cache_ring_register(struct igb_adapter *adapter) ...@@ -652,80 +652,6 @@ static void igb_cache_ring_register(struct igb_adapter *adapter)
} }
} }
static void igb_free_queues(struct igb_adapter *adapter)
{
int i;
for (i = 0; i < adapter->num_tx_queues; i++) {
kfree(adapter->tx_ring[i]);
adapter->tx_ring[i] = NULL;
}
for (i = 0; i < adapter->num_rx_queues; i++) {
kfree(adapter->rx_ring[i]);
adapter->rx_ring[i] = NULL;
}
adapter->num_rx_queues = 0;
adapter->num_tx_queues = 0;
}
/**
* igb_alloc_queues - Allocate memory for all rings
* @adapter: board private structure to initialize
*
* We allocate one ring per queue at run-time since we don't know the
* number of queues at compile-time.
**/
static int igb_alloc_queues(struct igb_adapter *adapter)
{
struct igb_ring *ring;
int i;
for (i = 0; i < adapter->num_tx_queues; i++) {
ring = kzalloc(sizeof(struct igb_ring), GFP_KERNEL);
if (!ring)
goto err;
ring->count = adapter->tx_ring_count;
ring->queue_index = i;
ring->dev = &adapter->pdev->dev;
ring->netdev = adapter->netdev;
/* For 82575, context index must be unique per ring. */
if (adapter->hw.mac.type == e1000_82575)
set_bit(IGB_RING_FLAG_TX_CTX_IDX, &ring->flags);
adapter->tx_ring[i] = ring;
}
for (i = 0; i < adapter->num_rx_queues; i++) {
ring = kzalloc(sizeof(struct igb_ring), GFP_KERNEL);
if (!ring)
goto err;
ring->count = adapter->rx_ring_count;
ring->queue_index = i;
ring->dev = &adapter->pdev->dev;
ring->netdev = adapter->netdev;
/* set flag indicating ring supports SCTP checksum offload */
if (adapter->hw.mac.type >= e1000_82576)
set_bit(IGB_RING_FLAG_RX_SCTP_CSUM, &ring->flags);
/*
* On i350, i210, and i211, loopback VLAN packets
* have the tag byte-swapped.
* */
if (adapter->hw.mac.type >= e1000_i350)
set_bit(IGB_RING_FLAG_RX_LB_VLAN_BSWAP, &ring->flags);
adapter->rx_ring[i] = ring;
}
igb_cache_ring_register(adapter);
return 0;
err:
igb_free_queues(adapter);
return -ENOMEM;
}
/** /**
* igb_write_ivar - configure ivar for given MSI-X vector * igb_write_ivar - configure ivar for given MSI-X vector
* @hw: pointer to the HW structure * @hw: pointer to the HW structure
...@@ -955,6 +881,35 @@ static void igb_reset_interrupt_capability(struct igb_adapter *adapter) ...@@ -955,6 +881,35 @@ static void igb_reset_interrupt_capability(struct igb_adapter *adapter)
} }
} }
/**
* igb_free_q_vector - Free memory allocated for specific interrupt vector
* @adapter: board private structure to initialize
* @v_idx: Index of vector to be freed
*
* This function frees the memory allocated to the q_vector. In addition if
* NAPI is enabled it will delete any references to the NAPI struct prior
* to freeing the q_vector.
**/
static void igb_free_q_vector(struct igb_adapter *adapter, int v_idx)
{
struct igb_q_vector *q_vector = adapter->q_vector[v_idx];
if (q_vector->tx.ring)
adapter->tx_ring[q_vector->tx.ring->queue_index] = NULL;
if (q_vector->rx.ring)
adapter->tx_ring[q_vector->rx.ring->queue_index] = NULL;
adapter->q_vector[v_idx] = NULL;
netif_napi_del(&q_vector->napi);
/*
* ixgbe_get_stats64() might access the rings on this vector,
* we must wait a grace period before freeing it.
*/
kfree_rcu(q_vector, rcu);
}
/** /**
* igb_free_q_vectors - Free memory allocated for interrupt vectors * igb_free_q_vectors - Free memory allocated for interrupt vectors
* @adapter: board private structure to initialize * @adapter: board private structure to initialize
...@@ -965,17 +920,14 @@ static void igb_reset_interrupt_capability(struct igb_adapter *adapter) ...@@ -965,17 +920,14 @@ static void igb_reset_interrupt_capability(struct igb_adapter *adapter)
**/ **/
static void igb_free_q_vectors(struct igb_adapter *adapter) static void igb_free_q_vectors(struct igb_adapter *adapter)
{ {
int v_idx; int v_idx = adapter->num_q_vectors;
for (v_idx = 0; v_idx < adapter->num_q_vectors; v_idx++) { adapter->num_tx_queues = 0;
struct igb_q_vector *q_vector = adapter->q_vector[v_idx]; adapter->num_rx_queues = 0;
adapter->q_vector[v_idx] = NULL;
if (!q_vector)
continue;
netif_napi_del(&q_vector->napi);
kfree(q_vector);
}
adapter->num_q_vectors = 0; adapter->num_q_vectors = 0;
while (v_idx--)
igb_free_q_vector(adapter, v_idx);
} }
/** /**
...@@ -986,7 +938,6 @@ static void igb_free_q_vectors(struct igb_adapter *adapter) ...@@ -986,7 +938,6 @@ static void igb_free_q_vectors(struct igb_adapter *adapter)
*/ */
static void igb_clear_interrupt_scheme(struct igb_adapter *adapter) static void igb_clear_interrupt_scheme(struct igb_adapter *adapter)
{ {
igb_free_queues(adapter);
igb_free_q_vectors(adapter); igb_free_q_vectors(adapter);
igb_reset_interrupt_capability(adapter); igb_reset_interrupt_capability(adapter);
} }
...@@ -1074,95 +1025,181 @@ static int igb_set_interrupt_capability(struct igb_adapter *adapter) ...@@ -1074,95 +1025,181 @@ static int igb_set_interrupt_capability(struct igb_adapter *adapter)
return err; return err;
} }
static void igb_add_ring(struct igb_ring *ring,
struct igb_ring_container *head)
{
head->ring = ring;
head->count++;
}
/** /**
* igb_alloc_q_vectors - Allocate memory for interrupt vectors * igb_alloc_q_vector - Allocate memory for a single interrupt vector
* @adapter: board private structure to initialize * @adapter: board private structure to initialize
* @v_count: q_vectors allocated on adapter, used for ring interleaving
* @v_idx: index of vector in adapter struct
* @txr_count: total number of Tx rings to allocate
* @txr_idx: index of first Tx ring to allocate
* @rxr_count: total number of Rx rings to allocate
* @rxr_idx: index of first Rx ring to allocate
* *
* We allocate one q_vector per queue interrupt. If allocation fails we * We allocate one q_vector. If allocation fails we return -ENOMEM.
* return -ENOMEM.
**/ **/
static int igb_alloc_q_vectors(struct igb_adapter *adapter) static int igb_alloc_q_vector(struct igb_adapter *adapter,
int v_count, int v_idx,
int txr_count, int txr_idx,
int rxr_count, int rxr_idx)
{ {
struct igb_q_vector *q_vector; struct igb_q_vector *q_vector;
struct e1000_hw *hw = &adapter->hw; struct igb_ring *ring;
int v_idx; int ring_count, size;
for (v_idx = 0; v_idx < adapter->num_q_vectors; v_idx++) { /* igb only supports 1 Tx and/or 1 Rx queue per vector */
q_vector = kzalloc(sizeof(struct igb_q_vector), if (txr_count > 1 || rxr_count > 1)
GFP_KERNEL); return -ENOMEM;
if (!q_vector)
goto err_out; ring_count = txr_count + rxr_count;
q_vector->adapter = adapter; size = sizeof(struct igb_q_vector) +
q_vector->itr_register = hw->hw_addr + E1000_EITR(0); (sizeof(struct igb_ring) * ring_count);
q_vector->itr_val = IGB_START_ITR;
netif_napi_add(adapter->netdev, &q_vector->napi, igb_poll, 64); /* allocate q_vector and rings */
adapter->q_vector[v_idx] = q_vector; q_vector = kzalloc(size, GFP_KERNEL);
if (!q_vector)
return -ENOMEM;
/* initialize NAPI */
netif_napi_add(adapter->netdev, &q_vector->napi,
igb_poll, 64);
/* tie q_vector and adapter together */
adapter->q_vector[v_idx] = q_vector;
q_vector->adapter = adapter;
/* initialize work limits */
q_vector->tx.work_limit = adapter->tx_work_limit;
/* initialize ITR configuration */
q_vector->itr_register = adapter->hw.hw_addr + E1000_EITR(0);
q_vector->itr_val = IGB_START_ITR;
/* initialize pointer to rings */
ring = q_vector->ring;
if (txr_count) {
/* assign generic ring traits */
ring->dev = &adapter->pdev->dev;
ring->netdev = adapter->netdev;
/* configure backlink on ring */
ring->q_vector = q_vector;
/* update q_vector Tx values */
igb_add_ring(ring, &q_vector->tx);
/* For 82575, context index must be unique per ring. */
if (adapter->hw.mac.type == e1000_82575)
set_bit(IGB_RING_FLAG_TX_CTX_IDX, &ring->flags);
/* apply Tx specific ring traits */
ring->count = adapter->tx_ring_count;
ring->queue_index = txr_idx;
/* assign ring to adapter */
adapter->tx_ring[txr_idx] = ring;
/* push pointer to next ring */
ring++;
} }
return 0; if (rxr_count) {
/* assign generic ring traits */
ring->dev = &adapter->pdev->dev;
ring->netdev = adapter->netdev;
err_out: /* configure backlink on ring */
igb_free_q_vectors(adapter); ring->q_vector = q_vector;
return -ENOMEM;
}
static void igb_map_rx_ring_to_vector(struct igb_adapter *adapter, /* update q_vector Rx values */
int ring_idx, int v_idx) igb_add_ring(ring, &q_vector->rx);
{
struct igb_q_vector *q_vector = adapter->q_vector[v_idx];
q_vector->rx.ring = adapter->rx_ring[ring_idx]; /* set flag indicating ring supports SCTP checksum offload */
q_vector->rx.ring->q_vector = q_vector; if (adapter->hw.mac.type >= e1000_82576)
q_vector->rx.count++; set_bit(IGB_RING_FLAG_RX_SCTP_CSUM, &ring->flags);
q_vector->itr_val = adapter->rx_itr_setting;
if (q_vector->itr_val && q_vector->itr_val <= 3)
q_vector->itr_val = IGB_START_ITR;
}
static void igb_map_tx_ring_to_vector(struct igb_adapter *adapter, /*
int ring_idx, int v_idx) * On i350, i210, and i211, loopback VLAN packets
{ * have the tag byte-swapped.
struct igb_q_vector *q_vector = adapter->q_vector[v_idx]; * */
if (adapter->hw.mac.type >= e1000_i350)
set_bit(IGB_RING_FLAG_RX_LB_VLAN_BSWAP, &ring->flags);
q_vector->tx.ring = adapter->tx_ring[ring_idx]; /* apply Rx specific ring traits */
q_vector->tx.ring->q_vector = q_vector; ring->count = adapter->rx_ring_count;
q_vector->tx.count++; ring->queue_index = rxr_idx;
q_vector->itr_val = adapter->tx_itr_setting;
q_vector->tx.work_limit = adapter->tx_work_limit; /* assign ring to adapter */
if (q_vector->itr_val && q_vector->itr_val <= 3) adapter->rx_ring[rxr_idx] = ring;
q_vector->itr_val = IGB_START_ITR; }
return 0;
} }
/** /**
* igb_map_ring_to_vector - maps allocated queues to vectors * igb_alloc_q_vectors - Allocate memory for interrupt vectors
* @adapter: board private structure to initialize
* *
* This function maps the recently allocated queues to vectors. * We allocate one q_vector per queue interrupt. If allocation fails we
* return -ENOMEM.
**/ **/
static int igb_map_ring_to_vector(struct igb_adapter *adapter) static int igb_alloc_q_vectors(struct igb_adapter *adapter)
{ {
int i; int q_vectors = adapter->num_q_vectors;
int v_idx = 0; int rxr_remaining = adapter->num_rx_queues;
int txr_remaining = adapter->num_tx_queues;
int rxr_idx = 0, txr_idx = 0, v_idx = 0;
int err;
if ((adapter->num_q_vectors < adapter->num_rx_queues) || if (q_vectors >= (rxr_remaining + txr_remaining)) {
(adapter->num_q_vectors < adapter->num_tx_queues)) for (; rxr_remaining; v_idx++) {
return -ENOMEM; err = igb_alloc_q_vector(adapter, q_vectors, v_idx,
0, 0, 1, rxr_idx);
if (adapter->num_q_vectors >= if (err)
(adapter->num_rx_queues + adapter->num_tx_queues)) { goto err_out;
for (i = 0; i < adapter->num_rx_queues; i++)
igb_map_rx_ring_to_vector(adapter, i, v_idx++); /* update counts and index */
for (i = 0; i < adapter->num_tx_queues; i++) rxr_remaining--;
igb_map_tx_ring_to_vector(adapter, i, v_idx++); rxr_idx++;
} else {
for (i = 0; i < adapter->num_rx_queues; i++) {
if (i < adapter->num_tx_queues)
igb_map_tx_ring_to_vector(adapter, i, v_idx);
igb_map_rx_ring_to_vector(adapter, i, v_idx++);
} }
for (; i < adapter->num_tx_queues; i++)
igb_map_tx_ring_to_vector(adapter, i, v_idx++);
} }
for (; v_idx < q_vectors; v_idx++) {
int rqpv = DIV_ROUND_UP(rxr_remaining, q_vectors - v_idx);
int tqpv = DIV_ROUND_UP(txr_remaining, q_vectors - v_idx);
err = igb_alloc_q_vector(adapter, q_vectors, v_idx,
tqpv, txr_idx, rqpv, rxr_idx);
if (err)
goto err_out;
/* update counts and index */
rxr_remaining -= rqpv;
txr_remaining -= tqpv;
rxr_idx++;
txr_idx++;
}
return 0; return 0;
err_out:
adapter->num_tx_queues = 0;
adapter->num_rx_queues = 0;
adapter->num_q_vectors = 0;
while (v_idx--)
igb_free_q_vector(adapter, v_idx);
return -ENOMEM;
} }
/** /**
...@@ -1185,24 +1222,10 @@ static int igb_init_interrupt_scheme(struct igb_adapter *adapter) ...@@ -1185,24 +1222,10 @@ static int igb_init_interrupt_scheme(struct igb_adapter *adapter)
goto err_alloc_q_vectors; goto err_alloc_q_vectors;
} }
err = igb_alloc_queues(adapter); igb_cache_ring_register(adapter);
if (err) {
dev_err(&pdev->dev, "Unable to allocate memory for queues\n");
goto err_alloc_queues;
}
err = igb_map_ring_to_vector(adapter);
if (err) {
dev_err(&pdev->dev, "Invalid q_vector to ring mapping\n");
goto err_map_queues;
}
return 0; return 0;
err_map_queues:
igb_free_queues(adapter);
err_alloc_queues:
igb_free_q_vectors(adapter);
err_alloc_q_vectors: err_alloc_q_vectors:
igb_reset_interrupt_capability(adapter); igb_reset_interrupt_capability(adapter);
return err; return err;
...@@ -1225,11 +1248,11 @@ static int igb_request_irq(struct igb_adapter *adapter) ...@@ -1225,11 +1248,11 @@ static int igb_request_irq(struct igb_adapter *adapter)
if (!err) if (!err)
goto request_done; goto request_done;
/* fall back to MSI */ /* fall back to MSI */
igb_free_all_tx_resources(adapter);
igb_free_all_rx_resources(adapter);
igb_clear_interrupt_scheme(adapter); igb_clear_interrupt_scheme(adapter);
if (!pci_enable_msi(pdev)) if (!pci_enable_msi(pdev))
adapter->flags |= IGB_FLAG_HAS_MSI; adapter->flags |= IGB_FLAG_HAS_MSI;
igb_free_all_tx_resources(adapter);
igb_free_all_rx_resources(adapter);
adapter->num_tx_queues = 1; adapter->num_tx_queues = 1;
adapter->num_rx_queues = 1; adapter->num_rx_queues = 1;
adapter->num_q_vectors = 1; adapter->num_q_vectors = 1;
...@@ -1239,13 +1262,6 @@ static int igb_request_irq(struct igb_adapter *adapter) ...@@ -1239,13 +1262,6 @@ static int igb_request_irq(struct igb_adapter *adapter)
"Unable to allocate memory for vectors\n"); "Unable to allocate memory for vectors\n");
goto request_done; goto request_done;
} }
err = igb_alloc_queues(adapter);
if (err) {
dev_err(&pdev->dev,
"Unable to allocate memory for queues\n");
igb_free_q_vectors(adapter);
goto request_done;
}
igb_setup_all_tx_resources(adapter); igb_setup_all_tx_resources(adapter);
igb_setup_all_rx_resources(adapter); igb_setup_all_rx_resources(adapter);
} }
...@@ -2633,10 +2649,8 @@ int igb_setup_tx_resources(struct igb_ring *tx_ring) ...@@ -2633,10 +2649,8 @@ int igb_setup_tx_resources(struct igb_ring *tx_ring)
tx_ring->size = tx_ring->count * sizeof(union e1000_adv_tx_desc); tx_ring->size = tx_ring->count * sizeof(union e1000_adv_tx_desc);
tx_ring->size = ALIGN(tx_ring->size, 4096); tx_ring->size = ALIGN(tx_ring->size, 4096);
tx_ring->desc = dma_alloc_coherent(dev, tx_ring->desc = dma_alloc_coherent(dev, tx_ring->size,
tx_ring->size, &tx_ring->dma, GFP_KERNEL);
&tx_ring->dma,
GFP_KERNEL);
if (!tx_ring->desc) if (!tx_ring->desc)
goto err; goto err;
...@@ -2773,15 +2787,12 @@ int igb_setup_rx_resources(struct igb_ring *rx_ring) ...@@ -2773,15 +2787,12 @@ int igb_setup_rx_resources(struct igb_ring *rx_ring)
if (!rx_ring->rx_buffer_info) if (!rx_ring->rx_buffer_info)
goto err; goto err;
/* Round up to nearest 4K */ /* Round up to nearest 4K */
rx_ring->size = rx_ring->count * sizeof(union e1000_adv_rx_desc); rx_ring->size = rx_ring->count * sizeof(union e1000_adv_rx_desc);
rx_ring->size = ALIGN(rx_ring->size, 4096); rx_ring->size = ALIGN(rx_ring->size, 4096);
rx_ring->desc = dma_alloc_coherent(dev, rx_ring->desc = dma_alloc_coherent(dev, rx_ring->size,
rx_ring->size, &rx_ring->dma, GFP_KERNEL);
&rx_ring->dma,
GFP_KERNEL);
if (!rx_ring->desc) if (!rx_ring->desc)
goto err; goto err;
......
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