Skip site navigation (1)Skip section navigation (2)
Date:      Sun, 23 Sep 2007 14:49:31 GMT
From:      Hans Petter Selasky <hselasky@FreeBSD.org>
To:        Perforce Change Reviews <perforce@FreeBSD.org>
Subject:   PERFORCE change 126734 for review
Message-ID:  <200709231449.l8NEnVSl087562@repoman.freebsd.org>

next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=126734

Change 126734 by hselasky@hselasky_laptop001 on 2007/09/23 14:49:06

	
	FYI; The comments follow the P4 diff from top to bottom.
	
	- renamed "sc_hub" into "sc_udev" to better match the analogy
	  of the other USB device drivers
	
	- convert kernel USB flags (USBD_XXX) into a bitfield (scripted)
	
	- all USB BULK/ISOC/INTR -transfers must setup
	  "xfer->frlengths[]" before calling "usbd_start_hardware()".
	  Else the actual length of the previous transfer will be 
	  used for transfer length of the next USB transfer.
	
	- new internal function "uhub_port_to_sub_device()" which gets
	  the USB device structure connected to a USB port.
	
	- factored out code into "uhub_explore_sub()"
	
	- USB port numbers and indexes are now 8-bit hence there
	  cannot be more than 255 ports according to the USB
	  specification.
	
	- passing a mutex to "usbd_do_request_flags()" and all
	  "usbreq_xxx()" functions is now mandatory.
	
	- remove redundant passing of "struct usbd_port *" to
	  "usbd_new_device()"
	
	- "struct usbd_device" now has a pointer directly to the
	  parent HUB, "paraent_hub"
	
	- be consequent on differating between port index and port
	  number which differ by 1 unit.

Affected files ...

.. //depot/projects/usb/src/sys/dev/usb/uhub.c#16 edit

Differences ...

==== //depot/projects/usb/src/sys/dev/usb/uhub.c#16 (text+ko) ====

@@ -70,14 +70,14 @@
 
 struct uhub_softc {
 	device_t		sc_dev;		/* base device */
-	struct usbd_device	*sc_hub;	/* USB device */
+	struct usbd_device	*sc_udev;	/* USB device */
 	struct usbd_xfer	*sc_xfer[2];	/* interrupt xfer */
 	uint8_t			sc_flags;
 #define	UHUB_FLAG_RUNNING 0x01
 #define	UHUB_FLAG_INTR_STALL 0x02
 	uint8_t			sc_name[32];
 };
-#define	UHUB_PROTO(sc) ((sc)->sc_hub->ddesc.bDeviceProtocol)
+#define	UHUB_PROTO(sc) ((sc)->sc_udev->ddesc.bDeviceProtocol)
 #define	UHUB_IS_HIGH_SPEED(sc) (UHUB_PROTO(sc) != UDPROTO_FSHUB)
 #define	UHUB_IS_SINGLE_TT(sc) (UHUB_PROTO(sc) == UDPROTO_HSHUBSTT)
 
@@ -101,7 +101,7 @@
       .endpoint  = UE_ADDR_ANY,
       .direction = UE_DIR_ANY,
       .timeout   = 0,
-      .flags     = (USBD_PIPE_BOF|USBD_SHORT_XFER_OK),
+      .flags     = { .pipe_bof = 1, .short_xfer_ok = 1, },
       .bufsize   = 0, /* use wMaxPacketSize */
       .callback  = &uhub_intr_callback,
       .interval  = UHUB_INTR_INTERVAL,
@@ -113,7 +113,7 @@
       .direction = UE_DIR_ANY,
       .timeout   = 1000, /* 1 second */
       .interval  = 50, /* 50ms */
-      .flags     = 0,
+      .flags     = { },
       .bufsize   = sizeof(usb_device_request_t),
       .callback  = &uhub_intr_clear_stall_callback,
     },
@@ -179,12 +179,13 @@
 	 * event handler thread that we need
 	 * to be explored again:
 	 */
-	usb_needs_explore(sc->sc_hub);
+	usb_needs_explore(sc->sc_udev);
 
  tr_setup:
 	if (sc->sc_flags & UHUB_FLAG_INTR_STALL) {
 	    usbd_transfer_start(sc->sc_xfer[1]);
 	} else {
+	    xfer->frlengths[0] = xfer->max_data_length;
 	    usbd_start_hardware(xfer);
 	}
 	return;
