[U-Boot] [RFC PATCH v2 3/6] dm: usb: Add UCLASS_USB_DEV_GENERIC shutdown

Jagan Teki jagan at amarulasolutions.com
Tue Aug 7 07:03:35 UTC 2018


On Tue, Jul 24, 2018 at 5:18 AM, Simon Glass <sjg at chromium.org> wrote:
> Hi Jagan,
>
> On 20 July 2018 at 01:13, Jagan Teki <jagan at amarulasolutions.com> wrote:
>> Some OTG controllers which operates on Peripheral
>> mode are registered as UCLASS_USB_DEV_GENERIC.
>>
>> So add support to shutdown them as well. shutdown
>> happened during 'usb reset' and U-Boot handoff code
>> for Linux boot.
>>
>> controller restarting in 'usb reset' like probing
>> again is still missing for this type of UCLASS.
>>
>> Cc: Simon Glass <sjg at chromium.org>
>> Signed-off-by: Jagan Teki <jagan at amarulasolutions.com>
>> ---
>> Note:
>> I tried of traversing for multiple UCLASS in removal
>> code in usb_stop, but couldn't come with proper solutions.
>> I don't think any other area of code need a requirement like
>> this, or may be I'm missing. any help would appreciate.
>>
>> Changes for v2:
>> - none
>>
>>  drivers/usb/host/usb-uclass.c | 43 +++++++++++++++++++++++++++++++++++
>>  1 file changed, 43 insertions(+)
>>
>> diff --git a/drivers/usb/host/usb-uclass.c b/drivers/usb/host/usb-uclass.c
>> index 611ea97a72..99cf3d2b49 100644
>> --- a/drivers/usb/host/usb-uclass.c
>> +++ b/drivers/usb/host/usb-uclass.c
>> @@ -158,6 +158,45 @@ int usb_get_max_xfer_size(struct usb_device *udev, size_t *size)
>>         return ops->get_max_xfer_size(bus, size);
>>  }
>>
>> +int __usb_stop(void)
>
> Why the __ ? I think it should be something like usb_remove_and_unbind_all()
>
>> +{
>> +       struct udevice *bus;
>> +       struct udevice *rh;
>> +       struct uclass *uc;
>> +       struct usb_uclass_priv *uc_priv;
>> +       int err = 0, ret;
>> +
>> +       /* De-activate any devices that have been activated */
>> +       ret = uclass_get(UCLASS_USB_DEV_GENERIC, &uc);
>> +       if (ret)
>> +               return ret;
>> +
>> +       uc_priv = uc->priv;
>> +
>> +       uclass_foreach_dev(bus, uc) {
>> +               ret = device_remove(bus, DM_REMOVE_NORMAL);
>> +               if (ret && !err)
>> +                       err = ret;
>> +
>> +               /* Locate root hub device */
>> +               device_find_first_child(bus, &rh);
>> +               if (rh) {
>> +                       /*
>> +                        * All USB devices are children of root hub.
>> +                        * Unbinding root hub will unbind all of its children.
>> +                        */
>> +                       ret = device_unbind(rh);
>> +                       if (ret && !err)
>> +                               err = ret;
>> +               }
>> +       }
>> +
>> +       uc_priv->companion_device_count = 0;
>> +       usb_started = 0;
>> +
>> +       return err;
>> +}
>> +
>>  int usb_stop(void)
>>  {
>>         struct udevice *bus;
>> @@ -166,6 +205,10 @@ int usb_stop(void)
>>         struct usb_uclass_priv *uc_priv;
>>         int err = 0, ret;
>>
>> +       ret = __usb_stop();
>> +       if (ret)
>> +               return ret;
>> +
>
> This looks like the same code that appears below here, or very
> similar. Why is this needed?

Here the usage-case it to remove/unbind UCLASS_USB and
UCLASS_USB_DEV_GENERIC and same code will use to do that, any
suggestions?


More information about the U-Boot mailing list