diff options
| author | Tom Rini <trini@ti.com> | 2013-08-27 09:49:43 -0400 | 
|---|---|---|
| committer | Tom Rini <trini@ti.com> | 2013-08-27 09:49:43 -0400 | 
| commit | 529c0d9b8cdfae232e4ed082fd8cde5416b266b7 (patch) | |
| tree | ea7b948c6faafb4ad9d754eca568a2ae2a0100fc /common | |
| parent | 880a41273265233004516351189aad126f964c86 (diff) | |
| parent | c95e2b9eaeadc0f985030ffa0638278acc2d8727 (diff) | |
| download | olio-uboot-2014.01-529c0d9b8cdfae232e4ed082fd8cde5416b266b7.tar.xz olio-uboot-2014.01-529c0d9b8cdfae232e4ed082fd8cde5416b266b7.zip | |
Merge branch 'master' of git://git.denx.de/u-boot-usb
Diffstat (limited to 'common')
| -rw-r--r-- | common/usb.c | 87 | ||||
| -rw-r--r-- | common/usb_hub.c | 26 | ||||
| -rw-r--r-- | common/usb_kbd.c | 15 | 
3 files changed, 102 insertions, 26 deletions
| diff --git a/common/usb.c b/common/usb.c index f740e5ec2..c97f522be 100644 --- a/common/usb.c +++ b/common/usb.c @@ -323,6 +323,7 @@ static int usb_set_maxpacket(struct usb_device *dev)  /*******************************************************************************   * Parse the config, located in buffer, and fills the dev->config structure.   * Note that all little/big endian swapping are done automatically. + * (wTotalLength has already been swapped and sanitized when it was read.)   */  static int usb_parse_config(struct usb_device *dev,  			unsigned char *buffer, int cfgno) @@ -343,24 +344,43 @@ static int usb_parse_config(struct usb_device *dev,  			head->bDescriptorType);  		return -1;  	} -	memcpy(&dev->config, buffer, buffer[0]); -	le16_to_cpus(&(dev->config.desc.wTotalLength)); +	if (head->bLength != USB_DT_CONFIG_SIZE) { +		printf("ERROR: Invalid USB CFG length (%d)\n", head->bLength); +		return -1; +	} +	memcpy(&dev->config, head, USB_DT_CONFIG_SIZE);  	dev->config.no_of_if = 0;  	index = dev->config.desc.bLength;  	/* Ok the first entry must be a configuration entry,  	 * now process the others */  	head = (struct usb_descriptor_header *) &buffer[index]; -	while (index + 1 < dev->config.desc.wTotalLength) { +	while (index + 1 < dev->config.desc.wTotalLength && head->bLength) {  		switch (head->bDescriptorType) {  		case USB_DT_INTERFACE: +			if (head->bLength != USB_DT_INTERFACE_SIZE) { +				printf("ERROR: Invalid USB IF length (%d)\n", +					head->bLength); +				break; +			} +			if (index + USB_DT_INTERFACE_SIZE > +			    dev->config.desc.wTotalLength) { +				puts("USB IF descriptor overflowed buffer!\n"); +				break; +			}  			if (((struct usb_interface_descriptor *) \ -			     &buffer[index])->bInterfaceNumber != curr_if_num) { +			     head)->bInterfaceNumber != curr_if_num) {  				/* this is a new interface, copy new desc */  				ifno = dev->config.no_of_if; +				if (ifno >= USB_MAXINTERFACES) { +					puts("Too many USB interfaces!\n"); +					/* try to go on with what we have */ +					return 1; +				}  				if_desc = &dev->config.if_desc[ifno];  				dev->config.no_of_if++; -				memcpy(if_desc,	&buffer[index], buffer[index]); +				memcpy(if_desc, head, +					USB_DT_INTERFACE_SIZE);  				if_desc->no_of_ep = 0;  				if_desc->num_altsetting = 1;  				curr_if_num = @@ -374,12 +394,31 @@ static int usb_parse_config(struct usb_device *dev,  			}  			break;  		case USB_DT_ENDPOINT: +			if (head->bLength != USB_DT_ENDPOINT_SIZE) { +				printf("ERROR: Invalid USB EP length (%d)\n", +					head->bLength); +				break; +			} +			if (index + USB_DT_ENDPOINT_SIZE > +			    dev->config.desc.wTotalLength) { +				puts("USB EP descriptor overflowed buffer!\n"); +				break; +			} +			if (ifno < 0) { +				puts("Endpoint descriptor out of order!\n"); +				break; +			}  			epno = dev->config.if_desc[ifno].no_of_ep;  			if_desc = &dev->config.if_desc[ifno]; +			if (epno > USB_MAXENDPOINTS) { +				printf("Interface %d has too many endpoints!\n", +					if_desc->desc.bInterfaceNumber); +				return 1; +			}  			/* found an endpoint */  			if_desc->no_of_ep++; -			memcpy(&if_desc->ep_desc[epno], -				&buffer[index], buffer[index]); +			memcpy(&if_desc->ep_desc[epno], head, +				USB_DT_ENDPOINT_SIZE);  			ep_wMaxPacketSize = get_unaligned(&dev->config.\  							if_desc[ifno].\  							ep_desc[epno].\ @@ -392,9 +431,23 @@ static int usb_parse_config(struct usb_device *dev,  			debug("if %d, ep %d\n", ifno, epno);  			break;  		case USB_DT_SS_ENDPOINT_COMP: +			if (head->bLength != USB_DT_SS_EP_COMP_SIZE) { +				printf("ERROR: Invalid USB EPC length (%d)\n", +					head->bLength); +				break; +			} +			if (index + USB_DT_SS_EP_COMP_SIZE > +			    dev->config.desc.wTotalLength) { +				puts("USB EPC descriptor overflowed buffer!\n"); +				break; +			} +			if (ifno < 0 || epno < 0) { +				puts("EPC descriptor out of order!\n"); +				break; +			}  			if_desc = &dev->config.if_desc[ifno]; -			memcpy(&if_desc->ss_ep_comp_desc[epno], -				&buffer[index], buffer[index]); +			memcpy(&if_desc->ss_ep_comp_desc[epno], head, +				USB_DT_SS_EP_COMP_SIZE);  			break;  		default:  			if (head->bLength == 0) @@ -473,7 +526,7 @@ int usb_get_configuration_no(struct usb_device *dev,  			     unsigned char *buffer, int cfgno)  {  	int result; -	unsigned int tmp; +	unsigned int length;  	struct usb_config_descriptor *config;  	config = (struct usb_config_descriptor *)&buffer[0]; @@ -487,16 +540,18 @@ int usb_get_configuration_no(struct usb_device *dev,  				"(expected %i, got %i)\n", 9, result);  		return -1;  	} -	tmp = le16_to_cpu(config->wTotalLength); +	length = le16_to_cpu(config->wTotalLength); -	if (tmp > USB_BUFSIZ) { -		printf("usb_get_configuration_no: failed to get " \ -		       "descriptor - too long: %d\n", tmp); +	if (length > USB_BUFSIZ) { +		printf("%s: failed to get descriptor - too long: %d\n", +			__func__, length);  		return -1;  	} -	result = usb_get_descriptor(dev, USB_DT_CONFIG, cfgno, buffer, tmp); -	debug("get_conf_no %d Result %d, wLength %d\n", cfgno, result, tmp); +	result = usb_get_descriptor(dev, USB_DT_CONFIG, cfgno, buffer, length); +	debug("get_conf_no %d Result %d, wLength %d\n", cfgno, result, length); +	config->wTotalLength = length; /* validated, with CPU byte order */ +  	return result;  } diff --git a/common/usb_hub.c b/common/usb_hub.c index a11b401e6..ffac0e743 100644 --- a/common/usb_hub.c +++ b/common/usb_hub.c @@ -44,6 +44,10 @@  static struct usb_hub_device hub_dev[USB_MAX_HUB];  static int usb_hub_index; +__weak void usb_hub_reset_devices(int port) +{ +	return; +}  static int usb_get_hub_descriptor(struct usb_device *dev, void *data, int size)  { @@ -302,7 +306,7 @@ void usb_hub_port_connect_change(struct usb_device *dev, int port)  static int usb_hub_configure(struct usb_device *dev)  { -	int i; +	int i, length;  	ALLOC_CACHE_ALIGN_BUFFER(unsigned char, buffer, USB_BUFSIZ);  	unsigned char *bitmap;  	short hubCharacteristics; @@ -323,20 +327,14 @@ static int usb_hub_configure(struct usb_device *dev)  	}  	descriptor = (struct usb_hub_descriptor *)buffer; -	/* silence compiler warning if USB_BUFSIZ is > 256 [= sizeof(char)] */ -	i = descriptor->bLength; -	if (i > USB_BUFSIZ) { -		debug("usb_hub_configure: failed to get hub " \ -		      "descriptor - too long: %d\n", descriptor->bLength); -		return -1; -	} +	length = min(descriptor->bLength, sizeof(struct usb_hub_descriptor)); -	if (usb_get_hub_descriptor(dev, buffer, descriptor->bLength) < 0) { +	if (usb_get_hub_descriptor(dev, buffer, length) < 0) {  		debug("usb_hub_configure: failed to get hub " \  		      "descriptor 2nd giving up %lX\n", dev->status);  		return -1;  	} -	memcpy((unsigned char *)&hub->desc, buffer, descriptor->bLength); +	memcpy((unsigned char *)&hub->desc, buffer, length);  	/* adjust 16bit values */  	put_unaligned(le16_to_cpu(get_unaligned(  			&descriptor->wHubCharacteristics)), @@ -426,6 +424,14 @@ static int usb_hub_configure(struct usb_device *dev)  	      "" : "no ");  	usb_hub_power_on(hub); +	/* +	 * Reset any devices that may be in a bad state when applying +	 * the power.  This is a __weak function.  Resetting of the devices +	 * should occur in the board file of the device. +	 */ +	for (i = 0; i < dev->maxchild; i++) +		usb_hub_reset_devices(i + 1); +  	for (i = 0; i < dev->maxchild; i++) {  		ALLOC_CACHE_ALIGN_BUFFER(struct usb_port_status, portsts, 1);  		unsigned short portstatus, portchange; diff --git a/common/usb_kbd.c b/common/usb_kbd.c index 2ca3767ec..1ad67caf1 100644 --- a/common/usb_kbd.c +++ b/common/usb_kbd.c @@ -104,6 +104,11 @@ struct usb_kbd_pdata {  	uint8_t		flags;  }; +extern int __maybe_unused net_busy_flag; + +/* The period of time between two calls of usb_kbd_testc(). */ +static unsigned long __maybe_unused kbd_testc_tms; +  /* Generic keyboard event polling. */  void usb_kbd_generic_poll(void)  { @@ -349,6 +354,16 @@ static int usb_kbd_testc(void)  	struct usb_device *usb_kbd_dev;  	struct usb_kbd_pdata *data; +#ifdef CONFIG_CMD_NET +	/* +	 * If net_busy_flag is 1, NET transfer is running, +	 * then we check key-pressed every second (first check may be +	 * less than 1 second) to improve TFTP booting performance. +	 */ +	if (net_busy_flag && (get_timer(kbd_testc_tms) < CONFIG_SYS_HZ)) +		return 0; +	kbd_testc_tms = get_timer(0); +#endif  	dev = stdio_get_by_name(DEVNAME);  	usb_kbd_dev = (struct usb_device *)dev->priv;  	data = usb_kbd_dev->privptr; |