Commit f17c34ff authored by Oliver Neukum's avatar Oliver Neukum Committed by Greg Kroah-Hartman

USB: hub: check for alternate port before enabling A_ALT_HNP_SUPPORT

The OTG 1.3 spec has the feature A_ALT_HNP_SUPPORT, which tells
a device that it is connected to the wrong port. Some devices
refuse to operate if you enable that feature, because it indicates
to them that they ought to request to be connected to another port.

According to the spec this feature may be used based only the following
three conditions:

6.5.3 a_alt_hnp_support
Setting this feature indicates to the B-device that it is connected to
an A-device port that is not capable of HNP, but that the A-device does
have an alternate port that is capable of HNP.
The A-device is required to set this feature under the following conditions:
• the A-device has multiple receptacles
• the A-device port that connects to the B-device does not support HNP
• the A-device has another port that does support HNP

A check for the third and first condition is missing. Add it.
Signed-off-by: default avatarOliver Neukum <oneukum@suse.com>
Cc: stable <stable@kernel.org>
Fixes: 7d2d641c ("usb: otg: don't set a_alt_hnp_support feature for OTG 2.0 device")
Link: https://lore.kernel.org/r/20240122153545.12284-1-oneukum@suse.comSigned-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent cc509b6a
...@@ -2398,17 +2398,25 @@ static int usb_enumerate_device_otg(struct usb_device *udev) ...@@ -2398,17 +2398,25 @@ static int usb_enumerate_device_otg(struct usb_device *udev)
} }
} else if (desc->bLength == sizeof } else if (desc->bLength == sizeof
(struct usb_otg_descriptor)) { (struct usb_otg_descriptor)) {
/* Set a_alt_hnp_support for legacy otg device */ /*
err = usb_control_msg(udev, * We are operating on a legacy OTP device
usb_sndctrlpipe(udev, 0), * These should be told that they are operating
USB_REQ_SET_FEATURE, 0, * on the wrong port if we have another port that does
USB_DEVICE_A_ALT_HNP_SUPPORT, * support HNP
0, NULL, 0, */
USB_CTRL_SET_TIMEOUT); if (bus->otg_port != 0) {
if (err < 0) /* Set a_alt_hnp_support for legacy otg device */
dev_err(&udev->dev, err = usb_control_msg(udev,
"set a_alt_hnp_support failed: %d\n", usb_sndctrlpipe(udev, 0),
err); USB_REQ_SET_FEATURE, 0,
USB_DEVICE_A_ALT_HNP_SUPPORT,
0, NULL, 0,
USB_CTRL_SET_TIMEOUT);
if (err < 0)
dev_err(&udev->dev,
"set a_alt_hnp_support failed: %d\n",
err);
}
} }
} }
#endif #endif
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment