756 lines
		
	
	
	
		
			21 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			756 lines
		
	
	
	
		
			21 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* usb.c - USB Hub Support.  */
 | ||
| /*
 | ||
|  *  GRUB  --  GRand Unified Bootloader
 | ||
|  *  Copyright (C) 2008  Free Software Foundation, Inc.
 | ||
|  *
 | ||
|  *  GRUB is free software: you can redistribute it and/or modify
 | ||
|  *  it under the terms of the GNU General Public License as published by
 | ||
|  *  the Free Software Foundation, either version 3 of the License, or
 | ||
|  *  (at your option) any later version.
 | ||
|  *
 | ||
|  *  GRUB is distributed in the hope that it will be useful,
 | ||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 | ||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | ||
|  *  GNU General Public License for more details.
 | ||
|  *
 | ||
|  *  You should have received a copy of the GNU General Public License
 | ||
|  *  along with GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | ||
|  */
 | ||
| 
 | ||
| #include <grub/dl.h>
 | ||
| #include <grub/mm.h>
 | ||
| #include <grub/usb.h>
 | ||
| #include <grub/misc.h>
 | ||
| #include <grub/time.h>
 | ||
| 
 | ||
| #define GRUB_USBHUB_MAX_DEVICES 128
 | ||
| 
 | ||
| /* USB Supports 127 devices, with device 0 as special case.  */
 | ||
| static struct grub_usb_device *grub_usb_devs[GRUB_USBHUB_MAX_DEVICES];
 | ||
| 
 | ||
| static int rescan = 0;
 | ||
| static int npending = 0;
 | ||
| 
 | ||
| struct grub_usb_hub
 | ||
| {
 | ||
|   struct grub_usb_hub *next;
 | ||
|   grub_usb_controller_t controller;
 | ||
|   int nports;
 | ||
|   struct grub_usb_device **devices;
 | ||
|   struct grub_usb_hub_port *ports;
 | ||
|   grub_usb_device_t dev;
 | ||
| };
 | ||
| 
 | ||
| static struct grub_usb_hub *hubs;
 | ||
| static grub_usb_controller_dev_t grub_usb_list;
 | ||
| 
 | ||
| /* Add a device that currently has device number 0 and resides on
 | ||
|    CONTROLLER, the Hub reported that the device speed is SPEED.  */
 | ||
| static grub_usb_device_t
 | ||
| grub_usb_hub_add_dev (grub_usb_controller_t controller,
 | ||
|                       grub_usb_speed_t speed,
 | ||
|                       int split_hubport, int split_hubaddr)
 | ||
| {
 | ||
|   grub_usb_device_t dev;
 | ||
|   int i;
 | ||
|   grub_usb_err_t err;
 | ||
| 
 | ||
|   grub_boot_time ("Attaching USB device");
 | ||
| 
 | ||
|   dev = grub_zalloc (sizeof (struct grub_usb_device));
 | ||
|   if (! dev)
 | ||
|     return NULL;
 | ||
| 
 | ||
|   dev->controller = *controller;
 | ||
|   dev->speed = speed;
 | ||
|   dev->split_hubport = split_hubport;
 | ||
|   dev->split_hubaddr = split_hubaddr;
 | ||
| 
 | ||
|   err = grub_usb_device_initialize (dev);
 | ||
|   if (err)
 | ||
|     {
 | ||
|       grub_free (dev);
 | ||
|       return NULL;
 | ||
|     }
 | ||
| 
 | ||
|   /* Assign a new address to the device.  */
 | ||
|   for (i = 1; i < GRUB_USBHUB_MAX_DEVICES; i++)
 | ||
|     {
 | ||
|       if (! grub_usb_devs[i])
 | ||
| 	break;
 | ||
|     }
 | ||
|   if (i == GRUB_USBHUB_MAX_DEVICES)
 | ||
|     {
 | ||
|       grub_error (GRUB_ERR_IO, "can't assign address to USB device");
 | ||
|       for (i = 0; i < 8; i++)
 | ||
|         grub_free (dev->config[i].descconf);
 | ||
|       grub_free (dev);
 | ||
|       return NULL;
 | ||
|     }
 | ||
| 
 | ||
|   err = grub_usb_control_msg (dev,
 | ||
| 			      (GRUB_USB_REQTYPE_OUT
 | ||
| 			       | GRUB_USB_REQTYPE_STANDARD
 | ||
| 			       | GRUB_USB_REQTYPE_TARGET_DEV),
 | ||
| 			      GRUB_USB_REQ_SET_ADDRESS,
 | ||
| 			      i, 0, 0, NULL);
 | ||
|   if (err)
 | ||
|     {
 | ||
|       for (i = 0; i < 8; i++)
 | ||
|         grub_free (dev->config[i].descconf);
 | ||
|       grub_free (dev);
 | ||
|       return NULL;
 | ||
|     }
 | ||
| 
 | ||
|   dev->addr = i;
 | ||
|   dev->initialized = 1;
 | ||
|   grub_usb_devs[i] = dev;
 | ||
| 
 | ||
|   grub_dprintf ("usb", "Added new usb device: %p, addr=%d\n",
 | ||
| 		dev, i);
 | ||
|   grub_dprintf ("usb", "speed=%d, split_hubport=%d, split_hubaddr=%d\n",
 | ||
| 		speed, split_hubport, split_hubaddr);
 | ||
| 
 | ||
|   /* Wait "recovery interval", spec. says 2ms */
 | ||
|   grub_millisleep (2);
 | ||
| 
 | ||
|   grub_boot_time ("Probing USB device driver");
 | ||
|   
 | ||
|   grub_usb_device_attach (dev);
 | ||
| 
 | ||
|   grub_boot_time ("Attached USB device");
 | ||
|   
 | ||
|   return dev;
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| static grub_usb_err_t
 | ||
| grub_usb_add_hub (grub_usb_device_t dev)
 | ||
| {
 | ||
|   struct grub_usb_usb_hubdesc hubdesc;
 | ||
|   grub_usb_err_t err;
 | ||
|   int i;
 | ||
|   
 | ||
|   err = grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_IN
 | ||
| 	  		            | GRUB_USB_REQTYPE_CLASS
 | ||
| 			            | GRUB_USB_REQTYPE_TARGET_DEV),
 | ||
|                               GRUB_USB_REQ_GET_DESCRIPTOR,
 | ||
| 			      (GRUB_USB_DESCRIPTOR_HUB << 8) | 0,
 | ||
| 			      0, sizeof (hubdesc), (char *) &hubdesc);
 | ||
