diff options
Diffstat (limited to 'drivers/usb/host')
| -rw-r--r-- | drivers/usb/host/ehci-dbg.c | 2 | ||||
| -rw-r--r-- | drivers/usb/host/xhci-debugfs.c | 16 | ||||
| -rw-r--r-- | drivers/usb/host/xhci-mem.c | 22 | ||||
| -rw-r--r-- | drivers/usb/host/xhci-pci.c | 3 | ||||
| -rw-r--r-- | drivers/usb/host/xhci-ring.c | 18 | ||||
| -rw-r--r-- | drivers/usb/host/xhci.c | 6 | 
6 files changed, 44 insertions, 23 deletions
| diff --git a/drivers/usb/host/ehci-dbg.c b/drivers/usb/host/ehci-dbg.c index 19f00424f53e..3ed75aaa09d9 100644 --- a/drivers/usb/host/ehci-dbg.c +++ b/drivers/usb/host/ehci-dbg.c @@ -827,7 +827,7 @@ static ssize_t fill_registers_buffer(struct debug_buffer *buf)  			default:		/* unknown */  				break;  			} -			temp = (cap >> 8) & 0xff; +			offset = (cap >> 8) & 0xff;  		}  	}  #endif diff --git a/drivers/usb/host/xhci-debugfs.c b/drivers/usb/host/xhci-debugfs.c index 4f7895dbcf88..e26e685d8a57 100644 --- a/drivers/usb/host/xhci-debugfs.c +++ b/drivers/usb/host/xhci-debugfs.c @@ -162,7 +162,7 @@ static void xhci_debugfs_extcap_regset(struct xhci_hcd *xhci, int cap_id,  static int xhci_ring_enqueue_show(struct seq_file *s, void *unused)  {  	dma_addr_t		dma; -	struct xhci_ring	*ring = s->private; +	struct xhci_ring	*ring = *(struct xhci_ring **)s->private;  	dma = xhci_trb_virt_to_dma(ring->enq_seg, ring->enqueue);  	seq_printf(s, "%pad\n", &dma); @@ -173,7 +173,7 @@ static int xhci_ring_enqueue_show(struct seq_file *s, void *unused)  static int xhci_ring_dequeue_show(struct seq_file *s, void *unused)  {  	dma_addr_t		dma; -	struct xhci_ring	*ring = s->private; +	struct xhci_ring	*ring = *(struct xhci_ring **)s->private;  	dma = xhci_trb_virt_to_dma(ring->deq_seg, ring->dequeue);  	seq_printf(s, "%pad\n", &dma); @@ -183,7 +183,7 @@ static int xhci_ring_dequeue_show(struct seq_file *s, void *unused)  static int xhci_ring_cycle_show(struct seq_file *s, void *unused)  { -	struct xhci_ring	*ring = s->private; +	struct xhci_ring	*ring = *(struct xhci_ring **)s->private;  	seq_printf(s, "%d\n", ring->cycle_state); @@ -346,7 +346,7 @@ static void xhci_debugfs_create_files(struct xhci_hcd *xhci,  }  static struct dentry *xhci_debugfs_create_ring_dir(struct xhci_hcd *xhci, -						   struct xhci_ring *ring, +						   struct xhci_ring **ring,  						   const char *name,  						   struct dentry *parent)  { @@ -387,7 +387,7 @@ void xhci_debugfs_create_endpoint(struct xhci_hcd *xhci,  	snprintf(epriv->name, sizeof(epriv->name), "ep%02d", ep_index);  	epriv->root = xhci_debugfs_create_ring_dir(xhci, -						   dev->eps[ep_index].new_ring, +						   &dev->eps[ep_index].new_ring,  						   epriv->name,  						   spriv->root);  	spriv->eps[ep_index] = epriv; @@ -423,7 +423,7 @@ void xhci_debugfs_create_slot(struct xhci_hcd *xhci, int slot_id)  	priv->dev = dev;  	dev->debugfs_private = priv; -	xhci_debugfs_create_ring_dir(xhci, dev->eps[0].ring, +	xhci_debugfs_create_ring_dir(xhci, &dev->eps[0].ring,  				     "ep00", priv->root);  	xhci_debugfs_create_context_files(xhci, priv->root, slot_id); @@ -488,11 +488,11 @@ void xhci_debugfs_init(struct xhci_hcd *xhci)  				   ARRAY_SIZE(xhci_extcap_dbc),  				   "reg-ext-dbc"); -	xhci_debugfs_create_ring_dir(xhci, xhci->cmd_ring, +	xhci_debugfs_create_ring_dir(xhci, &xhci->cmd_ring,  				     "command-ring",  				     xhci->debugfs_root); -	xhci_debugfs_create_ring_dir(xhci, xhci->event_ring, +	xhci_debugfs_create_ring_dir(xhci, &xhci->event_ring,  				     "event-ring",  				     xhci->debugfs_root); diff --git a/drivers/usb/host/xhci-mem.c b/drivers/usb/host/xhci-mem.c index e1fba4688509..3a29b32a3bd0 100644 --- a/drivers/usb/host/xhci-mem.c +++ b/drivers/usb/host/xhci-mem.c @@ -934,6 +934,12 @@ void xhci_free_virt_devices_depth_first(struct xhci_hcd *xhci, int slot_id)  	if (!vdev)  		return; +	if (vdev->real_port == 0 || +			vdev->real_port > HCS_MAX_PORTS(xhci->hcs_params1)) { +		xhci_dbg(xhci, "Bad vdev->real_port.\n"); +		goto out; +	} +  	tt_list_head = &(xhci->rh_bw[vdev->real_port - 1].tts);  	list_for_each_entry_safe(tt_info, next, tt_list_head, tt_list) {  		/* is this a hub device that added a tt_info to the tts list */ @@ -947,6 +953,7 @@ void xhci_free_virt_devices_depth_first(struct xhci_hcd *xhci, int slot_id)  			}  		}  	} +out:  	/* we are now at a leaf device */  	xhci_debugfs_remove_slot(xhci, slot_id);  	xhci_free_virt_device(xhci, slot_id); @@ -964,10 +971,9 @@ int xhci_alloc_virt_device(struct xhci_hcd *xhci, int slot_id,  		return 0;  	} -	xhci->devs[slot_id] = kzalloc(sizeof(*xhci->devs[slot_id]), flags); -	if (!xhci->devs[slot_id]) +	dev = kzalloc(sizeof(*dev), flags); +	if (!dev)  		return 0; -	dev = xhci->devs[slot_id];  	/* Allocate the (output) device context that will be used in the HC. */  	dev->out_ctx = xhci_alloc_container_ctx(xhci, XHCI_CTX_TYPE_DEVICE, flags); @@ -1008,9 +1014,17 @@ int xhci_alloc_virt_device(struct xhci_hcd *xhci, int slot_id,  	trace_xhci_alloc_virt_device(dev); +	xhci->devs[slot_id] = dev; +  	return 1;  fail: -	xhci_free_virt_device(xhci, slot_id); + +	if (dev->in_ctx) +		xhci_free_container_ctx(xhci, dev->in_ctx); +	if (dev->out_ctx) +		xhci_free_container_ctx(xhci, dev->out_ctx); +	kfree(dev); +  	return 0;  } diff --git a/drivers/usb/host/xhci-pci.c b/drivers/usb/host/xhci-pci.c index 7ef1274ef7f7..1aad89b8aba0 100644 --- a/drivers/usb/host/xhci-pci.c +++ b/drivers/usb/host/xhci-pci.c @@ -178,6 +178,9 @@ static void xhci_pci_quirks(struct device *dev, struct xhci_hcd *xhci)  		xhci->quirks |= XHCI_BROKEN_STREAMS;  	}  	if (pdev->vendor == PCI_VENDOR_ID_RENESAS && +			pdev->device == 0x0014) +		xhci->quirks |= XHCI_TRUST_TX_LENGTH; +	if (pdev->vendor == PCI_VENDOR_ID_RENESAS &&  			pdev->device == 0x0015)  		xhci->quirks |= XHCI_RESET_ON_RESUME;  	if (pdev->vendor == PCI_VENDOR_ID_VIA) diff --git a/drivers/usb/host/xhci-ring.c b/drivers/usb/host/xhci-ring.c index c239c688076c..c5cbc685c691 100644 --- a/drivers/usb/host/xhci-ring.c +++ b/drivers/usb/host/xhci-ring.c @@ -2477,12 +2477,16 @@ static int handle_tx_event(struct xhci_hcd *xhci,  		 */  		if (list_empty(&ep_ring->td_list)) {  			/* -			 * A stopped endpoint may generate an extra completion -			 * event if the device was suspended.  Don't print -			 * warnings. +			 * Don't print wanings if it's due to a stopped endpoint +			 * generating an extra completion event if the device +			 * was suspended. Or, a event for the last TRB of a +			 * short TD we already got a short event for. +			 * The short TD is already removed from the TD list.  			 */ +  			if (!(trb_comp_code == COMP_STOPPED || -				trb_comp_code == COMP_STOPPED_LENGTH_INVALID)) { +			      trb_comp_code == COMP_STOPPED_LENGTH_INVALID || +			      ep_ring->last_td_was_short)) {  				xhci_warn(xhci, "WARN Event TRB for slot %d ep %d with no TDs queued?\n",  						TRB_TO_SLOT_ID(le32_to_cpu(event->flags)),  						ep_index); @@ -3108,7 +3112,7 @@ static u32 xhci_td_remainder(struct xhci_hcd *xhci, int transferred,  {  	u32 maxp, total_packet_count; -	/* MTK xHCI is mostly 0.97 but contains some features from 1.0 */ +	/* MTK xHCI 0.96 contains some features from 1.0 */  	if (xhci->hci_version < 0x100 && !(xhci->quirks & XHCI_MTK_HOST))  		return ((td_total_len - transferred) >> 10); @@ -3117,8 +3121,8 @@ static u32 xhci_td_remainder(struct xhci_hcd *xhci, int transferred,  	    trb_buff_len == td_total_len)  		return 0; -	/* for MTK xHCI, TD size doesn't include this TRB */ -	if (xhci->quirks & XHCI_MTK_HOST) +	/* for MTK xHCI 0.96, TD size include this TRB, but not in 1.x */ +	if ((xhci->quirks & XHCI_MTK_HOST) && (xhci->hci_version < 0x100))  		trb_buff_len = 0;  	maxp = usb_endpoint_maxp(&urb->ep->desc); diff --git a/drivers/usb/host/xhci.c b/drivers/usb/host/xhci.c index 2424d3020ca3..da6dbe3ebd8b 100644 --- a/drivers/usb/host/xhci.c +++ b/drivers/usb/host/xhci.c @@ -3525,8 +3525,6 @@ static void xhci_free_dev(struct usb_hcd *hcd, struct usb_device *udev)  	struct xhci_slot_ctx *slot_ctx;  	int i, ret; -	xhci_debugfs_remove_slot(xhci, udev->slot_id); -  #ifndef CONFIG_USB_DEFAULT_PERSIST  	/*  	 * We called pm_runtime_get_noresume when the device was attached. @@ -3555,8 +3553,10 @@ static void xhci_free_dev(struct usb_hcd *hcd, struct usb_device *udev)  	}  	ret = xhci_disable_slot(xhci, udev->slot_id); -	if (ret) +	if (ret) { +		xhci_debugfs_remove_slot(xhci, udev->slot_id);  		xhci_free_virt_device(xhci, udev->slot_id); +	}  }  int xhci_disable_slot(struct xhci_hcd *xhci, u32 slot_id) |