From: Peter Chen <hzpeterchen-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
To: Joshua Clayton <stillcompiling-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
Cc: Peter Chen <peter.chen-3arQi8VN3Tc@public.gmane.org>,
gregkh-hQyY1W1yCW8ekmWlsbkhG0B+6BGkLq7r@public.gmane.org,
stern-nwvwT67g6+6dFdvTe/nMLpVzexx5G7lz@public.gmane.org,
ulf.hansson-QSEj5FYQhm4dnm+yROfE0A@public.gmane.org,
broonie-DgEjT+Ai2ygdnm+yROfE0A@public.gmane.org,
sre-DgEjT+Ai2ygdnm+yROfE0A@public.gmane.org,
robh+dt-DgEjT+Ai2ygdnm+yROfE0A@public.gmane.org,
shawnguo-DgEjT+Ai2ygdnm+yROfE0A@public.gmane.org,
dbaryshkov-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org,
dwmw2-wEGCiKHe2LqWVfeAwA7xHQ@public.gmane.org,
k.kozlowski-Sze3O3UU22JBDgjK7y7TUQ@public.gmane.org,
linux-arm-kernel-IAPFreCvJWM7uuMidbF8XUB+6BGkLq7r@public.gmane.org,
p.zabel-bIcnvbaLZ9MEGnE8C9+IrQ@public.gmane.org,
devicetree-u79uwXL29TY76Z2rM5mHXA@public.gmane.org,
pawel.moll-5wv7dgnIgG8@public.gmane.org,
mark.rutland-5wv7dgnIgG8@public.gmane.org,
linux-usb-u79uwXL29TY76Z2rM5mHXA@public.gmane.org,
arnd-r2nGTMty4D4@public.gmane.org,
s.hauer-bIcnvbaLZ9MEGnE8C9+IrQ@public.gmane.org,
mail-APzI5cXaD1zVlRWJc41N0YvC60bnQu0Y@public.gmane.org,
troy.kisky-Q5RJGjKts06CY9SHAMCTRUEOCMrvLtNR@public.gmane.org,
festevam-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org,
oscar-Bdbr4918Nnnk1uMJSBkQmQ@public.gmane.org,
stephen.boyd-QSEj5FYQhm4dnm+yROfE0A@public.gmane.org,
linux-pm-u79uwXL29TY76Z2rM5mHXA@public.gmane.org,
linux-kernel-u79uwXL29TY76Z2rM5mHXA@public.gmane.org
Subject: Re: [PATCH v3 4/6] usb: core: add power sequence handling for USB devices
Date: Thu, 28 Jul 2016 09:45:21 +0800 [thread overview]
Message-ID: <20160728014521.GB26186@shlinux2> (raw)
In-Reply-To: <5798E067.90207-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
On Wed, Jul 27, 2016 at 09:25:11AM -0700, Joshua Clayton wrote:
>
>
> On 07/20/2016 02:40 AM, Peter Chen wrote:
> > Some hard-wired USB devices need to do power sequence to let the
> > device work normally, the typical power sequence like: enable USB
> > PHY clock, toggle reset pin, etc. But current Linux USB driver
> > lacks of such code to do it, it may cause some hard-wired USB devices
> > works abnormal or can't be recognized by controller at all.
> >
> > In this patch, it calls power sequence library APIs to finish
> > the power sequence events. At first, it calls pwrseq_alloc_generic
> > to create a generic power sequence instance, then it will do power
> > on sequence at hub's probe for all devices under this hub
> > (includes root hub).
> >
> > At hub_disconnect, it will do power off sequence which is at powered
> > on list.
> >
> > Signed-off-by: Peter Chen <peter.chen-3arQi8VN3Tc@public.gmane.org>
> > ---
> > drivers/usb/core/Makefile | 1 +
> > drivers/usb/core/hub.c | 12 ++++--
> > drivers/usb/core/hub.h | 12 ++++++
> > drivers/usb/core/pwrseq.c | 100 ++++++++++++++++++++++++++++++++++++++++++++++
> > 4 files changed, 122 insertions(+), 3 deletions(-)
> > create mode 100644 drivers/usb/core/pwrseq.c
> >
> > diff --git a/drivers/usb/core/Makefile b/drivers/usb/core/Makefile
> > index da36b78..5fd2f54 100644
> > --- a/drivers/usb/core/Makefile
> > +++ b/drivers/usb/core/Makefile
> > @@ -10,5 +10,6 @@ usbcore-y += port.o
> > usbcore-$(CONFIG_OF) += of.o
> > usbcore-$(CONFIG_PCI) += hcd-pci.o
> > usbcore-$(CONFIG_ACPI) += usb-acpi.o
> > +usbcore-$(CONFIG_PWRSEQ_GENERIC) += pwrseq.o
> >
> > obj-$(CONFIG_USB) += usbcore.o
> > diff --git a/drivers/usb/core/hub.c b/drivers/usb/core/hub.c
> > index bee1351..a346a8b 100644
> > --- a/drivers/usb/core/hub.c
> > +++ b/drivers/usb/core/hub.c
> > @@ -1700,6 +1700,7 @@ static void hub_disconnect(struct usb_interface *intf)
> > hub->error = 0;
> > hub_quiesce(hub, HUB_DISCONNECT);
> >
> > + hub_pwrseq_off(hub);
> > mutex_lock(&usb_port_peer_mutex);
> >
> > /* Avoid races with recursively_mark_NOTATTACHED() */
> > @@ -1733,6 +1734,7 @@ static int hub_probe(struct usb_interface *intf, const struct usb_device_id *id)
> > struct usb_endpoint_descriptor *endpoint;
> > struct usb_device *hdev;
> > struct usb_hub *hub;
> > + int ret = -ENODEV;
> >
> > desc = intf->cur_altsetting;
> > hdev = interface_to_usbdev(intf);
> > @@ -1839,6 +1841,7 @@ descriptor_error:
> > INIT_DELAYED_WORK(&hub->leds, led_work);
> > INIT_DELAYED_WORK(&hub->init_work, NULL);
> > INIT_WORK(&hub->events, hub_event);
> > + INIT_LIST_HEAD(&hub->pwrseq_on_list);
> > usb_get_intf(intf);
> > usb_get_dev(hdev);
> >
> > @@ -1852,11 +1855,14 @@ descriptor_error:
> > if (id->driver_info & HUB_QUIRK_CHECK_PORT_AUTOSUSPEND)
> > hub->quirk_check_port_auto_suspend = 1;
> >
> > - if (hub_configure(hub, endpoint) >= 0)
> > - return 0;
> > + if (hub_configure(hub, endpoint) >= 0) {
> > + ret = hub_pwrseq_on(hub);
> > + if (!ret)
> > + return 0;
> > + }
> >
> > hub_disconnect(intf);
> > - return -ENODEV;
> > + return ret;
> > }
> >
> > static int
> > diff --git a/drivers/usb/core/hub.h b/drivers/usb/core/hub.h
> > index 34c1a7e..9473f6f 100644
> > --- a/drivers/usb/core/hub.h
> > +++ b/drivers/usb/core/hub.h
> > @@ -78,6 +78,7 @@ struct usb_hub {
> > struct delayed_work init_work;
> > struct work_struct events;
> > struct usb_port **ports;
> > + struct list_head pwrseq_on_list; /* powered pwrseq node list */
> > };
> >
> > /**
> > @@ -166,3 +167,14 @@ static inline int hub_port_debounce_be_stable(struct usb_hub *hub,
> > {
> > return hub_port_debounce(hub, port1, false);
> > }
> > +
> > +#if IS_ENABLED(CONFIG_PWRSEQ_GENERIC)
> > +int hub_pwrseq_on(struct usb_hub *hub);
> > +void hub_pwrseq_off(struct usb_hub *hub);
> > +#else
> > +static inline int hub_pwrseq_on(struct usb_hub *hub)
> > +{
> > + return 0;
> > +}
> > +static inline void hub_pwrseq_off(struct usb_hub *hub) {}
> > +#endif /* CONFIG_PWRSEQ_GENERIC */
> > diff --git a/drivers/usb/core/pwrseq.c b/drivers/usb/core/pwrseq.c
> > new file mode 100644
> > index 0000000..837fe66
> > --- /dev/null
> > +++ b/drivers/usb/core/pwrseq.c
> > @@ -0,0 +1,100 @@
> > +/*
> > + * pwrseq.c USB device power sequence management
> > + *
> > + * Copyright (C) 2016 Freescale Semiconductor, Inc.
> > + * Author: Peter Chen <peter.chen-3arQi8VN3Tc@public.gmane.org>
> > + *
> > + * This program is free software: you can redistribute it and/or modify
> > + * it under the terms of the GNU General Public License version 2 of
> > + * the License as published by the Free Software Foundation.
> > + *
> > + * This program is distributed in the hope that it will be useful,
> > + * but WITHOUT ANY WARRANTY; without even the implied warranty of
> > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
> > + * GNU General Public License for more details.
> > + *
> > + * You should have received a copy of the GNU General Public License
> > + * along with this program. If not, see <http://www.gnu.org/licenses/>.
> > + */
> > +
> > +#include <linux/list.h>
> > +#include <linux/of.h>
> > +#include <linux/power/pwrseq.h>
> > +#include <linux/slab.h>
> > +#include <linux/usb.h>
> > +#include <linux/usb/hcd.h>
> > +
> > +#include "hub.h"
> > +
> > +struct usb_pwrseq_node {
> > + struct pwrseq *pwrseq;
> > + struct list_head list;
> > +};
> > +
> > +static int hub_of_pwrseq_on(struct device_node *np, struct usb_hub *hub)
> > +{
> > + struct pwrseq *pwrseq;
> > + struct usb_pwrseq_node *pwrseq_node;
> > + int ret;
> > +
> > + pwrseq = pwrseq_alloc_generic();
> > + if (IS_ERR(pwrseq))
> > + return PTR_ERR(pwrseq);
> > +
> > + ret = pwrseq_get(np, pwrseq);
> > + if (ret)
> > + goto pwr_free;
> > +
> > + ret = pwrseq_on(np, pwrseq);
> > + if (ret)
> > + goto pwr_put;
> > +
> > + pwrseq_node = kzalloc(sizeof(*pwrseq_node), GFP_KERNEL);
> > + pwrseq_node->pwrseq = pwrseq;
> > + list_add(&pwrseq_node->list, &hub->pwrseq_on_list);
> > +
> > + return 0;
> > +
> > +pwr_put:
> > + pwrseq_put(pwrseq);
> > +pwr_free:
> > + pwrseq_free(pwrseq);
> > + return ret;
> > +}
> > +
> > +int hub_pwrseq_on(struct usb_hub *hub)
> > +{
> > + struct device *parent;
> > + struct usb_device *hdev = hub->hdev;
> > + struct device_node *np;
> > + int ret;
> > +
> > + if (hdev->parent)
> > + parent = &hdev->dev;
> > + else
> > + parent = bus_to_hcd(hdev->bus)->self.controller;
> > +
> > + for_each_child_of_node(parent->of_node, np) {
> > + ret = hub_of_pwrseq_on(np, hub);
> > + if (ret)
> > + return ret;
> > + }
> > +
> > + return 0;
> > +}
> > +
> > +void hub_pwrseq_off(struct usb_hub *hub)
> > +{
> > + struct pwrseq *pwrseq;
> > + struct usb_pwrseq_node *pwrseq_node, *tmp_node;
> > +
> > + list_for_each_entry_safe(pwrseq_node, tmp_node,
> > + &hub->pwrseq_on_list, list) {
> > + pwrseq = pwrseq_node->pwrseq;
> > + pwrseq_off(pwrseq);
> > + pwrseq_put(pwrseq);
> > + pwrseq_free(pwrseq);
> > + list_del(&pwrseq_node->list);
> > + kfree(pwrseq_node);
> > + }
> > +}
> Patch 4 does not apply for me
> (The Makefile has a "usbcore-$(CONFIG_OF) += of.o"
> line which I don't see in 4.7 or linux/next master).
> Is there another patch set this series depends on?
It seems below patch has not been queued:
https://git.kernel.org/cgit/linux/kernel/git/peter.chen/usb.git/commit/?h=add_pwrseq_for_usb&id=2efd73d0f935ffbc66135553da0ac807c0b4c3fe
This pwrseq-lib patch set does not depend on it, I will send patch set
on top of a clean mainline next time, sorry for confusing.
--
Best Regards,
Peter Chen
--
To unsubscribe from this list: send the line "unsubscribe linux-usb" in
the body of a message to majordomo-u79uwXL29TY76Z2rM5mHXA@public.gmane.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
WARNING: multiple messages have this Message-ID (diff)
From: hzpeterchen@gmail.com (Peter Chen)
To: linux-arm-kernel@lists.infradead.org
Subject: [PATCH v3 4/6] usb: core: add power sequence handling for USB devices
Date: Thu, 28 Jul 2016 09:45:21 +0800 [thread overview]
Message-ID: <20160728014521.GB26186@shlinux2> (raw)
In-Reply-To: <5798E067.90207@gmail.com>
On Wed, Jul 27, 2016 at 09:25:11AM -0700, Joshua Clayton wrote:
>
>
> On 07/20/2016 02:40 AM, Peter Chen wrote:
> > Some hard-wired USB devices need to do power sequence to let the
> > device work normally, the typical power sequence like: enable USB
> > PHY clock, toggle reset pin, etc. But current Linux USB driver
> > lacks of such code to do it, it may cause some hard-wired USB devices
> > works abnormal or can't be recognized by controller at all.
> >
> > In this patch, it calls power sequence library APIs to finish
> > the power sequence events. At first, it calls pwrseq_alloc_generic
> > to create a generic power sequence instance, then it will do power
> > on sequence at hub's probe for all devices under this hub
> > (includes root hub).
> >
> > At hub_disconnect, it will do power off sequence which is at powered
> > on list.
> >
> > Signed-off-by: Peter Chen <peter.chen@nxp.com>
> > ---
> > drivers/usb/core/Makefile | 1 +
> > drivers/usb/core/hub.c | 12 ++++--
> > drivers/usb/core/hub.h | 12 ++++++
> > drivers/usb/core/pwrseq.c | 100 ++++++++++++++++++++++++++++++++++++++++++++++
> > 4 files changed, 122 insertions(+), 3 deletions(-)
> > create mode 100644 drivers/usb/core/pwrseq.c
> >
> > diff --git a/drivers/usb/core/Makefile b/drivers/usb/core/Makefile
> > index da36b78..5fd2f54 100644
> > --- a/drivers/usb/core/Makefile
> > +++ b/drivers/usb/core/Makefile
> > @@ -10,5 +10,6 @@ usbcore-y += port.o
> > usbcore-$(CONFIG_OF) += of.o
> > usbcore-$(CONFIG_PCI) += hcd-pci.o
> > usbcore-$(CONFIG_ACPI) += usb-acpi.o
> > +usbcore-$(CONFIG_PWRSEQ_GENERIC) += pwrseq.o
> >
> > obj-$(CONFIG_USB) += usbcore.o
> > diff --git a/drivers/usb/core/hub.c b/drivers/usb/core/hub.c
> > index bee1351..a346a8b 100644
> > --- a/drivers/usb/core/hub.c
> > +++ b/drivers/usb/core/hub.c
> > @@ -1700,6 +1700,7 @@ static void hub_disconnect(struct usb_interface *intf)
> > hub->error = 0;
> > hub_quiesce(hub, HUB_DISCONNECT);
> >
> > + hub_pwrseq_off(hub);
> > mutex_lock(&usb_port_peer_mutex);
> >
> > /* Avoid races with recursively_mark_NOTATTACHED() */
> > @@ -1733,6 +1734,7 @@ static int hub_probe(struct usb_interface *intf, const struct usb_device_id *id)
> > struct usb_endpoint_descriptor *endpoint;
> > struct usb_device *hdev;
> > struct usb_hub *hub;
> > + int ret = -ENODEV;
> >
> > desc = intf->cur_altsetting;
> > hdev = interface_to_usbdev(intf);
> > @@ -1839,6 +1841,7 @@ descriptor_error:
> > INIT_DELAYED_WORK(&hub->leds, led_work);
> > INIT_DELAYED_WORK(&hub->init_work, NULL);
> > INIT_WORK(&hub->events, hub_event);
> > + INIT_LIST_HEAD(&hub->pwrseq_on_list);
> > usb_get_intf(intf);
> > usb_get_dev(hdev);
> >
> > @@ -1852,11 +1855,14 @@ descriptor_error:
> > if (id->driver_info & HUB_QUIRK_CHECK_PORT_AUTOSUSPEND)
> > hub->quirk_check_port_auto_suspend = 1;
> >
> > - if (hub_configure(hub, endpoint) >= 0)
> > - return 0;
> > + if (hub_configure(hub, endpoint) >= 0) {
> > + ret = hub_pwrseq_on(hub);
> > + if (!ret)
> > + return 0;
> > + }
> >
> > hub_disconnect(intf);
> > - return -ENODEV;
> > + return ret;
> > }
> >
> > static int
> > diff --git a/drivers/usb/core/hub.h b/drivers/usb/core/hub.h
> > index 34c1a7e..9473f6f 100644
> > --- a/drivers/usb/core/hub.h
> > +++ b/drivers/usb/core/hub.h
> > @@ -78,6 +78,7 @@ struct usb_hub {
> > struct delayed_work init_work;
> > struct work_struct events;
> > struct usb_port **ports;
> > + struct list_head pwrseq_on_list; /* powered pwrseq node list */
> > };
> >
> > /**
> > @@ -166,3 +167,14 @@ static inline int hub_port_debounce_be_stable(struct usb_hub *hub,
> > {
> > return hub_port_debounce(hub, port1, false);
> > }
> > +
> > +#if IS_ENABLED(CONFIG_PWRSEQ_GENERIC)
> > +int hub_pwrseq_on(struct usb_hub *hub);
> > +void hub_pwrseq_off(struct usb_hub *hub);
> > +#else
> > +static inline int hub_pwrseq_on(struct usb_hub *hub)
> > +{
> > + return 0;
> > +}
> > +static inline void hub_pwrseq_off(struct usb_hub *hub) {}
> > +#endif /* CONFIG_PWRSEQ_GENERIC */
> > diff --git a/drivers/usb/core/pwrseq.c b/drivers/usb/core/pwrseq.c
> > new file mode 100644
> > index 0000000..837fe66
> > --- /dev/null
> > +++ b/drivers/usb/core/pwrseq.c
> > @@ -0,0 +1,100 @@
> > +/*
> > + * pwrseq.c USB device power sequence management
> > + *
> > + * Copyright (C) 2016 Freescale Semiconductor, Inc.
> > + * Author: Peter Chen <peter.chen@nxp.com>
> > + *
> > + * This program is free software: you can redistribute it and/or modify
> > + * it under the terms of the GNU General Public License version 2 of
> > + * the License as published by the Free Software Foundation.
> > + *
> > + * This program is distributed in the hope that it will be useful,
> > + * but WITHOUT ANY WARRANTY; without even the implied warranty of
> > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
> > + * GNU General Public License for more details.
> > + *
> > + * You should have received a copy of the GNU General Public License
> > + * along with this program. If not, see <http://www.gnu.org/licenses/>.
> > + */
> > +
> > +#include <linux/list.h>
> > +#include <linux/of.h>
> > +#include <linux/power/pwrseq.h>
> > +#include <linux/slab.h>
> > +#include <linux/usb.h>
> > +#include <linux/usb/hcd.h>
> > +
> > +#include "hub.h"
> > +
> > +struct usb_pwrseq_node {
> > + struct pwrseq *pwrseq;
> > + struct list_head list;
> > +};
> > +
> > +static int hub_of_pwrseq_on(struct device_node *np, struct usb_hub *hub)
> > +{
> > + struct pwrseq *pwrseq;
> > + struct usb_pwrseq_node *pwrseq_node;
> > + int ret;
> > +
> > + pwrseq = pwrseq_alloc_generic();
> > + if (IS_ERR(pwrseq))
> > + return PTR_ERR(pwrseq);
> > +
> > + ret = pwrseq_get(np, pwrseq);
> > + if (ret)
> > + goto pwr_free;
> > +
> > + ret = pwrseq_on(np, pwrseq);
> > + if (ret)
> > + goto pwr_put;
> > +
> > + pwrseq_node = kzalloc(sizeof(*pwrseq_node), GFP_KERNEL);
> > + pwrseq_node->pwrseq = pwrseq;
> > + list_add(&pwrseq_node->list, &hub->pwrseq_on_list);
> > +
> > + return 0;
> > +
> > +pwr_put:
> > + pwrseq_put(pwrseq);
> > +pwr_free:
> > + pwrseq_free(pwrseq);
> > + return ret;
> > +}
> > +
> > +int hub_pwrseq_on(struct usb_hub *hub)
> > +{
> > + struct device *parent;
> > + struct usb_device *hdev = hub->hdev;
> > + struct device_node *np;
> > + int ret;
> > +
> > + if (hdev->parent)
> > + parent = &hdev->dev;
> > + else
> > + parent = bus_to_hcd(hdev->bus)->self.controller;
> > +
> > + for_each_child_of_node(parent->of_node, np) {
> > + ret = hub_of_pwrseq_on(np, hub);
> > + if (ret)
> > + return ret;
> > + }
> > +
> > + return 0;
> > +}
> > +
> > +void hub_pwrseq_off(struct usb_hub *hub)
> > +{
> > + struct pwrseq *pwrseq;
> > + struct usb_pwrseq_node *pwrseq_node, *tmp_node;
> > +
> > + list_for_each_entry_safe(pwrseq_node, tmp_node,
> > + &hub->pwrseq_on_list, list) {
> > + pwrseq = pwrseq_node->pwrseq;
> > + pwrseq_off(pwrseq);
> > + pwrseq_put(pwrseq);
> > + pwrseq_free(pwrseq);
> > + list_del(&pwrseq_node->list);
> > + kfree(pwrseq_node);
> > + }
> > +}
> Patch 4 does not apply for me
> (The Makefile has a "usbcore-$(CONFIG_OF) += of.o"
> line which I don't see in 4.7 or linux/next master).
> Is there another patch set this series depends on?
It seems below patch has not been queued:
https://git.kernel.org/cgit/linux/kernel/git/peter.chen/usb.git/commit/?h=add_pwrseq_for_usb&id=2efd73d0f935ffbc66135553da0ac807c0b4c3fe
This pwrseq-lib patch set does not depend on it, I will send patch set
on top of a clean mainline next time, sorry for confusing.
--
Best Regards,
Peter Chen
WARNING: multiple messages have this Message-ID (diff)
From: Peter Chen <hzpeterchen@gmail.com>
To: Joshua Clayton <stillcompiling@gmail.com>
Cc: Peter Chen <peter.chen@nxp.com>,
gregkh@linuxfoundation.org, stern@rowland.harvard.edu,
ulf.hansson@linaro.org, broonie@kernel.org, sre@kernel.org,
robh+dt@kernel.org, shawnguo@kernel.org, dbaryshkov@gmail.com,
dwmw2@infradead.org, k.kozlowski@samsung.com,
linux-arm-kernel@lists.infradead.org, p.zabel@pengutronix.de,
devicetree@vger.kernel.org, pawel.moll@arm.com,
mark.rutland@arm.com, linux-usb@vger.kernel.org, arnd@arndb.de,
s.hauer@pengutronix.de, mail@maciej.szmigiero.name,
troy.kisky@boundarydevices.com, festevam@gmail.com,
oscar@naiandei.net, stephen.boyd@linaro.org,
linux-pm@vger.kernel.org, linux-kernel@vger.kernel.org
Subject: Re: [PATCH v3 4/6] usb: core: add power sequence handling for USB devices
Date: Thu, 28 Jul 2016 09:45:21 +0800 [thread overview]
Message-ID: <20160728014521.GB26186@shlinux2> (raw)
In-Reply-To: <5798E067.90207@gmail.com>
On Wed, Jul 27, 2016 at 09:25:11AM -0700, Joshua Clayton wrote:
>
>
> On 07/20/2016 02:40 AM, Peter Chen wrote:
> > Some hard-wired USB devices need to do power sequence to let the
> > device work normally, the typical power sequence like: enable USB
> > PHY clock, toggle reset pin, etc. But current Linux USB driver
> > lacks of such code to do it, it may cause some hard-wired USB devices
> > works abnormal or can't be recognized by controller at all.
> >
> > In this patch, it calls power sequence library APIs to finish
> > the power sequence events. At first, it calls pwrseq_alloc_generic
> > to create a generic power sequence instance, then it will do power
> > on sequence at hub's probe for all devices under this hub
> > (includes root hub).
> >
> > At hub_disconnect, it will do power off sequence which is at powered
> > on list.
> >
> > Signed-off-by: Peter Chen <peter.chen@nxp.com>
> > ---
> > drivers/usb/core/Makefile | 1 +
> > drivers/usb/core/hub.c | 12 ++++--
> > drivers/usb/core/hub.h | 12 ++++++
> > drivers/usb/core/pwrseq.c | 100 ++++++++++++++++++++++++++++++++++++++++++++++
> > 4 files changed, 122 insertions(+), 3 deletions(-)
> > create mode 100644 drivers/usb/core/pwrseq.c
> >
> > diff --git a/drivers/usb/core/Makefile b/drivers/usb/core/Makefile
> > index da36b78..5fd2f54 100644
> > --- a/drivers/usb/core/Makefile
> > +++ b/drivers/usb/core/Makefile
> > @@ -10,5 +10,6 @@ usbcore-y += port.o
> > usbcore-$(CONFIG_OF) += of.o
> > usbcore-$(CONFIG_PCI) += hcd-pci.o
> > usbcore-$(CONFIG_ACPI) += usb-acpi.o
> > +usbcore-$(CONFIG_PWRSEQ_GENERIC) += pwrseq.o
> >
> > obj-$(CONFIG_USB) += usbcore.o
> > diff --git a/drivers/usb/core/hub.c b/drivers/usb/core/hub.c
> > index bee1351..a346a8b 100644
> > --- a/drivers/usb/core/hub.c
> > +++ b/drivers/usb/core/hub.c
> > @@ -1700,6 +1700,7 @@ static void hub_disconnect(struct usb_interface *intf)
> > hub->error = 0;
> > hub_quiesce(hub, HUB_DISCONNECT);
> >
> > + hub_pwrseq_off(hub);
> > mutex_lock(&usb_port_peer_mutex);
> >
> > /* Avoid races with recursively_mark_NOTATTACHED() */
> > @@ -1733,6 +1734,7 @@ static int hub_probe(struct usb_interface *intf, const struct usb_device_id *id)
> > struct usb_endpoint_descriptor *endpoint;
> > struct usb_device *hdev;
> > struct usb_hub *hub;
> > + int ret = -ENODEV;
> >
> > desc = intf->cur_altsetting;
> > hdev = interface_to_usbdev(intf);
> > @@ -1839,6 +1841,7 @@ descriptor_error:
> > INIT_DELAYED_WORK(&hub->leds, led_work);
> > INIT_DELAYED_WORK(&hub->init_work, NULL);
> > INIT_WORK(&hub->events, hub_event);
> > + INIT_LIST_HEAD(&hub->pwrseq_on_list);
> > usb_get_intf(intf);
> > usb_get_dev(hdev);
> >
> > @@ -1852,11 +1855,14 @@ descriptor_error:
> > if (id->driver_info & HUB_QUIRK_CHECK_PORT_AUTOSUSPEND)
> > hub->quirk_check_port_auto_suspend = 1;
> >
> > - if (hub_configure(hub, endpoint) >= 0)
> > - return 0;
> > + if (hub_configure(hub, endpoint) >= 0) {
> > + ret = hub_pwrseq_on(hub);
> > + if (!ret)
> > + return 0;
> > + }
> >
> > hub_disconnect(intf);
> > - return -ENODEV;
> > + return ret;
> > }
> >
> > static int
> > diff --git a/drivers/usb/core/hub.h b/drivers/usb/core/hub.h
> > index 34c1a7e..9473f6f 100644
> > --- a/drivers/usb/core/hub.h
> > +++ b/drivers/usb/core/hub.h
> > @@ -78,6 +78,7 @@ struct usb_hub {
> > struct delayed_work init_work;
> > struct work_struct events;
> > struct usb_port **ports;
> > + struct list_head pwrseq_on_list; /* powered pwrseq node list */
> > };
> >
> > /**
> > @@ -166,3 +167,14 @@ static inline int hub_port_debounce_be_stable(struct usb_hub *hub,
> > {
> > return hub_port_debounce(hub, port1, false);
> > }
> > +
> > +#if IS_ENABLED(CONFIG_PWRSEQ_GENERIC)
> > +int hub_pwrseq_on(struct usb_hub *hub);
> > +void hub_pwrseq_off(struct usb_hub *hub);
> > +#else
> > +static inline int hub_pwrseq_on(struct usb_hub *hub)
> > +{
> > + return 0;
> > +}
> > +static inline void hub_pwrseq_off(struct usb_hub *hub) {}
> > +#endif /* CONFIG_PWRSEQ_GENERIC */
> > diff --git a/drivers/usb/core/pwrseq.c b/drivers/usb/core/pwrseq.c
> > new file mode 100644
> > index 0000000..837fe66
> > --- /dev/null
> > +++ b/drivers/usb/core/pwrseq.c
> > @@ -0,0 +1,100 @@
> > +/*
> > + * pwrseq.c USB device power sequence management
> > + *
> > + * Copyright (C) 2016 Freescale Semiconductor, Inc.
> > + * Author: Peter Chen <peter.chen@nxp.com>
> > + *
> > + * This program is free software: you can redistribute it and/or modify
> > + * it under the terms of the GNU General Public License version 2 of
> > + * the License as published by the Free Software Foundation.
> > + *
> > + * This program is distributed in the hope that it will be useful,
> > + * but WITHOUT ANY WARRANTY; without even the implied warranty of
> > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
> > + * GNU General Public License for more details.
> > + *
> > + * You should have received a copy of the GNU General Public License
> > + * along with this program. If not, see <http://www.gnu.org/licenses/>.
> > + */
> > +
> > +#include <linux/list.h>
> > +#include <linux/of.h>
> > +#include <linux/power/pwrseq.h>
> > +#include <linux/slab.h>
> > +#include <linux/usb.h>
> > +#include <linux/usb/hcd.h>
> > +
> > +#include "hub.h"
> > +
> > +struct usb_pwrseq_node {
> > + struct pwrseq *pwrseq;
> > + struct list_head list;
> > +};
> > +
> > +static int hub_of_pwrseq_on(struct device_node *np, struct usb_hub *hub)
> > +{
> > + struct pwrseq *pwrseq;
> > + struct usb_pwrseq_node *pwrseq_node;
> > + int ret;
> > +
> > + pwrseq = pwrseq_alloc_generic();
> > + if (IS_ERR(pwrseq))
> > + return PTR_ERR(pwrseq);
> > +
> > + ret = pwrseq_get(np, pwrseq);
> > + if (ret)
> > + goto pwr_free;
> > +
> > + ret = pwrseq_on(np, pwrseq);
> > + if (ret)
> > + goto pwr_put;
> > +
> > + pwrseq_node = kzalloc(sizeof(*pwrseq_node), GFP_KERNEL);
> > + pwrseq_node->pwrseq = pwrseq;
> > + list_add(&pwrseq_node->list, &hub->pwrseq_on_list);
> > +
> > + return 0;
> > +
> > +pwr_put:
> > + pwrseq_put(pwrseq);
> > +pwr_free:
> > + pwrseq_free(pwrseq);
> > + return ret;
> > +}
> > +
> > +int hub_pwrseq_on(struct usb_hub *hub)
> > +{
> > + struct device *parent;
> > + struct usb_device *hdev = hub->hdev;
> > + struct device_node *np;
> > + int ret;
> > +
> > + if (hdev->parent)
> > + parent = &hdev->dev;
> > + else
> > + parent = bus_to_hcd(hdev->bus)->self.controller;
> > +
> > + for_each_child_of_node(parent->of_node, np) {
> > + ret = hub_of_pwrseq_on(np, hub);
> > + if (ret)
> > + return ret;
> > + }
> > +
> > + return 0;
> > +}
> > +
> > +void hub_pwrseq_off(struct usb_hub *hub)
> > +{
> > + struct pwrseq *pwrseq;
> > + struct usb_pwrseq_node *pwrseq_node, *tmp_node;
> > +
> > + list_for_each_entry_safe(pwrseq_node, tmp_node,
> > + &hub->pwrseq_on_list, list) {
> > + pwrseq = pwrseq_node->pwrseq;
> > + pwrseq_off(pwrseq);
> > + pwrseq_put(pwrseq);
> > + pwrseq_free(pwrseq);
> > + list_del(&pwrseq_node->list);
> > + kfree(pwrseq_node);
> > + }
> > +}
> Patch 4 does not apply for me
> (The Makefile has a "usbcore-$(CONFIG_OF) += of.o"
> line which I don't see in 4.7 or linux/next master).
> Is there another patch set this series depends on?
It seems below patch has not been queued:
https://git.kernel.org/cgit/linux/kernel/git/peter.chen/usb.git/commit/?h=add_pwrseq_for_usb&id=2efd73d0f935ffbc66135553da0ac807c0b4c3fe
This pwrseq-lib patch set does not depend on it, I will send patch set
on top of a clean mainline next time, sorry for confusing.
--
Best Regards,
Peter Chen
next prev parent reply other threads:[~2016-07-28 1:45 UTC|newest]
Thread overview: 70+ messages / expand[flat|nested] mbox.gz Atom feed top
2016-07-20 9:40 [PATCH v3 0/6] power: add power sequence library Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-20 9:40 ` [PATCH v3 1/6] binding-doc: power: pwrseq-generic: add binding doc for generic " Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-29 2:09 ` Peter Chen
2016-07-29 2:09 ` Peter Chen
2016-07-29 21:09 ` Rob Herring
2016-07-29 21:09 ` Rob Herring
2016-07-20 9:40 ` [PATCH v3 2/6] power: add " Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-20 9:40 ` Peter Chen
[not found] ` <1469007629-31757-3-git-send-email-peter.chen-3arQi8VN3Tc@public.gmane.org>
2016-07-29 20:06 ` [v3,2/6] " Matthias Kaehlcke
2016-07-29 20:06 ` Matthias Kaehlcke
2016-07-29 20:06 ` Matthias Kaehlcke
2016-08-01 1:58 ` Peter Chen
2016-08-01 1:58 ` Peter Chen
2016-08-01 1:58 ` Peter Chen
2016-08-02 3:32 ` Peter Chen
2016-08-02 3:32 ` Peter Chen
2016-08-02 3:32 ` Peter Chen
2016-08-02 17:29 ` Matthias Kaehlcke
2016-07-20 9:40 ` [PATCH v3 3/6] binding-doc: usb: usb-device: add optional properties for power sequence Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-29 21:11 ` Rob Herring
2016-07-29 21:11 ` Rob Herring
2016-07-20 9:40 ` [PATCH v3 4/6] usb: core: add power sequence handling for USB devices Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-27 16:25 ` Joshua Clayton
2016-07-27 16:25 ` Joshua Clayton
[not found] ` <5798E067.90207-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
2016-07-28 1:45 ` Peter Chen [this message]
2016-07-28 1:45 ` Peter Chen
2016-07-28 1:45 ` Peter Chen
2016-07-28 16:18 ` Joshua Clayton
2016-07-28 16:18 ` Joshua Clayton
[not found] ` <1469007629-31757-1-git-send-email-peter.chen-3arQi8VN3Tc@public.gmane.org>
2016-07-20 9:40 ` [PATCH v3 5/6] usb: chipidea: let chipidea core device of_node equal's glue layer device of_node Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-20 9:40 ` Peter Chen
[not found] ` <1469007629-31757-6-git-send-email-peter.chen-3arQi8VN3Tc@public.gmane.org>
2016-07-21 9:14 ` Russell King - ARM Linux
2016-07-21 9:14 ` Russell King - ARM Linux
2016-07-21 9:14 ` Russell King - ARM Linux
2016-07-21 9:20 ` Peter Chen
2016-07-21 9:20 ` Peter Chen
2016-07-21 9:41 ` Russell King - ARM Linux
2016-07-21 9:41 ` Russell King - ARM Linux
2016-07-21 9:41 ` Russell King - ARM Linux
2016-07-21 10:12 ` Peter Chen
2016-07-21 10:12 ` Peter Chen
2016-07-20 9:40 ` [PATCH v3 6/6] ARM: dts: imx6qdl-udoo.dtsi: fix onboard USB HUB property Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-20 9:40 ` Peter Chen
2016-07-28 15:56 ` [PATCH v3 0/6] power: add power sequence library Joshua Clayton
2016-07-28 15:56 ` Joshua Clayton
2016-07-28 16:41 ` Fabio Estevam
2016-07-28 16:41 ` Fabio Estevam
2016-07-28 16:41 ` Fabio Estevam
[not found] ` <CAOMZO5Dv_X-u6qKqAWtysAqmMw91bVoAaRvnc81yUGDxbd27rA-JsoAwUIsXosN+BqQ9rBEUg@public.gmane.org>
2016-08-01 14:55 ` Joshua Clayton
2016-08-01 14:55 ` Joshua Clayton
2016-08-01 14:55 ` Joshua Clayton
[not found] ` <579F62CE.1050900-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
2016-08-01 16:29 ` Fabio Estevam
2016-08-01 16:29 ` Fabio Estevam
2016-08-01 16:29 ` Fabio Estevam
2016-08-02 1:08 ` Peter Chen
2016-08-02 1:08 ` Peter Chen
2016-08-02 1:08 ` Peter Chen
2016-07-29 1:46 ` Peter Chen
2016-07-29 1:46 ` Peter Chen
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=20160728014521.GB26186@shlinux2 \
--to=hzpeterchen-re5jqeeqqe8avxtiumwx3w@public.gmane.org \
--cc=arnd-r2nGTMty4D4@public.gmane.org \
--cc=broonie-DgEjT+Ai2ygdnm+yROfE0A@public.gmane.org \
--cc=dbaryshkov-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org \
--cc=devicetree-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
--cc=dwmw2-wEGCiKHe2LqWVfeAwA7xHQ@public.gmane.org \
--cc=festevam-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org \
--cc=gregkh-hQyY1W1yCW8ekmWlsbkhG0B+6BGkLq7r@public.gmane.org \
--cc=k.kozlowski-Sze3O3UU22JBDgjK7y7TUQ@public.gmane.org \
--cc=linux-arm-kernel-IAPFreCvJWM7uuMidbF8XUB+6BGkLq7r@public.gmane.org \
--cc=linux-kernel-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
--cc=linux-pm-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
--cc=linux-usb-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
--cc=mail-APzI5cXaD1zVlRWJc41N0YvC60bnQu0Y@public.gmane.org \
--cc=mark.rutland-5wv7dgnIgG8@public.gmane.org \
--cc=oscar-Bdbr4918Nnnk1uMJSBkQmQ@public.gmane.org \
--cc=p.zabel-bIcnvbaLZ9MEGnE8C9+IrQ@public.gmane.org \
--cc=pawel.moll-5wv7dgnIgG8@public.gmane.org \
--cc=peter.chen-3arQi8VN3Tc@public.gmane.org \
--cc=robh+dt-DgEjT+Ai2ygdnm+yROfE0A@public.gmane.org \
--cc=s.hauer-bIcnvbaLZ9MEGnE8C9+IrQ@public.gmane.org \
--cc=shawnguo-DgEjT+Ai2ygdnm+yROfE0A@public.gmane.org \
--cc=sre-DgEjT+Ai2ygdnm+yROfE0A@public.gmane.org \
--cc=stephen.boyd-QSEj5FYQhm4dnm+yROfE0A@public.gmane.org \
--cc=stern-nwvwT67g6+6dFdvTe/nMLpVzexx5G7lz@public.gmane.org \
--cc=stillcompiling-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org \
--cc=troy.kisky-Q5RJGjKts06CY9SHAMCTRUEOCMrvLtNR@public.gmane.org \
--cc=ulf.hansson-QSEj5FYQhm4dnm+yROfE0A@public.gmane.org \
/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 an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.