|   if (err)
 | ||
|     return err;
 | ||
|   grub_dprintf ("usb", "Hub descriptor:\n\t\t len:%d, typ:0x%02x, cnt:%d, char:0x%02x, pwg:%d, curr:%d\n",
 | ||
|                 hubdesc.length, hubdesc.type, hubdesc.portcnt,
 | ||
|                 hubdesc.characteristics, hubdesc.pwdgood,
 | ||
|                 hubdesc.current);
 | ||
| 
 | ||
|   /* Activate the first configuration. Hubs should have only one conf. */
 | ||
|   grub_dprintf ("usb", "Hub set configuration\n");
 | ||
|   grub_usb_set_configuration (dev, 1);
 | ||
| 
 | ||
|   dev->nports = hubdesc.portcnt;
 | ||
|   dev->children = grub_zalloc (hubdesc.portcnt * sizeof (dev->children[0]));
 | ||
|   dev->ports = grub_zalloc (dev->nports * sizeof (dev->ports[0]));
 | ||
|   if (!dev->children || !dev->ports)
 | ||
|     {
 | ||
|       grub_free (dev->children);
 | ||
|       grub_free (dev->ports);
 | ||
|       return GRUB_USB_ERR_INTERNAL;
 | ||
|     }
 | ||
| 
 | ||
|   /* Power on all Hub ports.  */
 | ||
|   for (i = 1; i <= hubdesc.portcnt; i++)
 | ||
|     {
 | ||
|       grub_dprintf ("usb", "Power on - port %d\n", i);
 | ||
|       /* Power on the port and wait for possible device connect */
 | ||
|       grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_OUT
 | ||
| 				  | GRUB_USB_REQTYPE_CLASS
 | ||
| 				  | GRUB_USB_REQTYPE_TARGET_OTHER),
 | ||
| 			    GRUB_USB_REQ_SET_FEATURE,
 | ||
| 			    GRUB_USB_HUB_FEATURE_PORT_POWER,
 | ||
| 			    i, 0, NULL);
 | ||
|     }
 | ||
| 
 | ||
|   /* Rest will be done on next usb poll.  */
 | ||
|   for (i = 0; i < dev->config[0].interf[0].descif->endpointcnt;
 | ||
|        i++)
 | ||
|     {
 | ||
|       struct grub_usb_desc_endp *endp = NULL;
 | ||
|       endp = &dev->config[0].interf[0].descendp[i];
 | ||
| 
 | ||
|       if ((endp->endp_addr & 128) && grub_usb_get_ep_type(endp)
 | ||
| 	  == GRUB_USB_EP_INTERRUPT)
 | ||
| 	{
 | ||
| 	  grub_size_t len;
 | ||
| 	  dev->hub_endpoint = endp;
 | ||
| 	  len = endp->maxpacket;
 | ||
| 	  if (len > sizeof (dev->statuschange))
 | ||
| 	    len = sizeof (dev->statuschange);
 | ||
| 	  dev->hub_transfer
 | ||
| 	    = grub_usb_bulk_read_background (dev, endp, len,
 | ||
| 					     (char *) &dev->statuschange);
 | ||
| 	  break;
 | ||
| 	}
 | ||
|     }
 | ||
