Commit 40b01727 authored by Stephen Hemminger's avatar Stephen Hemminger Committed by Jeff Garzik

sky2: handle descriptor errors

There should never be descriptor error unless hardware or driver is buggy.
But if an error occurs, print useful information, clear irq, and recover.
Signed-off-by: default avatarStephen Hemminger <shemminger@linux-foundation.org>
Signed-off-by: default avatarJeff Garzik <jeff@garzik.org>
parent 0a17e4c2
...@@ -2343,26 +2343,22 @@ static void sky2_mac_intr(struct sky2_hw *hw, unsigned port) ...@@ -2343,26 +2343,22 @@ static void sky2_mac_intr(struct sky2_hw *hw, unsigned port)
} }
} }
/* This should never happen it is a fatal situation */ /* This should never happen it is a bug. */
static void sky2_descriptor_error(struct sky2_hw *hw, unsigned port, static void sky2_le_error(struct sky2_hw *hw, unsigned port,
const char *rxtx, u32 mask) u16 q, unsigned ring_size)
{ {
struct net_device *dev = hw->dev[port]; struct net_device *dev = hw->dev[port];
struct sky2_port *sky2 = netdev_priv(dev); struct sky2_port *sky2 = netdev_priv(dev);
u32 imask; unsigned idx;
const u64 *le = (q == Q_R1 || q == Q_R2)
printk(KERN_ERR PFX "%s: %s descriptor error (hardware problem)\n", ? (u64 *) sky2->rx_le : (u64 *) sky2->tx_le;
dev ? dev->name : "<not registered>", rxtx);
imask = sky2_read32(hw, B0_IMSK); idx = sky2_read16(hw, Y2_QADDR(q, PREF_UNIT_GET_IDX));
imask &= ~mask; printk(KERN_ERR PFX "%s: descriptor error q=%#x get=%u [%llx] put=%u\n",
sky2_write32(hw, B0_IMSK, imask); dev->name, (unsigned) q, idx, (unsigned long long) le[idx],
(unsigned) sky2_read16(hw, Y2_QADDR(q, PREF_UNIT_PUT_IDX)));
if (dev) { sky2_write32(hw, Q_ADDR(q, Q_CSR), BMU_CLR_IRQ_CHK);
spin_lock(&sky2->phy_lock);
sky2_link_down(sky2);
spin_unlock(&sky2->phy_lock);
}
} }
/* If idle then force a fake soft NAPI poll once a second /* If idle then force a fake soft NAPI poll once a second
...@@ -2386,23 +2382,15 @@ static void sky2_idle(unsigned long arg) ...@@ -2386,23 +2382,15 @@ static void sky2_idle(unsigned long arg)
mod_timer(&hw->idle_timer, jiffies + msecs_to_jiffies(idle_timeout)); mod_timer(&hw->idle_timer, jiffies + msecs_to_jiffies(idle_timeout));
} }
/* Hardware/software error handling */
static int sky2_poll(struct net_device *dev0, int *budget) static void sky2_err_intr(struct sky2_hw *hw, u32 status)
{ {
struct sky2_hw *hw = ((struct sky2_port *) netdev_priv(dev0))->hw; if (net_ratelimit())
int work_limit = min(dev0->quota, *budget); dev_warn(&hw->pdev->dev, "error interrupt status=%#x\n", status);
int work_done = 0;
u32 status = sky2_read32(hw, B0_Y2_SP_EISR);
if (status & Y2_IS_HW_ERR) if (status & Y2_IS_HW_ERR)
sky2_hw_intr(hw); sky2_hw_intr(hw);
if (status & Y2_IS_IRQ_PHY1)
sky2_phy_intr(hw, 0);
if (status & Y2_IS_IRQ_PHY2)
sky2_phy_intr(hw, 1);
if (status & Y2_IS_IRQ_MAC1) if (status & Y2_IS_IRQ_MAC1)
sky2_mac_intr(hw, 0); sky2_mac_intr(hw, 0);
...@@ -2410,16 +2398,33 @@ static int sky2_poll(struct net_device *dev0, int *budget) ...@@ -2410,16 +2398,33 @@ static int sky2_poll(struct net_device *dev0, int *budget)
sky2_mac_intr(hw, 1); sky2_mac_intr(hw, 1);
if (status & Y2_IS_CHK_RX1) if (status & Y2_IS_CHK_RX1)
sky2_descriptor_error(hw, 0, "receive", Y2_IS_CHK_RX1); sky2_le_error(hw, 0, Q_R1, RX_LE_SIZE);
if (status & Y2_IS_CHK_RX2) if (status & Y2_IS_CHK_RX2)
sky2_descriptor_error(hw, 1, "receive", Y2_IS_CHK_RX2); sky2_le_error(hw, 1, Q_R2, RX_LE_SIZE);
if (status & Y2_IS_CHK_TXA1) if (status & Y2_IS_CHK_TXA1)
sky2_descriptor_error(hw, 0, "transmit", Y2_IS_CHK_TXA1); sky2_le_error(hw, 0, Q_XA1, TX_RING_SIZE);
if (status & Y2_IS_CHK_TXA2) if (status & Y2_IS_CHK_TXA2)
sky2_descriptor_error(hw, 1, "transmit", Y2_IS_CHK_TXA2); sky2_le_error(hw, 1, Q_XA2, TX_RING_SIZE);
}
static int sky2_poll(struct net_device *dev0, int *budget)
{
struct sky2_hw *hw = ((struct sky2_port *) netdev_priv(dev0))->hw;
int work_limit = min(dev0->quota, *budget);
int work_done = 0;
u32 status = sky2_read32(hw, B0_Y2_SP_EISR);
if (unlikely(status & Y2_IS_ERROR))
sky2_err_intr(hw, status);
if (status & Y2_IS_IRQ_PHY1)
sky2_phy_intr(hw, 0);
if (status & Y2_IS_IRQ_PHY2)
sky2_phy_intr(hw, 1);
work_done = sky2_status_intr(hw, work_limit); work_done = sky2_status_intr(hw, work_limit);
if (work_done < work_limit) { if (work_done < work_limit) {
......
...@@ -288,6 +288,9 @@ enum { ...@@ -288,6 +288,9 @@ enum {
| Y2_IS_CHK_TXA1 | Y2_IS_CHK_RX1, | Y2_IS_CHK_TXA1 | Y2_IS_CHK_RX1,
Y2_IS_PORT_2 = Y2_IS_IRQ_PHY2 | Y2_IS_IRQ_MAC2 Y2_IS_PORT_2 = Y2_IS_IRQ_PHY2 | Y2_IS_IRQ_MAC2
| Y2_IS_CHK_TXA2 | Y2_IS_CHK_RX2, | Y2_IS_CHK_TXA2 | Y2_IS_CHK_RX2,
Y2_IS_ERROR = Y2_IS_HW_ERR |
Y2_IS_IRQ_MAC1 | Y2_IS_CHK_TXA1 | Y2_IS_CHK_RX1 |
Y2_IS_IRQ_MAC2 | Y2_IS_CHK_TXA2 | Y2_IS_CHK_RX2,
}; };
/* B2_IRQM_HWE_MSK 32 bit IRQ Moderation HW Error Mask */ /* B2_IRQM_HWE_MSK 32 bit IRQ Moderation HW Error Mask */
......
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