libpayload: usb: Fix several minor USB stack bugs

This patch fixes the following minor bugs in the USB stack:

1. Ensure that all dynamically allocated device structures are cleaned
on detachment, and that the device address is correctly released again.
2. Make sure MSC and HID drivers notice missing endpoints and actually
detach the device in that case (to prevent it from being used).
3. Make sure XHCI-specific set_address() cleans up all data structures
on failure.
4. Fix broken Slot ID range check that prevented XHCI devices from being
correctly cleaned up.

Change-Id: I7b2b9c8cd6c5e93cb19abcf01425bcd85d2e1f22
Signed-off-by: Julius Werner <jwerner@chromium.org>
Reviewed-on: https://chromium-review.googlesource.com/170665
Reviewed-by: Stefan Reinauer <reinauer@google.com>
Commit-Queue: Ronald Minnich <rminnich@chromium.org>
Tested-by: Ronald Minnich <rminnich@chromium.org>
(cherry picked from commit 9671472263ddd0c30400ae3b6da780a18cd21ded)
Signed-off-by: Isaac Christensen <isaac.christensen@se-eng.com>
Reviewed-on: http://review.coreboot.org/6701
Tested-by: build bot (Jenkins)
Reviewed-by: Paul Menzel <paulepanter@users.sourceforge.net>
Reviewed-by: Ronald G. Minnich <rminnich@gmail.com>
This commit is contained in:
Julius Werner 2013-09-25 12:30:07 -07:00 committed by Isaac Christensen
parent 14eb43be98
commit d609e89adf
4 changed files with 21 additions and 9 deletions

View File

@ -539,10 +539,14 @@ usb_detach_device(hci_t *controller, int devno)
been called yet by the usb class driver */ been called yet by the usb class driver */
if (controller->devices[devno]) { if (controller->devices[devno]) {
controller->devices[devno]->destroy (controller->devices[devno]); controller->devices[devno]->destroy (controller->devices[devno]);
free(controller->devices[devno]);
controller->devices[devno] = NULL;
if (controller->destroy_device) if (controller->destroy_device)
controller->destroy_device(controller, devno); controller->destroy_device(controller, devno);
if (controller->devices[devno]->configuration)
free(controller->devices[devno]->configuration);
if (controller->devices[devno]->descriptor)
free(controller->devices[devno]->descriptor);
free(controller->devices[devno]);
controller->devices[devno] = NULL;
} }
} }

View File

@ -468,15 +468,18 @@ usb_hid_init (usbdev_t *dev)
dev->destroy = usb_hid_destroy; dev->destroy = usb_hid_destroy;
dev->poll = usb_hid_poll; dev->poll = usb_hid_poll;
int i; int i;
for (i = 0; i <= dev->num_endp; i++) { for (i = 1; i < dev->num_endp; i++) {
if (dev->endpoints[i].endpoint == 0)
continue;
if (dev->endpoints[i].type != INTERRUPT) if (dev->endpoints[i].type != INTERRUPT)
continue; continue;
if (dev->endpoints[i].direction != IN) if (dev->endpoints[i].direction != IN)
continue; continue;
break; break;
} }
if (i >= dev->num_endp) {
usb_debug ("Could not find HID endpoint\n");
usb_detach_device (dev->controller, dev->address);
return;
}
usb_debug (" found endpoint %x for interrupt-in\n", i); usb_debug (" found endpoint %x for interrupt-in\n", i);
/* 20 buffers of 8 bytes, for every 10 msecs */ /* 20 buffers of 8 bytes, for every 10 msecs */
HID_INST(dev)->queue = dev->controller->create_intr_queue (&dev->endpoints[i], 8, 20, 10); HID_INST(dev)->queue = dev->controller->create_intr_queue (&dev->endpoints[i], 8, 20, 10);

View File

@ -598,6 +598,7 @@ usb_msc_init (usbdev_t *dev)
if (interface->bInterfaceProtocol != 0x50) { if (interface->bInterfaceProtocol != 0x50) {
usb_debug (" Protocol not supported.\n"); usb_debug (" Protocol not supported.\n");
usb_detach_device (dev->controller, dev->address);
return; return;
} }
@ -606,6 +607,7 @@ usb_msc_init (usbdev_t *dev)
(interface->bInterfaceSubClass != 6)) { // SCSI (interface->bInterfaceSubClass != 6)) { // SCSI
/* Other protocols, such as ATAPI don't seem to be very popular. looks like ATAPI would be really easy to add, if necessary. */ /* Other protocols, such as ATAPI don't seem to be very popular. looks like ATAPI would be really easy to add, if necessary. */
usb_debug (" Interface SubClass not supported.\n"); usb_debug (" Interface SubClass not supported.\n");
usb_detach_device (dev->controller, dev->address);
return; return;
} }
@ -632,11 +634,13 @@ usb_msc_init (usbdev_t *dev)
} }
if (MSC_INST (dev)->bulk_in == 0) { if (MSC_INST (dev)->bulk_in == 0) {
usb_debug("couldn't find bulk-in endpoint"); usb_debug("couldn't find bulk-in endpoint.\n");
usb_detach_device (dev->controller, dev->address);
return; return;
} }
if (MSC_INST (dev)->bulk_out == 0) { if (MSC_INST (dev)->bulk_out == 0) {
usb_debug("couldn't find bulk-out endpoint"); usb_debug("couldn't find bulk-out endpoint.\n");
usb_detach_device (dev->controller, dev->address);
return; return;
} }
usb_debug (" using endpoint %x as in, %x as out\n", usb_debug (" using endpoint %x as in, %x as out\n",

View File

@ -180,7 +180,7 @@ xhci_set_address (hci_t *controller, int speed, int hubport, int hubaddr)
di = &xhci->dev[slot_id]; di = &xhci->dev[slot_id];
void *dma_buffer = dma_memalign(64, NUM_EPS * ctxsize); void *dma_buffer = dma_memalign(64, NUM_EPS * ctxsize);
if (!dma_buffer) if (!dma_buffer)
goto _free_return; goto _disable_return;
memset(dma_buffer, 0, NUM_EPS * ctxsize); memset(dma_buffer, 0, NUM_EPS * ctxsize);
for (i = 0; i < NUM_EPS; i++, dma_buffer += ctxsize) for (i = 0; i < NUM_EPS; i++, dma_buffer += ctxsize)
di->ctx.ep[i] = dma_buffer; di->ctx.ep[i] = dma_buffer;
@ -249,6 +249,7 @@ xhci_set_address (hci_t *controller, int speed, int hubport, int hubaddr)
_disable_return: _disable_return:
xhci_cmd_disable_slot(xhci, slot_id); xhci_cmd_disable_slot(xhci, slot_id);
xhci->dcbaa[slot_id] = 0; xhci->dcbaa[slot_id] = 0;
usb_detach_device(controller, slot_id);
_free_return: _free_return:
if (tr) if (tr)
free((void *)tr->ring); free((void *)tr->ring);
@ -425,7 +426,7 @@ xhci_destroy_dev(hci_t *const controller, const int slot_id)
{ {
xhci_t *const xhci = XHCI_INST(controller); xhci_t *const xhci = XHCI_INST(controller);
if (slot_id <= 0 || xhci->max_slots_en > slot_id) if (slot_id <= 0 || slot_id > xhci->max_slots_en)
return; return;
int i; int i;