| 
 | ||
|   rescan = 1;
 | ||
| 
 | ||
|   return GRUB_ERR_NONE;
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| attach_root_port (struct grub_usb_hub *hub, int portno,
 | ||
| 		  grub_usb_speed_t speed)
 | ||
| {
 | ||
|   grub_usb_device_t dev;
 | ||
|   grub_err_t err;
 | ||
| 
 | ||
|   grub_boot_time ("After detect_dev");
 | ||
| 
 | ||
|   /* Enable the port.  */
 | ||
|   err = hub->controller->dev->portstatus (hub->controller, portno, 1);
 | ||
|   if (err)
 | ||
|     return;
 | ||
|   hub->controller->dev->pending_reset = grub_get_time_ms () + 5000;
 | ||
|   npending++;
 | ||
| 
 | ||
|   grub_millisleep (10);
 | ||
| 
 | ||
|   grub_boot_time ("Port enabled");
 | ||
| 
 | ||
|   /* Enable the port and create a device.  */
 | ||
|   /* High speed device needs not transaction translation
 | ||
|      and full/low speed device cannot be connected to EHCI root hub
 | ||
|      and full/low speed device connected to OHCI/UHCI needs not
 | ||
|      transaction translation - e.g. hubport and hubaddr should be
 | ||
|      always none (zero) for any device connected to any root hub. */
 | ||
|   dev = grub_usb_hub_add_dev (hub->controller, speed, 0, 0);
 | ||
|   hub->controller->dev->pending_reset = 0;
 | ||
|   npending--;
 | ||
|   if (! dev)
 | ||
|     return;
 | ||
| 
 | ||
|   hub->devices[portno] = dev;
 | ||
| 
 | ||
|   /* If the device is a Hub, scan it for more devices.  */
 | ||
|   if (dev->descdev.class == 0x09)
 | ||
|     grub_usb_add_hub (dev);
 | ||
| 
 | ||
|   grub_boot_time ("Attached root port");
 | ||
| }
 | ||
| 
 | ||
| /* Iterate over all controllers found by the driver.  */
 | ||
| static int
 | ||
| grub_usb_controller_dev_register_iter (grub_usb_controller_t controller, void *data)
 | ||
| {
 | ||
|   grub_usb_controller_dev_t usb = data;
 | ||
|   struct grub_usb_hub *hub;
 | ||
| 
 | ||
|   controller->dev = usb;
 | ||
| 
 | ||
|   grub_boot_time ("Registering USB root hub");
 | ||
| 
 | ||
|   hub = grub_malloc (sizeof (*hub));
 | ||
|   if (!hub)
 | ||
|     return GRUB_USB_ERR_INTERNAL;
 | ||
| 
 | ||
|   hub->next = hubs;
 | ||
|   hubs = hub;
 | ||
|   hub->controller = grub_malloc (sizeof (*controller));
 | ||
|   if (!hub->controller)
 | ||
|     {
 | ||
|       grub_free (hub);
 | ||
|       return GRUB_USB_ERR_INTERNAL;
 | ||
|     }
 | ||
| 
 | ||
|   grub_memcpy (hub->controller, controller, sizeof (*controller));
 | ||
|   hub->dev = 0;
 | ||
| 
 | ||
|   /* Query the number of ports the root Hub has.  */
 | ||
|   hub->nports = controller->dev->hubports (controller);
 | ||
|   hub->devices = grub_zalloc (sizeof (hub->devices[0]) * hub->nports);
 | ||
|   hub->ports = grub_zalloc (sizeof (hub->ports[0]) * hub->nports);
 | ||
|   if (!hub->devices || !hub->ports)
 | ||
|     {
 | ||
|       grub_free (hub->devices);
 | ||
|       grub_free (hub->ports);
 | ||
|       grub_free (hub->controller);
 | ||
|       grub_free (hub);
 | ||
|       grub_print_error ();
 | ||
|       return 0;
 | ||
|     }
 | ||
| 
 | ||
|   return 0;
 | ||
| }
 | ||
| 
 | ||
| void
 | ||
| grub_usb_controller_dev_unregister (grub_usb_controller_dev_t usb)
 | ||
| {
 | ||
|   grub_usb_controller_dev_t *p, q;
 | ||
| 
 | ||
|   for (p = &grub_usb_list, q = *p; q; p = &(q->next), q = q->next)
 | ||
|     if (q == usb)
 | ||
|       {
 | ||
| 	*p = q->next;
 | ||
| 	break;
 | ||
|       }
 | ||
| }
 | ||
| 
 | ||
| void
 | ||
| grub_usb_controller_dev_register (grub_usb_controller_dev_t usb)
 | ||
| {
 | ||
|   int portno;
 | ||
|   int continue_waiting = 0;
 | ||
|   struct grub_usb_hub *hub;
 | ||
| 
 | ||
|   usb->next = grub_usb_list;
 | ||
|   grub_usb_list = usb;
 | ||
| 
 | ||
|   if (usb->iterate)
 | ||
|     usb->iterate (grub_usb_controller_dev_register_iter, usb);
 | ||
| 
 | ||
|   grub_boot_time ("waiting for stable power on USB root\n");
 | ||
| 
 | ||
|   while (1)
 | ||
|     {
 | ||
|       for (hub = hubs; hub; hub = hub->next)
 | ||
| 	if (hub->controller->dev == usb)
 | ||
| 	  {
 | ||
| 	    /* Wait for completion of insertion and stable power (USB spec.)
 | ||
| 	     * Should be at least 100ms, some devices requires more...
 | ||
| 	     * There is also another thing - some devices have worse contacts
 | ||
| 	     * and connected signal is unstable for some time - we should handle
 | ||
| 	     * it - but prevent deadlock in case when device is too faulty... */
 | ||
| 	    for (portno = 0; portno < hub->nports; portno++)
 | ||
| 	      {
 | ||
| 		grub_usb_speed_t speed;
 | ||
| 		int changed = 0;
 | ||
| 
 | ||
| 		speed = hub->controller->dev->detect_dev (hub->controller, portno,
 | ||
| 							  &changed);
 | ||
|       
 | ||
| 		if (hub->ports[portno].state == PORT_STATE_NORMAL
 | ||
| 		    && speed != GRUB_USB_SPEED_NONE)
 | ||
| 		  {
 | ||
| 		    hub->ports[portno].soft_limit_time = grub_get_time_ms () + 250;
 | ||
| 		    hub->ports[portno].hard_limit_time = hub->ports[portno].soft_limit_time + 1750;
 | ||
| 		    hub->ports[portno].state = PORT_STATE_WAITING_FOR_STABLE_POWER;
 | ||
| 		    grub_boot_time ("Scheduling stable power wait for port %p:%d",
 | ||
| 				    usb, portno);
 | ||
| 		    continue_waiting++;
 | ||
| 		    continue;
 | ||
| 		  }
 | ||
| 
 | ||
| 		if (hub->ports[portno].state == PORT_STATE_WAITING_FOR_STABLE_POWER
 | ||
| 		    && speed == GRUB_USB_SPEED_NONE)
 | ||
| 		  {
 | ||
| 		    hub->ports[portno].soft_limit_time = grub_get_time_ms () + 250;
 | ||
| 		    continue;
 | ||
| 		  }
 | ||
| 		if (hub->ports[portno].state == PORT_STATE_WAITING_FOR_STABLE_POWER
 | ||
| 		    && grub_get_time_ms () > hub->ports[portno].soft_limit_time)
 | ||
| 		  {
 | ||
| 		    hub->ports[portno].state = PORT_STATE_STABLE_POWER;
 | ||
| 		    grub_boot_time ("Got stable power wait for port %p:%d",
 | ||
| 				    usb, portno);
 | ||
| 		    continue_waiting--;
 | ||
| 		    continue;
 | ||
| 		  }
 | ||
| 		if (hub->ports[portno].state == PORT_STATE_WAITING_FOR_STABLE_POWER
 | ||
| 		    && grub_get_time_ms () > hub->ports[portno].hard_limit_time)
 | ||
| 		  {
 | ||
| 		    hub->ports[portno].state = PORT_STATE_FAILED_DEVICE;
 | ||
| 		    continue_waiting--;
 | ||
| 		    continue;
 | ||
| 		  }
 | ||
| 	      }
 | ||
| 	  }
 | ||
|       if (!continue_waiting)
 | ||
| 	break;
 | ||
|       grub_millisleep (1);
 | ||
|     }
 | ||
| 
 | ||
|   grub_boot_time ("After the stable power wait on USB root");
 | ||
| 
 | ||
|   for (hub = hubs; hub; hub = hub->next)
 | ||
|     if (hub->controller->dev == usb)
 | ||
|       for (portno = 0; portno < hub->nports; portno++)
 | ||
| 	if (hub->ports[portno].state == PORT_STATE_STABLE_POWER)
 | ||
| 	  {
 | ||
| 	    grub_usb_speed_t speed;
 | ||
| 	    int changed = 0;
 | ||
| 	    hub->ports[portno].state = PORT_STATE_NORMAL;
 | ||
| 	    speed = hub->controller->dev->detect_dev (hub->controller, portno, &changed);
 | ||
| 	    attach_root_port (hub, portno, speed);
 | ||
| 	  }
 | ||
| 
 | ||
|   grub_boot_time ("USB root hub registered");
 | ||
| }
 | ||