@@ -198,17 +199,65 @@
 	return;
 }
 
+static struct usbd_device *
+uhub_port_to_sub_device(struct usbd_device *udev, struct usbd_port *up)
+{
+	if ((udev == NULL) || (up == NULL)) {
+	    /* be NULL safe */
+	    return NULL;
+	}
+
+	if (up->device_addr == USB_START_ADDR) {
+	    /* nothing to do */
+	    return NULL;
+	}
+
+	return udev->bus->devices[up->device_addr];
+}
+
 static usbd_status
+uhub_explore_sub(device_t dev, struct usbd_device *udev, struct usbd_port *up)
+{
+	struct usbd_device *child;
+	uint8_t refcount = usb_driver_added_refcount;
+	usbd_status err = 0;
+
+	child = uhub_port_to_sub_device(udev, up);
+
+	if (child == NULL) {
+	    /* nothing to do */
+	    return 0;
+	}
+
+	/* check if probe and attach should be done */
+
+	if (child->driver_added_refcount != refcount) {
+	    child->driver_added_refcount = refcount;
+	    err = usbd_probe_and_attach(dev, child);
+	}
+
+	/* if a HUB becomes present, do a recursive HUB explore */
+
+	if (child->hub) {
+	    (child->hub->explore)(child);
+	}
+
+	return err;
+}
+
+static usbd_status
 uhub_explore(struct usbd_device *udev)
 {
 	usb_hub_descriptor_t *hd = &udev->hub->hubdesc;
 	struct uhub_softc *sc = udev->hub->hubsoftc;
+	struct usbd_device *child;
 	struct usbd_port *up;
 	usbd_status err;
-	uint16_t port;
 	uint16_t change;
 	uint16_t status;
+	uint8_t portno;
 	uint8_t speed;
+	uint8_t x;
 
 	DPRINTF(sc, 10, "udev=%p addr=%d\n", udev, udev->address);
 
@@ -223,10 +272,12 @@
 		return (USBD_TOO_DEEP);
 	}
 
-	for(port = 1; port <= hd->bNbrPorts; port++)
+	for (x = 0; x < hd->bNbrPorts; x++)
 	{
-		up = &udev->hub->ports[port-1];
-		err = usbreq_get_port_status(udev, port, &up->status);
+		up = udev->hub->ports + x;
+		portno = x + 1;
+		err = usbreq_get_port_status
+		  (udev, &usb_global_lock, &up->status, portno);
 		if (err)
 		{
 			DPRINTF(sc, 0, "get port status failed, "
@@ -236,11 +287,12 @@
 		status = UGETW(up->status.wPortStatus);
 		change = UGETW(up->status.wPortChange);
 		DPRINTF(sc, 3, "port %d status 0x%04x 0x%04x\n",
-			port, status, change);
+			portno, status, change);
 		if(change & UPS_C_PORT_ENABLED)
 		{
 			DPRINTF(sc, 0, "C_PORT_ENABLED 0x%x\n", change);
-			usbreq_clear_port_feature(udev, port, UHF_C_PORT_ENABLE);
+			usbreq_clear_port_feature
+			  (udev, &usb_global_lock, portno, UHF_C_PORT_ENABLE);
 			if(change & UPS_C_CONNECT_STATUS)
 			{
 				/* ignore the port error
@@ -250,7 +302,7 @@
 			else if(status & UPS_PORT_ENABLED)
 			{
 				DPRINTF(sc, -1, "illegal enable change, "
-					"port %d\n", port);
+					"port %d\n", portno);
 			}
 			else
 			{
@@ -258,48 +310,35 @@
 				if(up->restartcnt) /* no message first time */
 				{
 					DPRINTF(sc, -1, "port error, restarting "
-						"port %d\n", port);
+						"port %d\n", portno);
 				}
 
 				if(up->restartcnt++ < USBD_RESTART_MAX) {
 					goto disconnect;
 				} else {
 					DPRINTF(sc, -1, "port error, giving up "
-						"port %d\n", port);
+						"port %d\n", portno);
 				}
 			}
 		}
 		if(!(change & UPS_C_CONNECT_STATUS))
 		{
 			DPRINTF(sc, 3, "port=%d !C_CONNECT_"
-				"STATUS\n", port);
+				"STATUS\n", portno);
+
 			/* no status change, just do recursive explore */
