net: dwc_eth_qos: Fix hang when freeing packet after stop
authorSamuel Holland <samuel.holland@sifive.com>
Mon, 14 Apr 2025 19:30:11 +0000 (12:30 -0700)
committerTom Rini <trini@konsulko.com>
Thu, 24 Apr 2025 14:22:59 +0000 (08:22 -0600)
If eqos_free_pkt() is called after eqos_stop(), eqos_stop_resets() will
have been called already. This may prevent accessing the MMIO space to
update the RX descriptor tail pointer, so we must skip the descriptor
maintenance logic. This is okay because the descriptors and tail pointer
will all be rewritten anyway during the next call to eqos_start().

This hang was observed after a failed TFTP transaction:

  eqos_recv(dev=000000047fb57330, flags=1):
  eqos_recv: *packetp=000000c3ffb5c080, length=151

  TFTP error: 'file <FILE> not found for <IP>' (1)
  Not retrying...
  eqos_stop(dev=000000047fb57330):
  eqos_stop: OK
  eqos_free_pkt(packet=000000c3ffb5c080, length=151)
  <HANG>

Fixes: ba4dfef1469f ("net: add driver for Synopsys Ethernet QoS device")
Signed-off-by: Samuel Holland <samuel.holland@sifive.com>
drivers/net/dwc_eth_qos.c

index b1bc422..0cfe093 100644 (file)
@@ -1173,7 +1173,7 @@ static int eqos_free_pkt(struct udevice *dev, uchar *packet, int length)
 
        eqos->config->ops->eqos_inval_buffer(packet, length);
 
-       if ((eqos->rx_desc_idx & idx_mask) == idx_mask) {
+       if (eqos->started && (eqos->rx_desc_idx & idx_mask) == idx_mask) {
                for (idx = eqos->rx_desc_idx - idx_mask;
                     idx <= eqos->rx_desc_idx;
                     idx++) {