| 
 | ||
| static void detach_device (grub_usb_device_t dev);
 | ||
| 
 | ||
| static void
 | ||
| detach_device (grub_usb_device_t dev)
 | ||
| {
 | ||
|   unsigned i;
 | ||
|   int k;
 | ||
|   if (!dev)
 | ||
|     return;
 | ||
|   if (dev->descdev.class == GRUB_USB_CLASS_HUB)
 | ||
|     {
 | ||
|       if (dev->hub_transfer)
 | ||
| 	grub_usb_cancel_transfer (dev->hub_transfer);
 | ||
| 
 | ||
|       for (i = 0; i < dev->nports; i++)
 | ||
| 	detach_device (dev->children[i]);
 | ||
|       grub_free (dev->children);
 | ||
|     }
 | ||
|   for (i = 0; i < ARRAY_SIZE (dev->config); i++)
 | ||
|     if (dev->config[i].descconf)
 | ||
|       for (k = 0; k < dev->config[i].descconf->numif; k++)
 | ||
| 	{
 | ||
| 	  struct grub_usb_interface *inter = &dev->config[i].interf[k];
 | ||
| 	  if (inter && inter->detach_hook)
 | ||
| 	    inter->detach_hook (dev, i, k);
 | ||
| 	}
 | ||
|   grub_usb_devs[dev->addr] = 0;
 | ||
| }
 | ||
| 
 | ||
| static int
 | ||
| wait_power_nonroot_hub (grub_usb_device_t dev)
 | ||
| {
 | ||
|   grub_usb_err_t err;
 | ||
|   int continue_waiting = 0;
 | ||
|   unsigned i;
 | ||
|   
 | ||
|   for (i = 1; i <= dev->nports; i++)
 | ||
|     if (dev->ports[i - 1].state == PORT_STATE_WAITING_FOR_STABLE_POWER)
 | ||
|       {
 | ||
| 	grub_uint64_t tm;
 | ||
| 	grub_uint32_t current_status = 0;
 | ||
| 
 | ||
| 	/* Get the port status.  */
 | ||
| 	err = grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_IN
 | ||
| 					  | GRUB_USB_REQTYPE_CLASS
 | ||
| 					  | GRUB_USB_REQTYPE_TARGET_OTHER),
 | ||
| 				    GRUB_USB_REQ_GET_STATUS,
 | ||
| 				    0, i,
 | ||
| 				    sizeof (current_status),
 | ||
| 				    (char *) ¤t_status);
 | ||
| 	if (err)
 | ||
| 	  {
 | ||
| 	    dev->ports[i - 1].state = PORT_STATE_FAILED_DEVICE;
 | ||
| 	    continue;
 | ||
| 	  }
 | ||
| 	tm = grub_get_time_ms ();
 | ||
| 	if (!(current_status & GRUB_USB_HUB_STATUS_PORT_CONNECTED))
 | ||
| 	  dev->ports[i - 1].soft_limit_time = tm + 250;
 | ||
| 	if (tm >= dev->ports[i - 1].soft_limit_time)
 | ||
| 	  {
 | ||
| 	    if (dev->controller.dev->pending_reset)
 | ||
| 	      continue;
 | ||
| 	    /* Now do reset of port. */
 | ||
| 	    grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_OUT
 | ||
| 					| GRUB_USB_REQTYPE_CLASS
 | ||
| 					| GRUB_USB_REQTYPE_TARGET_OTHER),
 | ||
| 				  GRUB_USB_REQ_SET_FEATURE,
 | ||
| 				  GRUB_USB_HUB_FEATURE_PORT_RESET,
 | ||
| 				  i, 0, 0);
 | ||
| 	    dev->ports[i - 1].state = PORT_STATE_NORMAL;
 | ||
| 	    grub_boot_time ("Resetting port %p:%d", dev, i - 1);
 | ||
| 
 | ||
| 	    rescan = 1;
 | ||
| 	    /* We cannot reset more than one device at the same time !
 | ||
| 	     * Resetting more devices together results in very bad
 | ||
| 	     * situation: more than one device has default address 0
 | ||
| 	     * at the same time !!!
 | ||
| 	     * Additionaly, we cannot perform another reset
 | ||
| 	     * anywhere on the same OHCI controller until
 | ||
| 	     * we will finish addressing of reseted device ! */
 | ||
| 	    dev->controller.dev->pending_reset = grub_get_time_ms () + 5000;
 | ||
| 	    npending++;
 | ||
| 	    continue;
 | ||
| 	  }
 | ||
| 	if (tm >= dev->ports[i - 1].hard_limit_time)
 | ||
| 	  {
 | ||
| 	    dev->ports[i - 1].state = PORT_STATE_FAILED_DEVICE;
 | ||
| 	    continue;
 | ||
| 	  }
 | ||
