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

PCI: pciehp: Ignore Link Down/Up caused by error-induced Hot Reset

Stuart Hayes reports that an error handled by DPC at a Root Port results
in pciehp gratuitously bringing down a subordinate hotplug port:

RP -- UP -- DP -- UP -- DP (hotplug) -- EP

pciehp brings the slot down because the Link to the Endpoint goes down.
That is caused by a Hot Reset being propagated as a result of DPC.
Per PCIe Base Spec 5.0, section 6.6.1 "Conventional Reset":

For a Switch, the following must cause a hot reset to be sent on all
Downstream Ports: [...]

* The Data Link Layer of the Upstream Port reporting DL_Down status.
In Switches that support Link speeds greater than 5.0 GT/s, the
Upstream Port must direct the LTSSM of each Downstream Port to the
Hot Reset state, but not hold the LTSSMs in that state. This permits
each Downstream Port to begin Link training immediately after its
hot reset completes. This behavior is recommended for all Switches.

* Receiving a hot reset on the Upstream Port.

Once DPC recovers, pcie_do_recovery() walks down the hierarchy and
invokes pcie_portdrv_slot_reset() to restore each port's config space.
At that point, a hotplug interrupt is signaled per PCIe Base Spec r5.0,
section 6.7.3.4 "Software Notification of Hot-Plug Events":

If the Port is enabled for edge-triggered interrupt signaling using
MSI or MSI-X, an interrupt message must be sent every time the logical
AND of the following conditions transitions from FALSE to TRUE: [...]

* The Hot-Plug Interrupt Enable bit in the Slot Control register is
set to 1b.

* At least one hot-plug event status bit in the Slot Status register
and its associated enable bit in the Slot Control register are both
set to 1b.

Prevent pciehp from gratuitously bringing down the slot by clearing the
error-induced Data Link Layer State Changed event before restoring
config space. Afterwards, check whether the link has unexpectedly
failed to retrain and synthesize a DLLSC event if so.

Allow each pcie_port_service_driver (one of them being pciehp) to define
a slot_reset callback and re-use the existing pm_iter() function to
iterate over the callbacks.

Thereby, the Endpoint driver remains bound throughout error recovery and
may restore the device to working state.

Surprise removal during error recovery is detected through a Presence
Detect Changed event. The hotplug port is expected to not signal that
event as a result of a Hot Reset.

The issue isn't DPC-specific, it also occurs when an error is handled by
AER through aer_root_reset(). So while the issue was noticed only now,
it's been around since 2006 when AER support was first introduced.

[bhelgaas: drop PCI_ERROR_RECOVERY Kconfig, split pm_iter() rename to
preparatory patch]
Link: https://lore.kernel.org/linux-pci/08c046b0-c9f2-3489-eeef-7e7aca435bb9@gmail.com/
Fixes: 6c2b374d7485 ("PCI-Express AER implemetation: AER core and aerdriver")
Link: https://lore.kernel.org/r/251f4edcc04c14f873ff1c967bc686169cd07d2d.1627638184.git.lukas@wunner.de
Reported-by: Stuart Hayes <stuart.w.hayes@gmail.com>
Tested-by: Stuart Hayes <stuart.w.hayes@gmail.com>
Signed-off-by: Lukas Wunner <lukas@wunner.de>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Cc: stable@vger.kernel.org # v2.6.19+: ba952824e6c1: PCI/portdrv: Report reset for frozen channel
Cc: Keith Busch <kbusch@kernel.org>

authored by

Lukas Wunner and committed by
Bjorn Helgaas
ea401499 3134689f

+35
+2
drivers/pci/hotplug/pciehp.h
··· 189 189 int pciehp_set_raw_indicator_status(struct hotplug_slot *h_slot, u8 status); 190 190 int pciehp_get_raw_indicator_status(struct hotplug_slot *h_slot, u8 *status); 191 191 192 + int pciehp_slot_reset(struct pcie_device *dev); 193 + 192 194 static inline const char *slot_name(struct controller *ctrl) 193 195 { 194 196 return hotplug_slot_name(&ctrl->hotplug_slot);
+2
drivers/pci/hotplug/pciehp_core.c
··· 351 351 .runtime_suspend = pciehp_runtime_suspend, 352 352 .runtime_resume = pciehp_runtime_resume, 353 353 #endif /* PM */ 354 + 355 + .slot_reset = pciehp_slot_reset, 354 356 }; 355 357 356 358 int __init pcie_hp_init(void)
+26
drivers/pci/hotplug/pciehp_hpc.c
··· 862 862 pcie_write_cmd(ctrl, 0, mask); 863 863 } 864 864 865 + /** 866 + * pciehp_slot_reset() - ignore link event caused by error-induced hot reset 867 + * @dev: PCI Express port service device 868 + * 869 + * Called from pcie_portdrv_slot_reset() after AER or DPC initiated a reset 870 + * further up in the hierarchy to recover from an error. The reset was 871 + * propagated down to this hotplug port. Ignore the resulting link flap. 872 + * If the link failed to retrain successfully, synthesize the ignored event. 873 + * Surprise removal during reset is detected through Presence Detect Changed. 874 + */ 875 + int pciehp_slot_reset(struct pcie_device *dev) 876 + { 877 + struct controller *ctrl = get_service_data(dev); 878 + 879 + if (ctrl->state != ON_STATE) 880 + return 0; 881 + 882 + pcie_capability_write_word(dev->port, PCI_EXP_SLTSTA, 883 + PCI_EXP_SLTSTA_DLLSC); 884 + 885 + if (!pciehp_check_link_active(ctrl)) 886 + pciehp_request(ctrl, PCI_EXP_SLTSTA_DLLSC); 887 + 888 + return 0; 889 + } 890 + 865 891 /* 866 892 * pciehp has a 1:1 bus:slot relationship so we ultimately want a secondary 867 893 * bus reset of the bridge, but at the same time we want to ensure that it is
+2
drivers/pci/pcie/portdrv.h
··· 85 85 int (*runtime_suspend)(struct pcie_device *dev); 86 86 int (*runtime_resume)(struct pcie_device *dev); 87 87 88 + int (*slot_reset)(struct pcie_device *dev); 89 + 88 90 /* Device driver may resume normal operations */ 89 91 void (*error_resume)(struct pci_dev *dev); 90 92
+3
drivers/pci/pcie/portdrv_pci.c
··· 160 160 161 161 static pci_ers_result_t pcie_portdrv_slot_reset(struct pci_dev *dev) 162 162 { 163 + size_t off = offsetof(struct pcie_port_service_driver, slot_reset); 164 + device_for_each_child(&dev->dev, &off, pcie_port_device_iter); 165 + 163 166 pci_restore_state(dev); 164 167 pci_save_state(dev); 165 168 return PCI_ERS_RESULT_RECOVERED;