• Lukas Wunner's avatar
    PCI: pciehp: Ignore Link Down/Up caused by error-induced Hot Reset · ea401499
    Lukas Wunner authored
    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: 6c2b374d ("PCI-Express AER implemetation: AER core and aerdriver")
    Link: https://lore.kernel.org/r/251f4edcc04c14f873ff1c967bc686169cd07d2d.1627638184.git.lukas@wunner.deReported-by: default avatarStuart Hayes <stuart.w.hayes@gmail.com>
    Tested-by: default avatarStuart Hayes <stuart.w.hayes@gmail.com>
    Signed-off-by: default avatarLukas Wunner <lukas@wunner.de>
    Signed-off-by: default avatarBjorn Helgaas <bhelgaas@google.com>
    Cc: stable@vger.kernel.org # v2.6.19+: ba952824: PCI/portdrv: Report reset for frozen channel
    Cc: Keith Busch <kbusch@kernel.org>
    ea401499
portdrv_pci.c 6.79 KB