dpaa2-eth: Poll Tx pending frames counter on if down
authorIoana Radulescu <ruxandra.radulescu@nxp.com>
Mon, 2 Sep 2019 10:23:19 +0000 (13:23 +0300)
committerDavid S. Miller <davem@davemloft.net>
Wed, 4 Sep 2019 22:24:06 +0000 (00:24 +0200)
Starting with firmware version MC10.18.0, a new counter for in flight
Tx frames is offered. Use it when bringing down the interface to
determine when all pending Tx frames have been processed by hardware
instead of sleeping a fixed amount of time.

Signed-off-by: Ioana Radulescu <ruxandra.radulescu@nxp.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/freescale/dpaa2/dpaa2-eth.c

index 5402867272be4a13b03b780258a86af7dda49380..162d7d8fb2950c643af4c6d68d1bdafe44ffcf46 100644 (file)
@@ -1348,7 +1348,7 @@ static u32 ingress_fq_count(struct dpaa2_eth_priv *priv)
        return total;
 }
 
-static void wait_for_fq_empty(struct dpaa2_eth_priv *priv)
+static void wait_for_ingress_fq_empty(struct dpaa2_eth_priv *priv)
 {
        int retries = 10;
        u32 pending;
@@ -1360,6 +1360,31 @@ static void wait_for_fq_empty(struct dpaa2_eth_priv *priv)
        } while (pending && --retries);
 }
 
+#define DPNI_TX_PENDING_VER_MAJOR      7
+#define DPNI_TX_PENDING_VER_MINOR      13
+static void wait_for_egress_fq_empty(struct dpaa2_eth_priv *priv)
+{
+       union dpni_statistics stats;
+       int retries = 10;
+       int err;
+
+       if (dpaa2_eth_cmp_dpni_ver(priv, DPNI_TX_PENDING_VER_MAJOR,
+                                  DPNI_TX_PENDING_VER_MINOR) < 0)
+               goto out;
+
+       do {
+               err = dpni_get_statistics(priv->mc_io, 0, priv->mc_token, 6,
+                                         &stats);
+               if (err)
+                       goto out;
+               if (stats.page_6.tx_pending_frames == 0)
+                       return;
+       } while (--retries);
+
+out:
+       msleep(500);
+}
+
 static int dpaa2_eth_stop(struct net_device *net_dev)
 {
        struct dpaa2_eth_priv *priv = netdev_priv(net_dev);
@@ -1379,7 +1404,7 @@ static int dpaa2_eth_stop(struct net_device *net_dev)
         * on WRIOP. After it finishes, wait until all remaining frames on Rx
         * and Tx conf queues are consumed on NAPI poll.
         */
-       msleep(500);
+       wait_for_egress_fq_empty(priv);
 
        do {
                dpni_disable(priv->mc_io, 0, priv->mc_token);
@@ -1395,7 +1420,7 @@ static int dpaa2_eth_stop(struct net_device *net_dev)
                 */
        }
 
-       wait_for_fq_empty(priv);
+       wait_for_ingress_fq_empty(priv);
        disable_ch_napi(priv);
 
        /* Empty the buffer pool */