linux-kernel.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: Baolin Wang <baolin.wang@linaro.org>
To: Felipe Balbi <balbi@kernel.org>
Cc: mathias.nyman@intel.com, Greg KH <gregkh@linuxfoundation.org>,
	Mark Brown <broonie@kernel.org>, USB <linux-usb@vger.kernel.org>,
	LKML <linux-kernel@vger.kernel.org>
Subject: Re: [PATCH v4 2/2] usb: dwc3: core: Support the dwc3 host suspend/resume
Date: Thu, 8 Dec 2016 18:20:08 +0800	[thread overview]
Message-ID: <CAMz4kuJVOCq6-iN4k4xkZXWGch_YROeEC1mQBFhE4xc75y-vww@mail.gmail.com> (raw)
In-Reply-To: <87oa0mkbd6.fsf@linux.intel.com>

Hi,

On 8 December 2016 at 17:40, Felipe Balbi <balbi@kernel.org> wrote:
>
> Hi,
>
> Baolin Wang <baolin.wang@linaro.org> writes:
>> Hi Felipe,
>>
>> On 28 November 2016 at 14:43, Baolin Wang <baolin.wang@linaro.org> wrote:
>>> For some mobile devices with strict power management, we also want to suspend
>>> the host when the slave is detached for power saving. Thus we add the host
>>> suspend/resume functions to support this requirement.
>>>
>>> Signed-off-by: Baolin Wang <baolin.wang@linaro.org>
>>> ---
>>> Changes since v3:
>>>  - No updates.
>>>
>>> Changes since v2:
>>>  - Remove pm_children_suspended() and other unused macros.
>>>
>>>  Changes since v1:
>>>    - Add pm_runtime.h head file to avoid kbuild error.
>>> ---
>>>  drivers/usb/dwc3/Kconfig |    7 +++++++
>>>  drivers/usb/dwc3/core.c  |   26 +++++++++++++++++++++++++-
>>>  drivers/usb/dwc3/core.h  |   15 +++++++++++++++
>>>  drivers/usb/dwc3/host.c  |   37 +++++++++++++++++++++++++++++++++++++
>>>  4 files changed, 84 insertions(+), 1 deletion(-)
>>>
>>> diff --git a/drivers/usb/dwc3/Kconfig b/drivers/usb/dwc3/Kconfig
>>> index a45b4f1..47bb2f3 100644
>>> --- a/drivers/usb/dwc3/Kconfig
>>> +++ b/drivers/usb/dwc3/Kconfig
>>> @@ -47,6 +47,13 @@ config USB_DWC3_DUAL_ROLE
>>>
>>>  endchoice
>>>
>>> +config USB_DWC3_HOST_SUSPEND
>>> +       bool "Choose if the DWC3 host (xhci) can be suspend/resume"
>>> +       depends on USB_DWC3_HOST=y || USB_DWC3_DUAL_ROLE=y
>
> why do you need another Kconfig for this? Just enable it already :-p

I assume some platforms may do not need this feature. But okay, I can
remove this kconfig and enable it.

>
>>> diff --git a/drivers/usb/dwc3/core.c b/drivers/usb/dwc3/core.c
>>> index 9a4a5e4..7ad4bc3 100644
>>> --- a/drivers/usb/dwc3/core.c
>>> +++ b/drivers/usb/dwc3/core.c
>>> @@ -1091,6 +1091,7 @@ static int dwc3_probe(struct platform_device *pdev)
>>>         pm_runtime_use_autosuspend(dev);
>>>         pm_runtime_set_autosuspend_delay(dev, DWC3_DEFAULT_AUTOSUSPEND_DELAY);
>>>         pm_runtime_enable(dev);
>>> +       pm_suspend_ignore_children(dev, true);
>
> why do you need this?

