|
@@ -2343,26 +2343,22 @@ static void sky2_mac_intr(struct sky2_hw *hw, unsigned port)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-/* This should never happen it is a fatal situation */
|
|
|
|
-static void sky2_descriptor_error(struct sky2_hw *hw, unsigned port,
|
|
|
|
- const char *rxtx, u32 mask)
|
|
|
|
|
|
+/* This should never happen it is a bug. */
|
|
|
|
+static void sky2_le_error(struct sky2_hw *hw, unsigned port,
|
|
|
|
+ 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;
|
|
|
|
-
|
|
|
|
- printk(KERN_ERR PFX "%s: %s descriptor error (hardware problem)\n",
|
|
|
|
- dev ? dev->name : "<not registered>", rxtx);
|
|
|
|
|
|
+ unsigned idx;
|
|
|
|
+ const u64 *le = (q == Q_R1 || q == Q_R2)
|
|
|
|
+ ? (u64 *) sky2->rx_le : (u64 *) sky2->tx_le;
|
|
|
|
|
|
- imask = sky2_read32(hw, B0_IMSK);
|
|
|
|
- imask &= ~mask;
|
|
|
|
- sky2_write32(hw, B0_IMSK, imask);
|
|
|
|
|
|
+ idx = sky2_read16(hw, Y2_QADDR(q, PREF_UNIT_GET_IDX));
|
|
|
|
+ printk(KERN_ERR PFX "%s: descriptor error q=%#x get=%u [%llx] put=%u\n",
|
|
|
|
+ dev->name, (unsigned) q, idx, (unsigned long long) le[idx],
|
|
|
|
+ (unsigned) sky2_read16(hw, Y2_QADDR(q, PREF_UNIT_PUT_IDX)));
|
|
|
|
|
|
- if (dev) {
|
|
|
|
- spin_lock(&sky2->phy_lock);
|
|
|
|
- sky2_link_down(sky2);
|
|
|
|
- spin_unlock(&sky2->phy_lock);
|
|
|
|
- }
|
|
|
|
|
|
+ sky2_write32(hw, Q_ADDR(q, Q_CSR), BMU_CLR_IRQ_CHK);
|
|
}
|
|
}
|
|
|
|
|
|
/* 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)
|
|
mod_timer(&hw->idle_timer, jiffies + msecs_to_jiffies(idle_timeout));
|
|
mod_timer(&hw->idle_timer, jiffies + msecs_to_jiffies(idle_timeout));
|
|
}
|
|
}
|
|
|
|
|
|
-
|
|
|
|
-static int sky2_poll(struct net_device *dev0, int *budget)
|
|
|
|
|
|
+/* Hardware/software error handling */
|
|
|
|
+static void sky2_err_intr(struct sky2_hw *hw, u32 status)
|
|
{
|
|
{
|
|
- 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 (net_ratelimit())
|
|
|
|
+ dev_warn(&hw->pdev->dev, "error interrupt status=%#x\n", status);
|
|
|
|
|
|
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)
|
|
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) {
|