net: ethernet: mtk_eth_soc: add support for page_pool_get_stats
authorLorenzo Bianconi <lorenzo@kernel.org>
Fri, 22 Jul 2022 07:19:40 +0000 (09:19 +0200)
committerDavid S. Miller <davem@davemloft.net>
Mon, 25 Jul 2022 09:38:57 +0000 (10:38 +0100)
Introduce support for the page_pool stats API into mtk_eth_soc driver.
Report page_pool stats through ethtool.

Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/mediatek/Kconfig
drivers/net/ethernet/mediatek/mtk_eth_soc.c

index d2422c7b31b0cdac61199c3861bdada370e252f5..97374fb3ee7933da2ab09ecacb859ff5334ccd07 100644 (file)
@@ -18,6 +18,7 @@ config NET_MEDIATEK_SOC
        select PHYLINK
        select DIMLIB
        select PAGE_POOL
+       select PAGE_POOL_STATS
        help
          This driver supports the gigabit ethernet MACs in the
          MediaTek SoC family.
index cfbdcf68f9b96cd9a38e20623b3ee80043c79855..c370d6589596f9ee811b33b3791fa87c419ccbbb 100644 (file)
@@ -3528,11 +3528,18 @@ static void mtk_get_strings(struct net_device *dev, u32 stringset, u8 *data)
        int i;
 
        switch (stringset) {
-       case ETH_SS_STATS:
+       case ETH_SS_STATS: {
+               struct mtk_mac *mac = netdev_priv(dev);
+
                for (i = 0; i < ARRAY_SIZE(mtk_ethtool_stats); i++) {
                        memcpy(data, mtk_ethtool_stats[i].str, ETH_GSTRING_LEN);
                        data += ETH_GSTRING_LEN;
                }
+               if (mtk_page_pool_enabled(mac->hw))
+                       page_pool_ethtool_stats_get_strings(data);
+               break;
+       }
+       default:
                break;
        }
 }
@@ -3540,13 +3547,35 @@ static void mtk_get_strings(struct net_device *dev, u32 stringset, u8 *data)
 static int mtk_get_sset_count(struct net_device *dev, int sset)
 {
        switch (sset) {
-       case ETH_SS_STATS:
-               return ARRAY_SIZE(mtk_ethtool_stats);
+       case ETH_SS_STATS: {
+               int count = ARRAY_SIZE(mtk_ethtool_stats);
+               struct mtk_mac *mac = netdev_priv(dev);
+
+               if (mtk_page_pool_enabled(mac->hw))
+                       count += page_pool_ethtool_stats_get_count();
+               return count;
+       }
        default:
                return -EOPNOTSUPP;
        }
 }
 
+static void mtk_ethtool_pp_stats(struct mtk_eth *eth, u64 *data)
+{
+       struct page_pool_stats stats = {};
+       int i;
+
+       for (i = 0; i < ARRAY_SIZE(eth->rx_ring); i++) {
+               struct mtk_rx_ring *ring = &eth->rx_ring[i];
+
+               if (!ring->page_pool)
+                       continue;
+
+               page_pool_get_stats(ring->page_pool, &stats);
+       }
+       page_pool_ethtool_stats_get(data, &stats);
+}
+
 static void mtk_get_ethtool_stats(struct net_device *dev,
                                  struct ethtool_stats *stats, u64 *data)
 {
@@ -3574,6 +3603,8 @@ static void mtk_get_ethtool_stats(struct net_device *dev,
 
                for (i = 0; i < ARRAY_SIZE(mtk_ethtool_stats); i++)
                        *data_dst++ = *(data_src + mtk_ethtool_stats[i].offset);
+               if (mtk_page_pool_enabled(mac->hw))
+                       mtk_ethtool_pp_stats(mac->hw, data_dst);
        } while (u64_stats_fetch_retry_irq(&hwstats->syncp, start));
 }