Count per-queue RX stats in poll, replenish, and the IRQ handler:
packets, bytes, polls, large_packets, invalid_buffers, no_buffer_drops,
and interrupts. Stop updating netdev->stats.rx_* in poll; totals are
summed from rx_qstats[] in get_stats64(). Per-queue TX stats follow in
the next patch.

Expose the counters via:

- ethtool -S: per-queue rxN_* strings and aggregated invalid/large
  packet globals via ibmveth_aggregate_rx_qstats(). pool%d_* reports
  queue-0 pool geometry (size, active, available) only: static probe
  config used as the template for every queue. Live per-queue pool
  usage is exported through sysfs in the next patch.
- get_stats64: sum rx_qstats[] so ip -s and /proc/net/dev report total RX
- ethtool hcall_stats counters and count send_lan on successful TX hcalls

Fix get_channels() reporting: max_rx is IBMVETH_MAX_RX_QUEUES only when
MQ firmware is enabled, rx_count tracks adapter->num_rx_queues.

Signed-off-by: Mingming Cao <[email protected]>
Reviewed-by: Dave Marquardt <[email protected]>
---
 drivers/net/ethernet/ibm/ibmveth.c | 152 ++++++++++++++++++++++++++---
 1 file changed, 141 insertions(+), 11 deletions(-)

diff --git a/drivers/net/ethernet/ibm/ibmveth.c 
b/drivers/net/ethernet/ibm/ibmveth.c
index 863e5c68b42c..1c08082ffbd6 100644
--- a/drivers/net/ethernet/ibm/ibmveth.c
+++ b/drivers/net/ethernet/ibm/ibmveth.c
@@ -98,7 +98,15 @@ static struct ibmveth_stat ibmveth_stats[] = {
        { "fw_enabled_ipv6_csum", IBMVETH_STAT_OFF(fw_ipv6_csum_support) },
        { "tx_large_packets", IBMVETH_STAT_OFF(tx_large_packets) },
        { "rx_large_packets", IBMVETH_STAT_OFF(rx_large_packets) },
-       { "fw_enabled_large_send", IBMVETH_STAT_OFF(fw_large_send_support) }
+       { "fw_enabled_large_send", IBMVETH_STAT_OFF(fw_large_send_support) },
+       { "hcall_reg_lan_queue", IBMVETH_STAT_OFF(hcall_stats.reg_lan_queue) },
+       { "hcall_reg_lan", IBMVETH_STAT_OFF(hcall_stats.reg_lan) },
+       { "hcall_add_bufs_queue", IBMVETH_STAT_OFF(hcall_stats.add_bufs_queue) 
},
+       { "hcall_add_bufs", IBMVETH_STAT_OFF(hcall_stats.add_bufs) },
+       { "hcall_add_buf", IBMVETH_STAT_OFF(hcall_stats.add_buf) },
+       { "hcall_free_lan_queue", IBMVETH_STAT_OFF(hcall_stats.free_lan_queue) 
},
+       { "hcall_free_lan", IBMVETH_STAT_OFF(hcall_stats.free_lan) },
+       { "hcall_send_lan", IBMVETH_STAT_OFF(hcall_stats.send_lan) },
 };
 
 /* simple methods of getting data from the current rxq entry */
@@ -847,6 +855,8 @@ static void ibmveth_update_rx_no_buffer(struct 
ibmveth_adapter *adapter)
                __be64 *p = adapter->buffer_list_addr[i] + 4096 - 8;
                u64 drops = be64_to_cpup(p);
 
+               if (adapter->rx_qstats)
+                       adapter->rx_qstats[i].no_buffer_drops = drops;
                if (i == 0)
                        adapter->rx_no_buffer = drops;
        }
@@ -1925,22 +1935,71 @@ static int ibmveth_set_features(struct net_device *dev,
        return rc1 ? rc1 : rc2;
 }
 
