alx: add ability to allocate and free alx_napi structures
authorTobias Regnery <tobias.regnery@gmail.com>
Tue, 15 Nov 2016 11:43:10 +0000 (12:43 +0100)
committerDavid S. Miller <davem@davemloft.net>
Wed, 16 Nov 2016 03:46:30 +0000 (22:46 -0500)
Add new functions to allocate and free the alx_napi structures and use them
in __alx_open and __alx_stop. We only allocate one of these structures for
now, as the rest of the driver is not yet ready for multiple queues.

We switch over the setup of the interrupt mask and the call to netif_napi_add
to the new function because we must adjust these later on a per queue basis.

Based on the downstream driver at github.com/qca/alx

Signed-off-by: Tobias Regnery <tobias.regnery@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/atheros/alx/main.c

index 8935766829b1aa5feb59a6c23ba3dfe7e3e2a4d2..193da6799979f0e4a6f9f260939a50b9fbe4d757 100644 (file)
@@ -632,45 +632,96 @@ static int alx_alloc_rings(struct alx_priv *alx)
        offset = alx_alloc_tx_ring(alx, &alx->txq, offset);
        if (offset < 0) {
                netdev_err(alx->dev, "Allocation of tx buffer failed!\n");
-               goto out_free;
+               return -ENOMEM;
        }
 
        offset = alx_alloc_rx_ring(alx, &alx->rxq, offset);
        if (offset < 0) {
                netdev_err(alx->dev, "Allocation of rx buffer failed!\n");
-               goto out_free;
+               return -ENOMEM;
        }
 
-       alx->int_mask &= ~ALX_ISR_ALL_QUEUES;
-       alx->int_mask |= ALX_ISR_TX_Q0 | ALX_ISR_RX_Q0;
-
-       netif_napi_add(alx->dev, &alx->napi, alx_poll, 64);
-
        alx_reinit_rings(alx);
 
        return 0;
-out_free:
-       kfree(alx->txq.bufs);
-       kfree(alx->rxq.bufs);
-       dma_free_coherent(&alx->hw.pdev->dev,
-                         alx->descmem.size,
-                         alx->descmem.virt,
-                         alx->descmem.dma);
-       return -ENOMEM;
 }
 
 static void alx_free_rings(struct alx_priv *alx)
 {
-       netif_napi_del(&alx->napi);
        alx_free_buffers(alx);
 
        kfree(alx->txq.bufs);
        kfree(alx->rxq.bufs);
 
-       dma_free_coherent(&alx->hw.pdev->dev,
-                         alx->descmem.size,
-                         alx->descmem.virt,
-                         alx->descmem.dma);
+       if (!alx->descmem.virt)
+               dma_free_coherent(&alx->hw.pdev->dev,
+                                 alx->descmem.size,
+                                 alx->descmem.virt,
+                                 alx->descmem.dma);
+}
+
+static void alx_free_napis(struct alx_priv *alx)
+{
+       struct alx_napi *np;
+
+       np = alx->qnapi[0];
+       if (!np)
+               return;
+
+       netif_napi_del(&alx->napi);
+       kfree(np->txq);
+       kfree(np->rxq);
+       kfree(np);
+       alx->qnapi[0] = NULL;
+}
+
+static int alx_alloc_napis(struct alx_priv *alx)
+{
+       struct alx_napi *np;
+       struct alx_rx_queue *rxq;
+       struct alx_tx_queue *txq;
+
+       alx->int_mask &= ~ALX_ISR_ALL_QUEUES;
+       alx->int_mask |= ALX_ISR_TX_Q0 | ALX_ISR_RX_Q0;
+
+       /* allocate alx_napi structures */
+       np = kzalloc(sizeof(struct alx_napi), GFP_KERNEL);
+       if (!np)
+               goto err_out;
+
+       np->alx = alx;
+       netif_napi_add(alx->dev, &alx->napi, alx_poll, 64);
+       alx->qnapi[0] = np;
+
+       /* allocate tx queues */
+       np = alx->qnapi[0];
+       txq = kzalloc(sizeof(*txq), GFP_KERNEL);
+       if (!txq)
+               goto err_out;
+
+       np->txq = txq;
+       txq->count = alx->tx_ringsz;
+       txq->netdev = alx->dev;
+       txq->dev = &alx->hw.pdev->dev;
+
+       /* allocate rx queues */
+       np = alx->qnapi[0];
+       rxq = kzalloc(sizeof(*rxq), GFP_KERNEL);
+       if (!rxq)
+               goto err_out;
+
+       np->rxq = rxq;
+       rxq->np = alx->qnapi[0];
+       rxq->count = alx->rx_ringsz;
+       rxq->netdev = alx->dev;
+       rxq->dev = &alx->hw.pdev->dev;
+
+       return 0;
+
+err_out:
+       netdev_err(alx->dev, "error allocating internal structures\n");
+       alx_free_napis(alx);
+       return -ENOMEM;
 }
 
 static void alx_config_vector_mapping(struct alx_priv *alx)
@@ -1031,10 +1082,14 @@ static int __alx_open(struct alx_priv *alx, bool resume)
        if (!resume)
                netif_carrier_off(alx->dev);
 
-       err = alx_alloc_rings(alx);
+       err = alx_alloc_napis(alx);
        if (err)
                goto out_disable_adv_intr;
 
+       err = alx_alloc_rings(alx);
+       if (err)
+               goto out_free_rings;
+
        alx_configure(alx);
 
        err = alx_request_irq(alx);
@@ -1054,6 +1109,7 @@ static int __alx_open(struct alx_priv *alx, bool resume)
 
 out_free_rings:
        alx_free_rings(alx);
+       alx_free_napis(alx);
 out_disable_adv_intr:
        alx_disable_advanced_intr(alx);
        return err;
@@ -1064,6 +1120,7 @@ static void __alx_stop(struct alx_priv *alx)
        alx_halt(alx);
        alx_free_irq(alx);
        alx_free_rings(alx);
+       alx_free_napis(alx);
 }
 
 static const char *alx_speed_desc(struct alx_hw *hw)