| 	continue_waiting = 1;
 | ||
|       }
 | ||
|   return continue_waiting && dev->controller.dev->pending_reset == 0;
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| poll_nonroot_hub (grub_usb_device_t dev)
 | ||
| {
 | ||
|   grub_usb_err_t err;
 | ||
|   unsigned i;
 | ||
|   grub_uint32_t changed;
 | ||
|   grub_size_t actual, len;
 | ||
| 
 | ||
|   if (!dev->hub_transfer)
 | ||
|     return;
 | ||
| 
 | ||
|   err = grub_usb_check_transfer (dev->hub_transfer, &actual);
 | ||
| 
 | ||
|   if (err == GRUB_USB_ERR_WAIT)
 | ||
|     return;
 | ||
| 
 | ||
|   changed = dev->statuschange;
 | ||
| 
 | ||
|   len = dev->hub_endpoint->maxpacket;
 | ||
|   if (len > sizeof (dev->statuschange))
 | ||
|     len = sizeof (dev->statuschange);
 | ||
|   dev->hub_transfer
 | ||
|     = grub_usb_bulk_read_background (dev, dev->hub_endpoint, len,
 | ||
| 				     (char *) &dev->statuschange);
 | ||
| 
 | ||
|   if (err || actual == 0 || changed == 0)
 | ||
|     return;
 | ||
| 
 | ||
|   /* Iterate over the Hub ports.  */
 | ||
|   for (i = 1; i <= dev->nports; i++)
 | ||
|     {
 | ||
|       grub_uint32_t status;
 | ||
| 
 | ||
|       if (!(changed & (1 << i))
 | ||
| 	  || dev->ports[i - 1].state == PORT_STATE_WAITING_FOR_STABLE_POWER)
 | ||
| 	continue;
 | ||
| 
 | ||
|       /* Get the port status.  */
 | ||
|       err = grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_IN
 | ||
| 					| GRUB_USB_REQTYPE_CLASS
 | ||
| 					| GRUB_USB_REQTYPE_TARGET_OTHER),
 | ||
| 				  GRUB_USB_REQ_GET_STATUS,
 | ||
| 				  0, i, sizeof (status), (char *) &status);
 | ||
| 
 | ||
|       grub_dprintf ("usb", "dev = %p, i = %d, status = %08x\n",
 | ||
|                    dev, i, status);
 | ||
| 
 | ||
|       if (err)
 | ||
| 	continue;
 | ||
| 
 | ||
|       /* FIXME: properly handle these conditions.  */
 | ||
|       if (status & GRUB_USB_HUB_STATUS_C_PORT_ENABLED)
 | ||
| 	grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_OUT
 | ||
| 				    | GRUB_USB_REQTYPE_CLASS
 | ||
| 				    | GRUB_USB_REQTYPE_TARGET_OTHER),
 | ||
| 			      GRUB_USB_REQ_CLEAR_FEATURE,
 | ||
| 			      GRUB_USB_HUB_FEATURE_C_PORT_ENABLED, i, 0, 0);
 | ||
| 
 | ||
|       if (status & GRUB_USB_HUB_STATUS_C_PORT_SUSPEND)
 | ||
| 	grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_OUT
 | ||
| 				    | GRUB_USB_REQTYPE_CLASS
 | ||
| 				    | GRUB_USB_REQTYPE_TARGET_OTHER),
 | ||
| 			      GRUB_USB_REQ_CLEAR_FEATURE,
 | ||
| 			      GRUB_USB_HUB_FEATURE_C_PORT_SUSPEND, i, 0, 0);
 | ||
| 
 | ||
|       if (status & GRUB_USB_HUB_STATUS_C_PORT_OVERCURRENT)
 | ||
| 	grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_OUT
 | ||
| 				    | GRUB_USB_REQTYPE_CLASS
 | ||
| 				    | GRUB_USB_REQTYPE_TARGET_OTHER),
 | ||
| 			      GRUB_USB_REQ_CLEAR_FEATURE,
 | ||
| 			      GRUB_USB_HUB_FEATURE_C_PORT_OVERCURRENT, i, 0, 0);
 | ||
| 
 | ||
|       if (!dev->controller.dev->pending_reset &&
 | ||
|           (status & GRUB_USB_HUB_STATUS_C_PORT_CONNECTED))
 | ||
| 	{
 | ||
| 	  grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_OUT
 | ||
| 				      | GRUB_USB_REQTYPE_CLASS
 | ||
| 				      | GRUB_USB_REQTYPE_TARGET_OTHER),
 | ||
| 				GRUB_USB_REQ_CLEAR_FEATURE,
 | ||
| 				GRUB_USB_HUB_FEATURE_C_PORT_CONNECTED, i, 0, 0);
 | ||