-			if(up->device != NULL)
-			{
-				if(up->device->hub != NULL)
-				{
-					(up->device->hub->explore)(up->device);
-				}
-				else
-				{
-					/* allow drivers to be hot-plugged */
-
-					if(up->last_refcount != usb_driver_added_refcount)
-					{
-						usbd_probe_and_attach
-						  (sc->sc_dev, port, up);
-					}
-				}
-			}
+			err = uhub_explore_sub(sc->sc_dev, udev, up);
 			continue;
 		}
 
 		/* we have a connect status change, handle it */
 
 		DPRINTF(sc, 0, "status change hub=%d port=%d\n",
-			udev->address, port);
-		usbreq_clear_port_feature(udev, port, UHF_C_PORT_CONNECTION);
-		/*usbreq_clear_port_feature(udev, port, UHF_C_PORT_ENABLE);*/
+			udev->address, portno);
+		usbreq_clear_port_feature
+		  (udev, &usb_global_lock, portno, UHF_C_PORT_CONNECTION);
+		/* usbreq_clear_port_feature
+		     (udev, &usb_global_lock, portno, UHF_C_PORT_ENABLE); */
 		/*
 		 * If there is already a device on the port the change status
 		 * must mean that is has disconnected.  Looking at the
@@ -308,20 +347,21 @@
 		 * the disconnect.
 		 */
 	disconnect:
-		if(up->device != NULL)
-		{
+		child = uhub_port_to_sub_device(udev, up);
+		if (child) {
 			/* disconnected */
 			DPRINTF(sc, 0, "device addr=%d disappeared "
-				"on port %d\n", up->device->address, port);
-			usbd_free_device(up, 1);
-			usbreq_clear_port_feature(udev, port,
-						  UHF_C_PORT_CONNECTION);
+				"on port %d\n", child->address,
+				portno);
+			usbd_free_device(child, 1);
+			usbreq_clear_port_feature
+			  (udev, &usb_global_lock, portno, UHF_C_PORT_CONNECTION);
 		}
 		if(!(status & UPS_CURRENT_CONNECT_STATUS))
 		{
 			/* nothing connected, just ignore it */
 			DPRINTF(sc, 3, "port=%d !CURRENT_CONNECT_STATUS\n",
-				port);
+				portno);
 			continue;
 		}
 
@@ -330,22 +370,27 @@
 		if(!(status & UPS_PORT_POWER))
 		{
 			DPRINTF(sc, -1, "strange, connected port %d "
-				"has no power\n", port);
+				"has no power\n", portno);
 		}
 
 		/* wait for maximum device power up time */
 		usbd_delay_ms(udev, USB_PORT_POWERUP_DELAY);
 
 		/* reset port, which implies enabling it */
-		if(usbreq_reset_port(udev, port, &up->status))
-		{
+		err = usbreq_reset_port
+		  (udev, &usb_global_lock, &up->status, portno);
+
+		if (err) {
 			DPRINTF(sc, -1, "port %d reset "
-				"failed\n", port);
+				"failed, error=%s\n",
+				portno, usbd_errstr(err));
 			continue;
 		}
 
 		/* get port status again, it might have changed during reset */
