Commit bf5295bb authored by Michael Chan's avatar Michael Chan Committed by David S. Miller

[BNX2]: Fix link change handling

Fix some link-related problems by doing a coalesce_now after link
change interrupt to flush out the transient link status.

To facilitate this, the host coalesce cmd register value is cached in
the device structure.
Signed-off-by: default avatarMichael Chan <mchan@broadcom.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent ca6549af
...@@ -313,8 +313,6 @@ bnx2_disable_int(struct bnx2 *bp) ...@@ -313,8 +313,6 @@ bnx2_disable_int(struct bnx2 *bp)
static void static void
bnx2_enable_int(struct bnx2 *bp) bnx2_enable_int(struct bnx2 *bp)
{ {
u32 val;
REG_WR(bp, BNX2_PCICFG_INT_ACK_CMD, REG_WR(bp, BNX2_PCICFG_INT_ACK_CMD,
BNX2_PCICFG_INT_ACK_CMD_INDEX_VALID | BNX2_PCICFG_INT_ACK_CMD_INDEX_VALID |
BNX2_PCICFG_INT_ACK_CMD_MASK_INT | bp->last_status_idx); BNX2_PCICFG_INT_ACK_CMD_MASK_INT | bp->last_status_idx);
...@@ -322,8 +320,7 @@ bnx2_enable_int(struct bnx2 *bp) ...@@ -322,8 +320,7 @@ bnx2_enable_int(struct bnx2 *bp)
REG_WR(bp, BNX2_PCICFG_INT_ACK_CMD, REG_WR(bp, BNX2_PCICFG_INT_ACK_CMD,
BNX2_PCICFG_INT_ACK_CMD_INDEX_VALID | bp->last_status_idx); BNX2_PCICFG_INT_ACK_CMD_INDEX_VALID | bp->last_status_idx);
val = REG_RD(bp, BNX2_HC_COMMAND); REG_WR(bp, BNX2_HC_COMMAND, bp->hc_cmd | BNX2_HC_COMMAND_COAL_NOW);
REG_WR(bp, BNX2_HC_COMMAND, val | BNX2_HC_COMMAND_COAL_NOW);
} }
static void static void
...@@ -1926,6 +1923,13 @@ bnx2_poll(struct net_device *dev, int *budget) ...@@ -1926,6 +1923,13 @@ bnx2_poll(struct net_device *dev, int *budget)
spin_lock(&bp->phy_lock); spin_lock(&bp->phy_lock);
bnx2_phy_int(bp); bnx2_phy_int(bp);
spin_unlock(&bp->phy_lock); spin_unlock(&bp->phy_lock);
/* This is needed to take care of transient status
* during link changes.
*/
REG_WR(bp, BNX2_HC_COMMAND,
bp->hc_cmd | BNX2_HC_COMMAND_COAL_NOW_WO_INT);
REG_RD(bp, BNX2_HC_COMMAND);
} }
if (bp->status_blk->status_tx_quick_consumer_index0 != bp->hw_tx_cons) if (bp->status_blk->status_tx_quick_consumer_index0 != bp->hw_tx_cons)
...@@ -3307,6 +3311,8 @@ bnx2_init_chip(struct bnx2 *bp) ...@@ -3307,6 +3311,8 @@ bnx2_init_chip(struct bnx2 *bp)
udelay(20); udelay(20);
bp->hc_cmd = REG_RD(bp, BNX2_HC_COMMAND);
return rc; return rc;
} }
...@@ -3746,7 +3752,6 @@ bnx2_run_loopback(struct bnx2 *bp, int loopback_mode) ...@@ -3746,7 +3752,6 @@ bnx2_run_loopback(struct bnx2 *bp, int loopback_mode)
struct sk_buff *skb, *rx_skb; struct sk_buff *skb, *rx_skb;
unsigned char *packet; unsigned char *packet;
u16 rx_start_idx, rx_idx; u16 rx_start_idx, rx_idx;
u32 val;
dma_addr_t map; dma_addr_t map;
struct tx_bd *txbd; struct tx_bd *txbd;
struct sw_bd *rx_buf; struct sw_bd *rx_buf;
...@@ -3777,8 +3782,9 @@ bnx2_run_loopback(struct bnx2 *bp, int loopback_mode) ...@@ -3777,8 +3782,9 @@ bnx2_run_loopback(struct bnx2 *bp, int loopback_mode)
map = pci_map_single(bp->pdev, skb->data, pkt_size, map = pci_map_single(bp->pdev, skb->data, pkt_size,
PCI_DMA_TODEVICE); PCI_DMA_TODEVICE);
val = REG_RD(bp, BNX2_HC_COMMAND); REG_WR(bp, BNX2_HC_COMMAND,
REG_WR(bp, BNX2_HC_COMMAND, val | BNX2_HC_COMMAND_COAL_NOW_WO_INT); bp->hc_cmd | BNX2_HC_COMMAND_COAL_NOW_WO_INT);
REG_RD(bp, BNX2_HC_COMMAND); REG_RD(bp, BNX2_HC_COMMAND);
udelay(5); udelay(5);
...@@ -3802,8 +3808,9 @@ bnx2_run_loopback(struct bnx2 *bp, int loopback_mode) ...@@ -3802,8 +3808,9 @@ bnx2_run_loopback(struct bnx2 *bp, int loopback_mode)
udelay(100); udelay(100);
val = REG_RD(bp, BNX2_HC_COMMAND); REG_WR(bp, BNX2_HC_COMMAND,
REG_WR(bp, BNX2_HC_COMMAND, val | BNX2_HC_COMMAND_COAL_NOW_WO_INT); bp->hc_cmd | BNX2_HC_COMMAND_COAL_NOW_WO_INT);
REG_RD(bp, BNX2_HC_COMMAND); REG_RD(bp, BNX2_HC_COMMAND);
udelay(5); udelay(5);
...@@ -3939,7 +3946,6 @@ static int ...@@ -3939,7 +3946,6 @@ static int
bnx2_test_intr(struct bnx2 *bp) bnx2_test_intr(struct bnx2 *bp)
{ {
int i; int i;
u32 val;
u16 status_idx; u16 status_idx;
if (!netif_running(bp->dev)) if (!netif_running(bp->dev))
...@@ -3948,8 +3954,7 @@ bnx2_test_intr(struct bnx2 *bp) ...@@ -3948,8 +3954,7 @@ bnx2_test_intr(struct bnx2 *bp)
status_idx = REG_RD(bp, BNX2_PCICFG_INT_ACK_CMD) & 0xffff; status_idx = REG_RD(bp, BNX2_PCICFG_INT_ACK_CMD) & 0xffff;
/* This register is not touched during run-time. */ /* This register is not touched during run-time. */
val = REG_RD(bp, BNX2_HC_COMMAND); REG_WR(bp, BNX2_HC_COMMAND, bp->hc_cmd | BNX2_HC_COMMAND_COAL_NOW);
REG_WR(bp, BNX2_HC_COMMAND, val | BNX2_HC_COMMAND_COAL_NOW);
REG_RD(bp, BNX2_HC_COMMAND); REG_RD(bp, BNX2_HC_COMMAND);
for (i = 0; i < 10; i++) { for (i = 0; i < 10; i++) {
......
...@@ -4038,6 +4038,7 @@ struct bnx2 { ...@@ -4038,6 +4038,7 @@ struct bnx2 {
struct statistics_block *stats_blk; struct statistics_block *stats_blk;
dma_addr_t stats_blk_mapping; dma_addr_t stats_blk_mapping;
u32 hc_cmd;
u32 rx_mode; u32 rx_mode;
u16 req_line_speed; u16 req_line_speed;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment