Commit 8535ef05 authored by Mike Mason's avatar Mike Mason Committed by Benjamin Herrenschmidt

powerpc/eeh: Only disable/enable LSI interrupts in EEH

The EEH code disables and enables interrupts during the
device recovery process.  This is unnecessary for MSI
and MSI-X interrupts because they are effectively disabled
by the DMA Stopped state when an EEH error occurs.  The
current code is also incorrect for MSI-X interrupts.  It
doesn't take into account that MSI-X interrupts are tracked
in a different way than LSI/MSI interrupts.  This patch
ensures only LSI interrupts are disabled/enabled.
Signed-off-by: default avatarMike Mason <mmlnx@us.ibm.com>
Acked-by: default avatarLinas Vepstas <linasvepstas@gmail.com>
Signed-off-by: default avatarBenjamin Herrenschmidt <benh@kernel.crashing.org>
parent 10156cea
...@@ -79,6 +79,40 @@ static int irq_in_use(unsigned int irq) ...@@ -79,6 +79,40 @@ static int irq_in_use(unsigned int irq)
return rc; return rc;
} }
/**
* eeh_disable_irq - disable interrupt for the recovering device
*/
static void eeh_disable_irq(struct pci_dev *dev)
{
struct device_node *dn = pci_device_to_OF_node(dev);
/* Don't disable MSI and MSI-X interrupts. They are
* effectively disabled by the DMA Stopped state
* when an EEH error occurs.
*/
if (dev->msi_enabled || dev->msix_enabled)
return;
if (!irq_in_use(dev->irq))
return;
PCI_DN(dn)->eeh_mode |= EEH_MODE_IRQ_DISABLED;
disable_irq_nosync(dev->irq);
}
/**
* eeh_enable_irq - enable interrupt for the recovering device
*/
static void eeh_enable_irq(struct pci_dev *dev)
{
struct device_node *dn = pci_device_to_OF_node(dev);
if ((PCI_DN(dn)->eeh_mode) & EEH_MODE_IRQ_DISABLED) {
PCI_DN(dn)->eeh_mode &= ~EEH_MODE_IRQ_DISABLED;
enable_irq(dev->irq);
}
}
/* ------------------------------------------------------- */ /* ------------------------------------------------------- */
/** /**
* eeh_report_error - report pci error to each device driver * eeh_report_error - report pci error to each device driver
...@@ -98,11 +132,8 @@ static void eeh_report_error(struct pci_dev *dev, void *userdata) ...@@ -98,11 +132,8 @@ static void eeh_report_error(struct pci_dev *dev, void *userdata)
if (!driver) if (!driver)
return; return;
if (irq_in_use (dev->irq)) { eeh_disable_irq(dev);
struct device_node *dn = pci_device_to_OF_node(dev);
PCI_DN(dn)->eeh_mode |= EEH_MODE_IRQ_DISABLED;
disable_irq_nosync(dev->irq);
}
if (!driver->err_handler || if (!driver->err_handler ||
!driver->err_handler->error_detected) !driver->err_handler->error_detected)
return; return;
...@@ -147,15 +178,12 @@ static void eeh_report_reset(struct pci_dev *dev, void *userdata) ...@@ -147,15 +178,12 @@ static void eeh_report_reset(struct pci_dev *dev, void *userdata)
{ {
enum pci_ers_result rc, *res = userdata; enum pci_ers_result rc, *res = userdata;
struct pci_driver *driver = dev->driver; struct pci_driver *driver = dev->driver;
struct device_node *dn = pci_device_to_OF_node(dev);
if (!driver) if (!driver)
return; return;
if ((PCI_DN(dn)->eeh_mode) & EEH_MODE_IRQ_DISABLED) { eeh_enable_irq(dev);
PCI_DN(dn)->eeh_mode &= ~EEH_MODE_IRQ_DISABLED;
enable_irq(dev->irq);
}
if (!driver->err_handler || if (!driver->err_handler ||
!driver->err_handler->slot_reset) !driver->err_handler->slot_reset)
return; return;
...@@ -174,17 +202,14 @@ static void eeh_report_reset(struct pci_dev *dev, void *userdata) ...@@ -174,17 +202,14 @@ static void eeh_report_reset(struct pci_dev *dev, void *userdata)
static void eeh_report_resume(struct pci_dev *dev, void *userdata) static void eeh_report_resume(struct pci_dev *dev, void *userdata)
{ {
struct pci_driver *driver = dev->driver; struct pci_driver *driver = dev->driver;
struct device_node *dn = pci_device_to_OF_node(dev);
dev->error_state = pci_channel_io_normal; dev->error_state = pci_channel_io_normal;
if (!driver) if (!driver)
return; return;
if ((PCI_DN(dn)->eeh_mode) & EEH_MODE_IRQ_DISABLED) { eeh_enable_irq(dev);
PCI_DN(dn)->eeh_mode &= ~EEH_MODE_IRQ_DISABLED;
enable_irq(dev->irq);
}
if (!driver->err_handler || if (!driver->err_handler ||
!driver->err_handler->resume) !driver->err_handler->resume)
return; return;
...@@ -208,15 +233,12 @@ static void eeh_report_failure(struct pci_dev *dev, void *userdata) ...@@ -208,15 +233,12 @@ static void eeh_report_failure(struct pci_dev *dev, void *userdata)
if (!driver) if (!driver)
return; return;
if (irq_in_use (dev->irq)) { eeh_disable_irq(dev);
struct device_node *dn = pci_device_to_OF_node(dev);
PCI_DN(dn)->eeh_mode |= EEH_MODE_IRQ_DISABLED; if (!driver->err_handler ||
disable_irq_nosync(dev->irq); !driver->err_handler->error_detected)
}
if (!driver->err_handler)
return;
if (!driver->err_handler->error_detected)
return; return;
driver->err_handler->error_detected(dev, pci_channel_io_perm_failure); driver->err_handler->error_detected(dev, pci_channel_io_perm_failure);
} }
......
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