summaryrefslogtreecommitdiff
path: root/sys/dev/usb/uhub.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/usb/uhub.c')
-rw-r--r--sys/dev/usb/uhub.c82
1 files changed, 81 insertions, 1 deletions
diff --git a/sys/dev/usb/uhub.c b/sys/dev/usb/uhub.c
index f7e0b5bc69b..56033142051 100644
--- a/sys/dev/usb/uhub.c
+++ b/sys/dev/usb/uhub.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: uhub.c,v 1.26 2004/07/11 05:32:38 deraadt Exp $ */
+/* $OpenBSD: uhub.c,v 1.27 2004/12/09 11:49:55 dlg Exp $ */
/* $NetBSD: uhub.c,v 1.64 2003/02/08 03:32:51 ichiro Exp $ */
/* $FreeBSD: src/sys/dev/usb/uhub.c,v 1.18 1999/11/17 22:33:43 n_hibma Exp $ */
@@ -83,6 +83,7 @@ struct uhub_softc {
};
Static usbd_status uhub_explore(usbd_device_handle hub);
+Static usbd_status uhub_reset_device(usbd_device_handle, int);
Static void uhub_intr(usbd_xfer_handle, usbd_private_handle,usbd_status);
#if defined(__FreeBSD__)
@@ -211,6 +212,7 @@ USB_ATTACH(uhub)
dev->hub = hub;
dev->hub->hubsoftc = sc;
hub->explore = uhub_explore;
+ hub->reset_device = uhub_reset_device;
hub->hubdesc = hubdesc;
DPRINTFN(1,("usbhub_init_hub: selfpowered=%d, parent=%p, "
@@ -323,6 +325,84 @@ USB_ATTACH(uhub)
USB_ATTACH_ERROR_RETURN;
}
+Static usbd_status
+uhub_reset_device(usbd_device_handle dev, int port)
+{
+ struct uhub_softc *sc = dev->hub->hubsoftc;
+ struct usbd_port *up;
+ int status, change;
+ int speed;
+ usbd_status err;
+
+ DPRINTFN(10, ("uhub_reset_device port=%d\n", port));
+
+ if (!sc->sc_running)
+ return (USBD_NOT_STARTED);
+
+ up = &dev->hub->ports[port-1];
+
+ /* Disconnect */
+ DPRINTF(("uhub_reset_device: disconnecting device on port %d\n", port));
+ usb_disconnect_port(up, USBDEV(sc->sc_dev));
+ usbd_clear_port_feature(dev, port, UHF_C_PORT_CONNECTION);
+
+ /* Wait for maximum device power up time. */
+ usbd_delay_ms(dev, USB_PORT_POWERUP_DELAY);
+
+ /* Reset port, which implies enabling it. */
+ err = usbd_reset_port(dev, port, &up->status);
+ if (err) {
+ printf("%s: port %d reset failed\n",
+ USBDEVNAME(sc->sc_dev), port);
+ return(err);
+ }
+
+ /* connect */
+ err = usbd_get_port_status(dev, port, &up->status);
+ if (err) {
+ DPRINTF(("uhub_reset_device: get port status failed, "
+ "error=%s\n", usbd_errstr(err)));
+ return (err);
+ }
+ status = UGETW(up->status.wPortStatus);
+ change = UGETW(up->status.wPortChange);
+
+ /* Figure out device speed */
+ if (status & UPS_HIGH_SPEED)
+ speed = USB_SPEED_HIGH;
+ else if (status & UPS_LOW_SPEED)
+ speed = USB_SPEED_LOW;
+ else
+ speed = USB_SPEED_FULL;
+ /* Get device info and set its address. */
+ err = usbd_new_device(USBDEV(sc->sc_dev), dev->bus, dev->depth + 1,
+ speed, port, up);
+ /* XXX retry a few times? */
+ if (err) {
+ DPRINTFN(-1,("uhub_explore: usb_new_device failed, "
+ "error=%s\n", usbd_errstr(err)));
+ /* Avoid addressing problems by disabling. */
+ /* usbd_reset_port(dev, port, &up->status); */
+
+ /*
+ * The unit refused to accept a new address, or had
+ * some other serious problem. Since we cannot leave
+ * at 0 we have to disable the port instead.
+ */
+ printf("%s: device problem, disabling port %d\n",
+ USBDEVNAME(sc->sc_dev), port);
+ usbd_clear_port_feature(dev, port, UHF_PORT_ENABLE);
+ } else {
+ /* The port set up succeeded, reset error count. */
+ up->restartcnt = 0;
+
+ if (up->device->hub)
+ up->device->hub->explore(up->device);
+ }
+
+ return (USBD_NORMAL_COMPLETION);
+}
+
usbd_status
uhub_explore(usbd_device_handle dev)
{