Commit 38892570 authored by Manfred Spraul's avatar Manfred Spraul Committed by Greg Kroah-Hartman

forcedeth msi bugfix

patch a7475906 in mainline.

pci_enable_msi() replaces the INTx irq number in pci_dev->irq with the
new MSI irq number.
The forcedeth driver did not update the copy in netdevice->irq and
parts of the driver used the stale copy.
See bugzilla.kernel.org, bug 9047.

The patch
- updates netdevice->irq
- replaces all accesses to netdevice->irq with pci_dev->irq.

The patch is against 2.6.23.1. IMHO suitable for both 2.6.23 and 2.6.24
Signed-off-by: default avatarManfred Spraul <manfred@colorfullife.com>
Signed-off-by: default avatarJeff Garzik <jeff@garzik.org>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent c5b130d0
...@@ -988,7 +988,7 @@ static void nv_enable_irq(struct net_device *dev) ...@@ -988,7 +988,7 @@ static void nv_enable_irq(struct net_device *dev)
if (np->msi_flags & NV_MSI_X_ENABLED) if (np->msi_flags & NV_MSI_X_ENABLED)
enable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector); enable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector);
else else
enable_irq(dev->irq); enable_irq(np->pci_dev->irq);
} else { } else {
enable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_RX].vector); enable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_RX].vector);
enable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_TX].vector); enable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_TX].vector);
...@@ -1004,7 +1004,7 @@ static void nv_disable_irq(struct net_device *dev) ...@@ -1004,7 +1004,7 @@ static void nv_disable_irq(struct net_device *dev)
if (np->msi_flags & NV_MSI_X_ENABLED) if (np->msi_flags & NV_MSI_X_ENABLED)
disable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector); disable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector);
else else
disable_irq(dev->irq); disable_irq(np->pci_dev->irq);
} else { } else {
disable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_RX].vector); disable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_RX].vector);
disable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_TX].vector); disable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_TX].vector);
...@@ -1601,7 +1601,7 @@ static void nv_do_rx_refill(unsigned long data) ...@@ -1601,7 +1601,7 @@ static void nv_do_rx_refill(unsigned long data)
if (np->msi_flags & NV_MSI_X_ENABLED) if (np->msi_flags & NV_MSI_X_ENABLED)
disable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector); disable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector);
else else
disable_irq(dev->irq); disable_irq(np->pci_dev->irq);
} else { } else {
disable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_RX].vector); disable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_RX].vector);
} }
...@@ -1619,7 +1619,7 @@ static void nv_do_rx_refill(unsigned long data) ...@@ -1619,7 +1619,7 @@ static void nv_do_rx_refill(unsigned long data)
if (np->msi_flags & NV_MSI_X_ENABLED) if (np->msi_flags & NV_MSI_X_ENABLED)
enable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector); enable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector);
else else
enable_irq(dev->irq); enable_irq(np->pci_dev->irq);
} else { } else {
enable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_RX].vector); enable_irq(np->msi_x_entry[NV_MSI_X_VECTOR_RX].vector);
} }
...@@ -3557,10 +3557,12 @@ static int nv_request_irq(struct net_device *dev, int intr_test) ...@@ -3557,10 +3557,12 @@ static int nv_request_irq(struct net_device *dev, int intr_test)
if (ret != 0 && np->msi_flags & NV_MSI_CAPABLE) { if (ret != 0 && np->msi_flags & NV_MSI_CAPABLE) {
if ((ret = pci_enable_msi(np->pci_dev)) == 0) { if ((ret = pci_enable_msi(np->pci_dev)) == 0) {
np->msi_flags |= NV_MSI_ENABLED; np->msi_flags |= NV_MSI_ENABLED;
dev->irq = np->pci_dev->irq;
if (request_irq(np->pci_dev->irq, handler, IRQF_SHARED, dev->name, dev) != 0) { if (request_irq(np->pci_dev->irq, handler, IRQF_SHARED, dev->name, dev) != 0) {
printk(KERN_INFO "forcedeth: request_irq failed %d\n", ret); printk(KERN_INFO "forcedeth: request_irq failed %d\n", ret);
pci_disable_msi(np->pci_dev); pci_disable_msi(np->pci_dev);
np->msi_flags &= ~NV_MSI_ENABLED; np->msi_flags &= ~NV_MSI_ENABLED;
dev->irq = np->pci_dev->irq;
goto out_err; goto out_err;
} }
...@@ -3623,7 +3625,7 @@ static void nv_do_nic_poll(unsigned long data) ...@@ -3623,7 +3625,7 @@ static void nv_do_nic_poll(unsigned long data)
if (np->msi_flags & NV_MSI_X_ENABLED) if (np->msi_flags & NV_MSI_X_ENABLED)
disable_irq_lockdep(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector); disable_irq_lockdep(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector);
else else
disable_irq_lockdep(dev->irq); disable_irq_lockdep(np->pci_dev->irq);
mask = np->irqmask; mask = np->irqmask;
} else { } else {
if (np->nic_poll_irq & NVREG_IRQ_RX_ALL) { if (np->nic_poll_irq & NVREG_IRQ_RX_ALL) {
...@@ -3641,6 +3643,8 @@ static void nv_do_nic_poll(unsigned long data) ...@@ -3641,6 +3643,8 @@ static void nv_do_nic_poll(unsigned long data)
} }
np->nic_poll_irq = 0; np->nic_poll_irq = 0;
/* disable_irq() contains synchronize_irq, thus no irq handler can run now */
if (np->recover_error) { if (np->recover_error) {
np->recover_error = 0; np->recover_error = 0;
printk(KERN_INFO "forcedeth: MAC in recoverable error state\n"); printk(KERN_INFO "forcedeth: MAC in recoverable error state\n");
...@@ -3677,7 +3681,6 @@ static void nv_do_nic_poll(unsigned long data) ...@@ -3677,7 +3681,6 @@ static void nv_do_nic_poll(unsigned long data)
} }
} }
/* FIXME: Do we need synchronize_irq(dev->irq) here? */
writel(mask, base + NvRegIrqMask); writel(mask, base + NvRegIrqMask);
pci_push(base); pci_push(base);
...@@ -3690,7 +3693,7 @@ static void nv_do_nic_poll(unsigned long data) ...@@ -3690,7 +3693,7 @@ static void nv_do_nic_poll(unsigned long data)
if (np->msi_flags & NV_MSI_X_ENABLED) if (np->msi_flags & NV_MSI_X_ENABLED)
enable_irq_lockdep(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector); enable_irq_lockdep(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector);
else else
enable_irq_lockdep(dev->irq); enable_irq_lockdep(np->pci_dev->irq);
} else { } else {
if (np->nic_poll_irq & NVREG_IRQ_RX_ALL) { if (np->nic_poll_irq & NVREG_IRQ_RX_ALL) {
nv_nic_irq_rx(0, dev); nv_nic_irq_rx(0, dev);
...@@ -4943,7 +4946,7 @@ static int nv_close(struct net_device *dev) ...@@ -4943,7 +4946,7 @@ static int nv_close(struct net_device *dev)
np->in_shutdown = 1; np->in_shutdown = 1;
spin_unlock_irq(&np->lock); spin_unlock_irq(&np->lock);
netif_poll_disable(dev); netif_poll_disable(dev);
synchronize_irq(dev->irq); synchronize_irq(np->pci_dev->irq);
del_timer_sync(&np->oom_kick); del_timer_sync(&np->oom_kick);
del_timer_sync(&np->nic_poll); del_timer_sync(&np->nic_poll);
......
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