-		err = usbreq_get_port_status(udev, port, &up->status);
+		err = usbreq_get_port_status
+		  (udev, &usb_global_lock, &up->status, portno);
+
 		if(err)
 		{
 			DPRINTF(sc, 0, "get port status failed, "
@@ -358,7 +403,7 @@
 		{
 			/* nothing connected, just ignore it */
 			DPRINTF(sc, 1, "port %d, device disappeared "
-				"after reset\n", port);
+				"after reset\n", portno);
 			continue;
 		}
 
@@ -368,14 +413,18 @@
 		  (status & UPS_LOW_SPEED) ? USB_SPEED_LOW : USB_SPEED_FULL;
 
 		/* get device info and set its address */
-		err = usbd_new_device(sc->sc_dev, udev->bus,
-				      udev->depth + 1, speed, port, up);
-		if (err)
-		{
+		err = usbd_new_device(sc->sc_dev, udev->bus, udev,
+				      udev->depth + 1, speed, x, portno);
+		if (err == 0) {
+		    err = uhub_explore_sub(sc->sc_dev, udev, up);
+		}
+
+		if (err) {
 			DPRINTF(sc, -1, "usb_new_device failed, "
 				"error=%s\n", usbd_errstr(err));
 			/* Avoid addressing problems by disabling. */
-			/* usbd_reset_port(udev, port, &up->status); */
+			/* usbreq_reset_port
+			   (udev, &usb_global_lock, &up->status, portno); */
 
 			/*
 			 * The unit refused to accept a new address, or had
@@ -383,19 +432,18 @@
 			 * at 0 we have to disable the port instead.
 			 */
 			DPRINTF(sc, -1, "device problem (%s), "
-				"disabling port %d\n", usbd_errstr(err), port);
-			usbreq_clear_port_feature(udev, port, UHF_PORT_ENABLE);
+				"disabling port %d\n", usbd_errstr(err), portno);
+			usbreq_clear_port_feature
+			  (udev, &usb_global_lock, portno, UHF_PORT_ENABLE);
+			continue;
 		}
-		else
-		{
-			/* the port set up succeeded, reset error count */
-			up->restartcnt = 0;
+
+		/* 
+		 * The port setup succeeded, reset error count and
+		 * do recursive explore, if any:
+		 */
+		up->restartcnt = 0;
 
-			if(up->device->hub)
-			{
-				(up->device->hub->explore)(up->device);
-			}
-		}
 	}
 	return (USBD_NORMAL_COMPLETION);
 }
@@ -424,20 +472,22 @@
 	struct uhub_softc *sc = device_get_softc(dev);
 	struct usb_attach_arg *uaa = device_get_ivars(dev);
 	struct usbd_device *udev = uaa->device;
+	struct usbd_device *parent_hub = udev->parent_hub;
 	struct usbd_hub *hub;
 	usb_device_request_t req;
 	usb_hub_descriptor_t hubdesc;
-	uint16_t port;
-	uint16_t nports;
-	uint16_t removable;
 	uint16_t pwrdly;
+	uint8_t x;
+	uint8_t nports;
+	uint8_t portno;
+	uint8_t removable;
 	usbd_status err;
 
 	if (sc == NULL) {
 	    return ENOMEM;
 	}
 
-	sc->sc_hub = udev;
+	sc->sc_udev = udev;
 	sc->sc_dev = dev; 
 
 	snprintf(sc->sc_name, sizeof(sc->sc_name), "%s",
@@ -460,9 +510,9 @@
 		    "parent->selfpowered=%d\n",
 		    udev->depth,
 		    udev->self_powered, 
-		    udev->powersrc->parent,
-		    udev->powersrc->parent ?
-		    udev->powersrc->parent->self_powered : 0);
+		    parent_hub,
+		    parent_hub ?
+		    parent_hub->self_powered : 0);
 
 	if(udev->depth > USB_HUB_MAX_DEPTH)
 	{
@@ -471,9 +521,8 @@
 		goto error;
 	}
 
-	if(!udev->self_powered && 
-	    (udev->powersrc->parent != NULL) &&
-	    (!udev->powersrc->parent->self_powered))
+	if (!udev->self_powered && parent_hub &&
+	    (!parent_hub->self_powered))
 	{
 		DPRINTF(sc, -1, "bus powered hub connected to "
 			"bus powered hub. HUB ignored!\n");
@@ -490,7 +539,7 @@
 	USETW(req.wIndex, 0);
 	USETW(req.wLength, USB_HUB_DESCRIPTOR_SIZE);
 
-	err = usbd_do_request(udev, &req, &hubdesc);
+	err = usbd_do_request(udev, &usb_global_lock, &req, &hubdesc);
 
 	nports = hubdesc.bNbrPorts;
 
@@ -498,7 +547,7 @@
 	{
 		uint16_t len = (USB_HUB_DESCRIPTOR_SIZE-1) + ((nports+7) / 8);
 		USETW(req.wLength, len);
-		err = usbd_do_request(udev, &req, &hubdesc);
+		err = usbd_do_request(udev, &usb_global_lock, &req, &hubdesc);
 	}
 
 	if(err)
@@ -520,16 +569,31 @@
 		goto error;
 	}
 
