Commit 37781fd2 authored by David S. Miller's avatar David S. Miller

Merge branch 'gve-fixes'

David Awogbemila says:

====================
GVE bug fixes

This patch series includes fixes to some bugs in the gve driver.
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents 1d482e66 fbd4a28b
...@@ -180,7 +180,7 @@ static int gve_napi_poll(struct napi_struct *napi, int budget) ...@@ -180,7 +180,7 @@ static int gve_napi_poll(struct napi_struct *napi, int budget)
/* Double check we have no extra work. /* Double check we have no extra work.
* Ensure unmask synchronizes with checking for work. * Ensure unmask synchronizes with checking for work.
*/ */
dma_rmb(); mb();
if (block->tx) if (block->tx)
reschedule |= gve_tx_poll(block, -1); reschedule |= gve_tx_poll(block, -1);
if (block->rx) if (block->rx)
...@@ -220,6 +220,7 @@ static int gve_alloc_notify_blocks(struct gve_priv *priv) ...@@ -220,6 +220,7 @@ static int gve_alloc_notify_blocks(struct gve_priv *priv)
int vecs_left = new_num_ntfy_blks % 2; int vecs_left = new_num_ntfy_blks % 2;
priv->num_ntfy_blks = new_num_ntfy_blks; priv->num_ntfy_blks = new_num_ntfy_blks;
priv->mgmt_msix_idx = priv->num_ntfy_blks;
priv->tx_cfg.max_queues = min_t(int, priv->tx_cfg.max_queues, priv->tx_cfg.max_queues = min_t(int, priv->tx_cfg.max_queues,
vecs_per_type); vecs_per_type);
priv->rx_cfg.max_queues = min_t(int, priv->rx_cfg.max_queues, priv->rx_cfg.max_queues = min_t(int, priv->rx_cfg.max_queues,
...@@ -300,20 +301,22 @@ static void gve_free_notify_blocks(struct gve_priv *priv) ...@@ -300,20 +301,22 @@ static void gve_free_notify_blocks(struct gve_priv *priv)
{ {
int i; int i;
/* Free the irqs */ if (priv->msix_vectors) {
for (i = 0; i < priv->num_ntfy_blks; i++) { /* Free the irqs */
struct gve_notify_block *block = &priv->ntfy_blocks[i]; for (i = 0; i < priv->num_ntfy_blks; i++) {
int msix_idx = i; struct gve_notify_block *block = &priv->ntfy_blocks[i];
int msix_idx = i;
irq_set_affinity_hint(priv->msix_vectors[msix_idx].vector, irq_set_affinity_hint(priv->msix_vectors[msix_idx].vector,
NULL); NULL);
free_irq(priv->msix_vectors[msix_idx].vector, block); free_irq(priv->msix_vectors[msix_idx].vector, block);
}
free_irq(priv->msix_vectors[priv->mgmt_msix_idx].vector, priv);
} }
dma_free_coherent(&priv->pdev->dev, dma_free_coherent(&priv->pdev->dev,
priv->num_ntfy_blks * sizeof(*priv->ntfy_blocks), priv->num_ntfy_blks * sizeof(*priv->ntfy_blocks),
priv->ntfy_blocks, priv->ntfy_block_bus); priv->ntfy_blocks, priv->ntfy_block_bus);
priv->ntfy_blocks = NULL; priv->ntfy_blocks = NULL;
free_irq(priv->msix_vectors[priv->mgmt_msix_idx].vector, priv);
pci_disable_msix(priv->pdev); pci_disable_msix(priv->pdev);
kvfree(priv->msix_vectors); kvfree(priv->msix_vectors);
priv->msix_vectors = NULL; priv->msix_vectors = NULL;
......
...@@ -212,10 +212,11 @@ static int gve_tx_alloc_ring(struct gve_priv *priv, int idx) ...@@ -212,10 +212,11 @@ static int gve_tx_alloc_ring(struct gve_priv *priv, int idx)
tx->dev = &priv->pdev->dev; tx->dev = &priv->pdev->dev;
if (!tx->raw_addressing) { if (!tx->raw_addressing) {
tx->tx_fifo.qpl = gve_assign_tx_qpl(priv); tx->tx_fifo.qpl = gve_assign_tx_qpl(priv);
if (!tx->tx_fifo.qpl)
goto abort_with_desc;
/* map Tx FIFO */ /* map Tx FIFO */
if (gve_tx_fifo_init(priv, &tx->tx_fifo)) if (gve_tx_fifo_init(priv, &tx->tx_fifo))
goto abort_with_desc; goto abort_with_qpl;
} }
tx->q_resources = tx->q_resources =
...@@ -236,6 +237,9 @@ static int gve_tx_alloc_ring(struct gve_priv *priv, int idx) ...@@ -236,6 +237,9 @@ static int gve_tx_alloc_ring(struct gve_priv *priv, int idx)
abort_with_fifo: abort_with_fifo:
if (!tx->raw_addressing) if (!tx->raw_addressing)
gve_tx_fifo_release(priv, &tx->tx_fifo); gve_tx_fifo_release(priv, &tx->tx_fifo);
abort_with_qpl:
if (!tx->raw_addressing)
gve_unassign_qpl(priv, tx->tx_fifo.qpl->id);
abort_with_desc: abort_with_desc:
dma_free_coherent(hdev, bytes, tx->desc, tx->bus); dma_free_coherent(hdev, bytes, tx->desc, tx->bus);
tx->desc = NULL; tx->desc = NULL;
...@@ -589,7 +593,7 @@ netdev_tx_t gve_tx(struct sk_buff *skb, struct net_device *dev) ...@@ -589,7 +593,7 @@ netdev_tx_t gve_tx(struct sk_buff *skb, struct net_device *dev)
struct gve_tx_ring *tx; struct gve_tx_ring *tx;
int nsegs; int nsegs;
WARN(skb_get_queue_mapping(skb) > priv->tx_cfg.num_queues, WARN(skb_get_queue_mapping(skb) >= priv->tx_cfg.num_queues,
"skb queue index out of range"); "skb queue index out of range");
tx = &priv->tx[skb_get_queue_mapping(skb)]; tx = &priv->tx[skb_get_queue_mapping(skb)];
if (unlikely(gve_maybe_stop_tx(tx, skb))) { if (unlikely(gve_maybe_stop_tx(tx, skb))) {
......
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