Blame 0613-usb3-superspeed-endpoint-companion.patch

5544c1
From e0354b4f91dd198b5bfe1ddf649588d6af84ea9c Mon Sep 17 00:00:00 2001
c8dfc6
From: Gerd Hoffmann <kraxel@redhat.com>
c8dfc6
Date: Tue, 28 Aug 2012 17:28:03 +0200
5544c1
Subject: [PATCH] usb3: superspeed endpoint companion
c8dfc6
c8dfc6
Add support for building superspeed endpoint companion descriptors,
c8dfc6
create them for superspeed usb devices.
c8dfc6
c8dfc6
Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
c8dfc6
---
c8dfc6
 hw/usb.h      |  1 +
c8dfc6
 hw/usb/desc.c | 55 ++++++++++++++++++++++++++++++++++++++++---------------
c8dfc6
 hw/usb/desc.h | 26 +++++++++++++++++++++-----
c8dfc6
 3 files changed, 62 insertions(+), 20 deletions(-)
c8dfc6
c8dfc6
diff --git a/hw/usb.h b/hw/usb.h
c8dfc6
index 684e3f4..78ffdf4 100644
c8dfc6
--- a/hw/usb.h
c8dfc6
+++ b/hw/usb.h
c8dfc6
@@ -137,6 +137,7 @@
c8dfc6
 #define USB_DT_INTERFACE_ASSOC          0x0B
c8dfc6
 #define USB_DT_CS_INTERFACE             0x24
c8dfc6
 #define USB_DT_CS_ENDPOINT              0x25
c8dfc6
+#define USB_DT_ENDPOINT_COMPANION       0x30
c8dfc6
 
c8dfc6
 #define USB_ENDPOINT_XFER_CONTROL	0
c8dfc6
 #define USB_ENDPOINT_XFER_ISOC		1
c8dfc6
diff --git a/hw/usb/desc.c b/hw/usb/desc.c
c8dfc6
index 3e8c6cb..8f5a8e5 100644
c8dfc6
--- a/hw/usb/desc.c
c8dfc6
+++ b/hw/usb/desc.c
c8dfc6
@@ -76,7 +76,8 @@ int usb_desc_device_qualifier(const USBDescDevice *dev,
c8dfc6
     return bLength;
c8dfc6
 }
c8dfc6
 
