svn commit: r359120 - head/sys/dev/usb
Hans Petter Selasky
hselasky at FreeBSD.org
Thu Mar 19 09:16:01 UTC 2020
Author: hselasky
Date: Thu Mar 19 09:15:59 2020
New Revision: 359120
URL: https://svnweb.freebsd.org/changeset/base/359120
Log:
Correctly implement support for remote wakeup for USB 3.0 device.
Submitted by: Horse Ma <Shichun.Ma at dell.com>
MFC after: 1 week
Sponsored by: Mellanox Technologies
Modified:
head/sys/dev/usb/usb.h
head/sys/dev/usb/usb_hub.c
Modified: head/sys/dev/usb/usb.h
==============================================================================
--- head/sys/dev/usb/usb.h Thu Mar 19 08:13:51 2020 (r359119)
+++ head/sys/dev/usb/usb.h Thu Mar 19 09:15:59 2020 (r359120)
@@ -274,6 +274,11 @@ typedef struct usb_device_request usb_device_request_t
#define UHF_C_BH_PORT_RESET 29
#define UHF_FORCE_LINKPM_ACCEPT 30
+/* SuperSpeed suspend support */
+#define USB_INTERFACE_FUNC_SUSPEND 0
+#define USB_INTERFACE_FUNC_SUSPEND_LP (1 << 8)
+#define USB_INTERFACE_FUNC_SUSPEND_RW (1 << 9)
+
struct usb_descriptor {
uByte bLength;
uByte bDescriptorType;
Modified: head/sys/dev/usb/usb_hub.c
==============================================================================
--- head/sys/dev/usb/usb_hub.c Thu Mar 19 08:13:51 2020 (r359119)
+++ head/sys/dev/usb/usb_hub.c Thu Mar 19 09:15:59 2020 (r359120)
@@ -2571,6 +2571,50 @@ usb_bus_powerd(struct usb_bus *bus)
}
#endif
+static usb_error_t
+usbd_device_30_remote_wakeup(struct usb_device *udev, uint8_t bRequest)
+{
+ struct usb_device_request req = {};
+
+ req.bmRequestType = UT_WRITE_INTERFACE;
+ req.bRequest = bRequest;
+ USETW(req.wValue, USB_INTERFACE_FUNC_SUSPEND);
+ USETW(req.wIndex, USB_INTERFACE_FUNC_SUSPEND_LP |
+ USB_INTERFACE_FUNC_SUSPEND_RW);
+
+ return (usbd_do_request(udev, NULL, &req, 0));
+}
+
+static usb_error_t
+usbd_clear_dev_wakeup(struct usb_device *udev)
+{
+ usb_error_t err;
+
+ if (usb_device_20_compatible(udev)) {
+ err = usbd_req_clear_device_feature(udev,
+ NULL, UF_DEVICE_REMOTE_WAKEUP);
+ } else {
+ err = usbd_device_30_remote_wakeup(udev,
+ UR_CLEAR_FEATURE);
+ }
+ return (err);
+}
+
+static usb_error_t
+usbd_set_dev_wakeup(struct usb_device *udev)
+{
+ usb_error_t err;
+
+ if (usb_device_20_compatible(udev)) {
+ err = usbd_req_set_device_feature(udev,
+ NULL, UF_DEVICE_REMOTE_WAKEUP);
+ } else {
+ err = usbd_device_30_remote_wakeup(udev,
+ UR_SET_FEATURE);
+ }
+ return (err);
+}
+
/*------------------------------------------------------------------------*
* usb_dev_resume_peer
*
@@ -2674,8 +2718,7 @@ usb_dev_resume_peer(struct usb_device *udev)
/* check if peer has wakeup capability */
if (usb_peer_can_wakeup(udev)) {
/* clear remote wakeup */
- err = usbd_req_clear_device_feature(udev,
- NULL, UF_DEVICE_REMOTE_WAKEUP);
+ err = usbd_clear_dev_wakeup(udev);
if (err) {
DPRINTFN(0, "Clearing device "
"remote wakeup failed: %s\n",
@@ -2740,8 +2783,7 @@ repeat:
*/
/* allow device to do remote wakeup */
- err = usbd_req_set_device_feature(udev,
- NULL, UF_DEVICE_REMOTE_WAKEUP);
+ err = usbd_set_dev_wakeup(udev);
if (err) {
DPRINTFN(0, "Setting device "
"remote wakeup failed\n");
@@ -2767,8 +2809,7 @@ repeat:
if (err != 0) {
if (usb_peer_can_wakeup(udev)) {
/* allow device to do remote wakeup */
- err = usbd_req_clear_device_feature(udev,
- NULL, UF_DEVICE_REMOTE_WAKEUP);
+ err = usbd_clear_dev_wakeup(udev);
if (err) {
DPRINTFN(0, "Setting device "
"remote wakeup failed\n");
More information about the svn-src-head
mailing list