+/**
+ * ibmveth_aggregate_rx_qstats - Sum per-queue RX stats into globals
+ * @adapter: ibmveth adapter
+ *
+ * Cold path only (ethtool). Keeps legacy global counters meaningful for
+ * tools that read the adapter-level fields in ibmveth_stats[].
+ */
+static void ibmveth_aggregate_rx_qstats(struct ibmveth_adapter *adapter)
+{
+       u64 total_invalid = 0;
+       u64 total_large = 0;
+       int i;
+
+       if (!adapter->rx_qstats)
+               return;
+
+       for (i = 0; i < adapter->num_rx_queues; i++) {
+               total_invalid += adapter->rx_qstats[i].invalid_buffers;
+               total_large += adapter->rx_qstats[i].large_packets;
+       }
+
+       adapter->rx_invalid_buffer = total_invalid;
+       adapter->rx_large_packets = total_large;
+}
+
 static void ibmveth_get_strings(struct net_device *dev, u32 stringset, u8 
*data)
 {
+       struct ibmveth_adapter *adapter = netdev_priv(dev);
+       u8 *p = data;
        int i;
 
        if (stringset != ETH_SS_STATS)
                return;
 
-       for (i = 0; i < ARRAY_SIZE(ibmveth_stats); i++, data += ETH_GSTRING_LEN)
-               memcpy(data, ibmveth_stats[i].name, ETH_GSTRING_LEN);
+       for (i = 0; i < ARRAY_SIZE(ibmveth_stats); i++) {
+               memcpy(p, ibmveth_stats[i].name, ETH_GSTRING_LEN);
+               p += ETH_GSTRING_LEN;
+       }
+
+       for (i = 0; i < adapter->num_rx_queues; i++) {
+               ethtool_sprintf(&p, "rx%d_packets", i);
+               ethtool_sprintf(&p, "rx%d_bytes", i);
+               ethtool_sprintf(&p, "rx%d_interrupts", i);
+               ethtool_sprintf(&p, "rx%d_polls", i);
+               ethtool_sprintf(&p, "rx%d_large_packets", i);
+               ethtool_sprintf(&p, "rx%d_invalid_buffers", i);
+               ethtool_sprintf(&p, "rx%d_no_buffer_drops", i);
+       }
+
+       for (i = 0; i < IBMVETH_NUM_BUFF_POOLS; i++) {
+               ethtool_sprintf(&p, "pool%d_size", i);
+               ethtool_sprintf(&p, "pool%d_active", i);
+               ethtool_sprintf(&p, "pool%d_available", i);
+       }
 }
 
 static int ibmveth_get_sset_count(struct net_device *dev, int sset)
 {
+       struct ibmveth_adapter *adapter = netdev_priv(dev);
+
        switch (sset) {
        case ETH_SS_STATS:
-               return ARRAY_SIZE(ibmveth_stats);
+               return ARRAY_SIZE(ibmveth_stats) +
+                      adapter->num_rx_queues * IBMVETH_NUM_RX_QSTATS +
+                      IBMVETH_NUM_BUFF_POOLS * 3;
        default:
                return -EOPNOTSUPP;
        }
@@ -1949,21 +2008,48 @@ static int ibmveth_get_sset_count(struct net_device 
*dev, int sset)
 static void ibmveth_get_ethtool_stats(struct net_device *dev,
                                      struct ethtool_stats *stats, u64 *data)
 {
-       int i;
        struct ibmveth_adapter *adapter = netdev_priv(dev);
+       int i, j;
+
+       ibmveth_aggregate_rx_qstats(adapter);
 
        for (i = 0; i < ARRAY_SIZE(ibmveth_stats); i++)
                data[i] = IBMVETH_GET_STAT(adapter, ibmveth_stats[i].offset);
+
+       for (j = 0; j < adapter->num_rx_queues; j++) {
+               if (adapter->rx_qstats) {
+                       data[i++] = adapter->rx_qstats[j].packets;
+                       data[i++] = adapter->rx_qstats[j].bytes;
+                       data[i++] = adapter->rx_qstats[j].interrupts;
+                       data[i++] = adapter->rx_qstats[j].polls;
+                       data[i++] = adapter->rx_qstats[j].large_packets;
+                       data[i++] = adapter->rx_qstats[j].invalid_buffers;
+                       data[i++] = adapter->rx_qstats[j].no_buffer_drops;
+               } else {
+                       i += IBMVETH_NUM_RX_QSTATS;
+               }
+       }
+
+       for (j = 0; j < IBMVETH_NUM_BUFF_POOLS; j++) {
+               data[i++] = adapter->rx_buff_pool[0][j].size;
+               data[i++] = adapter->rx_buff_pool[0][j].active;
+               data[i++] = atomic_read(&adapter->rx_buff_pool[0][j].available);
+       }
 }
 
 static void ibmveth_get_channels(struct net_device *netdev,
                                 struct ethtool_channels *channels)
 {
+       struct ibmveth_adapter *adapter = netdev_priv(netdev);
+
        channels->max_tx = ibmveth_real_max_tx_queues();
        channels->tx_count = netdev->real_num_tx_queues;
 
-       channels->max_rx = netdev->real_num_rx_queues;
-       channels->rx_count = netdev->real_num_rx_queues;
+       if (adapter->multi_queue)
+               channels->max_rx = IBMVETH_MAX_RX_QUEUES;
+       else
+               channels->max_rx = 1;
+       channels->rx_count = adapter->num_rx_queues;
 }
 
 static int ibmveth_set_channels(struct net_device *netdev,
@@ -2061,6 +2147,7 @@ static int ibmveth_send(struct ibmveth_adapter *adapter,
                return 1;
        }
 
+       adapter->hcall_stats.send_lan++;
        return 0;
 }
 
@@ -2311,6 +2398,9 @@ static int ibmveth_poll(struct napi_struct *napi, int 
budget)
        if (WARN_ON(queue_index < 0 || queue_index >= adapter->num_rx_queues))
                return 0;
 
+       if (adapter->rx_qstats)
+               adapter->rx_qstats[queue_index].polls++;
+
 restart_poll:
        while (frames_processed < budget) {
                if (!ibmveth_rxq_pending_buffer(adapter, queue_index))
@@ -2319,7 +2409,10 @@ static int ibmveth_poll(struct napi_struct *napi, int 
budget)
                smp_rmb();
                if (!ibmveth_rxq_buffer_valid(adapter, queue_index)) {
                        wmb(); /* suggested by larson1 */
-                       adapter->rx_invalid_buffer++;
+                       if (adapter->rx_qstats)
+                               
adapter->rx_qstats[queue_index].invalid_buffers++;
+                       else
+                               adapter->rx_invalid_buffer++;
                        netdev_dbg(netdev, "recycling invalid buffer\n");
                        rc = ibmveth_rxq_harvest_buffer(adapter, queue_index, 
true);
                        if (unlikely(rc))
@@ -2384,7 +2477,10 @@ static int ibmveth_poll(struct napi_struct *napi, int 
budget)
                        if ((length > netdev->mtu + ETH_HLEN) ||
                            lrg_pkt || iph_check == 0xffff) {
                                ibmveth_rx_mss_helper(skb, mss, lrg_pkt);
-                               adapter->rx_large_packets++;
+                               if (adapter->rx_qstats)
+                                       
adapter->rx_qstats[queue_index].large_packets++;
+                               else
+                                       adapter->rx_large_packets++;
                        }
 
                        if (csum_good) {
@@ -2394,8 +2490,11 @@ static int ibmveth_poll(struct napi_struct *napi, int 
budget)
 
                        napi_gro_receive(napi, skb);    /* send it up */
 
-                       netdev->stats.rx_packets++;
-                       netdev->stats.rx_bytes += length;
+                       if (adapter->rx_qstats) {
+                               adapter->rx_qstats[queue_index].packets++;
+                               adapter->rx_qstats[queue_index].bytes += length;
+                       }
+
                        frames_processed++;
                }
        }
@@ -2444,6 +2543,9 @@ static irqreturn_t ibmveth_interrupt(int irq, void 
*dev_instance)
        if (WARN_ON(qindex < 0 || qindex >= adapter->num_rx_queues))
                return IRQ_NONE;
 
+       if (adapter->rx_qstats)
+               adapter->rx_qstats[qindex].interrupts++;
+
        if (napi_schedule_prep(napi)) {
                lpar_rc = ibmveth_disable_irq(adapter, qindex);
                WARN_ON(lpar_rc != H_SUCCESS);
@@ -2656,6 +2758,33 @@ static netdev_features_t ibmveth_features_check(struct 
sk_buff *skb,
        return vlan_features_check(skb, features);
 }
 
+/**
+ * ibmveth_get_stats64 - Return aggregated per-queue RX statistics
+ * @dev: network device
+ * @stats: rtnl link statistics storage
+ *
+ * Sums per-queue rx_qstats into rx_packets/rx_bytes for multi-queue mode.
+ * TX counters continue to come from netdev->stats (updated in start_xmit).
+ */
+static void ibmveth_get_stats64(struct net_device *dev,
+                               struct rtnl_link_stats64 *stats)
+{
+       struct ibmveth_adapter *adapter = netdev_priv(dev);
+       int i;
+
+       if (adapter->rx_qstats) {
+               for (i = 0; i < adapter->num_rx_queues; i++) {
+                       stats->rx_packets += adapter->rx_qstats[i].packets;
+                       stats->rx_bytes += adapter->rx_qstats[i].bytes;
+               }
+       }
+
+       stats->tx_packets = dev->stats.tx_packets;
+       stats->tx_bytes = dev->stats.tx_bytes;
+       stats->tx_dropped = dev->stats.tx_dropped;
+       stats->tx_errors = dev->stats.tx_errors;
+}
+
 static const struct net_device_ops ibmveth_netdev_ops = {
        .ndo_open               = ibmveth_open,
        .ndo_stop               = ibmveth_close,
@@ -2668,6 +2797,7 @@ static const struct net_device_ops ibmveth_netdev_ops = {
        .ndo_validate_addr      = eth_validate_addr,
        .ndo_set_mac_address    = ibmveth_set_mac_addr,
        .ndo_features_check     = ibmveth_features_check,
+       .ndo_get_stats64        = ibmveth_get_stats64,
 #ifdef CONFIG_NET_POLL_CONTROLLER
        .ndo_poll_controller    = ibmveth_poll_controller,
 #endif
-- 
2.39.3 (Apple Git-146)


Reply via email to