Skip to content

Commit 5883425

Browse files
Wei Fangmehmetb0
Wei Fang
authored andcommitted
net: enetc: block concurrent XDP transmissions during ring reconfiguration
BugLink: https://bugs.launchpad.net/bugs/2097393 commit c728a95ccf2a8ba544facfc30a4418d4c68c39f0 upstream. When testing the XDP_REDIRECT function on the LS1028A platform, we found a very reproducible issue that the Tx frames can no longer be sent out even if XDP_REDIRECT is turned off. Specifically, if there is a lot of traffic on Rx direction, when XDP_REDIRECT is turned on, the console may display some warnings like "timeout for tx ring #6 clear", and all redirected frames will be dropped, the detailed log is as follows. root@ls1028ardb:~# ./xdp-bench redirect eno0 eno2 Redirecting from eno0 (ifindex 3; driver fsl_enetc) to eno2 (ifindex 4; driver fsl_enetc) [203.849809] fsl_enetc 0000:00:00.2 eno2: timeout for tx ring #5 clear [204.006051] fsl_enetc 0000:00:00.2 eno2: timeout for tx ring #6 clear [204.161944] fsl_enetc 0000:00:00.2 eno2: timeout for tx ring #7 clear eno0->eno2 1420505 rx/s 1420590 err,drop/s 0 xmit/s xmit eno0->eno2 0 xmit/s 1420590 drop/s 0 drv_err/s 15.71 bulk-avg eno0->eno2 1420484 rx/s 1420485 err,drop/s 0 xmit/s xmit eno0->eno2 0 xmit/s 1420485 drop/s 0 drv_err/s 15.71 bulk-avg By analyzing the XDP_REDIRECT implementation of enetc driver, the driver will reconfigure Tx and Rx BD rings when a bpf program is installed or uninstalled, but there is no mechanisms to block the redirected frames when enetc driver reconfigures rings. Similarly, XDP_TX verdicts on received frames can also lead to frames being enqueued in the Tx rings. Because XDP ignores the state set by the netif_tx_wake_queue() API, so introduce the ENETC_TX_DOWN flag to suppress transmission of XDP frames. Fixes: c33bfaf ("net: enetc: set up XDP program under enetc_reconfigure()") Cc: [email protected] Signed-off-by: Wei Fang <[email protected]> Reviewed-by: Vladimir Oltean <[email protected]> Link: https://patch.msgid.link/[email protected] Signed-off-by: Jakub Kicinski <[email protected]> Signed-off-by: Greg Kroah-Hartman <[email protected]> Signed-off-by: Manuel Diewald <[email protected]> Signed-off-by: Koichiro Den <[email protected]>
1 parent 1e6bfcf commit 5883425

File tree

2 files changed

+15
-0
lines changed

2 files changed

+15
-0
lines changed

drivers/net/ethernet/freescale/enetc/enetc.c

Lines changed: 14 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -902,6 +902,7 @@ static bool enetc_clean_tx_ring(struct enetc_bdr *tx_ring, int napi_budget)
902902

903903
if (unlikely(tx_frm_cnt && netif_carrier_ok(ndev) &&
904904
__netif_subqueue_stopped(ndev, tx_ring->index) &&
905+
!test_bit(ENETC_TX_DOWN, &priv->flags) &&
905906
(enetc_bd_unused(tx_ring) >= ENETC_TXBDS_MAX_NEEDED))) {
906907
netif_wake_subqueue(ndev, tx_ring->index);
907908
}
@@ -1380,6 +1381,9 @@ int enetc_xdp_xmit(struct net_device *ndev, int num_frames,
13801381
int xdp_tx_bd_cnt, i, k;
13811382
int xdp_tx_frm_cnt = 0;
13821383

1384+
if (unlikely(test_bit(ENETC_TX_DOWN, &priv->flags)))
1385+
return -ENETDOWN;
1386+
13831387
enetc_lock_mdio();
13841388

13851389
tx_ring = priv->xdp_tx_ring[smp_processor_id()];
@@ -1605,6 +1609,12 @@ static int enetc_clean_rx_ring_xdp(struct enetc_bdr *rx_ring,
16051609
break;
16061610
case XDP_TX:
16071611
tx_ring = priv->xdp_tx_ring[rx_ring->index];
1612+
if (unlikely(test_bit(ENETC_TX_DOWN, &priv->flags))) {
1613+
enetc_xdp_drop(rx_ring, orig_i, i);
1614+
tx_ring->stats.xdp_tx_drops++;
1615+
break;
1616+
}
1617+
16081618
xdp_tx_bd_cnt = enetc_rx_swbd_to_xdp_tx_swbd(xdp_tx_arr,
16091619
rx_ring,
16101620
orig_i, i);
@@ -2466,6 +2476,8 @@ void enetc_start(struct net_device *ndev)
24662476
enetc_enable_bdrs(priv);
24672477

24682478
netif_tx_start_all_queues(ndev);
2479+
2480+
clear_bit(ENETC_TX_DOWN, &priv->flags);
24692481
}
24702482
EXPORT_SYMBOL_GPL(enetc_start);
24712483

@@ -2523,6 +2535,8 @@ void enetc_stop(struct net_device *ndev)
25232535
struct enetc_ndev_priv *priv = netdev_priv(ndev);
25242536
int i;
25252537

2538+
set_bit(ENETC_TX_DOWN, &priv->flags);
2539+
25262540
netif_tx_stop_all_queues(ndev);
25272541

25282542
enetc_disable_bdrs(priv);

drivers/net/ethernet/freescale/enetc/enetc.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -328,6 +328,7 @@ enum enetc_active_offloads {
328328

329329
enum enetc_flags_bit {
330330
ENETC_TX_ONESTEP_TSTAMP_IN_PROGRESS = 0,
331+
ENETC_TX_DOWN,
331332
};
332333

333334
/* interrupt coalescing modes */

0 commit comments

Comments
 (0)