-	udev->hub = malloc((sizeof(hub[0]) + (sizeof(hub[0].ports[0]) * nports)),
-			   M_USBDEV, M_WAITOK|M_ZERO);
+	hub = malloc(sizeof(hub[0]) + (sizeof(hub->ports[0]) * nports),
+		     M_USBDEV, M_WAITOK|M_ZERO);
 
-	if(udev->hub == NULL)
+	if (hub == NULL)
 	{
 		goto error;
 	}
 
+	udev->hub = hub;
+
 	/* init FULL-speed ISOCHRONOUS schedule */
-	usbd_fs_isoc_schedule_init_all(udev->hub->fs_isoc_schedule);
+	usbd_fs_isoc_schedule_init_all(hub->fs_isoc_schedule);
+
+	/* initialize HUB structure */
+	hub->hubsoftc = sc;
+	hub->explore = &uhub_explore;
+	hub->hubdesc = hubdesc;
+	hub->hubudev = udev;
+
+	/* if self powered hub, give ports maximum current */
+	if (udev->self_powered) {
+	    hub->portpower = USB_MAX_POWER;
+	} else {
+	    hub->portpower = USB_MIN_POWER;
+	}
 
 	/* set up interrupt pipe */
 	err = usbd_transfer_setup(udev, 0, sc->sc_xfer, 
@@ -541,12 +605,6 @@
 		goto error;
 	}
 
-	hub = udev->hub;
-
-	hub->hubsoftc = sc;
-	hub->explore = uhub_explore;
-	hub->hubdesc = hubdesc;
-
 	/* wait with power off for a while */
 	usbd_delay_ms(udev, USB_POWER_DOWN_TIME);
 
@@ -581,46 +639,37 @@
 	pwrdly = ((hubdesc.bPwrOn2PwrGood * UHD_PWRON_FACTOR) +
 		  USB_EXTRA_POWER_UP_TIME);
 
-	for(port = 1;
-	    port <= nports;
-	    port++)
-	{
+	for(x = 0; x < nports; x++) {
 		/* set up data structures */
-		struct usbd_port *up = &hub->ports[port-1];
-		up->device = NULL;
-		up->parent = udev;
-		up->portno = port;
-
-		/* self powered hub, give ports maximum current */
-		up->power = (udev->self_powered) ? 
-		  USB_MAX_POWER :
-		  USB_MIN_POWER ;
-
+		struct usbd_port *up = hub->ports + x;
+		up->device_addr = USB_START_ADDR;
 		up->restartcnt = 0;
+		portno = x + 1;
 
 		/* check if port is removable */
-		if(!UHD_NOT_REMOV(&hubdesc, port))
+		if(!UHD_NOT_REMOV(&hubdesc, portno))
 		{
 			removable++;
 		}
 
 		/* turn the power on */
-		err = usbreq_set_port_feature(udev, port, UHF_PORT_POWER);
-		if(err)
-		{
+		err = usbreq_set_port_feature
+		  (udev, &usb_global_lock, portno, UHF_PORT_POWER);
+
+		if (err) {
 			DPRINTF(sc, -1, "port %d power on failed, %s\n",
-				port, usbd_errstr(err));
+				portno, usbd_errstr(err));
 		}
-		DPRINTF(sc, 0, "turn on port %d power\n", port);
+		DPRINTF(sc, 0, "turn on port %d power\n",
+			portno);
 
 		/* wait for stable power */
 		usbd_delay_ms(udev, pwrdly);
 	}
 
 	device_printf(dev, "%d port%s with %d "
-		      "removable, %s powered\n",
-		      nports, (nports != 1) ? "s" : "",
-		      removable, udev->self_powered ? "self" : "bus");
+	      "removable, %s powered\n", nports, (nports != 1) ? "s" : "",
+	      removable, udev->self_powered ? "self" : "bus");
 
 	/* the usual exploration will finish the setup */
 
