From: Michael Grzeschik <mgr@pengutronix.de>
To: Alan Stern <stern@rowland.harvard.edu>
Cc: linux-usb@vger.kernel.org, gregkh@linuxfoundation.org,
kernel@pengutronix.de
Subject: Re: [PATCH v3] usb: hub: port: add sysfs entry to switch port power
Date: Fri, 3 Jun 2022 19:22:09 +0200 [thread overview]
Message-ID: <20220603172209.GD26638@pengutronix.de> (raw)
In-Reply-To: <YpovuqKtQBKQoVos@rowland.harvard.edu>
[-- Attachment #1: Type: text/plain, Size: 7421 bytes --]
On Fri, Jun 03, 2022 at 11:58:50AM -0400, Alan Stern wrote:
>On Fri, Jun 03, 2022 at 01:52:22PM +0200, Michael Grzeschik wrote:
>> In some cases the port of an hub needs to be disabled or switched off
>> and on again. E.g. when the connected device needs to be re-enumerated.
>> Or it needs to be explicitly disabled while the rest of the usb tree
>> stays working.
>>
>> For this purpose this patch adds an sysfs switch to enable/disable the
>> port on any hub. In the case the hub is supporting power switching, the
>> power line will be disabled to the connected device.
>>
>> When the port gets disabled, the associated device gets disconnected and
>> removed from the logical usb tree. No further device will be enumerated
>> on that port until the port gets enabled again.
>>
>> Signed-off-by: Michael Grzeschik <m.grzeschik@pengutronix.de>
>>
>> ---
>
>This is looking a lot better. I have only a few small comments.
>
>> v1 -> v2:
>> - improved patch description
>> - moved usb_hub_set_port_power to end of function
>> - renamed value to set
>> - removed udev variable
>> - added usb_set_configuration set to -1 before removing device
>> - calling autosuspend of udev before usb_disconnect, ensuring hub_suspend succeeds
>> - removed port_dev->child = NULL assignment
>> - directly returning count on no failure success
>> - removed test for hub->in_reset
>> - using usb_autopm_get_interface/usb_autopm_put_interface around hub handling
>> - locking usb_disconnect call
>> - using &port_dev->child instead of local udev pointer
>> - added Documentation/ABI
>>
>> v2 -> v3:
>> - renamed sysfs file to disable instead of port_power
>> - added disable_show function to read out the current port state
>> - moved usb_lock/unlock_device near put/get_interface
>> - removed unnecessary usb_set_configuration of port_dev->child before disconnect
>> - removed unnecessary usb_autosuspend of port_dev->child before disconnect
>> - moved clearing of port_feature flags to be done after usb_hub_set_port_power
>> - checking for hub->disconnected after locking hdev
>> - updated the ABI documentation
>>
>> Documentation/ABI/testing/sysfs-bus-usb | 11 +++++
>> drivers/usb/core/port.c | 61 +++++++++++++++++++++++++
>> 2 files changed, 72 insertions(+)
>>
>> diff --git a/Documentation/ABI/testing/sysfs-bus-usb b/Documentation/ABI/testing/sysfs-bus-usb
>> index 7efe31ed3a25c7..d907123ac5d0f9 100644
>> --- a/Documentation/ABI/testing/sysfs-bus-usb
>> +++ b/Documentation/ABI/testing/sysfs-bus-usb
>> @@ -253,6 +253,17 @@ Description:
>> only if the system firmware is capable of describing the
>> connection between a port and its connector.
>>
>> +What: /sys/bus/usb/devices/.../<hub_interface>/port<X>/disable
>> +Date: June 2022
>> +Contact: Michael Grzeschik <m.grzeschik@pengutronix.de>
>> +Description:
>> + This file controls the state to USB ports, including
>
>s/to USB ports/of a USB port/
Okay
>> + Vbus power output (but only on hubs that support
>> + power switching -- most hubs don't support it). When
>> + turning a port off, the port is unusable: Devices
>
>s/When turning a port off/If a port is disabled/
Okay
>> + attached to the port will not be detected, initialized,
>> + or enumerated.
>> +
>> What: /sys/bus/usb/devices/.../power/usb2_lpm_l1_timeout
>> Date: May 2013
>> Contact: Mathias Nyman <mathias.nyman@linux.intel.com>
>> diff --git a/drivers/usb/core/port.c b/drivers/usb/core/port.c
>> index d5bc36ca5b1f77..8343590c3800f8 100644
>> --- a/drivers/usb/core/port.c
>> +++ b/drivers/usb/core/port.c
>> @@ -17,6 +17,66 @@ static int usb_port_block_power_off;
>>
>> static const struct attribute_group *port_dev_group[];
>>
>> +static ssize_t disable_show(struct device *dev,
>> + struct device_attribute *attr, char *buf)
>> +{
>> + struct usb_port *port_dev = to_usb_port(dev);
>> + struct usb_device *hdev = to_usb_device(dev->parent->parent);
>> + struct usb_hub *hub = usb_hub_to_struct_hub(hdev);
>> + int port1 = port_dev->portnum;
>> + bool state = test_bit(port1, hub->power_bits);
>> +
>> + return sprintf(buf, "%s\n", state ? "0" : "1");
>
>Maybe "false" and "true" instead of "0" and "1"?
I prefer 0 and 1 since we also feed this to the file.
Also, I just found out that just parsing power_bits is not enough.
E.g. when we use other tools to set clear PORT_POWER on the hub like
uhubctl to disable a port. The value does not represent the real state
of the port.
I think it is better to use hub_port_status and port_is_power_on from
hub.c to test the real state by sending a GET_STATUS command.
For this, the functions need to be unset static and exported via hub.h.
I will add this in v4.
>> +}
>> +
>> +static ssize_t disable_store(struct device *dev, struct device_attribute *attr,
>> + const char *buf, size_t count)
>> +{
>> + struct usb_port *port_dev = to_usb_port(dev);
>> + struct usb_device *hdev = to_usb_device(dev->parent->parent);
>> + struct usb_hub *hub = usb_hub_to_struct_hub(hdev);
>> + struct usb_interface *intf = to_usb_interface(hub->intfdev);
>> + int port1 = port_dev->portnum;
>> + bool set;
>> + int rc;
>> +
>> + if (!hub)
>> + return -EINVAL;
>
>Is this test needed? If it is then it should be present in
>disable_show() as well, and the line above that calls to_usb_interface()
>should not run until after the test has been done.
>
>However, I suspect the test isn't needed.
Okay, I will remove it.
>> +
>> + rc = strtobool(buf, &set);
>> + if (rc)
>> + return rc;
>> +
>> + rc = usb_autopm_get_interface(intf);
>> + if (rc < 0)
>> + return rc;
>> +
>> + usb_lock_device(hdev);
>> + if (unlikely(hub->disconnected))
>
>Add: rc = -ENODEV;
>
Right.
>> + goto out_hdev_lock;
>> +
>> + if (set && port_dev->child)
>> + usb_disconnect(&port_dev->child);
>
>I think the logic will be easier to understand if you rename "set" to
>"disabled".
Much better!
>> +
>> + rc = usb_hub_set_port_power(hdev, hub, port1, !set);
>> +
>> + if (set) {
>> + usb_clear_port_feature(hdev, port1, USB_PORT_FEAT_C_CONNECTION);
>> + if (!port_dev->is_superspeed)
>> + usb_clear_port_feature(hdev, port1, USB_PORT_FEAT_C_ENABLE);
>> + }
>> +
>> + if (!rc)
>> + rc = count;
>> +
>> +out_hdev_lock:
>> + usb_unlock_device(hdev);
>> + usb_autopm_put_interface(intf);
>> +
>> + return rc;
>> +}
>> +static DEVICE_ATTR_RW(disable);
>> +
>> static ssize_t location_show(struct device *dev,
>> struct device_attribute *attr, char *buf)
>> {
>> @@ -153,6 +213,7 @@ static struct attribute *port_dev_attrs[] = {
>> &dev_attr_location.attr,
>> &dev_attr_quirks.attr,
>> &dev_attr_over_current_count.attr,
>> + &dev_attr_disable.attr,
>> NULL,
>> };
>
>Alan Stern
Thanks,
Michael
--
Pengutronix e.K. | |
Steuerwalder Str. 21 | http://www.pengutronix.de/ |
31137 Hildesheim, Germany | Phone: +49-5121-206917-0 |
Amtsgericht Hildesheim, HRA 2686 | Fax: +49-5121-206917-5555 |
[-- Attachment #2: signature.asc --]
[-- Type: application/pgp-signature, Size: 833 bytes --]
next prev parent reply other threads:[~2022-06-03 17:22 UTC|newest]
Thread overview: 6+ messages / expand[flat|nested] mbox.gz Atom feed top
2022-06-03 11:52 [PATCH v3] usb: hub: port: add sysfs entry to switch port power Michael Grzeschik
2022-06-03 15:58 ` Alan Stern
2022-06-03 17:22 ` Michael Grzeschik [this message]
2022-06-03 19:37 ` Alan Stern
2022-06-03 20:29 ` Michael Grzeschik
2022-06-03 21:24 ` Alan Stern
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=20220603172209.GD26638@pengutronix.de \
--to=mgr@pengutronix.de \
--cc=gregkh@linuxfoundation.org \
--cc=kernel@pengutronix.de \
--cc=linux-usb@vger.kernel.org \
--cc=stern@rowland.harvard.edu \
/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