Skip to content

Commit 0a93f2c

Browse files
Wei Fangkuba-moo
Wei Fang
authored andcommitted
net: enetc: disable Tx BD rings after they are empty
The Tx BD rings are disabled first in enetc_stop() and the driver waits for them to become empty. This operation is not safe while the ring is actively transmitting frames, and will cause the ring to not be empty and hardware exception. As described in the NETC block guide, software should only disable an active Tx ring after all pending ring entries have been consumed (i.e. when PI = CI). Disabling a transmit ring that is actively processing BDs risks a HW-SW race hazard whereby a hardware resource becomes assigned to work on one or more ring entries only to have those entries be removed due to the ring becoming disabled. When testing XDP_REDIRECT feautre, although all frames were blocked from being put into Tx rings during ring reconfiguration, the similar warning log was still encountered: fsl_enetc 0000:00:00.2 eno2: timeout for tx ring #6 clear fsl_enetc 0000:00:00.2 eno2: timeout for tx ring #7 clear The reason is that when there are still unsent frames in the Tx ring, disabling the Tx ring causes the remaining frames to be unable to be sent out. And the Tx ring cannot be restored, which means that even if the xdp program is uninstalled, the Tx frames cannot be sent out anymore. Therefore, correct the operation order in enect_start() and enect_stop(). Fixes: ff58fda ("net: enetc: prioritize ability to go down over packet processing") Cc: stable@vger.kernel.org Signed-off-by: Wei Fang <wei.fang@nxp.com> Reviewed-by: Vladimir Oltean <vladimir.oltean@nxp.com> Link: https://patch.msgid.link/20241010092056.298128-4-wei.fang@nxp.com Signed-off-by: Jakub Kicinski <kuba@kernel.org>
1 parent c728a95 commit 0a93f2c

File tree

1 file changed

+26
-10
lines changed
  • drivers/net/ethernet/freescale/enetc

1 file changed

+26
-10
lines changed

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

+26-10
Original file line numberDiff line numberDiff line change
@@ -2233,18 +2233,24 @@ static void enetc_enable_rxbdr(struct enetc_hw *hw, struct enetc_bdr *rx_ring)
22332233
enetc_rxbdr_wr(hw, idx, ENETC_RBMR, rbmr);
22342234
}
22352235

2236-
static void enetc_enable_bdrs(struct enetc_ndev_priv *priv)
2236+
static void enetc_enable_rx_bdrs(struct enetc_ndev_priv *priv)
22372237
{
22382238
struct enetc_hw *hw = &priv->si->hw;
22392239
int i;
22402240

2241-
for (i = 0; i < priv->num_tx_rings; i++)
2242-
enetc_enable_txbdr(hw, priv->tx_ring[i]);
2243-
22442241
for (i = 0; i < priv->num_rx_rings; i++)
22452242
enetc_enable_rxbdr(hw, priv->rx_ring[i]);
22462243
}
22472244

2245+
static void enetc_enable_tx_bdrs(struct enetc_ndev_priv *priv)
2246+
{
2247+
struct enetc_hw *hw = &priv->si->hw;
2248+
int i;
2249+
2250+
for (i = 0; i < priv->num_tx_rings; i++)
2251+
enetc_enable_txbdr(hw, priv->tx_ring[i]);
2252+
}
2253+
22482254
static void enetc_disable_rxbdr(struct enetc_hw *hw, struct enetc_bdr *rx_ring)
22492255
{
22502256
int idx = rx_ring->index;
@@ -2261,18 +2267,24 @@ static void enetc_disable_txbdr(struct enetc_hw *hw, struct enetc_bdr *rx_ring)
22612267
enetc_txbdr_wr(hw, idx, ENETC_TBMR, 0);
22622268
}
22632269

2264-
static void enetc_disable_bdrs(struct enetc_ndev_priv *priv)
2270+
static void enetc_disable_rx_bdrs(struct enetc_ndev_priv *priv)
22652271
{
22662272
struct enetc_hw *hw = &priv->si->hw;
22672273
int i;
22682274

2269-
for (i = 0; i < priv->num_tx_rings; i++)
2270-
enetc_disable_txbdr(hw, priv->tx_ring[i]);
2271-
22722275
for (i = 0; i < priv->num_rx_rings; i++)
22732276
enetc_disable_rxbdr(hw, priv->rx_ring[i]);
22742277
}
22752278

2279+
static void enetc_disable_tx_bdrs(struct enetc_ndev_priv *priv)
2280+
{
2281+
struct enetc_hw *hw = &priv->si->hw;
2282+
int i;
2283+
2284+
for (i = 0; i < priv->num_tx_rings; i++)
2285+
enetc_disable_txbdr(hw, priv->tx_ring[i]);
2286+
}
2287+
22762288
static void enetc_wait_txbdr(struct enetc_hw *hw, struct enetc_bdr *tx_ring)
22772289
{
22782290
int delay = 8, timeout = 100;
@@ -2462,6 +2474,8 @@ void enetc_start(struct net_device *ndev)
24622474

24632475
enetc_setup_interrupts(priv);
24642476

2477+
enetc_enable_tx_bdrs(priv);
2478+
24652479
for (i = 0; i < priv->bdr_int_num; i++) {
24662480
int irq = pci_irq_vector(priv->si->pdev,
24672481
ENETC_BDR_INT_BASE_IDX + i);
@@ -2470,7 +2484,7 @@ void enetc_start(struct net_device *ndev)
24702484
enable_irq(irq);
24712485
}
24722486

2473-
enetc_enable_bdrs(priv);
2487+
enetc_enable_rx_bdrs(priv);
24742488

24752489
netif_tx_start_all_queues(ndev);
24762490

@@ -2536,7 +2550,7 @@ void enetc_stop(struct net_device *ndev)
25362550

25372551
netif_tx_stop_all_queues(ndev);
25382552

2539-
enetc_disable_bdrs(priv);
2553+
enetc_disable_rx_bdrs(priv);
25402554

25412555
for (i = 0; i < priv->bdr_int_num; i++) {
25422556
int irq = pci_irq_vector(priv->si->pdev,
@@ -2549,6 +2563,8 @@ void enetc_stop(struct net_device *ndev)
25492563

25502564
enetc_wait_bdrs(priv);
25512565

2566+
enetc_disable_tx_bdrs(priv);
2567+
25522568
enetc_clear_interrupts(priv);
25532569
}
25542570
EXPORT_SYMBOL_GPL(enetc_stop);

0 commit comments

Comments
 (0)