Re: [RFC PATCH] PCIe: Add PCIe runtime D3cold support

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Thu, Apr 19, 2012 at 5:00 AM, Rafael J. Wysocki <rjw@xxxxxxx> wrote:
> On Wednesday, April 18, 2012, huang ying wrote:
>> On Wed, Apr 18, 2012 at 5:03 AM, Rafael J. Wysocki <rjw@xxxxxxx> wrote:
>> > On Tuesday, April 17, 2012, huang ying wrote:
>> >> On Tue, Apr 17, 2012 at 5:30 AM, Rafael J. Wysocki <rjw@xxxxxxx> wrote:
>> >> > On Monday, April 16, 2012, huang ying wrote:
>> >> >> Hi,
>> >> >>
>> >> >> On Sat, Apr 14, 2012 at 3:41 AM, Rafael J. Wysocki <rjw@xxxxxxx> wrote:
>> >> >> > Hi,
>> >> >> >
>> >> >> > On Friday, April 13, 2012, Yan, Zheng wrote:
>> >> >> >> Hi all,
>> >> >> >>
>> >> >> >> This patch adds PCIe runtime D3cold support, namely cut power supply for functions
>> >> >> >> beneath a PCIe port when they all have entered D3. A device in D3cold can only
>> >> >> >> generate wake event through the WAKE# pin. Because we can not access to a device's
>> >> >> >> configure space while it's in D3cold, pme_poll is disabled for devices in D3cold.
>> >> >> >>
>> >> >> >> Any comment will be appreciated.
>> >> >> >>
>> >> >> >> Signed-off-by: Zheng Yan <zheng.z.yan@xxxxxxxxx>
>> >> >> >> ---
>> >> >> >> diff --git a/drivers/pci/pci-acpi.c b/drivers/pci/pci-acpi.c
>> >> >> >> index 0f150f2..e210e8cb 100644
>> >> >> >> --- a/drivers/pci/pci-acpi.c
>> >> >> >> +++ b/drivers/pci/pci-acpi.c
>> >> >> >> @@ -224,7 +224,7 @@ static int acpi_pci_set_power_state(struct pci_dev *dev, pci_power_t state)
>> >> >> >>               [PCI_D1] = ACPI_STATE_D1,
>> >> >> >>               [PCI_D2] = ACPI_STATE_D2,
>> >> >> >>               [PCI_D3hot] = ACPI_STATE_D3,
>> >> >> >> -             [PCI_D3cold] = ACPI_STATE_D3
>> >> >> >> +             [PCI_D3cold] = ACPI_STATE_D3_COLD
>> >> >> >>       };
>> >> >> >>       int error = -EINVAL;
>> >> >> >>
>> >> >> >
>> >> >> > Please don't use that ACPI_STATE_D3_COLD thing, it's not defined correctly.
>> >> >> >
>> >> >> > We should define ACPI_STATE_D3_COLD == ACPI_STATE_D3 and add ACPI_STATE_D3_HOT
>> >> >> > instead.  I'll prepare a patch for that over the weekend if no one has done
>> >> >> > that already.
>> >> >> >
>> >> >> >> @@ -296,7 +296,8 @@ static void acpi_pci_propagate_run_wake(struct pci_bus *bus, bool enable)
>> >> >> >>
>> >> >> >>  static int acpi_pci_run_wake(struct pci_dev *dev, bool enable)
>> >> >> >>  {
>> >> >> >> -     if (dev->pme_interrupt)
>> >> >> >> +     /* PME interrupt isn't available in the D3cold case */
>> >> >> >> +     if (dev->pme_interrupt && !dev->runtime_d3cold)
>> >> >> >
>> >> >> > This whole thing is wrong.  First off, I don't think that the runtime_d3cold
>> >> >> > flag makes any sense.  We already cover that in dev->pme_support.
>> >> >>
>> >> >> PCIe devices and PCIe root port may have proper PME interrupt support
>> >> >> (that is, dev->pme_interrupt = true), but the process of remote wakeup
>> >> >> from D3cold is as follow:
>> >> >>
>> >> >> 1) In D3cold, the power of the main link is turned off, aux power is
>> >> >> provided (PCIe L2 link state)
>> >> >> 2) Device detect condition to resume, then assert #WAKE pin
>> >> >> 2) ACPI circuit linked with #WAKE pin, and will generate a GPE for that
>> >> >> 3) GPE handler will resume device with ACPI (via _PS3 and _PR0), the
>> >> >> power of the main link is turned on, after a while, link goes into L0
>> >> >> state
>> >> >> 4) The PME message is sent to root port, pme interrupt generated
>> >> >
>> >> > This isn't how it's supposed to work in theory.  If the device can signal PME
>> >> > from D3cold, it should be able to reestablish the link and send a PME
>> >> > message from there.  dev->pme_interrupt set means exactly that.
>> >> >
>> >> > ACPI is only supposed to be needed for things that don't send PME
>> >> > messages (in your case the PME interrupt generated by the port is essentially
>> >> > useless, because the wakeup event has already been signaled through ACPI).
>> >> >
>> >> >> So, for deivce, dev->pme_interrupt = true and dev->pme_support
>> >> >> advocate it support PME in D3cold.  But we still need ACPI to setup
>> >> >> run wake for the device.
>> >> >
>> >> > OK, so this is nonstandard.
>> >>
>> >> This is the standard behavior.  Please refer to PCI Express Base
>> >> Sepcification Revision 2.0, section 5.3.3.2 Link Wakeup.  In D1, D2
>> >> and D3hot state, PCIe device can transit the link from L1 to L0 state,
>> >> and send the PME message.  In D3cold, the main link is powered off,
>> >> PCIe device will use a STANDARD sideband signal WAKE# to signal wakeup
>> >> firstly, then platform (power controller in spec) will power on the
>> >> main link for the device, after main link is back to L0, the PME
>> >> message is send to root port, pme interrupt is generated.  So in
>> >> theory, the wake up process can be divided into platform part (which
>> >> power on the main link) and PCIe part (which send PME).
>> >
>> > That's fine.  However, the platform part should be completely transparent
>> > to the PCI bus type, then.  It just should power up the link to allow a
>> > PME message to be transmitted through it.
>>
>> Yes.
>>
>> >[...]
>> >
>> >> > So don't use pci_set_power_state() for that, because it's essentially
>> >> > a different operation.  You need a pci_platform_remove_power() helper or
>> >> > similar for that.
>> >> >
>> >> > What ACPI method exactly is used to remove power from the device?
>> >>
>> >> The ACPI method executed is as follow:
>> >>
>> >> - _PS3 (if exist)
>> >> - Power resources in _PR3 is turned on
>> >> - Power resources in _PR0 is turned off
>> >> - Power resources in _PR3 is turned off
>> >
>> > I'd rather think
>> >
>> > - make sure that _PR3 resources are referenced
>> > - drop references (from this device) for all other power resources
>> > - execute _PS3 (if any)
>> > - drop references for _PR3 resources
>> >
>> > if Section 7.2.11 of ACPI 5.0 is to be followed.
>>
>> Yes.  You are right.
>>
>> >> I think the process can fit pci_set_power_state() quite well, so why
>> >> invent another helper for that?
>> >
>> > OK, we can special case it, perhaps.
>> >
>> > Suppose we have a "this device may be put into D3_cold" flag.
>> >
>> > Who's going to decide whether to put it into D3_hot or D3_cold?
>>
>> In most cases, I think it is OK to put device into D3_cold if that is
>> supported.
>
> Well, there may be PM QoS latency requirements preventing us from doing so.