| 
 | ||
| 	  detach_device (dev->children[i - 1]);
 | ||
| 	  dev->children[i - 1] = NULL;
 | ||
|       	    
 | ||
| 	  /* Connected and status of connection changed ? */
 | ||
| 	  if (status & GRUB_USB_HUB_STATUS_PORT_CONNECTED)
 | ||
| 	    {
 | ||
| 	      grub_boot_time ("Before the stable power wait portno=%d", i);
 | ||
| 	      /* A device is actually connected to this port. */
 | ||
| 	      /* Wait for completion of insertion and stable power (USB spec.)
 | ||
| 	       * Should be at least 100ms, some devices requires more...
 | ||
| 	       * There is also another thing - some devices have worse contacts
 | ||
| 	       * and connected signal is unstable for some time - we should handle
 | ||
| 	       * it - but prevent deadlock in case when device is too faulty... */
 | ||
| 	      dev->ports[i - 1].soft_limit_time = grub_get_time_ms () + 250;
 | ||
| 	      dev->ports[i - 1].hard_limit_time = dev->ports[i - 1].soft_limit_time + 1750;
 | ||
| 	      dev->ports[i - 1].state = PORT_STATE_WAITING_FOR_STABLE_POWER;
 | ||
| 	      grub_boot_time ("Scheduling stable power wait for port %p:%d",
 | ||
| 			      dev, i - 1);
 | ||
| 	      continue;
 | ||
| 	    }
 | ||
| 	}
 | ||
| 
 | ||
|       if (status & GRUB_USB_HUB_STATUS_C_PORT_RESET)
 | ||
| 	{
 | ||
| 	  grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_OUT
 | ||
| 				      | GRUB_USB_REQTYPE_CLASS
 | ||
| 				      | GRUB_USB_REQTYPE_TARGET_OTHER),
 | ||
| 				GRUB_USB_REQ_CLEAR_FEATURE,
 | ||
| 				GRUB_USB_HUB_FEATURE_C_PORT_RESET, i, 0, 0);
 | ||
| 
 | ||
| 	  grub_boot_time ("Port %d reset", i);
 | ||
| 
 | ||
| 	  if (status & GRUB_USB_HUB_STATUS_PORT_CONNECTED)
 | ||
| 	    {
 | ||
| 	      grub_usb_speed_t speed;
 | ||
| 	      grub_usb_device_t next_dev;
 | ||
| 	      int split_hubport = 0;
 | ||
| 	      int split_hubaddr = 0;
 | ||
| 
 | ||
| 	      /* Determine the device speed.  */
 | ||
| 	      if (status & GRUB_USB_HUB_STATUS_PORT_LOWSPEED)
 | ||
| 		speed = GRUB_USB_SPEED_LOW;
 | ||
| 	      else
 | ||
| 		{
 | ||
| 		  if (status & GRUB_USB_HUB_STATUS_PORT_HIGHSPEED)
 | ||
| 		    speed = GRUB_USB_SPEED_HIGH;
 | ||
| 		  else
 | ||
| 		    speed = GRUB_USB_SPEED_FULL;
 | ||
| 		}
 | ||
| 
 | ||
| 	      /* Wait a recovery time after reset, spec. says 10ms */
 | ||
| 	      grub_millisleep (10);
 | ||
| 
 | ||
|               /* Find correct values for SPLIT hubport and hubaddr */
 | ||
| 	      if (speed == GRUB_USB_SPEED_HIGH)
 | ||
| 	        {
 | ||
| 		  /* HIGH speed device needs not transaction translation */
 | ||
| 		  split_hubport = 0;
 | ||
| 		  split_hubaddr = 0;
 | ||
| 		}
 | ||
| 	      else
 | ||
| 	        /* FULL/LOW device needs hub port and hub address
 | ||
| 		   for transaction translation (if connected to EHCI) */
 | ||
| 	        if (dev->speed == GRUB_USB_SPEED_HIGH)
 | ||
| 	          {
 | ||
| 		    /* This port is the first FULL/LOW speed port
 | ||
| 		       in the chain from root hub. Attached device
 | ||
| 		       should use its port number and hub address */
 | ||
| 		    split_hubport = i;
 | ||
| 		    split_hubaddr = dev->addr;
 | ||
| 		  }
 | ||
| 	        else
 | ||
| 	          {
 | ||
| 		    /* This port is NOT the first FULL/LOW speed port
 | ||
| 		       in the chain from root hub. Attached device
 | ||
| 		       should use values inherited from some parent
 | ||
| 		       HIGH speed hub - if any. */
 | ||
| 		    split_hubport = dev->split_hubport;
 | ||
| 		    split_hubaddr = dev->split_hubaddr;
 | ||
| 		  }
 | ||
| 		
 | ||
| 	      /* Add the device and assign a device address to it.  */
 | ||
| 	      next_dev = grub_usb_hub_add_dev (&dev->controller, speed,
 | ||
| 					       split_hubport, split_hubaddr);
 | ||
| 	      if (dev->controller.dev->pending_reset)
 | ||
| 		{
 | ||
| 		  dev->controller.dev->pending_reset = 0;
 | ||
| 		  npending--;
 | ||
| 		}
 | ||
| 	      if (! next_dev)
 | ||
| 		continue;
 | ||
| 
 | ||
| 	      dev->children[i - 1] = next_dev;
 | ||
| 
 | ||
| 	      /* If the device is a Hub, scan it for more devices.  */
 | ||
| 	      if (next_dev->descdev.class == 0x09)
 | ||
| 		grub_usb_add_hub (next_dev);
 | ||
| 	    }
 | ||
