Commit 446ab1284e0648eb9e2bfd08edb769bcd22dccb2
1 parent
9d0efc88
husb: Make control transactions asynchronous (Max Krasnyansky)
USB is 99.8% async now :). 0.2% is the three control requests that we need to execute synchronously. We could off-load that to a thread or something but it's not worth the pain since those requests are performed only during device initialization (ie when device is connected to the VM). The change is a bit bigger than I wanted due to the fact that generic handle_packet()/handle_control() interface was not designed for async transactions. So I ended up adding custom handle_packet() code to usb-linux. We can make that generic if/when some other component needs it. Signed-off-by: Max Krasnyansky <maxk@kernel.org> Signed-off-by: Anthony Liguori <aliguori@us.ibm.com> git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@5204 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
1 changed file
with
371 additions
and
93 deletions
usb-linux.c
... | ... | @@ -25,28 +25,20 @@ |
25 | 25 | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
26 | 26 | * THE SOFTWARE. |
27 | 27 | */ |
28 | + | |
28 | 29 | #include "qemu-common.h" |
29 | 30 | #include "qemu-timer.h" |
30 | -#include "hw/usb.h" | |
31 | 31 | #include "console.h" |
32 | 32 | |
33 | 33 | #if defined(__linux__) |
34 | 34 | #include <dirent.h> |
35 | 35 | #include <sys/ioctl.h> |
36 | -#include <linux/usbdevice_fs.h> | |
37 | -#include <linux/version.h> | |
38 | 36 | #include <signal.h> |
39 | 37 | |
40 | -/* We redefine it to avoid version problems */ | |
41 | -struct usb_ctrltransfer { | |
42 | - uint8_t bRequestType; | |
43 | - uint8_t bRequest; | |
44 | - uint16_t wValue; | |
45 | - uint16_t wIndex; | |
46 | - uint16_t wLength; | |
47 | - uint32_t timeout; | |
48 | - void *data; | |
49 | -}; | |
38 | +#include <linux/usb/ch9.h> | |
39 | +#include <linux/usbdevice_fs.h> | |
40 | +#include <linux/version.h> | |
41 | +#include "hw/usb.h" | |
50 | 42 | |
51 | 43 | typedef int USBScanFunc(void *opaque, int bus_num, int addr, int class_id, |
52 | 44 | int vendor_id, int product_id, |
... | ... | @@ -54,7 +46,6 @@ typedef int USBScanFunc(void *opaque, int bus_num, int addr, int class_id, |
54 | 46 | static int usb_host_find_device(int *pbus_num, int *paddr, |
55 | 47 | char *product_name, int product_name_size, |
56 | 48 | const char *devname); |
57 | - | |
58 | 49 | //#define DEBUG |
59 | 50 | |
60 | 51 | #ifdef DEBUG |
... | ... | @@ -67,14 +58,32 @@ static int usb_host_find_device(int *pbus_num, int *paddr, |
67 | 58 | #define PRODUCT_NAME_SZ 32 |
68 | 59 | #define MAX_ENDPOINTS 16 |
69 | 60 | |
70 | -struct sigaction sigact; | |
71 | - | |
72 | 61 | /* endpoint association data */ |
73 | 62 | struct endp_data { |
74 | 63 | uint8_t type; |
75 | 64 | uint8_t halted; |
76 | 65 | }; |
77 | 66 | |
67 | +enum { | |
68 | + CTRL_STATE_IDLE = 0, | |
69 | + CTRL_STATE_SETUP, | |
70 | + CTRL_STATE_DATA, | |
71 | + CTRL_STATE_ACK | |
72 | +}; | |
73 | + | |
74 | +/* | |
75 | + * Control transfer state. | |
76 | + * Note that 'buffer' _must_ follow 'req' field because | |
77 | + * we need contigious buffer when we submit control URB. | |
78 | + */ | |
79 | +struct ctrl_struct { | |
80 | + uint16_t len; | |
81 | + uint16_t offset; | |
82 | + uint8_t state; | |
83 | + struct usb_ctrlrequest req; | |
84 | + uint8_t buffer[1024]; | |
85 | +}; | |
86 | + | |
78 | 87 | typedef struct USBHostDevice { |
79 | 88 | USBDevice dev; |
80 | 89 | int fd; |
... | ... | @@ -82,8 +91,10 @@ typedef struct USBHostDevice { |
82 | 91 | uint8_t descr[1024]; |
83 | 92 | int descr_len; |
84 | 93 | int configuration; |
94 | + int ninterfaces; | |
85 | 95 | int closing; |
86 | 96 | |
97 | + struct ctrl_struct ctrl; | |
87 | 98 | struct endp_data endp_table[MAX_ENDPOINTS]; |
88 | 99 | |
89 | 100 | /* Host side address */ |
... | ... | @@ -172,6 +183,26 @@ static void async_free(AsyncURB *aurb) |
172 | 183 | qemu_free(aurb); |
173 | 184 | } |
174 | 185 | |
186 | +static void async_complete_ctrl(USBHostDevice *s, USBPacket *p) | |
187 | +{ | |
188 | + switch(s->ctrl.state) { | |
189 | + case CTRL_STATE_SETUP: | |
190 | + if (p->len < s->ctrl.len) | |
191 | + s->ctrl.len = p->len; | |
192 | + s->ctrl.state = CTRL_STATE_DATA; | |
193 | + p->len = 8; | |
194 | + break; | |
195 | + | |
196 | + case CTRL_STATE_ACK: | |
197 | + s->ctrl.state = CTRL_STATE_IDLE; | |
198 | + p->len = 0; | |
199 | + break; | |
200 | + | |
201 | + default: | |
202 | + break; | |
203 | + } | |
204 | +} | |
205 | + | |
175 | 206 | static void async_complete(void *opaque) |
176 | 207 | { |
177 | 208 | USBHostDevice *s = opaque; |
... | ... | @@ -204,6 +235,8 @@ static void async_complete(void *opaque) |
204 | 235 | switch (aurb->urb.status) { |
205 | 236 | case 0: |
206 | 237 | p->len = aurb->urb.actual_length; |
238 | + if (aurb->urb.type == USBDEVFS_URB_TYPE_CONTROL) | |
239 | + async_complete_ctrl(s, p); | |
207 | 240 | break; |
208 | 241 | |
209 | 242 | case -EPIPE: |
... | ... | @@ -237,7 +270,7 @@ static void async_cancel(USBPacket *unused, void *opaque) |
237 | 270 | } |
238 | 271 | } |
239 | 272 | |
240 | -static int usb_host_update_interfaces(USBHostDevice *dev, int configuration) | |
273 | +static int usb_host_claim_interfaces(USBHostDevice *dev, int configuration) | |
241 | 274 | { |
242 | 275 | int dev_descr_len, config_descr_len; |
243 | 276 | int interface, nb_interfaces, nb_configurations; |
... | ... | @@ -246,6 +279,8 @@ static int usb_host_update_interfaces(USBHostDevice *dev, int configuration) |
246 | 279 | if (configuration == 0) /* address state - ignore */ |
247 | 280 | return 1; |
248 | 281 | |
282 | + dprintf("husb: claiming interfaces. config %d\n", configuration); | |
283 | + | |
249 | 284 | i = 0; |
250 | 285 | dev_descr_len = dev->descr[0]; |
251 | 286 | if (dev_descr_len > dev->descr_len) |
... | ... | @@ -265,8 +300,10 @@ static int usb_host_update_interfaces(USBHostDevice *dev, int configuration) |
265 | 300 | |
266 | 301 | printf("husb: config #%d need %d\n", dev->descr[i + 5], configuration); |
267 | 302 | |
268 | - if (configuration < 0 || configuration == dev->descr[i + 5]) | |
303 | + if (configuration < 0 || configuration == dev->descr[i + 5]) { | |
304 | + configuration = dev->descr[i + 5]; | |
269 | 305 | break; |
306 | + } | |
270 | 307 | |
271 | 308 | i += config_descr_len; |
272 | 309 | } |
... | ... | @@ -310,17 +347,37 @@ static int usb_host_update_interfaces(USBHostDevice *dev, int configuration) |
310 | 347 | printf("husb: %d interfaces claimed for configuration %d\n", |
311 | 348 | nb_interfaces, configuration); |
312 | 349 | |
350 | + dev->ninterfaces = nb_interfaces; | |
351 | + dev->configuration = configuration; | |
352 | + return 1; | |
353 | +} | |
354 | + | |
355 | +static int usb_host_release_interfaces(USBHostDevice *s) | |
356 | +{ | |
357 | + int ret, i; | |
358 | + | |
359 | + dprintf("husb: releasing interfaces\n"); | |
360 | + | |
361 | + for (i = 0; i < s->ninterfaces; i++) { | |
362 | + ret = ioctl(s->fd, USBDEVFS_RELEASEINTERFACE, &i); | |
363 | + if (ret < 0) { | |
364 | + perror("husb: failed to release interface"); | |
365 | + return 0; | |
366 | + } | |
367 | + } | |
368 | + | |
313 | 369 | return 1; |
314 | 370 | } |
315 | 371 | |
316 | 372 | static void usb_host_handle_reset(USBDevice *dev) |
317 | 373 | { |
318 | - USBHostDevice *s = (USBHostDevice *)dev; | |
374 | + USBHostDevice *s = (USBHostDevice *) dev; | |
319 | 375 | |
320 | 376 | dprintf("husb: reset device %u.%u\n", s->bus_num, s->addr); |
321 | 377 | |
322 | 378 | ioctl(s->fd, USBDEVFS_RESET); |
323 | - usb_host_update_interfaces(s, s->configuration); | |
379 | + | |
380 | + usb_host_claim_interfaces(s, s->configuration); | |
324 | 381 | } |
325 | 382 | |
326 | 383 | static void usb_host_handle_destroy(USBDevice *dev) |
... | ... | @@ -343,73 +400,10 @@ static void usb_host_handle_destroy(USBDevice *dev) |
343 | 400 | |
344 | 401 | static int usb_linux_update_endp_table(USBHostDevice *s); |
345 | 402 | |
346 | -static int usb_host_handle_control(USBDevice *dev, | |
347 | - int request, | |
348 | - int value, | |
349 | - int index, | |
350 | - int length, | |
351 | - uint8_t *data) | |
352 | -{ | |
353 | - USBHostDevice *s = (USBHostDevice *)dev; | |
354 | - struct usb_ctrltransfer ct; | |
355 | - struct usbdevfs_setinterface si; | |
356 | - int intf_update_required = 0; | |
357 | - int ret; | |
358 | - | |
359 | - if (request == (DeviceOutRequest | USB_REQ_SET_ADDRESS)) { | |
360 | - /* specific SET_ADDRESS support */ | |
361 | - dev->addr = value; | |
362 | - return 0; | |
363 | - } else if (request == ((USB_RECIP_INTERFACE << 8) | | |
364 | - USB_REQ_SET_INTERFACE)) { | |
365 | - /* set alternate setting for the interface */ | |
366 | - si.interface = index; | |
367 | - si.altsetting = value; | |
368 | - ret = ioctl(s->fd, USBDEVFS_SETINTERFACE, &si); | |
369 | - usb_linux_update_endp_table(s); | |
370 | - } else if (request == (DeviceOutRequest | USB_REQ_SET_CONFIGURATION)) { | |
371 | - dprintf("husb: ctrl set config %d\n", value & 0xff); | |
372 | - if (s->configuration != (value & 0xff)) { | |
373 | - s->configuration = (value & 0xff); | |
374 | - intf_update_required = 1; | |
375 | - } | |
376 | - goto do_request; | |
377 | - } else { | |
378 | - do_request: | |
379 | - ct.bRequestType = request >> 8; | |
380 | - ct.bRequest = request; | |
381 | - ct.wValue = value; | |
382 | - ct.wIndex = index; | |
383 | - ct.wLength = length; | |
384 | - ct.timeout = 50; | |
385 | - ct.data = data; | |
386 | - ret = ioctl(s->fd, USBDEVFS_CONTROL, &ct); | |
387 | - | |
388 | - dprintf("husb: ctrl req 0x%x val 0x%x index %u len %u ret %d\n", | |
389 | - ct.bRequest, ct.wValue, ct.wIndex, ct.wLength, ret); | |
390 | - } | |
391 | - | |
392 | - if (ret < 0) { | |
393 | - switch(errno) { | |
394 | - case ETIMEDOUT: | |
395 | - return USB_RET_NAK; | |
396 | - default: | |
397 | - return USB_RET_STALL; | |
398 | - } | |
399 | - } else { | |
400 | - if (intf_update_required) { | |
401 | - dprintf("husb: updating interfaces\n"); | |
402 | - usb_host_update_interfaces(s, value & 0xff); | |
403 | - } | |
404 | - return ret; | |
405 | - } | |
406 | -} | |
407 | - | |
408 | -static int usb_host_handle_data(USBDevice *dev, USBPacket *p) | |
403 | +static int usb_host_handle_data(USBHostDevice *s, USBPacket *p) | |
409 | 404 | { |
410 | - USBHostDevice *s = (USBHostDevice *) dev; | |
411 | - AsyncURB *aurb; | |
412 | 405 | struct usbdevfs_urb *urb; |
406 | + AsyncURB *aurb; | |
413 | 407 | int ret; |
414 | 408 | |
415 | 409 | aurb = async_alloc(); |
... | ... | @@ -474,12 +468,292 @@ static int usb_host_handle_data(USBDevice *dev, USBPacket *p) |
474 | 468 | return USB_RET_ASYNC; |
475 | 469 | } |
476 | 470 | |
471 | +static int ctrl_error(void) | |
472 | +{ | |
473 | + if (errno == ETIMEDOUT) | |
474 | + return USB_RET_NAK; | |
475 | + else | |
476 | + return USB_RET_STALL; | |
477 | +} | |
478 | + | |
479 | +static int usb_host_set_address(USBHostDevice *s, int addr) | |
480 | +{ | |
481 | + dprintf("husb: ctrl set addr %u\n", addr); | |
482 | + s->dev.addr = addr; | |
483 | + return 0; | |
484 | +} | |
485 | + | |
486 | +static int usb_host_set_config(USBHostDevice *s, int config) | |
487 | +{ | |
488 | + usb_host_release_interfaces(s); | |
489 | + | |
490 | + int ret = ioctl(s->fd, USBDEVFS_SETCONFIGURATION, &config); | |
491 | + | |
492 | + dprintf("husb: ctrl set config %d ret %d errno %d\n", config, ret, errno); | |
493 | + | |
494 | + if (ret < 0) | |
495 | + return ctrl_error(); | |
496 | + | |
497 | + usb_host_claim_interfaces(s, config); | |
498 | + return 0; | |
499 | +} | |
500 | + | |
501 | +static int usb_host_set_interface(USBHostDevice *s, int iface, int alt) | |
502 | +{ | |
503 | + struct usbdevfs_setinterface si; | |
504 | + int ret; | |
505 | + | |
506 | + si.interface = iface; | |
507 | + si.altsetting = alt; | |
508 | + ret = ioctl(s->fd, USBDEVFS_SETINTERFACE, &si); | |
509 | + | |
510 | + dprintf("husb: ctrl set iface %d altset %d ret %d errno %d\n", | |
511 | + iface, alt, ret, errno); | |
512 | + | |
513 | + if (ret < 0) | |
514 | + return ctrl_error(); | |
515 | + | |
516 | + usb_linux_update_endp_table(s); | |
517 | + return 0; | |
518 | +} | |
519 | + | |
520 | +static int usb_host_handle_control(USBHostDevice *s, USBPacket *p) | |
521 | +{ | |
522 | + struct usbdevfs_urb *urb; | |
523 | + AsyncURB *aurb; | |
524 | + int ret, value, index; | |
525 | + | |
526 | + /* | |
527 | + * Process certain standard device requests. | |
528 | + * These are infrequent and are processed synchronously. | |
529 | + */ | |
530 | + value = le16_to_cpu(s->ctrl.req.wValue); | |
531 | + index = le16_to_cpu(s->ctrl.req.wIndex); | |
532 | + | |
533 | + dprintf("husb: ctrl type 0x%x req 0x%x val 0x%x index %u len %u\n", | |
534 | + s->ctrl.req.bRequestType, s->ctrl.req.bRequest, value, index, | |
535 | + s->ctrl.len); | |
536 | + | |
537 | + if (s->ctrl.req.bRequestType == 0) { | |
538 | + switch (s->ctrl.req.bRequest) { | |
539 | + case USB_REQ_SET_ADDRESS: | |
540 | + return usb_host_set_address(s, value); | |
541 | + | |
542 | + case USB_REQ_SET_CONFIGURATION: | |
543 | + return usb_host_set_config(s, value & 0xff); | |
544 | + } | |
545 | + } | |
546 | + | |
547 | + if (s->ctrl.req.bRequestType == 1 && | |
548 | + s->ctrl.req.bRequest == USB_REQ_SET_INTERFACE) | |
549 | + return usb_host_set_interface(s, index, value); | |
550 | + | |
551 | + /* The rest are asynchronous */ | |
552 | + | |
553 | + aurb = async_alloc(); | |
554 | + if (!aurb) { | |
555 | + dprintf("husb: async malloc failed\n"); | |
556 | + return USB_RET_NAK; | |
557 | + } | |
558 | + aurb->hdev = s; | |
559 | + aurb->packet = p; | |
560 | + | |
561 | + /* | |
562 | + * Setup ctrl transfer. | |
563 | + * | |
564 | + * s->ctrl is layed out such that data buffer immediately follows | |
565 | + * 'req' struct which is exactly what usbdevfs expects. | |
566 | + */ | |
567 | + urb = &aurb->urb; | |
568 | + | |
569 | + urb->type = USBDEVFS_URB_TYPE_CONTROL; | |
570 | + urb->endpoint = p->devep; | |
571 | + | |
572 | + urb->buffer = &s->ctrl.req; | |
573 | + urb->buffer_length = 8 + s->ctrl.len; | |
574 | + | |
575 | + urb->usercontext = s; | |
576 | + | |
577 | + ret = ioctl(s->fd, USBDEVFS_SUBMITURB, urb); | |
578 | + | |
579 | + dprintf("husb: submit ctrl. len %u aurb %p\n", urb->buffer_length, aurb); | |
580 | + | |
581 | + if (ret < 0) { | |
582 | + dprintf("husb: submit failed. errno %d\n", errno); | |
583 | + async_free(aurb); | |
584 | + | |
585 | + switch(errno) { | |
586 | + case ETIMEDOUT: | |
587 | + return USB_RET_NAK; | |
588 | + case EPIPE: | |
589 | + default: | |
590 | + return USB_RET_STALL; | |
591 | + } | |
592 | + } | |
593 | + | |
594 | + usb_defer_packet(p, async_cancel, aurb); | |
595 | + return USB_RET_ASYNC; | |
596 | +} | |
597 | + | |
598 | +static int do_token_setup(USBDevice *dev, USBPacket *p) | |
599 | +{ | |
600 | + USBHostDevice *s = (USBHostDevice *) dev; | |
601 | + int ret = 0; | |
602 | + | |
603 | + if (p->len != 8) | |
604 | + return USB_RET_STALL; | |
605 | + | |
606 | + memcpy(&s->ctrl.req, p->data, 8); | |
607 | + s->ctrl.len = le16_to_cpu(s->ctrl.req.wLength); | |
608 | + s->ctrl.offset = 0; | |
609 | + s->ctrl.state = CTRL_STATE_SETUP; | |
610 | + | |
611 | + if (s->ctrl.req.bRequestType & USB_DIR_IN) { | |
612 | + ret = usb_host_handle_control(s, p); | |
613 | + if (ret < 0) | |
614 | + return ret; | |
615 | + | |
616 | + if (ret < s->ctrl.len) | |
617 | + s->ctrl.len = ret; | |
618 | + s->ctrl.state = CTRL_STATE_DATA; | |
619 | + } else { | |
620 | + if (s->ctrl.len == 0) | |
621 | + s->ctrl.state = CTRL_STATE_ACK; | |
622 | + else | |
623 | + s->ctrl.state = CTRL_STATE_DATA; | |
624 | + } | |
625 | + | |
626 | + return ret; | |
627 | +} | |
628 | + | |
629 | +static int do_token_in(USBDevice *dev, USBPacket *p) | |
630 | +{ | |
631 | + USBHostDevice *s = (USBHostDevice *) dev; | |
632 | + int ret = 0; | |
633 | + | |
634 | + if (p->devep != 0) | |
635 | + return usb_host_handle_data(s, p); | |
636 | + | |
637 | + switch(s->ctrl.state) { | |
638 | + case CTRL_STATE_ACK: | |
639 | + if (!(s->ctrl.req.bRequestType & USB_DIR_IN)) { | |
640 | + ret = usb_host_handle_control(s, p); | |
641 | + if (ret == USB_RET_ASYNC) | |
642 | + return USB_RET_ASYNC; | |
643 | + | |
644 | + s->ctrl.state = CTRL_STATE_IDLE; | |
645 | + return ret > 0 ? 0 : ret; | |
646 | + } | |
647 | + | |
648 | + return 0; | |
649 | + | |
650 | + case CTRL_STATE_DATA: | |
651 | + if (s->ctrl.req.bRequestType & USB_DIR_IN) { | |
652 | + int len = s->ctrl.len - s->ctrl.offset; | |
653 | + if (len > p->len) | |
654 | + len = p->len; | |
655 | + memcpy(p->data, s->ctrl.buffer + s->ctrl.offset, len); | |
656 | + s->ctrl.offset += len; | |
657 | + if (s->ctrl.offset >= s->ctrl.len) | |
658 | + s->ctrl.state = CTRL_STATE_ACK; | |
659 | + return len; | |
660 | + } | |
661 | + | |
662 | + s->ctrl.state = CTRL_STATE_IDLE; | |
663 | + return USB_RET_STALL; | |
664 | + | |
665 | + default: | |
666 | + return USB_RET_STALL; | |
667 | + } | |
668 | +} | |
669 | + | |
670 | +static int do_token_out(USBDevice *dev, USBPacket *p) | |
671 | +{ | |
672 | + USBHostDevice *s = (USBHostDevice *) dev; | |
673 | + | |
674 | + if (p->devep != 0) | |
675 | + return usb_host_handle_data(s, p); | |
676 | + | |
677 | + switch(s->ctrl.state) { | |
678 | + case CTRL_STATE_ACK: | |
679 | + if (s->ctrl.req.bRequestType & USB_DIR_IN) { | |
680 | + s->ctrl.state = CTRL_STATE_IDLE; | |
681 | + /* transfer OK */ | |
682 | + } else { | |
683 | + /* ignore additional output */ | |
684 | + } | |
685 | + return 0; | |
686 | + | |
687 | + case CTRL_STATE_DATA: | |
688 | + if (!(s->ctrl.req.bRequestType & USB_DIR_IN)) { | |
689 | + int len = s->ctrl.len - s->ctrl.offset; | |
690 | + if (len > p->len) | |
691 | + len = p->len; | |
692 | + memcpy(s->ctrl.buffer + s->ctrl.offset, p->data, len); | |
693 | + s->ctrl.offset += len; | |
694 | + if (s->ctrl.offset >= s->ctrl.len) | |
695 | + s->ctrl.state = CTRL_STATE_ACK; | |
696 | + return len; | |
697 | + } | |
698 | + | |
699 | + s->ctrl.state = CTRL_STATE_IDLE; | |
700 | + return USB_RET_STALL; | |
701 | + | |
702 | + default: | |
703 | + return USB_RET_STALL; | |
704 | + } | |
705 | +} | |
706 | + | |
707 | +/* | |
708 | + * Packet handler. | |
709 | + * Called by the HC (host controller). | |
710 | + * | |
711 | + * Returns length of the transaction or one of the USB_RET_XXX codes. | |
712 | + */ | |
713 | +int usb_host_handle_packet(USBDevice *s, USBPacket *p) | |
714 | +{ | |
715 | + switch(p->pid) { | |
716 | + case USB_MSG_ATTACH: | |
717 | + s->state = USB_STATE_ATTACHED; | |
718 | + return 0; | |
719 | + | |
720 | + case USB_MSG_DETACH: | |
721 | + s->state = USB_STATE_NOTATTACHED; | |
722 | + return 0; | |
723 | + | |
724 | + case USB_MSG_RESET: | |
725 | + s->remote_wakeup = 0; | |
726 | + s->addr = 0; | |
727 | + s->state = USB_STATE_DEFAULT; | |
728 | + s->handle_reset(s); | |
729 | + return 0; | |
730 | + } | |
731 | + | |
732 | + /* Rest of the PIDs must match our address */ | |
733 | + if (s->state < USB_STATE_DEFAULT || p->devaddr != s->addr) | |
734 | + return USB_RET_NODEV; | |
735 | + | |
736 | + switch (p->pid) { | |
737 | + case USB_TOKEN_SETUP: | |
738 | + return do_token_setup(s, p); | |
739 | + | |
740 | + case USB_TOKEN_IN: | |
741 | + return do_token_in(s, p); | |
742 | + | |
743 | + case USB_TOKEN_OUT: | |
744 | + return do_token_out(s, p); | |
745 | + | |
746 | + default: | |
747 | + return USB_RET_STALL; | |
748 | + } | |
749 | +} | |
750 | + | |
477 | 751 | /* returns 1 on problem encountered or 0 for success */ |
478 | 752 | static int usb_linux_update_endp_table(USBHostDevice *s) |
479 | 753 | { |
480 | 754 | uint8_t *descriptors; |
481 | 755 | uint8_t devep, type, configuration, alt_interface; |
482 | - struct usb_ctrltransfer ct; | |
756 | + struct usbdevfs_ctrltransfer ct; | |
483 | 757 | int interface, ret, length, i; |
484 | 758 | |
485 | 759 | ct.bRequestType = USB_DIR_IN; |
... | ... | @@ -624,10 +898,14 @@ static USBDevice *usb_host_device_open_addr(int bus_num, int addr, const char *p |
624 | 898 | #endif |
625 | 899 | |
626 | 900 | dev->fd = fd; |
627 | - dev->configuration = 1; | |
628 | 901 | |
629 | - /* XXX - do something about initial configuration */ | |
630 | - if (!usb_host_update_interfaces(dev, -1)) | |
902 | + /* | |
903 | + * Initial configuration is -1 which makes us claim first | |
904 | + * available config. We used to start with 1, which does not | |
905 | + * always work. I've seen devices where first config starts | |
906 | + * with 2. | |
907 | + */ | |
908 | + if (!usb_host_claim_interfaces(dev, -1)) | |
631 | 909 | goto fail; |
632 | 910 | |
633 | 911 | ret = ioctl(fd, USBDEVFS_CONNECTINFO, &ci); |
... | ... | @@ -646,11 +924,9 @@ static USBDevice *usb_host_device_open_addr(int bus_num, int addr, const char *p |
646 | 924 | dev->dev.speed = USB_SPEED_LOW; |
647 | 925 | else |
648 | 926 | dev->dev.speed = USB_SPEED_HIGH; |
649 | - dev->dev.handle_packet = usb_generic_handle_packet; | |
650 | 927 | |
651 | - dev->dev.handle_reset = usb_host_handle_reset; | |
652 | - dev->dev.handle_control = usb_host_handle_control; | |
653 | - dev->dev.handle_data = usb_host_handle_data; | |
928 | + dev->dev.handle_packet = usb_host_handle_packet; | |
929 | + dev->dev.handle_reset = usb_host_handle_reset; | |
654 | 930 | dev->dev.handle_destroy = usb_host_handle_destroy; |
655 | 931 | |
656 | 932 | if (!prod_name || prod_name[0] == '\0') |
... | ... | @@ -1055,6 +1331,8 @@ void usb_host_info(void) |
1055 | 1331 | |
1056 | 1332 | #else |
1057 | 1333 | |
1334 | +#include "hw/usb.h" | |
1335 | + | |
1058 | 1336 | void usb_host_info(void) |
1059 | 1337 | { |
1060 | 1338 | term_printf("USB host devices not supported\n"); | ... | ... |