Yes.

>> But there should be some special case where D3_cold is not
>> desirable, for example, we can put SSD into D3_cold safely, but it is
>> not quite safe to put HDD into D3_cold.  So we want to introduce a
>> flag: "may_power_off" like in the following patch
>>
>> https://lkml.org/lkml/2012/3/29/41
>>
>> It gives device driver a chance to prevent the device to be put into D3_cold.
>
> I see.  So your proposal is that the flag might be used to indicate to
> whoever carries out power transitions of devices that power must not be
> removed from this particular device, right?

Yes.

> In that case we can put that flag into struct dev_pm_info after all, but
> perhaps the name should indicate more precisely what it is about.  Something
> like "power_must_be_on" maybe?

I am not good at naming in English :)
I will accept your proposal.

>> > [...]
>> >
>> >> >> > So now please tell me what exactly you want to achieve and why you want to do
>> >> >> > that in the first place.
>> >> >
>> >> > Well, is there any chance to get that information?
>> >>
>> >> You mean the runtime_d3cold flag? That flag is used to tell
>> >> acpi_pci_run_wake() that we need ACPI wakeup setup for the device
>> >> because that is needed by D3cold.  The ACPI wakeup setup here means
>> >> turn on power resources needed by wake up (_PRW) and execute _DSW.
>> >>
>> >> If you mean the whole patch, we want to implement runtime D3cold
>> >> support, which can save more power than D3hot.
>> >
>> > So, do I think correctly that you'd like to put devices into D3_cold
>> > if that's possible via ACPI and to be able to wake it up from that state
>> > using remote wakeup?
>>
>> Yes.  Support both remote wakeup and host wakeup.
>
> OK, so we need to clean up the ACPI D3_HOT/D3_COLD mess first.  Then, we need
> to change PCI so that devices are not put into D3cold (by ACPI) if only D3hot
> is requested by the caller of pci_set_power_state().
>
> Having done that, we can modify pci_set_power_state() to handle D3cold as
> a special case (essentially, it should check that case before doing anything
> else).  Finally, we need to teach the ACPI notify handler about the WAKE#
> event and we need to add the 100 ms wait to the device resume code path
> somewhere (I guess in pci_set_power_state() for the D3cold->D0 transition).

Yes.  Sound good to me.

> Now, there's one more thing to consider.  Namely, if a PCIe endpoint is put
> into D3hot (via native PM) and then the port it is connected to is put into
> D3_hot (via native PM), does that transfer the endpoint into D3cold?

No.

But if a PCIe endpoint is put into D3hot and then the port it is
connected is put into D3_cold (via ACPI), this will transfer the
endpoint into D3_cold, and if the port is put into D0 afterwards, all
subordinate endpoint devices will be put into D0 (because of power on
reset).  I think what we need to do here is:

- when choose power state, if any subordinate device has
power_must_be_on set, will not choose D3_cold

- when put PCIe port from D3_cold to D0, resume all subordinate
devices too.  We design a method to do that in following patch:

https://lkml.org/lkml/2012/3/29/38

Where we will register all subordinate devices via
acpi_power_resource_register_device(endpoint_device,
bridget_acpi_handle).

Best Regards,
Huang Ying
--
To unsubscribe from this list: send the line "unsubscribe linux-pci" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [DMA Engine]     [Linux Coverity]     [Linux USB]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [Greybus]

  Powered by Linux