| 	}
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| void
 | ||
| grub_usb_poll_devices (int wait_for_completion)
 | ||
| {
 | ||
|   struct grub_usb_hub *hub;
 | ||
|   int i;
 | ||
| 
 | ||
|   for (hub = hubs; hub; hub = hub->next)
 | ||
|     {
 | ||
|       /* Do we have to recheck number of ports?  */
 | ||
|       /* No, it should be never changed, it should be constant. */
 | ||
|       for (i = 0; i < hub->nports; i++)
 | ||
| 	{
 | ||
| 	  grub_usb_speed_t speed = GRUB_USB_SPEED_NONE;
 | ||
| 	  int changed = 0;
 | ||
| 
 | ||
|           if (hub->controller->dev->pending_reset)
 | ||
|             {
 | ||
|               /* Check for possible timeout */
 | ||
|               if (grub_get_time_ms () > hub->controller->dev->pending_reset)
 | ||
|                 {
 | ||
|                   /* Something went wrong, reset device was not
 | ||
|                    * addressed properly, timeout happened */
 | ||
| 	          hub->controller->dev->pending_reset = 0;
 | ||
| 		  npending--;
 | ||
|                 }
 | ||
|             }
 | ||
|           if (!hub->controller->dev->pending_reset)
 | ||
| 	    speed = hub->controller->dev->detect_dev (hub->controller,
 | ||
| 						      i, &changed);
 | ||
| 
 | ||
| 	  if (changed)
 | ||
| 	    {
 | ||
| 	      detach_device (hub->devices[i]);
 | ||
| 	      hub->devices[i] = NULL;
 | ||
| 	      if (speed != GRUB_USB_SPEED_NONE)
 | ||
|                 attach_root_port (hub, i, speed);
 | ||
| 	    }
 | ||
| 	}
 | ||
|     }
 | ||
| 
 | ||
|   while (1)
 | ||
|     {
 | ||
|       rescan = 0;
 | ||
|       
 | ||
|       /* We should check changes of non-root hubs too. */
 | ||
|       for (i = 0; i < GRUB_USBHUB_MAX_DEVICES; i++)
 | ||
| 	{
 | ||
| 	  grub_usb_device_t dev = grub_usb_devs[i];
 | ||
| 	  
 | ||
| 	  if (dev && dev->descdev.class == 0x09)
 | ||
| 	    poll_nonroot_hub (dev);
 | ||
| 	}
 | ||
| 
 | ||
|       while (1)
 | ||
| 	{
 | ||
| 	  int continue_waiting = 0;
 | ||
| 	  for (i = 0; i < GRUB_USBHUB_MAX_DEVICES; i++)
 | ||
| 	    {
 | ||
| 	      grub_usb_device_t dev = grub_usb_devs[i];
 | ||
| 	    
 | ||
| 	      if (dev && dev->descdev.class == 0x09)
 | ||
| 		continue_waiting = continue_waiting || wait_power_nonroot_hub (dev);
 | ||
| 	    }
 | ||
| 	  if (!continue_waiting)
 | ||
| 	    break;
 | ||
| 	  grub_millisleep (1);
 | ||
| 	}
 | ||
| 
 | ||
|       if (!(rescan || (npending && wait_for_completion)))
 | ||
| 	break;
 | ||
|       grub_millisleep (25);
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| int
 | ||
| grub_usb_iterate (grub_usb_iterate_hook_t hook, void *hook_data)
 | ||
| {
 | ||
|   int i;
 | ||
| 
 | ||
|   for (i = 0; i < GRUB_USBHUB_MAX_DEVICES; i++)
 | ||
|     {
 | ||
|       if (grub_usb_devs[i])
 | ||
| 	{
 | ||
| 	  if (hook (grub_usb_devs[i], hook_data))
 | ||
| 	      return 1;
 | ||
| 	}
 | ||
|     }
 | ||
| 
 | ||
|   return 0;
 | ||
| }
 |