Skip to content

Commit 2182153

Browse files
committed
Merge branch 'there-are-some-bugfix-for-hibmcge-ethernet-driver'
Jijie Shao says: ==================== There are some bugfix for hibmcge ethernet driver This patch set is intended to fix several issues for hibmcge driver: 1. Holding the rtnl_lock in pci_error_handlers->reset_prepare() may lead to a deadlock issue. 2. A division by zero issue caused by debugfs when the port is down. 3. A probabilistic false positive issue with np_link_fail. v2: https://lore.kernel.org/[email protected] v1: https://lore.kernel.org/[email protected] ==================== Link: https://patch.msgid.link/[email protected] Signed-off-by: Jakub Kicinski <[email protected]>
2 parents f6a2a31 + 62c5018 commit 2182153

File tree

3 files changed

+24
-12
lines changed

3 files changed

+24
-12
lines changed

drivers/net/ethernet/hisilicon/hibmcge/hbg_err.c

Lines changed: 5 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -53,9 +53,11 @@ static int hbg_reset_prepare(struct hbg_priv *priv, enum hbg_reset_type type)
5353
{
5454
int ret;
5555

56-
ASSERT_RTNL();
56+
if (test_and_set_bit(HBG_NIC_STATE_RESETTING, &priv->state))
57+
return -EBUSY;
5758

5859
if (netif_running(priv->netdev)) {
60+
clear_bit(HBG_NIC_STATE_RESETTING, &priv->state);
5961
dev_warn(&priv->pdev->dev,
6062
"failed to reset because port is up\n");
6163
return -EBUSY;
@@ -64,7 +66,6 @@ static int hbg_reset_prepare(struct hbg_priv *priv, enum hbg_reset_type type)
6466
netif_device_detach(priv->netdev);
6567

6668
priv->reset_type = type;
67-
set_bit(HBG_NIC_STATE_RESETTING, &priv->state);
6869
clear_bit(HBG_NIC_STATE_RESET_FAIL, &priv->state);
6970
ret = hbg_hw_event_notify(priv, HBG_HW_EVENT_RESET);
7071
if (ret) {
@@ -84,29 +85,26 @@ static int hbg_reset_done(struct hbg_priv *priv, enum hbg_reset_type type)
8485
type != priv->reset_type)
8586
return 0;
8687

87-
ASSERT_RTNL();
88-
89-
clear_bit(HBG_NIC_STATE_RESETTING, &priv->state);
9088
ret = hbg_rebuild(priv);
9189
if (ret) {
9290
priv->stats.reset_fail_cnt++;
9391
set_bit(HBG_NIC_STATE_RESET_FAIL, &priv->state);
92+
clear_bit(HBG_NIC_STATE_RESETTING, &priv->state);
9493
dev_err(&priv->pdev->dev, "failed to rebuild after reset\n");
9594
return ret;
9695
}
9796

9897
netif_device_attach(priv->netdev);
98+
clear_bit(HBG_NIC_STATE_RESETTING, &priv->state);
9999

100100
dev_info(&priv->pdev->dev, "reset done\n");
101101
return ret;
102102
}
103103

104-
/* must be protected by rtnl lock */
105104
int hbg_reset(struct hbg_priv *priv)
106105
{
107106
int ret;
108107

109-
ASSERT_RTNL();
110108
ret = hbg_reset_prepare(priv, HBG_RESET_TYPE_FUNCTION);
111109
if (ret)
112110
return ret;
@@ -171,7 +169,6 @@ static void hbg_pci_err_reset_prepare(struct pci_dev *pdev)
171169
struct net_device *netdev = pci_get_drvdata(pdev);
172170
struct hbg_priv *priv = netdev_priv(netdev);
173171

174-
rtnl_lock();
175172
hbg_reset_prepare(priv, HBG_RESET_TYPE_FLR);
176173
}
177174

@@ -181,7 +178,6 @@ static void hbg_pci_err_reset_done(struct pci_dev *pdev)
181178
struct hbg_priv *priv = netdev_priv(netdev);
182179

183180
hbg_reset_done(priv, HBG_RESET_TYPE_FLR);
184-
rtnl_unlock();
185181
}
186182

187183
static const struct pci_error_handlers hbg_pci_err_handler = {

drivers/net/ethernet/hisilicon/hibmcge/hbg_hw.c

Lines changed: 13 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,8 @@
1212

1313
#define HBG_HW_EVENT_WAIT_TIMEOUT_US (2 * 1000 * 1000)
1414
#define HBG_HW_EVENT_WAIT_INTERVAL_US (10 * 1000)
15+
#define HBG_MAC_LINK_WAIT_TIMEOUT_US (500 * 1000)
16+
#define HBG_MAC_LINK_WAIT_INTERVAL_US (5 * 1000)
1517
/* little endian or big endian.
1618
* ctrl means packet description, data means skb packet data
1719
*/
@@ -228,6 +230,9 @@ void hbg_hw_fill_buffer(struct hbg_priv *priv, u32 buffer_dma_addr)
228230

229231
void hbg_hw_adjust_link(struct hbg_priv *priv, u32 speed, u32 duplex)
230232
{
233+
u32 link_status;
234+
int ret;
235+
231236
hbg_hw_mac_enable(priv, HBG_STATUS_DISABLE);
232237

233238
hbg_reg_write_field(priv, HBG_REG_PORT_MODE_ADDR,
@@ -239,8 +244,14 @@ void hbg_hw_adjust_link(struct hbg_priv *priv, u32 speed, u32 duplex)
239244

240245
hbg_hw_mac_enable(priv, HBG_STATUS_ENABLE);
241246

242-
if (!hbg_reg_read_field(priv, HBG_REG_AN_NEG_STATE_ADDR,
243-
HBG_REG_AN_NEG_STATE_NP_LINK_OK_B))
247+
/* wait MAC link up */
248+
ret = readl_poll_timeout(priv->io_base + HBG_REG_AN_NEG_STATE_ADDR,
249+
link_status,
250+
FIELD_GET(HBG_REG_AN_NEG_STATE_NP_LINK_OK_B,
251+
link_status),
252+
HBG_MAC_LINK_WAIT_INTERVAL_US,
253+
HBG_MAC_LINK_WAIT_TIMEOUT_US);
254+
if (ret)
244255
hbg_np_link_fail_task_schedule(priv);
245256
}
246257

drivers/net/ethernet/hisilicon/hibmcge/hbg_txrx.h

Lines changed: 6 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,12 @@ static inline bool hbg_fifo_is_full(struct hbg_priv *priv, enum hbg_dir dir)
2929

3030
static inline u32 hbg_get_queue_used_num(struct hbg_ring *ring)
3131
{
32-
return (ring->ntu + ring->len - ring->ntc) % ring->len;
32+
u32 len = READ_ONCE(ring->len);
33+
34+
if (!len)
35+
return 0;
36+
37+
return (READ_ONCE(ring->ntu) + len - READ_ONCE(ring->ntc)) % len;
3338
}
3439

3540
netdev_tx_t hbg_net_start_xmit(struct sk_buff *skb, struct net_device *netdev);

0 commit comments

Comments
 (0)