]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
veth: fix data race in veth_get_ethtool_stats
authorDavid Yang <mmyangfl@gmail.com>
Wed, 14 Jan 2026 12:24:45 +0000 (20:24 +0800)
committerJakub Kicinski <kuba@kernel.org>
Sun, 18 Jan 2026 00:22:18 +0000 (16:22 -0800)
In veth_get_ethtool_stats(), some statistics protected by
u64_stats_sync, are read and accumulated in ignorance of possible
u64_stats_fetch_retry() events. These statistics, peer_tq_xdp_xmit and
peer_tq_xdp_xmit_err, are already accumulated by veth_xdp_xmit(). Fix
this by reading them into a temporary buffer first.

Fixes: 5fe6e56776ba ("veth: rely on peer veth_rq for ndo_xdp_xmit accounting")
Signed-off-by: David Yang <mmyangfl@gmail.com>
Link: https://patch.msgid.link/20260114122450.227982-1-mmyangfl@gmail.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
drivers/net/veth.c

index 14e6f2a2fb7783334d8f6afd54e658cf9a0e6f3d..9982412fd7f238e996ccdff24342974cb25094bf 100644 (file)
@@ -228,16 +228,20 @@ static void veth_get_ethtool_stats(struct net_device *dev,
                const struct veth_rq_stats *rq_stats = &rcv_priv->rq[i].stats;
                const void *base = (void *)&rq_stats->vs;
                unsigned int start, tx_idx = idx;
+               u64 buf[VETH_TQ_STATS_LEN];
                size_t offset;
 
-               tx_idx += (i % dev->real_num_tx_queues) * VETH_TQ_STATS_LEN;
                do {
                        start = u64_stats_fetch_begin(&rq_stats->syncp);
                        for (j = 0; j < VETH_TQ_STATS_LEN; j++) {
                                offset = veth_tq_stats_desc[j].offset;
-                               data[tx_idx + j] += *(u64 *)(base + offset);
+                               buf[j] = *(u64 *)(base + offset);
                        }
                } while (u64_stats_fetch_retry(&rq_stats->syncp, start));
+
+               tx_idx += (i % dev->real_num_tx_queues) * VETH_TQ_STATS_LEN;
+               for (j = 0; j < VETH_TQ_STATS_LEN; j++)
+                       data[tx_idx + j] += buf[j];
        }
        pp_idx = idx + dev->real_num_tx_queues * VETH_TQ_STATS_LEN;