diff options
| author | Thomas Gleixner <tglx@linutronix.de> | 2010-12-09 18:17:25 +0100 | 
|---|---|---|
| committer | Thomas Gleixner <tglx@linutronix.de> | 2010-12-09 18:17:25 +0100 | 
| commit | d834a9dcecae834cd6b2bc5e50e1907738d9cf6a (patch) | |
| tree | 0589d753465d3fe359ba451ba6cb7798df03aaa2 /drivers/usb/core | |
| parent | a38c5380ef9f088be9f49b6e4c5d80af8b1b5cd4 (diff) | |
| parent | f658bcfb2607bf0808966a69cf74135ce98e5c2d (diff) | |
Merge branch 'x86/amd-nb' into x86/apic-cleanups
Reason: apic cleanup series depends on x86/apic, x86/amd-nb x86/platform
Conflicts:
	arch/x86/include/asm/io_apic.h
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Diffstat (limited to 'drivers/usb/core')
| -rw-r--r-- | drivers/usb/core/Makefile | 21 | ||||
| -rw-r--r-- | drivers/usb/core/devices.c | 11 | ||||
| -rw-r--r-- | drivers/usb/core/devio.c | 7 | ||||
| -rw-r--r-- | drivers/usb/core/driver.c | 2 | ||||
| -rw-r--r-- | drivers/usb/core/endpoint.c | 2 | ||||
| -rw-r--r-- | drivers/usb/core/file.c | 1 | ||||
| -rw-r--r-- | drivers/usb/core/hcd-pci.c | 4 | ||||
| -rw-r--r-- | drivers/usb/core/hcd.c | 19 | ||||
| -rw-r--r-- | drivers/usb/core/hub.c | 41 | ||||
| -rw-r--r-- | drivers/usb/core/inode.c | 9 | ||||
| -rw-r--r-- | drivers/usb/core/message.c | 14 | ||||
| -rw-r--r-- | drivers/usb/core/urb.c | 5 | 
12 files changed, 77 insertions, 59 deletions
| diff --git a/drivers/usb/core/Makefile b/drivers/usb/core/Makefile index ec16e602990..507a4e1b636 100644 --- a/drivers/usb/core/Makefile +++ b/drivers/usb/core/Makefile @@ -2,20 +2,13 @@  # Makefile for USB Core files and filesystem  # -usbcore-objs	:= usb.o hub.o hcd.o urb.o message.o driver.o \ -			config.o file.o buffer.o sysfs.o endpoint.o \ -			devio.o notify.o generic.o quirks.o devices.o +ccflags-$(CONFIG_USB_DEBUG) := -DDEBUG -ifeq ($(CONFIG_PCI),y) -	usbcore-objs	+= hcd-pci.o -endif +usbcore-y := usb.o hub.o hcd.o urb.o message.o driver.o +usbcore-y += config.o file.o buffer.o sysfs.o endpoint.o +usbcore-y += devio.o notify.o generic.o quirks.o devices.o -ifeq ($(CONFIG_USB_DEVICEFS),y) -	usbcore-objs	+= inode.o -endif +usbcore-$(CONFIG_PCI)		+= hcd-pci.o +usbcore-$(CONFIG_USB_DEVICEFS)	+= inode.o -obj-$(CONFIG_USB)	+= usbcore.o - -ifeq ($(CONFIG_USB_DEBUG),y) -EXTRA_CFLAGS += -DDEBUG -endif +obj-$(CONFIG_USB)		+= usbcore.o diff --git a/drivers/usb/core/devices.c b/drivers/usb/core/devices.c index 3449742c00e..ddb4dc98092 100644 --- a/drivers/usb/core/devices.c +++ b/drivers/usb/core/devices.c @@ -66,8 +66,8 @@  #define ALLOW_SERIAL_NUMBER  static const char *format_topo = -/* T:  Bus=dd Lev=dd Prnt=dd Port=dd Cnt=dd Dev#=ddd Spd=ddd MxCh=dd */ -"\nT:  Bus=%2.2d Lev=%2.2d Prnt=%2.2d Port=%2.2d Cnt=%2.2d Dev#=%3d Spd=%3s MxCh=%2d\n"; +/* T:  Bus=dd Lev=dd Prnt=dd Port=dd Cnt=dd Dev#=ddd Spd=dddd MxCh=dd */ +"\nT:  Bus=%2.2d Lev=%2.2d Prnt=%2.2d Port=%2.2d Cnt=%2.2d Dev#=%3d Spd=%-4s MxCh=%2d\n";  static const char *format_string_manufacturer =  /* S:  Manufacturer=xxxx */ @@ -520,11 +520,14 @@ static ssize_t usb_device_dump(char __user **buffer, size_t *nbytes,  		speed = "1.5"; break;  	case USB_SPEED_UNKNOWN:		/* usb 1.1 root hub code */  	case USB_SPEED_FULL: -		speed = "12 "; break; +		speed = "12"; break; +	case USB_SPEED_WIRELESS:	/* Wireless has no real fixed speed */  	case USB_SPEED_HIGH:  		speed = "480"; break; +	case USB_SPEED_SUPER: +		speed = "5000"; break;  	default: -		speed = "?? "; +		speed = "??";  	}  	data_end = pages_start + sprintf(pages_start, format_topo,  			bus->busnum, level, parent_devnum, diff --git a/drivers/usb/core/devio.c b/drivers/usb/core/devio.c index f1aaff6202a..045bb4b823e 100644 --- a/drivers/usb/core/devio.c +++ b/drivers/usb/core/devio.c @@ -965,10 +965,11 @@ static int proc_getdriver(struct dev_state *ps, void __user *arg)  static int proc_connectinfo(struct dev_state *ps, void __user *arg)  { -	struct usbdevfs_connectinfo ci; +	struct usbdevfs_connectinfo ci = { +		.devnum = ps->dev->devnum, +		.slow = ps->dev->speed == USB_SPEED_LOW +	}; -	ci.devnum = ps->dev->devnum; -	ci.slow = ps->dev->speed == USB_SPEED_LOW;  	if (copy_to_user(arg, &ci, sizeof(ci)))  		return -EFAULT;  	return 0; diff --git a/drivers/usb/core/driver.c b/drivers/usb/core/driver.c index d7a4401ef01..c0e60fbcb04 100644 --- a/drivers/usb/core/driver.c +++ b/drivers/usb/core/driver.c @@ -1337,7 +1337,7 @@ int usb_resume(struct device *dev, pm_message_t msg)  	/* Avoid PM error messages for devices disconnected while suspended  	 * as we'll display regular disconnect messages just a bit later.  	 */ -	if (status == -ENODEV) +	if (status == -ENODEV || status == -ESHUTDOWN)  		status = 0;  	return status;  } diff --git a/drivers/usb/core/endpoint.c b/drivers/usb/core/endpoint.c index 3788e738e26..9da25056302 100644 --- a/drivers/usb/core/endpoint.c +++ b/drivers/usb/core/endpoint.c @@ -202,7 +202,7 @@ int usb_create_ep_devs(struct device *parent,  	return retval;  error_register: -	kfree(ep_dev); +	put_device(&ep_dev->dev);  exit:  	return retval;  } diff --git a/drivers/usb/core/file.c b/drivers/usb/core/file.c index 1e6ccef2cf0..9fe34fb78ef 100644 --- a/drivers/usb/core/file.c +++ b/drivers/usb/core/file.c @@ -59,6 +59,7 @@ static int usb_open(struct inode * inode, struct file * file)  static const struct file_operations usb_fops = {  	.owner =	THIS_MODULE,  	.open =		usb_open, +	.llseek =	noop_llseek,  };  static struct usb_class { diff --git a/drivers/usb/core/hcd-pci.c b/drivers/usb/core/hcd-pci.c index c3f98543caa..3799573bd38 100644 --- a/drivers/usb/core/hcd-pci.c +++ b/drivers/usb/core/hcd-pci.c @@ -329,8 +329,10 @@ void usb_hcd_pci_shutdown(struct pci_dev *dev)  		return;  	if (test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags) && -			hcd->driver->shutdown) +			hcd->driver->shutdown) {  		hcd->driver->shutdown(hcd); +		pci_disable_device(dev); +	}  }  EXPORT_SYMBOL_GPL(usb_hcd_pci_shutdown); diff --git a/drivers/usb/core/hcd.c b/drivers/usb/core/hcd.c index 5cca00a6d09..61800f77dac 100644 --- a/drivers/usb/core/hcd.c +++ b/drivers/usb/core/hcd.c @@ -1263,10 +1263,8 @@ static void hcd_free_coherent(struct usb_bus *bus, dma_addr_t *dma_handle,  	*dma_handle = 0;  } -static void unmap_urb_for_dma(struct usb_hcd *hcd, struct urb *urb) +void unmap_urb_setup_for_dma(struct usb_hcd *hcd, struct urb *urb)  { -	enum dma_data_direction dir; -  	if (urb->transfer_flags & URB_SETUP_MAP_SINGLE)  		dma_unmap_single(hcd->self.controller,  				urb->setup_dma, @@ -1279,6 +1277,17 @@ static void unmap_urb_for_dma(struct usb_hcd *hcd, struct urb *urb)  				sizeof(struct usb_ctrlrequest),  				DMA_TO_DEVICE); +	/* Make it safe to call this routine more than once */ +	urb->transfer_flags &= ~(URB_SETUP_MAP_SINGLE | URB_SETUP_MAP_LOCAL); +} +EXPORT_SYMBOL_GPL(unmap_urb_setup_for_dma); + +void unmap_urb_for_dma(struct usb_hcd *hcd, struct urb *urb) +{ +	enum dma_data_direction dir; + +	unmap_urb_setup_for_dma(hcd, urb); +  	dir = usb_urb_dir_in(urb) ? DMA_FROM_DEVICE : DMA_TO_DEVICE;  	if (urb->transfer_flags & URB_DMA_MAP_SG)  		dma_unmap_sg(hcd->self.controller, @@ -1303,10 +1312,10 @@ static void unmap_urb_for_dma(struct usb_hcd *hcd, struct urb *urb)  				dir);  	/* Make it safe to call this routine more than once */ -	urb->transfer_flags &= ~(URB_SETUP_MAP_SINGLE | URB_SETUP_MAP_LOCAL | -			URB_DMA_MAP_SG | URB_DMA_MAP_PAGE | +	urb->transfer_flags &= ~(URB_DMA_MAP_SG | URB_DMA_MAP_PAGE |  			URB_DMA_MAP_SINGLE | URB_MAP_LOCAL);  } +EXPORT_SYMBOL_GPL(unmap_urb_for_dma);  static int map_urb_for_dma(struct usb_hcd *hcd, struct urb *urb,  			   gfp_t mem_flags) diff --git a/drivers/usb/core/hub.c b/drivers/usb/core/hub.c index 84c1897188d..27115b45edc 100644 --- a/drivers/usb/core/hub.c +++ b/drivers/usb/core/hub.c @@ -758,6 +758,9 @@ static void hub_activate(struct usb_hub *hub, enum hub_activation_type type)  				clear_port_feature(hdev, port1,  						   USB_PORT_FEAT_ENABLE);  				portstatus &= ~USB_PORT_STAT_ENABLE; +			} else { +				/* Pretend that power was lost for USB3 devs */ +				portstatus &= ~USB_PORT_STAT_ENABLE;  			}  		} @@ -2594,16 +2597,14 @@ static int hub_set_address(struct usb_device *udev, int devnum)  		return 0;  	if (udev->state != USB_STATE_DEFAULT)  		return -EINVAL; -	if (hcd->driver->address_device) { +	if (hcd->driver->address_device)  		retval = hcd->driver->address_device(hcd, udev); -	} else { +	else  		retval = usb_control_msg(udev, usb_sndaddr0pipe(),  				USB_REQ_SET_ADDRESS, 0, devnum, 0,  				NULL, 0, USB_CTRL_SET_TIMEOUT); -		if (retval == 0) -			update_address(udev, devnum); -	}  	if (retval == 0) { +		update_address(udev, devnum);  		/* Device now using proper address. */  		usb_set_device_state(udev, USB_STATE_ADDRESS);  		usb_ep0_reinit(udev); @@ -2860,13 +2861,16 @@ hub_port_init (struct usb_hub *hub, struct usb_device *udev, int port1,  	else  		i = udev->descriptor.bMaxPacketSize0;  	if (le16_to_cpu(udev->ep0.desc.wMaxPacketSize) != i) { -		if (udev->speed != USB_SPEED_FULL || +		if (udev->speed == USB_SPEED_LOW ||  				!(i == 8 || i == 16 || i == 32 || i == 64)) { -			dev_err(&udev->dev, "ep0 maxpacket = %d\n", i); +			dev_err(&udev->dev, "Invalid ep0 maxpacket: %d\n", i);  			retval = -EMSGSIZE;  			goto fail;  		} -		dev_dbg(&udev->dev, "ep0 maxpacket = %d\n", i); +		if (udev->speed == USB_SPEED_FULL) +			dev_dbg(&udev->dev, "ep0 maxpacket = %d\n", i); +		else +			dev_warn(&udev->dev, "Using ep0 maxpacket: %d\n", i);  		udev->ep0.desc.wMaxPacketSize = cpu_to_le16(i);  		usb_ep0_reinit(udev);  	} @@ -3097,16 +3101,17 @@ static void hub_port_connect_change(struct usb_hub *hub, int port1,  			udev->speed = USB_SPEED_UNKNOWN;  		/* -		 * xHCI needs to issue an address device command later -		 * in the hub_port_init sequence for SS/HS/FS/LS devices. +		 * Set the address. +		 * Note xHCI needs to issue an address device command later +		 * in the hub_port_init sequence for SS/HS/FS/LS devices, +		 * and xHC will assign an address to the device. But use +		 * kernel assigned address here, to avoid any address conflict +		 * issue.  		 */ -		if (!(hcd->driver->flags & HCD_USB3)) { -			/* set the address */ -			choose_address(udev); -			if (udev->devnum <= 0) { -				status = -ENOTCONN;	/* Don't retry */ -				goto loop; -			} +		choose_address(udev); +		if (udev->devnum <= 0) { +			status = -ENOTCONN;	/* Don't retry */ +			goto loop;  		}  		/* reset (non-USB 3.0 devices) and get descriptor */ @@ -3629,7 +3634,7 @@ static int usb_reset_and_verify_device(struct usb_device *udev)  	}  	if (!parent_hdev) { -		/* this requires hcd-specific logic; see OHCI hc_restart() */ +		/* this requires hcd-specific logic; see ohci_restart() */  		dev_dbg(&udev->dev, "%s for root hub!\n", __func__);  		return -EISDIR;  	} diff --git a/drivers/usb/core/inode.c b/drivers/usb/core/inode.c index 095fa536669..9819a4cc3b2 100644 --- a/drivers/usb/core/inode.c +++ b/drivers/usb/core/inode.c @@ -276,6 +276,7 @@ static struct inode *usbfs_get_inode (struct super_block *sb, int mode, dev_t de  	struct inode *inode = new_inode(sb);  	if (inode) { +		inode->i_ino = get_next_ino();  		inode->i_mode = mode;  		inode->i_uid = current_fsuid();  		inode->i_gid = current_fsgid(); @@ -573,16 +574,16 @@ static void fs_remove_file (struct dentry *dentry)  /* --------------------------------------------------------------------- */ -static int usb_get_sb(struct file_system_type *fs_type, -	int flags, const char *dev_name, void *data, struct vfsmount *mnt) +static struct dentry *usb_mount(struct file_system_type *fs_type, +	int flags, const char *dev_name, void *data)  { -	return get_sb_single(fs_type, flags, data, usbfs_fill_super, mnt); +	return mount_single(fs_type, flags, data, usbfs_fill_super);  }  static struct file_system_type usb_fs_type = {  	.owner =	THIS_MODULE,  	.name =		"usbfs", -	.get_sb =	usb_get_sb, +	.mount =	usb_mount,  	.kill_sb =	kill_litter_super,  }; diff --git a/drivers/usb/core/message.c b/drivers/usb/core/message.c index 9f0ce7de0e3..d6e3e410477 100644 --- a/drivers/usb/core/message.c +++ b/drivers/usb/core/message.c @@ -1140,13 +1140,6 @@ void usb_disable_device(struct usb_device *dev, int skip_ep0)  {  	int i; -	dev_dbg(&dev->dev, "%s nuking %s URBs\n", __func__, -		skip_ep0 ? "non-ep0" : "all"); -	for (i = skip_ep0; i < 16; ++i) { -		usb_disable_endpoint(dev, i, true); -		usb_disable_endpoint(dev, i + USB_DIR_IN, true); -	} -  	/* getting rid of interfaces will disconnect  	 * any drivers bound to them (a key side effect)  	 */ @@ -1176,6 +1169,13 @@ void usb_disable_device(struct usb_device *dev, int skip_ep0)  		if (dev->state == USB_STATE_CONFIGURED)  			usb_set_device_state(dev, USB_STATE_ADDRESS);  	} + +	dev_dbg(&dev->dev, "%s nuking %s URBs\n", __func__, +		skip_ep0 ? "non-ep0" : "all"); +	for (i = skip_ep0; i < 16; ++i) { +		usb_disable_endpoint(dev, i, true); +		usb_disable_endpoint(dev, i + USB_DIR_IN, true); +	}  }  /** diff --git a/drivers/usb/core/urb.c b/drivers/usb/core/urb.c index 419e6b34e2f..c14fc082864 100644 --- a/drivers/usb/core/urb.c +++ b/drivers/usb/core/urb.c @@ -401,8 +401,11 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags)  	};  	/* Check that the pipe's type matches the endpoint's type */ -	if (usb_pipetype(urb->pipe) != pipetypes[xfertype]) +	if (usb_pipetype(urb->pipe) != pipetypes[xfertype]) { +		dev_err(&dev->dev, "BOGUS urb xfer, pipe %x != type %x\n", +			usb_pipetype(urb->pipe), pipetypes[xfertype]);  		return -EPIPE;		/* The most suitable error code :-) */ +	}  	/* enforce simple/standard policy */  	allowed = (URB_NO_TRANSFER_DMA_MAP | URB_NO_INTERRUPT | URB_DIR_MASK | | 
