5 /* ------------------------------------------------------------------ */
7 static uint8_t usb_lo(uint16_t val
)
12 static uint8_t usb_hi(uint16_t val
)
14 return (val
>> 8) & 0xff;
17 int usb_desc_device(const USBDescID
*id
, const USBDescDevice
*dev
,
18 uint8_t *dest
, size_t len
)
20 uint8_t bLength
= 0x12;
27 dest
[0x01] = USB_DT_DEVICE
;
29 dest
[0x02] = usb_lo(dev
->bcdUSB
);
30 dest
[0x03] = usb_hi(dev
->bcdUSB
);
31 dest
[0x04] = dev
->bDeviceClass
;
32 dest
[0x05] = dev
->bDeviceSubClass
;
33 dest
[0x06] = dev
->bDeviceProtocol
;
34 dest
[0x07] = dev
->bMaxPacketSize0
;
36 dest
[0x08] = usb_lo(id
->idVendor
);
37 dest
[0x09] = usb_hi(id
->idVendor
);
38 dest
[0x0a] = usb_lo(id
->idProduct
);
39 dest
[0x0b] = usb_hi(id
->idProduct
);
40 dest
[0x0c] = usb_lo(id
->bcdDevice
);
41 dest
[0x0d] = usb_hi(id
->bcdDevice
);
42 dest
[0x0e] = id
->iManufacturer
;
43 dest
[0x0f] = id
->iProduct
;
44 dest
[0x10] = id
->iSerialNumber
;
46 dest
[0x11] = dev
->bNumConfigurations
;
51 int usb_desc_device_qualifier(const USBDescDevice
*dev
,
52 uint8_t *dest
, size_t len
)
54 uint8_t bLength
= 0x0a;
61 dest
[0x01] = USB_DT_DEVICE_QUALIFIER
;
63 dest
[0x02] = usb_lo(dev
->bcdUSB
);
64 dest
[0x03] = usb_hi(dev
->bcdUSB
);
65 dest
[0x04] = dev
->bDeviceClass
;
66 dest
[0x05] = dev
->bDeviceSubClass
;
67 dest
[0x06] = dev
->bDeviceProtocol
;
68 dest
[0x07] = dev
->bMaxPacketSize0
;
69 dest
[0x08] = dev
->bNumConfigurations
;
70 dest
[0x09] = 0; /* reserved */
75 int usb_desc_config(const USBDescConfig
*conf
, uint8_t *dest
, size_t len
)
77 uint8_t bLength
= 0x09;
78 uint16_t wTotalLength
= 0;
86 dest
[0x01] = USB_DT_CONFIG
;
87 dest
[0x04] = conf
->bNumInterfaces
;
88 dest
[0x05] = conf
->bConfigurationValue
;
89 dest
[0x06] = conf
->iConfiguration
;
90 dest
[0x07] = conf
->bmAttributes
;
91 dest
[0x08] = conf
->bMaxPower
;
92 wTotalLength
+= bLength
;
94 count
= conf
->nif
? conf
->nif
: conf
->bNumInterfaces
;
95 for (i
= 0; i
< count
; i
++) {
96 rc
= usb_desc_iface(conf
->ifs
+ i
, dest
+ wTotalLength
, len
- wTotalLength
);
103 dest
[0x02] = usb_lo(wTotalLength
);
104 dest
[0x03] = usb_hi(wTotalLength
);
108 int usb_desc_iface(const USBDescIface
*iface
, uint8_t *dest
, size_t len
)
110 uint8_t bLength
= 0x09;
117 dest
[0x00] = bLength
;
118 dest
[0x01] = USB_DT_INTERFACE
;
119 dest
[0x02] = iface
->bInterfaceNumber
;
120 dest
[0x03] = iface
->bAlternateSetting
;
121 dest
[0x04] = iface
->bNumEndpoints
;
122 dest
[0x05] = iface
->bInterfaceClass
;
123 dest
[0x06] = iface
->bInterfaceSubClass
;
124 dest
[0x07] = iface
->bInterfaceProtocol
;
125 dest
[0x08] = iface
->iInterface
;
128 for (i
= 0; i
< iface
->ndesc
; i
++) {
129 rc
= usb_desc_other(iface
->descs
+ i
, dest
+ pos
, len
- pos
);
136 for (i
= 0; i
< iface
->bNumEndpoints
; i
++) {
137 rc
= usb_desc_endpoint(iface
->eps
+ i
, dest
+ pos
, len
- pos
);
147 int usb_desc_endpoint(const USBDescEndpoint
*ep
, uint8_t *dest
, size_t len
)
149 uint8_t bLength
= 0x07;
155 dest
[0x00] = bLength
;
156 dest
[0x01] = USB_DT_ENDPOINT
;
157 dest
[0x02] = ep
->bEndpointAddress
;
158 dest
[0x03] = ep
->bmAttributes
;
159 dest
[0x04] = usb_lo(ep
->wMaxPacketSize
);
160 dest
[0x05] = usb_hi(ep
->wMaxPacketSize
);
161 dest
[0x06] = ep
->bInterval
;
166 int usb_desc_other(const USBDescOther
*desc
, uint8_t *dest
, size_t len
)
168 int bLength
= desc
->length
? desc
->length
: desc
->data
[0];
174 memcpy(dest
, desc
->data
, bLength
);
178 /* ------------------------------------------------------------------ */
180 static void usb_desc_setdefaults(USBDevice
*dev
)
182 const USBDesc
*desc
= dev
->info
->usb_desc
;
184 assert(desc
!= NULL
);
185 switch (dev
->speed
) {
188 dev
->device
= desc
->full
;
191 dev
->device
= desc
->high
;
194 dev
->config
= dev
->device
->confs
;
197 void usb_desc_init(USBDevice
*dev
)
199 dev
->speed
= USB_SPEED_FULL
;
200 usb_desc_setdefaults(dev
);
203 void usb_desc_attach(USBDevice
*dev
)
205 const USBDesc
*desc
= dev
->info
->usb_desc
;
207 assert(desc
!= NULL
);
208 if (desc
->high
&& (dev
->port
->speedmask
& USB_SPEED_MASK_HIGH
)) {
209 dev
->speed
= USB_SPEED_HIGH
;
210 } else if (desc
->full
&& (dev
->port
->speedmask
& USB_SPEED_MASK_FULL
)) {
211 dev
->speed
= USB_SPEED_FULL
;
213 fprintf(stderr
, "usb: port/device speed mismatch for \"%s\"\n",
214 dev
->info
->product_desc
);
217 usb_desc_setdefaults(dev
);
220 void usb_desc_set_string(USBDevice
*dev
, uint8_t index
, const char *str
)
224 QLIST_FOREACH(s
, &dev
->strings
, next
) {
225 if (s
->index
== index
) {
230 s
= qemu_mallocz(sizeof(*s
));
232 QLIST_INSERT_HEAD(&dev
->strings
, s
, next
);
235 s
->str
= qemu_strdup(str
);
238 const char *usb_desc_get_string(USBDevice
*dev
, uint8_t index
)
242 QLIST_FOREACH(s
, &dev
->strings
, next
) {
243 if (s
->index
== index
) {
250 int usb_desc_string(USBDevice
*dev
, int index
, uint8_t *dest
, size_t len
)
252 uint8_t bLength
, pos
, i
;
262 dest
[1] = USB_DT_STRING
;
268 str
= usb_desc_get_string(dev
, index
);
270 str
= dev
->info
->usb_desc
->str
[index
];
276 bLength
= strlen(str
) * 2 + 2;
278 dest
[1] = USB_DT_STRING
;
280 while (pos
+1 < bLength
&& pos
+1 < len
) {
281 dest
[pos
++] = str
[i
++];
287 int usb_desc_get_descriptor(USBDevice
*dev
, int value
, uint8_t *dest
, size_t len
)
289 const USBDesc
*desc
= dev
->info
->usb_desc
;
290 const USBDescDevice
*other_dev
;
292 uint8_t type
= value
>> 8;
293 uint8_t index
= value
& 0xff;
296 if (dev
->speed
== USB_SPEED_HIGH
) {
297 other_dev
= dev
->info
->usb_desc
->full
;
299 other_dev
= dev
->info
->usb_desc
->high
;
304 ret
= usb_desc_device(&desc
->id
, dev
->device
, buf
, sizeof(buf
));
305 trace_usb_desc_device(dev
->addr
, len
, ret
);
308 if (index
< dev
->device
->bNumConfigurations
) {
309 ret
= usb_desc_config(dev
->device
->confs
+ index
, buf
, sizeof(buf
));
311 trace_usb_desc_config(dev
->addr
, index
, len
, ret
);
314 ret
= usb_desc_string(dev
, index
, buf
, sizeof(buf
));
315 trace_usb_desc_string(dev
->addr
, index
, len
, ret
);
318 case USB_DT_DEVICE_QUALIFIER
:
319 if (other_dev
!= NULL
) {
320 ret
= usb_desc_device_qualifier(other_dev
, buf
, sizeof(buf
));
322 trace_usb_desc_device_qualifier(dev
->addr
, len
, ret
);
324 case USB_DT_OTHER_SPEED_CONFIG
:
325 if (other_dev
!= NULL
&& index
< other_dev
->bNumConfigurations
) {
326 ret
= usb_desc_config(other_dev
->confs
+ index
, buf
, sizeof(buf
));
327 buf
[0x01] = USB_DT_OTHER_SPEED_CONFIG
;
329 trace_usb_desc_other_speed_config(dev
->addr
, index
, len
, ret
);
333 fprintf(stderr
, "%s: %d unknown type %d (len %zd)\n", __FUNCTION__
,
334 dev
->addr
, type
, len
);
342 memcpy(dest
, buf
, ret
);
347 int usb_desc_handle_control(USBDevice
*dev
, int request
, int value
,
348 int index
, int length
, uint8_t *data
)
350 const USBDesc
*desc
= dev
->info
->usb_desc
;
353 assert(desc
!= NULL
);
355 case DeviceOutRequest
| USB_REQ_SET_ADDRESS
:
357 trace_usb_set_addr(dev
->addr
);
361 case DeviceRequest
| USB_REQ_GET_DESCRIPTOR
:
362 ret
= usb_desc_get_descriptor(dev
, value
, data
, length
);
365 case DeviceRequest
| USB_REQ_GET_CONFIGURATION
:
366 data
[0] = dev
->config
->bConfigurationValue
;
369 case DeviceOutRequest
| USB_REQ_SET_CONFIGURATION
:
370 for (i
= 0; i
< dev
->device
->bNumConfigurations
; i
++) {
371 if (dev
->device
->confs
[i
].bConfigurationValue
== value
) {
372 dev
->config
= dev
->device
->confs
+ i
;
376 trace_usb_set_config(dev
->addr
, value
, ret
);
379 case DeviceRequest
| USB_REQ_GET_STATUS
:
381 if (dev
->config
->bmAttributes
& 0x40) {
382 data
[0] |= 1 << USB_DEVICE_SELF_POWERED
;
384 if (dev
->remote_wakeup
) {
385 data
[0] |= 1 << USB_DEVICE_REMOTE_WAKEUP
;
390 case DeviceOutRequest
| USB_REQ_CLEAR_FEATURE
:
391 if (value
== USB_DEVICE_REMOTE_WAKEUP
) {
392 dev
->remote_wakeup
= 0;
395 trace_usb_clear_device_feature(dev
->addr
, value
, ret
);
397 case DeviceOutRequest
| USB_REQ_SET_FEATURE
:
398 if (value
== USB_DEVICE_REMOTE_WAKEUP
) {
399 dev
->remote_wakeup
= 1;
402 trace_usb_set_device_feature(dev
->addr
, value
, ret
);