Re: [PATCH v2 2/2] nvme-pci: Allow PCI bus-level PM to be used if ASPM is disabled

From: Bjorn Helgaas
Date: Thu Aug 08 2019 - 14:39:58 EST


On Thu, Aug 08, 2019 at 04:47:45PM +0200, Rafael J. Wysocki wrote:
> On Thu, Aug 8, 2019 at 3:43 PM Bjorn Helgaas <helgaas@xxxxxxxxxx> wrote:
> > On Thu, Aug 08, 2019 at 12:10:06PM +0200, Rafael J. Wysocki wrote:
> > > From: Rafael J. Wysocki <rafael.j.wysocki@xxxxxxxxx>
> > >
> > > One of the modifications made by commit d916b1be94b6 ("nvme-pci: use
> > > host managed power state for suspend") was adding a pci_save_state()
> > > call to nvme_suspend() in order to prevent the PCI bus-level PM from
> > > being applied to the suspended NVMe devices, but if ASPM is not
> > > enabled for the target NVMe device, that causes its PCIe link to stay
> > > up and the platform may not be able to get into its optimum low-power
> > > state because of that.
> > >
> > > For example, if ASPM is disabled for the NVMe drive (PC401 NVMe SK
> > > hynix 256GB) in my Dell XPS13 9380, leaving it in D0 during
> > > suspend-to-idle prevents the SoC from reaching package idle states
> > > deeper than PC3, which is way insufficient for system suspend.
> >
> > Just curious: I assume the SoC you reference is some part of the NVMe
> > drive?
>
> No, the SoC is what contains the Intel processor and PCH (formerly "chipset").
>
> > > To address this shortcoming, make nvme_suspend() check if ASPM is
> > > enabled for the target device and fall back to full device shutdown
> > > and PCI bus-level PM if that is not the case.
> > >
> > > Fixes: d916b1be94b6 ("nvme-pci: use host managed power state for suspend")
> > > Link: https://lore.kernel.org/linux-pm/2763495.NmdaWeg79L@kreacher/T/#t
> > > Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@xxxxxxxxx>
> > > ---
> > >
> > > -> v2:
> > > * Move the PCI/PCIe ASPM changes to a separate patch.
> > > * Do not add a redundant ndev->last_ps == U32_MAX check in nvme_suspend().
> > >
> > > ---
> > > drivers/nvme/host/pci.c | 13 ++++++++++---
> > > 1 file changed, 10 insertions(+), 3 deletions(-)
> > >
> > > Index: linux-pm/drivers/nvme/host/pci.c
> > > ===================================================================
> > > --- linux-pm.orig/drivers/nvme/host/pci.c
> > > +++ linux-pm/drivers/nvme/host/pci.c
> > > @@ -2846,7 +2846,7 @@ static int nvme_resume(struct device *de
> > > struct nvme_dev *ndev = pci_get_drvdata(to_pci_dev(dev));
> > > struct nvme_ctrl *ctrl = &ndev->ctrl;
> > >
> > > - if (pm_resume_via_firmware() || !ctrl->npss ||
> > > + if (ndev->last_ps == U32_MAX ||
> > > nvme_set_power_state(ctrl, ndev->last_ps) != 0)
> > > nvme_reset_ctrl(ctrl);
> > > return 0;
> > > @@ -2859,6 +2859,8 @@ static int nvme_suspend(struct device *d
> > > struct nvme_ctrl *ctrl = &ndev->ctrl;
> > > int ret = -EBUSY;
> > >
> > > + ndev->last_ps = U32_MAX;
> > > +
> > > /*
> > > * The platform does not remove power for a kernel managed suspend so
> > > * use host managed nvme power settings for lowest idle power if
> > > @@ -2866,8 +2868,14 @@ static int nvme_suspend(struct device *d
> > > * shutdown. But if the firmware is involved after the suspend or the
> > > * device does not support any non-default power states, shut down the
> > > * device fully.
> > > + *
> > > + * If ASPM is not enabled for the device, shut down the device and allow
> > > + * the PCI bus layer to put it into D3 in order to take the PCIe link
> > > + * down, so as to allow the platform to achieve its minimum low-power
> > > + * state (which may not be possible if the link is up).
> > > */
> > > - if (pm_suspend_via_firmware() || !ctrl->npss) {
> > > + if (pm_suspend_via_firmware() || !ctrl->npss ||
> > > + !pcie_aspm_enabled_mask(pdev)) {
> >
> > This seems like a layering violation, in the sense that ASPM is
> > supposed to be hardware-autonomous and invisible to software.
>
> But software has to enable it.
>
> If it is not enabled, it will not be used, and that's what the check
> is about.
>
> > IIUC the NVMe device will go to the desired package idle state if
> > the link is in L0s or L1, but not if the link is in L0. I don't
> > understand that connection; AFAIK that would be something outside
> > the scope of the PCIe spec.
>
> Yes, it is outside of the PCIe spec.
>
> No, this is not about the NVMe device, it is about the Intel SoC
> (System-on-a-Chip) the platform is based on.

Ah. So this problem could occur with any device, not just NVMe? If
so, how do you address that? Obviously you don't want to patch all
drivers this way.

> The background really is commit d916b1be94b6 and its changelog is
> kind of misleading, unfortunately. What it did, among other things,
> was to cause the NVMe driver to prevent the PCI bus type from
> applying the standard PCI PM to the devices handled by it in the
> suspend-to-idle flow.

This is more meaningful to you than to most people because "applying
the standard PCI PM" doesn't tell us what that means in terms of the
device. Presumably it has something to do with a D-state transition?
I *assume* a suspend might involve the D0 -> D3hot transition you
mention below?

> The reason for doing that was a (reportedly) widespread failure to
> take the PCIe link down during D0 -> D3hot transitions of NVMe
> devices,

I don't know any of the details, but "failure to take the link down
during D0 -> D3hot transitions" is phrased as though it might be a
hardware erratum. If this *is* related to an NVMe erratum, that would
explain why you only need to patch the nvme driver, and it would be
useful to mention that in the commit log, since otherwise it sounds
like something that might be needed in other drivers, too.

According to PCIe r5.0 sec 5.3.2, the only legal link states for D3hot
are L1, L2/L3 Ready. So if you put a device in D3hot and its link
stays in L0, that sounds like a defect. Is that what happens?

Obviously I'm still confused. I think it would help if you could
describe the problem in terms of the specific PCIe states involved
(D0, D3hot, L0, L1, L2, L3, etc) because then the spec would help
explain what's happening.

> which then prevented the platform from going into a deep enough
> low-power state while suspended (because it was not sure whether or
> not the NVMe device was really "sufficiently" inactive). [I guess I
> should mention that in the changelog of the $subject patch.] So the
> idea was to put the (NVMe) device into a low-power state internally
> and then let ASPM take care of the PCIe link.
>
> Of course, that can only work if ASPM is enabled at all for the
> device in question, even though it may not be sufficient as you say
> below.
>
> > The spec (PCIe r5.0, sec 5.4.1.1.1 for L0s, 5.4.1.2.1 for L1) is
> > careful to say that when the conditions are right, devices
> > "should" enter L0s but it is never mandatory, or "may" enter L1.
> >
> > And this patch assumes that if ASPM is enabled, the link will
> > eventually go to L0s or L1.
>
> No, it doesn't.
>
> It avoids failure in the case in which it is guaranteed to happen
> (disabled ASPM) and that's it.
>
> > Because the PCIe spec doesn't mandate that transition, I think
> > this patch makes the driver dependent on device-specific behavior.
>
> IMO not really. It just adds a "don't do it if you are going to
> fail" kind of check.
>
> >
> > > nvme_dev_disable(ndev, true);
> > > return 0;
> > > }
> > > @@ -2880,7 +2888,6 @@ static int nvme_suspend(struct device *d
> > > ctrl->state != NVME_CTRL_ADMIN_ONLY)
> > > goto unfreeze;
> > >
> > > - ndev->last_ps = 0;
> > > ret = nvme_get_power_state(ctrl, &ndev->last_ps);
> > > if (ret < 0)
> > > goto unfreeze;
> > >
> > >
> > >