aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGerd Hoffmann <kraxel@redhat.com>2013-08-21 13:59:21 +0200
committerGerd Hoffmann <kraxel@redhat.com>2013-09-19 13:04:53 +0200
commit63cbab1628dc406ed5d765bd85bae3a1a525a2c0 (patch)
tree460f6dd97dcdb8852c731e088b31fae79e861bdf
parentee9b84f61c55a67b477a019e937460be13ccbfa1 (diff)
downloadseabios-63cbab1628dc406ed5d765bd85bae3a1a525a2c0.tar.gz
usb: add usb_update_pipe()
Preparation for better xhci support: allows to notify host controllers instead of going through a free+alloc cycle. Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
-rw-r--r--src/hw/usb.c22
1 files changed, 17 insertions, 5 deletions
diff --git a/src/hw/usb.c b/src/hw/usb.c
index 59d2a93..b4d0ab8 100644
--- a/src/hw/usb.c
+++ b/src/hw/usb.c
@@ -44,6 +44,18 @@ usb_alloc_pipe(struct usbdevice_s *usbdev
}
}
+// Update an pipe (used for control only)
+struct usb_pipe *
+usb_update_pipe(struct usbdevice_s *usbdev, struct usb_pipe *pipe
+ , struct usb_endpoint_descriptor *epdesc)
+{
+ switch (usbdev->hub->cntl->type) {
+ default:
+ free_pipe(pipe);
+ return usb_alloc_pipe(usbdev, epdesc);
+ }
+}
+
// Send a message on a control pipe using the default control descriptor.
static int
send_control(struct usb_pipe *pipe, int dir, const void *cmd, int cmdsize
@@ -262,15 +274,16 @@ usb_set_address(struct usbdevice_s *usbdev)
req.wIndex = 0;
req.wLength = 0;
int ret = send_default_control(usbdev->defpipe, &req, NULL);
- free_pipe(usbdev->defpipe);
- if (ret)
+ if (ret) {
+ free_pipe(usbdev->defpipe);
return -1;
+ }
msleep(USB_TIME_SETADDR_RECOVERY);
cntl->maxaddr++;
usbdev->devaddr = cntl->maxaddr;
- usbdev->defpipe = usb_alloc_pipe(usbdev, &epdesc);
+ usbdev->defpipe = usb_update_pipe(usbdev, usbdev->defpipe, &epdesc);
if (!usbdev->defpipe)
return -1;
return 0;
@@ -294,12 +307,11 @@ configure_usb_device(struct usbdevice_s *usbdev)
, dinfo.bDeviceProtocol, dinfo.bMaxPacketSize0);
if (dinfo.bMaxPacketSize0 < 8 || dinfo.bMaxPacketSize0 > 64)
return 0;
- free_pipe(usbdev->defpipe);
struct usb_endpoint_descriptor epdesc = {
.wMaxPacketSize = dinfo.bMaxPacketSize0,
.bmAttributes = USB_ENDPOINT_XFER_CONTROL,
};
- usbdev->defpipe = usb_alloc_pipe(usbdev, &epdesc);
+ usbdev->defpipe = usb_update_pipe(usbdev, usbdev->defpipe, &epdesc);
if (!usbdev->defpipe)
return -1;