Since the dwc3 device is the parent deive of xhci device, if we want
to suspend dwc3 device, we need to suspend child device (xhci device)
manually by issuing pm_runtime_put_sync() in dwc3_host_suspend(). In
pm_runtime_put_sync(), it will check if the children (xhci device) of
dwc3 device have been in suspend state, if not we will suspend dwc3
device failed.

We get/put the child device manually in parent device's runtime
callback, we need to ignore the child device's runtime state, or it
will failed due to the dependency.

>
>>> @@ -1215,15 +1216,27 @@ static int dwc3_remove(struct platform_device *pdev)
>>>  static int dwc3_suspend_common(struct dwc3 *dwc)
>>>  {
>>>         unsigned long   flags;
>>> +       int             ret;
>>>
>>>         switch (dwc->dr_mode) {
>>>         case USB_DR_MODE_PERIPHERAL:
>>> +               spin_lock_irqsave(&dwc->lock, flags);
>>> +               dwc3_gadget_suspend(dwc);
>>> +               spin_unlock_irqrestore(&dwc->lock, flags);
>>> +               break;
>>>         case USB_DR_MODE_OTG:
>>> +               ret = dwc3_host_suspend(dwc);
>>> +               if (ret)
>>> +                       return ret;
>>> +
>>>                 spin_lock_irqsave(&dwc->lock, flags);
>>>                 dwc3_gadget_suspend(dwc);
>>>                 spin_unlock_irqrestore(&dwc->lock, flags);
>>>                 break;
>>>         case USB_DR_MODE_HOST:
>>> +               ret = dwc3_host_suspend(dwc);
>>> +               if (ret)
>>> +                       return ret;
>>>         default:
>>>                 /* do nothing */
>>>                 break;
>>> @@ -1245,12 +1258,23 @@ static int dwc3_resume_common(struct dwc3 *dwc)
>>>
>>>         switch (dwc->dr_mode) {
>>>         case USB_DR_MODE_PERIPHERAL:
>>> +               spin_lock_irqsave(&dwc->lock, flags);
>>> +               dwc3_gadget_resume(dwc);
>>> +               spin_unlock_irqrestore(&dwc->lock, flags);
>>> +               break;
>>>         case USB_DR_MODE_OTG:
>>> +               ret = dwc3_host_resume(dwc);
>>> +               if (ret)
>>> +                       return ret;
>>> +
>>>                 spin_lock_irqsave(&dwc->lock, flags);
>>>                 dwc3_gadget_resume(dwc);
>>>                 spin_unlock_irqrestore(&dwc->lock, flags);
>>> -               /* FALLTHROUGH */
>>> +               break;
>>>         case USB_DR_MODE_HOST:
>>> +               ret = dwc3_host_resume(dwc);
>>> +               if (ret)
>>> +                       return ret;
>>>         default:
>>>                 /* do nothing */
>>>                 break;
>>> diff --git a/drivers/usb/dwc3/core.h b/drivers/usb/dwc3/core.h
>>> index b585a30..db41908 100644
>>> --- a/drivers/usb/dwc3/core.h
>>> +++ b/drivers/usb/dwc3/core.h
>>> @@ -1226,4 +1226,19 @@ static inline void dwc3_ulpi_exit(struct dwc3 *dwc)
>>>  { }
>>>  #endif
>>>
>>> +#if IS_ENABLED(CONFIG_USB_DWC3_HOST_SUSPEND)
>>> +int dwc3_host_suspend(struct dwc3 *dwc);
>>> +int dwc3_host_resume(struct dwc3 *dwc);
>>> +#else
>>> +static inline int dwc3_host_suspend(struct dwc3 *dwc)
>>> +{
>>> +       return 0;
>>> +}
>>> +
>>> +static inline int dwc3_host_resume(struct dwc3 *dwc)
>>> +{
>>> +       return 0;
>>> +}
>>> +#endif
>>> +
>>>  #endif /* __DRIVERS_USB_DWC3_CORE_H */
>>> diff --git a/drivers/usb/dwc3/host.c b/drivers/usb/dwc3/host.c
>>> index ed82464..8e5309d6 100644
>>> --- a/drivers/usb/dwc3/host.c
>>> +++ b/drivers/usb/dwc3/host.c
>>> @@ -16,6 +16,7 @@
>>>   */
>>>
>>>  #include <linux/platform_device.h>
>>> +#include <linux/pm_runtime.h>
>>>
>>>  #include "core.h"
>>>
>>> @@ -130,3 +131,39 @@ void dwc3_host_exit(struct dwc3 *dwc)
>>>                           dev_name(&dwc->xhci->dev));
>>>         platform_device_unregister(dwc->xhci);
>>>  }
>>> +
>>> +#ifdef CONFIG_USB_DWC3_HOST_SUSPEND
>>> +int dwc3_host_suspend(struct dwc3 *dwc)
>>> +{
>>> +       struct device *xhci = &dwc->xhci->dev;
>>> +       int ret;
>>> +
>>> +       /*
>>> +        * Note: if we get the -EBUSY, which means the xHCI children devices are
>>> +        * not in suspend state yet, the glue layer need to wait for a while and
>>> +        * try to suspend xHCI device again.
>>> +        */
>>> +       ret = pm_runtime_put_sync(xhci);
>>> +       if (ret) {
>>> +               dev_err(xhci, "failed to suspend xHCI device\n");
>>> +               return ret;
>>> +       }
>
> return pm_runtime_put_sync() is probably enough here.

OK.

>
>>> +
>>> +       return 0;
>>> +}
>>> +
>>> +int dwc3_host_resume(struct dwc3 *dwc)
>>> +{
>>> +       struct device *xhci = &dwc->xhci->dev;
>>> +       int ret;
>>> +
>>> +       /* Resume the xHCI device synchronously. */
>>> +       ret = pm_runtime_get_sync(xhci);
>
> likewise.

OK. I will fix these in next version. Thanks.

>
>>> +       if (ret) {
>>> +               dev_err(xhci, "failed to resume xHCI device\n");
>>> +               return ret;
>>> +       }
>>> +
>>> +       return 0;
>>> +}
>>> +#endif
>>> --
>>> 1.7.9.5
>>>
>>
>> How do you think this patch and do you have any suggestion? Thanks.
>>
>> --
>> Baolin.wang
>> Best Regards
>> --
>> To unsubscribe from this list: send the line "unsubscribe linux-usb" in
>> the body of a message to majordomo@vger.kernel.org
>> More majordomo info at  http://vger.kernel.org/majordomo-info.html
>
> --
> balbi



-- 
Baolin.wang
Best Regards

  reply	other threads:[~2016-12-08 10:20 UTC|newest]

Thread overview: 10+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2016-11-28  6:43 [PATCH v4 1/2] usb: host: plat: Enable xhci plat runtime PM Baolin Wang
2016-11-28  6:43 ` [PATCH v4 2/2] usb: dwc3: core: Support the dwc3 host suspend/resume Baolin Wang
2016-12-08  7:05   ` Baolin Wang
2016-12-08  9:40     ` Felipe Balbi
2016-12-08 10:20       ` Baolin Wang [this message]
2016-12-08 11:02         ` Felipe Balbi
2016-12-08 11:14           ` Baolin Wang
2016-12-08 17:52             ` Felipe Balbi
2016-12-09  3:23               ` Baolin Wang
2016-12-08  7:04 ` [PATCH v4 1/2] usb: host: plat: Enable xhci plat runtime PM Baolin Wang

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=CAMz4kuJVOCq6-iN4k4xkZXWGch_YROeEC1mQBFhE4xc75y-vww@mail.gmail.com \
    --to=baolin.wang@linaro.org \
    --cc=balbi@kernel.org \
    --cc=broonie@kernel.org \
    --cc=gregkh@linuxfoundation.org \
    --cc=linux-kernel@vger.kernel.org \
    --cc=linux-usb@vger.kernel.org \
    --cc=mathias.nyman@intel.com \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).