@@ -637,8 +686,7 @@
  error:
 	usbd_transfer_unsetup(sc->sc_xfer, 2);
 
-	if(udev->hub)
-	{
+	if (udev->hub) {
 		free(udev->hub, M_USBDEV);
 		udev->hub = NULL;
 	}
@@ -653,10 +701,9 @@
 uhub_detach(device_t dev)
 {
         struct uhub_softc *sc = device_get_softc(dev);
-	struct usbd_hub *hub = sc->sc_hub->hub;
-	struct usbd_port *up;
-	uint16_t port;
-	uint16_t nports;
+	struct usbd_hub *hub = sc->sc_udev->hub;
+	struct usbd_device *child;
+	uint8_t x;
 
 	/* detach all children first */
 	bus_generic_detach(dev);
@@ -666,26 +713,22 @@
 		return (0);
 	}
 
-	nports = hub->hubdesc.bNbrPorts;
-	for(port = 0;
-	    port < nports;
-	    port++)
-	{
-		up = &hub->ports[port];
-		if(up->device)
-		{
-			/* subdevices are not freed, because
-			 * the caller of uhub_detach() will
-			 * do that
-			 */
-			usbd_free_device(up, 0);
-		}
+	for (x = 0; x < hub->hubdesc.bNbrPorts; x++) {
+
+	    child = uhub_port_to_sub_device(sc->sc_udev, hub->ports + x);
+
+	    /* Subdevices are not freed, because
+	     * the caller of uhub_detach() will
+	     * do that. The function we are calling
+	     * is NULL safe.
+	     */
+	    usbd_free_device(child, 0);
 	}
 
 	usbd_transfer_unsetup(sc->sc_xfer, 2);
 
 	free(hub, M_USBDEV);
-	sc->sc_hub->hub = NULL;
+	sc->sc_udev->hub = NULL;
 	return (0);
 }
 
@@ -701,20 +744,18 @@
 			   char *buf, size_t buflen)
 {
 	struct uhub_softc *sc = device_get_softc(parent);
-	struct usbd_hub *hub = sc->sc_hub->hub;
+	struct usbd_hub *hub = sc->sc_udev->hub;
 	struct usbd_device *udev;
-	uint16_t port;
-	uint16_t nports;
+	uint8_t x;
+	uint8_t nports;
 	uint8_t iface_index;
 
 	mtx_lock(&usb_global_lock);
 
 	nports = hub->hubdesc.bNbrPorts;
-        for(port = 0;
-            port < nports;
-            port++)
+        for (x = 0; x < nports; x++)
         {
-		udev = hub->ports[port].device;
+		udev = uhub_port_to_sub_device(sc->sc_udev, hub->ports + x);
 		if(udev)
 		{
 			device_t * subdev = 
@@ -754,11 +795,11 @@
 	if(udev->probed == USBD_PROBED_IFACE_AND_FOUND)
 	{
 		snprintf(buf, buflen, "port=%i interface=%i",
-			 port, iface_index);
+			 x + 1, iface_index);
 	}
 	else
 	{
-                snprintf(buf, buflen, "port=%i", port);
+                snprintf(buf, buflen, "port=%i", x + 1);
 	}
 
 	mtx_unlock(&usb_global_lock);
@@ -771,21 +812,19 @@
 			  char *buf, size_t buflen)
 {
 	struct uhub_softc *sc = device_get_softc(parent);
-	struct usbd_hub *hub = sc->sc_hub->hub;
+	struct usbd_hub *hub = sc->sc_udev->hub;
         struct usbd_interface *iface;
 	struct usbd_device *udev;
-	uint16_t port;
-	uint16_t nports;
+	uint8_t x;
+	uint8_t nports;
 	uint8_t iface_index;
 
 	mtx_lock(&usb_global_lock);
 
 	nports = hub->hubdesc.bNbrPorts;
-        for(port = 0;
-            port < nports;
-            port++)
+        for (x = 0; x < nports; x++)
         {
-		udev = hub->ports[port].device;
+		udev = uhub_port_to_sub_device(sc->sc_udev, hub->ports + x);
 		if(udev)
 		{
 			device_t * subdev = 



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200709231449.l8NEnVSl087562>