Commit 39a70d79 authored by Arjan van de Ven's avatar Arjan van de Ven Committed by Jeff Garzik

Update eepro100 net driver to issue soft rx reset for certain cases, fixing several

reports of hangs or lockups (of the NIC, not the entire system).

Author: Steve Parker @ Sun
parent 8903bc1e
...@@ -1552,82 +1552,39 @@ static void speedo_interrupt(int irq, void *dev_instance, struct pt_regs *regs) ...@@ -1552,82 +1552,39 @@ static void speedo_interrupt(int irq, void *dev_instance, struct pt_regs *regs)
if ((status & 0xfc00) == 0) if ((status & 0xfc00) == 0)
break; break;
/* Always check if all rx buffers are allocated. --SAW */
speedo_refill_rx_buffers(dev, 0);
if ((status & 0x5000) || /* Packet received, or Rx error. */ if ((status & 0x5000) || /* Packet received, or Rx error. */
(sp->rx_ring_state&(RrNoMem|RrPostponed)) == RrPostponed) (sp->rx_ring_state&(RrNoMem|RrPostponed)) == RrPostponed)
/* Need to gather the postponed packet. */ /* Need to gather the postponed packet. */
speedo_rx(dev); speedo_rx(dev);
if (status & 0x1000) { /* Always check if all rx buffers are allocated. --SAW */
spin_lock(&sp->lock); speedo_refill_rx_buffers(dev, 0);
if ((status & 0x003c) == 0x0028) { /* No more Rx buffers. */
struct RxFD *rxf;
printk(KERN_WARNING "%s: card reports no RX buffers.\n",
dev->name);
rxf = sp->rx_ringp[sp->cur_rx % RX_RING_SIZE];
if (rxf == NULL) {
if (speedo_debug > 2)
printk(KERN_DEBUG
"%s: NULL cur_rx in speedo_interrupt().\n",
dev->name);
sp->rx_ring_state |= RrNoMem|RrNoResources;
} else if (rxf == sp->last_rxf) {
if (speedo_debug > 2)
printk(KERN_DEBUG
"%s: cur_rx is last in speedo_interrupt().\n",
dev->name);
sp->rx_ring_state |= RrNoMem|RrNoResources;
} else
outb(RxResumeNoResources, ioaddr + SCBCmd);
} else if ((status & 0x003c) == 0x0008) { /* No resources. */
struct RxFD *rxf;
printk(KERN_WARNING "%s: card reports no resources.\n",
dev->name);
rxf = sp->rx_ringp[sp->cur_rx % RX_RING_SIZE];
if (rxf == NULL) {
if (speedo_debug > 2)
printk(KERN_DEBUG
"%s: NULL cur_rx in speedo_interrupt().\n",
dev->name);
sp->rx_ring_state |= RrNoMem|RrNoResources;
} else if (rxf == sp->last_rxf) {
if (speedo_debug > 2)
printk(KERN_DEBUG
"%s: cur_rx is last in speedo_interrupt().\n",
dev->name);
sp->rx_ring_state |= RrNoMem|RrNoResources;
} else {
/* Restart the receiver. */
outl(sp->rx_ring_dma[sp->cur_rx % RX_RING_SIZE],
ioaddr + SCBPointer);
outb(RxStart, ioaddr + SCBCmd);
}
}
sp->stats.rx_errors++;
spin_unlock(&sp->lock);
}
if ((sp->rx_ring_state&(RrNoMem|RrNoResources)) == RrNoResources) {
printk(KERN_WARNING
"%s: restart the receiver after a possible hang.\n",
dev->name);
spin_lock(&sp->lock); spin_lock(&sp->lock);
/* Restart the receiver. /*
I'm not sure if it's always right to restart the receiver * The chip may have suspended reception for various reasons.
here but I don't know another way to prevent receiver hangs. * Check for that, and re-prime it should this be the case.
1999/12/25 SAW */ */
outl(sp->rx_ring_dma[sp->cur_rx % RX_RING_SIZE], switch ((status >> 2) & 0xf) {
ioaddr + SCBPointer); case 0: /* Idle */
outb(RxStart, ioaddr + SCBCmd); break;
sp->rx_ring_state &= ~RrNoResources; case 1: /* Suspended */
spin_unlock(&sp->lock); case 2: /* No resources (RxFDs) */
case 9: /* Suspended with no more RBDs */
case 10: /* No resources due to no RBDs */
case 12: /* Ready with no RBDs */
speedo_rx_soft_reset(dev);
break;
case 3: case 5: case 6: case 7: case 8:
case 11: case 13: case 14: case 15:
/* these are all reserved values */
break;
} }
/* User interrupt, Command/Tx unit interrupt or CU not active. */ /* User interrupt, Command/Tx unit interrupt or CU not active. */
if (status & 0xA400) { if (status & 0xA400) {
spin_lock(&sp->lock);
speedo_tx_buffer_gc(dev); speedo_tx_buffer_gc(dev);
if (sp->tx_full if (sp->tx_full
&& (int)(sp->cur_tx - sp->dirty_tx) < TX_QUEUE_UNFULL) { && (int)(sp->cur_tx - sp->dirty_tx) < TX_QUEUE_UNFULL) {
...@@ -1635,9 +1592,10 @@ static void speedo_interrupt(int irq, void *dev_instance, struct pt_regs *regs) ...@@ -1635,9 +1592,10 @@ static void speedo_interrupt(int irq, void *dev_instance, struct pt_regs *regs)
sp->tx_full = 0; sp->tx_full = 0;
netif_wake_queue(dev); /* Attention: under a spinlock. --SAW */ netif_wake_queue(dev); /* Attention: under a spinlock. --SAW */
} }
spin_unlock(&sp->lock);
} }
spin_unlock(&sp->lock);
if (--boguscnt < 0) { if (--boguscnt < 0) {
printk(KERN_ERR "%s: Too much work at interrupt, status=0x%4.4x.\n", printk(KERN_ERR "%s: Too much work at interrupt, status=0x%4.4x.\n",
dev->name, status); dev->name, status);
......
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