c8dfc6
-int usb_desc_config(const USBDescConfig *conf, uint8_t *dest, size_t len)
c8dfc6
+int usb_desc_config(const USBDescConfig *conf, int flags,
c8dfc6
+                    uint8_t *dest, size_t len)
c8dfc6
 {
c8dfc6
     uint8_t  bLength = 0x09;
c8dfc6
     uint16_t wTotalLength = 0;
c8dfc6
@@ -99,7 +100,7 @@ int usb_desc_config(const USBDescConfig *conf, uint8_t *dest, size_t len)
c8dfc6
 
c8dfc6
     /* handle grouped interfaces if any */
c8dfc6
     for (i = 0; i < conf->nif_groups; i++) {
c8dfc6
-        rc = usb_desc_iface_group(&(conf->if_groups[i]),
c8dfc6
+        rc = usb_desc_iface_group(&(conf->if_groups[i]), flags,
c8dfc6
                                   dest + wTotalLength,
c8dfc6
                                   len - wTotalLength);
c8dfc6
         if (rc < 0) {
c8dfc6
@@ -110,7 +111,8 @@ int usb_desc_config(const USBDescConfig *conf, uint8_t *dest, size_t len)
c8dfc6
 
c8dfc6
     /* handle normal (ungrouped / no IAD) interfaces if any */
c8dfc6
     for (i = 0; i < conf->nif; i++) {
c8dfc6
-        rc = usb_desc_iface(conf->ifs + i, dest + wTotalLength, len - wTotalLength);
c8dfc6
+        rc = usb_desc_iface(conf->ifs + i, flags,
c8dfc6
+                            dest + wTotalLength, len - wTotalLength);
c8dfc6
         if (rc < 0) {
c8dfc6
             return rc;
c8dfc6
         }
c8dfc6
@@ -122,8 +124,8 @@ int usb_desc_config(const USBDescConfig *conf, uint8_t *dest, size_t len)
c8dfc6
     return wTotalLength;
c8dfc6
 }
c8dfc6
 
c8dfc6
-int usb_desc_iface_group(const USBDescIfaceAssoc *iad, uint8_t *dest,
c8dfc6
-                         size_t len)
c8dfc6
+int usb_desc_iface_group(const USBDescIfaceAssoc *iad, int flags,
c8dfc6
+                         uint8_t *dest, size_t len)
c8dfc6
 {
c8dfc6
     int pos = 0;
c8dfc6
     int i = 0;
c8dfc6
@@ -147,7 +149,7 @@ int usb_desc_iface_group(const USBDescIfaceAssoc *iad, uint8_t *dest,
c8dfc6
 
c8dfc6
     /* handle associated interfaces in this group */
c8dfc6
     for (i = 0; i < iad->nif; i++) {
c8dfc6
-        int rc = usb_desc_iface(&(iad->ifs[i]), dest + pos, len - pos);
c8dfc6
+        int rc = usb_desc_iface(&(iad->ifs[i]), flags, dest + pos, len - pos);
c8dfc6
         if (rc < 0) {
c8dfc6
             return rc;
c8dfc6
         }
c8dfc6
@@ -157,7 +159,8 @@ int usb_desc_iface_group(const USBDescIfaceAssoc *iad, uint8_t *dest,
c8dfc6
     return pos;
c8dfc6
 }
c8dfc6
 
c8dfc6
-int usb_desc_iface(const USBDescIface *iface, uint8_t *dest, size_t len)
c8dfc6
+int usb_desc_iface(const USBDescIface *iface, int flags,
c8dfc6
+                   uint8_t *dest, size_t len)
c8dfc6
 {
c8dfc6
     uint8_t bLength = 0x09;
c8dfc6
     int i, rc, pos = 0;
c8dfc6
@@ -188,7 +191,7 @@ int usb_desc_iface(const USBDescIface *iface, uint8_t *dest, size_t len)
c8dfc6
     }
c8dfc6
 
c8dfc6
     for (i = 0; i < iface->bNumEndpoints; i++) {
c8dfc6
-        rc = usb_desc_endpoint(iface->eps + i, dest + pos, len - pos);
c8dfc6
+        rc = usb_desc_endpoint(iface->eps + i, flags, dest + pos, len - pos);
c8dfc6
         if (rc < 0) {
c8dfc6
             return rc;
c8dfc6
         }
c8dfc6
@@ -198,13 +201,15 @@ int usb_desc_iface(const USBDescIface *iface, uint8_t *dest, size_t len)
c8dfc6
     return pos;
c8dfc6
 }
c8dfc6
 
c8dfc6
-int usb_desc_endpoint(const USBDescEndpoint *ep, uint8_t *dest, size_t len)
c8dfc6
+int usb_desc_endpoint(const USBDescEndpoint *ep, int flags,
c8dfc6
+                      uint8_t *dest, size_t len)
c8dfc6
 {
c8dfc6
     uint8_t bLength = ep->is_audio ? 0x09 : 0x07;
c8dfc6
     uint8_t extralen = ep->extra ? ep->extra[0] : 0;
c8dfc6
+    uint8_t superlen = (flags & USB_DESC_FLAG_SUPER) ? 0x06 : 0;
c8dfc6
     USBDescriptor *d = (void *)dest;
c8dfc6
 
c8dfc6
-    if (len < bLength + extralen) {
c8dfc6
+    if (len < bLength + extralen + superlen) {
c8dfc6
         return -1;
c8dfc6
     }
c8dfc6
 
c8dfc6
@@ -224,7 +229,21 @@ int usb_desc_endpoint(const USBDescEndpoint *ep, uint8_t *dest, size_t len)
c8dfc6
         memcpy(dest + bLength, ep->extra, extralen);
c8dfc6
     }
c8dfc6
 
c8dfc6
-    return bLength + extralen;
c8dfc6
+    if (superlen) {
c8dfc6
+        USBDescriptor *d = (void *)(dest + bLength + extralen);
c8dfc6
+
c8dfc6
+        d->bLength                       = 0x06;
c8dfc6
+        d->bDescriptorType               = USB_DT_ENDPOINT_COMPANION;
c8dfc6
+
c8dfc6
+        d->u.super_endpoint.bMaxBurst    = ep->bMaxBurst;
c8dfc6
+        d->u.super_endpoint.bmAttributes = ep->bmAttributes_super;
c8dfc6
+        d->u.super_endpoint.wBytesPerInterval_lo =
c8dfc6
+            usb_lo(ep->wBytesPerInterval);
c8dfc6
+        d->u.super_endpoint.wBytesPerInterval_hi =
c8dfc6
+            usb_hi(ep->wBytesPerInterval);
c8dfc6
+    }
c8dfc6
+
c8dfc6
+    return bLength + extralen + superlen;
c8dfc6
 }
c8dfc6
 
c8dfc6
 int usb_desc_other(const USBDescOther *desc, uint8_t *dest, size_t len)
c8dfc6
@@ -509,7 +528,7 @@ int usb_desc_get_descriptor(USBDevice *dev, int value, uint8_t *dest, size_t len
c8dfc6
     uint8_t buf[256];
c8dfc6
     uint8_t type = value >> 8;
c8dfc6
     uint8_t index = value & 0xff;
c8dfc6
-    int ret = -1;
c8dfc6
+    int flags, ret = -1;
c8dfc6
 
c8dfc6
     if (dev->speed == USB_SPEED_HIGH) {
c8dfc6
         other_dev = usb_device_get_usb_desc(dev)->full;
c8dfc6
@@ -517,6 +536,11 @@ int usb_desc_get_descriptor(USBDevice *dev, int value, uint8_t *dest, size_t len
c8dfc6
         other_dev = usb_device_get_usb_desc(dev)->high;
c8dfc6
     }
c8dfc6
 
c8dfc6
+    flags = 0;
c8dfc6
+    if (dev->device->bcdUSB >= 0x0300) {
c8dfc6
+        flags |= USB_DESC_FLAG_SUPER;
c8dfc6
+    }
c8dfc6
+
c8dfc6
     switch(type) {
c8dfc6
     case USB_DT_DEVICE:
c8dfc6
         ret = usb_desc_device(&desc->id, dev->device, buf, sizeof(buf));
c8dfc6
@@ -524,7 +548,8 @@ int usb_desc_get_descriptor(USBDevice *dev, int value, uint8_t *dest, size_t len
c8dfc6
         break;
c8dfc6
     case USB_DT_CONFIG:
c8dfc6
         if (index < dev->device->bNumConfigurations) {
c8dfc6
-            ret = usb_desc_config(dev->device->confs + index, buf, sizeof(buf));
c8dfc6
+            ret = usb_desc_config(dev->device->confs + index, flags,
c8dfc6
+                                  buf, sizeof(buf));
c8dfc6
         }
c8dfc6
         trace_usb_desc_config(dev->addr, index, len, ret);
c8dfc6
         break;
c8dfc6
@@ -532,7 +557,6 @@ int usb_desc_get_descriptor(USBDevice *dev, int value, uint8_t *dest, size_t len
c8dfc6
         ret = usb_desc_string(dev, index, buf, sizeof(buf));
c8dfc6
         trace_usb_desc_string(dev->addr, index, len, ret);
c8dfc6
         break;
c8dfc6
-
c8dfc6
     case USB_DT_DEVICE_QUALIFIER:
c8dfc6
         if (other_dev != NULL) {
c8dfc6
             ret = usb_desc_device_qualifier(other_dev, buf, sizeof(buf));
c8dfc6
@@ -541,7 +565,8 @@ int usb_desc_get_descriptor(USBDevice *dev, int value, uint8_t *dest, size_t len
c8dfc6
         break;
c8dfc6
     case USB_DT_OTHER_SPEED_CONFIG:
c8dfc6
         if (other_dev != NULL && index < other_dev->bNumConfigurations) {
c8dfc6
-            ret = usb_desc_config(other_dev->confs + index, buf, sizeof(buf));
c8dfc6
+            ret = usb_desc_config(other_dev->confs + index, flags,
c8dfc6
+                                  buf, sizeof(buf));
c8dfc6
             buf[0x01] = USB_DT_OTHER_SPEED_CONFIG;
c8dfc6
         }
c8dfc6
         trace_usb_desc_other_speed_config(dev->addr, index, len, ret);
c8dfc6
diff --git a/hw/usb/desc.h b/hw/usb/desc.h
c8dfc6
index d89fa41..4b5e88d 100644
c8dfc6
--- a/hw/usb/desc.h
c8dfc6
+++ b/hw/usb/desc.h
c8dfc6
@@ -63,6 +63,12 @@ typedef struct USBDescriptor {
c8dfc6
             uint8_t           bRefresh;        /* only audio ep */
c8dfc6
             uint8_t           bSynchAddress;   /* only audio ep */
c8dfc6
         } endpoint;
c8dfc6
+        struct {
c8dfc6
+            uint8_t           bMaxBurst;
c8dfc6
+            uint8_t           bmAttributes;
c8dfc6
+            uint8_t           wBytesPerInterval_lo;
c8dfc6
+            uint8_t           wBytesPerInterval_hi;
c8dfc6
+        } super_endpoint;
c8dfc6
     } u;
c8dfc6
 } QEMU_PACKED USBDescriptor;
c8dfc6
 
c8dfc6
@@ -139,6 +145,11 @@ struct USBDescEndpoint {
c8dfc6
 
c8dfc6
     uint8_t                   is_audio; /* has bRefresh + bSynchAddress */
c8dfc6
     uint8_t                   *extra;
c8dfc6
+
c8dfc6
+    /* superspeed endpoint companion */
c8dfc6
+    uint8_t                   bMaxBurst;
c8dfc6
+    uint8_t                   bmAttributes_super;
c8dfc6
+    uint16_t                  wBytesPerInterval;
c8dfc6
 };
c8dfc6
 
c8dfc6
 struct USBDescOther {
c8dfc6
@@ -156,16 +167,21 @@ struct USBDesc {
c8dfc6
     const char* const         *str;
c8dfc6
 };
c8dfc6
 
c8dfc6
+#define USB_DESC_FLAG_SUPER (1 << 1)
c8dfc6
+
c8dfc6
 /* generate usb packages from structs */
c8dfc6
 int usb_desc_device(const USBDescID *id, const USBDescDevice *dev,
c8dfc6
                     uint8_t *dest, size_t len);
c8dfc6
 int usb_desc_device_qualifier(const USBDescDevice *dev,
c8dfc6
                               uint8_t *dest, size_t len);
c8dfc6
-int usb_desc_config(const USBDescConfig *conf, uint8_t *dest, size_t len);
c8dfc6
-int usb_desc_iface_group(const USBDescIfaceAssoc *iad, uint8_t *dest,
c8dfc6
-                         size_t len);
c8dfc6
-int usb_desc_iface(const USBDescIface *iface, uint8_t *dest, size_t len);
c8dfc6
-int usb_desc_endpoint(const USBDescEndpoint *ep, uint8_t *dest, size_t len);
c8dfc6
+int usb_desc_config(const USBDescConfig *conf, int flags,
c8dfc6
+                    uint8_t *dest, size_t len);
c8dfc6
+int usb_desc_iface_group(const USBDescIfaceAssoc *iad, int flags,
c8dfc6
+                         uint8_t *dest, size_t len);
c8dfc6
+int usb_desc_iface(const USBDescIface *iface, int flags,
c8dfc6
+                   uint8_t *dest, size_t len);
c8dfc6
+int usb_desc_endpoint(const USBDescEndpoint *ep, int flags,
c8dfc6
+                      uint8_t *dest, size_t len);
c8dfc6
 int usb_desc_other(const USBDescOther *desc, uint8_t *dest, size_t len);
c8dfc6
 
c8dfc6
 /* control message emulation helpers */
c8dfc6
-- 
5544c1
1.7.12.1
c8dfc6