Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

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: Mike Mason <mmlnx@us.ibm.com>
Acked-by: Linas Vepstas <linasvepstas@gmail.com>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>

authored by

Mike Mason and committed by
Benjamin Herrenschmidt
8535ef05 10156cea

+45 -23
+45 -23
arch/powerpc/platforms/pseries/eeh_driver.c
··· 79 79 return rc; 80 80 } 81 81 82 + /** 83 + * eeh_disable_irq - disable interrupt for the recovering device 84 + */ 85 + static void eeh_disable_irq(struct pci_dev *dev) 86 + { 87 + struct device_node *dn = pci_device_to_OF_node(dev); 88 + 89 + /* Don't disable MSI and MSI-X interrupts. They are 90 + * effectively disabled by the DMA Stopped state 91 + * when an EEH error occurs. 92 + */ 93 + if (dev->msi_enabled || dev->msix_enabled) 94 + return; 95 + 96 + if (!irq_in_use(dev->irq)) 97 + return; 98 + 99 + PCI_DN(dn)->eeh_mode |= EEH_MODE_IRQ_DISABLED; 100 + disable_irq_nosync(dev->irq); 101 + } 102 + 103 + /** 104 + * eeh_enable_irq - enable interrupt for the recovering device 105 + */ 106 + static void eeh_enable_irq(struct pci_dev *dev) 107 + { 108 + struct device_node *dn = pci_device_to_OF_node(dev); 109 + 110 + if ((PCI_DN(dn)->eeh_mode) & EEH_MODE_IRQ_DISABLED) { 111 + PCI_DN(dn)->eeh_mode &= ~EEH_MODE_IRQ_DISABLED; 112 + enable_irq(dev->irq); 113 + } 114 + } 115 + 82 116 /* ------------------------------------------------------- */ 83 117 /** 84 118 * eeh_report_error - report pci error to each device driver ··· 132 98 if (!driver) 133 99 return; 134 100 135 - if (irq_in_use (dev->irq)) { 136 - struct device_node *dn = pci_device_to_OF_node(dev); 137 - PCI_DN(dn)->eeh_mode |= EEH_MODE_IRQ_DISABLED; 138 - disable_irq_nosync(dev->irq); 139 - } 101 + eeh_disable_irq(dev); 102 + 140 103 if (!driver->err_handler || 141 104 !driver->err_handler->error_detected) 142 105 return; ··· 178 147 { 179 148 enum pci_ers_result rc, *res = userdata; 180 149 struct pci_driver *driver = dev->driver; 181 - struct device_node *dn = pci_device_to_OF_node(dev); 182 150 183 151 if (!driver) 184 152 return; 185 153 186 - if ((PCI_DN(dn)->eeh_mode) & EEH_MODE_IRQ_DISABLED) { 187 - PCI_DN(dn)->eeh_mode &= ~EEH_MODE_IRQ_DISABLED; 188 - enable_irq(dev->irq); 189 - } 154 + eeh_enable_irq(dev); 155 + 190 156 if (!driver->err_handler || 191 157 !driver->err_handler->slot_reset) 192 158 return; ··· 202 174 static void eeh_report_resume(struct pci_dev *dev, void *userdata) 203 175 { 204 176 struct pci_driver *driver = dev->driver; 205 - struct device_node *dn = pci_device_to_OF_node(dev); 206 177 207 178 dev->error_state = pci_channel_io_normal; 208 179 209 180 if (!driver) 210 181 return; 211 182 212 - if ((PCI_DN(dn)->eeh_mode) & EEH_MODE_IRQ_DISABLED) { 213 - PCI_DN(dn)->eeh_mode &= ~EEH_MODE_IRQ_DISABLED; 214 - enable_irq(dev->irq); 215 - } 183 + eeh_enable_irq(dev); 184 + 216 185 if (!driver->err_handler || 217 186 !driver->err_handler->resume) 218 187 return; ··· 233 208 if (!driver) 234 209 return; 235 210 236 - if (irq_in_use (dev->irq)) { 237 - struct device_node *dn = pci_device_to_OF_node(dev); 238 - PCI_DN(dn)->eeh_mode |= EEH_MODE_IRQ_DISABLED; 239 - disable_irq_nosync(dev->irq); 240 - } 241 - if (!driver->err_handler) 211 + eeh_disable_irq(dev); 212 + 213 + if (!driver->err_handler || 214 + !driver->err_handler->error_detected) 242 215 return; 243 - if (!driver->err_handler->error_detected) 244 - return; 216 + 245 217 driver->err_handler->error_detected(dev, pci_channel_io_perm_failure); 246 218 } 247 219