diff options
Diffstat (limited to 'drivers/usb/host')
| -rw-r--r-- | drivers/usb/host/xhci-hub.c | 13 | ||||
| -rw-r--r-- | drivers/usb/host/xhci-mtk-sch.c | 15 | ||||
| -rw-r--r-- | drivers/usb/host/xhci-plat.c | 11 | ||||
| -rw-r--r-- | drivers/usb/host/xhci.c | 19 | ||||
| -rw-r--r-- | drivers/usb/host/xhci.h | 4 | 
5 files changed, 30 insertions, 32 deletions
| diff --git a/drivers/usb/host/xhci-hub.c b/drivers/usb/host/xhci-hub.c index 0fdc014c9401..4619d5e89d5b 100644 --- a/drivers/usb/host/xhci-hub.c +++ b/drivers/usb/host/xhci-hub.c @@ -652,7 +652,7 @@ struct xhci_hub *xhci_get_rhub(struct usb_hcd *hcd)   * It will release and re-aquire the lock while calling ACPI   * method.   */ -void xhci_set_port_power(struct xhci_hcd *xhci, struct usb_hcd *hcd, +static void xhci_set_port_power(struct xhci_hcd *xhci, struct usb_hcd *hcd,  				u16 index, bool on, unsigned long *flags)  	__must_hold(&xhci->lock)  { @@ -1648,6 +1648,17 @@ int xhci_hub_status_data(struct usb_hcd *hcd, char *buf)  	status = bus_state->resuming_ports; +	/* +	 * SS devices are only visible to roothub after link training completes. +	 * Keep polling roothubs for a grace period after xHC start +	 */ +	if (xhci->run_graceperiod) { +		if (time_before(jiffies, xhci->run_graceperiod)) +			status = 1; +		else +			xhci->run_graceperiod = 0; +	} +  	mask = PORT_CSC | PORT_PEC | PORT_OCC | PORT_PLC | PORT_WRC | PORT_CEC;  	/* For each port, did anything change?  If so, set that bit in buf. */ diff --git a/drivers/usb/host/xhci-mtk-sch.c b/drivers/usb/host/xhci-mtk-sch.c index 06a6b19acaae..579899eb24c1 100644 --- a/drivers/usb/host/xhci-mtk-sch.c +++ b/drivers/usb/host/xhci-mtk-sch.c @@ -425,7 +425,6 @@ static int check_fs_bus_bw(struct mu3h_sch_ep_info *sch_ep, int offset)  static int check_sch_tt(struct mu3h_sch_ep_info *sch_ep, u32 offset)  { -	u32 extra_cs_count;  	u32 start_ss, last_ss;  	u32 start_cs, last_cs; @@ -461,18 +460,12 @@ static int check_sch_tt(struct mu3h_sch_ep_info *sch_ep, u32 offset)  		if (last_cs > 7)  			return -ESCH_CS_OVERFLOW; -		if (sch_ep->ep_type == ISOC_IN_EP) -			extra_cs_count = (last_cs == 7) ? 1 : 2; -		else /*  ep_type : INTR IN / INTR OUT */ -			extra_cs_count = 1; - -		cs_count += extra_cs_count;  		if (cs_count > 7)  			cs_count = 7; /* HW limit */  		sch_ep->cs_count = cs_count; -		/* one for ss, the other for idle */ -		sch_ep->num_budget_microframes = cs_count + 2; +		/* ss, idle are ignored */ +		sch_ep->num_budget_microframes = cs_count;  		/*  		 * if interval=1, maxp >752, num_budge_micoframe is larger @@ -771,8 +764,8 @@ int xhci_mtk_drop_ep(struct usb_hcd *hcd, struct usb_device *udev,  	if (ret)  		return ret; -	if (ep->hcpriv) -		drop_ep_quirk(hcd, udev, ep); +	/* needn't check @ep->hcpriv, xhci_endpoint_disable set it NULL */ +	drop_ep_quirk(hcd, udev, ep);  	return 0;  } diff --git a/drivers/usb/host/xhci-plat.c b/drivers/usb/host/xhci-plat.c index 044855818cb1..a8641b6536ee 100644 --- a/drivers/usb/host/xhci-plat.c +++ b/drivers/usb/host/xhci-plat.c @@ -398,12 +398,17 @@ static int xhci_plat_remove(struct platform_device *dev)  	pm_runtime_get_sync(&dev->dev);  	xhci->xhc_state |= XHCI_STATE_REMOVING; -	usb_remove_hcd(shared_hcd); -	xhci->shared_hcd = NULL; +	if (shared_hcd) { +		usb_remove_hcd(shared_hcd); +		xhci->shared_hcd = NULL; +	} +  	usb_phy_shutdown(hcd->usb_phy);  	usb_remove_hcd(hcd); -	usb_put_hcd(shared_hcd); + +	if (shared_hcd) +		usb_put_hcd(shared_hcd);  	clk_disable_unprepare(clk);  	clk_disable_unprepare(reg_clk); diff --git a/drivers/usb/host/xhci.c b/drivers/usb/host/xhci.c index 65858f607437..38649284ff88 100644 --- a/drivers/usb/host/xhci.c +++ b/drivers/usb/host/xhci.c @@ -151,9 +151,11 @@ int xhci_start(struct xhci_hcd *xhci)  		xhci_err(xhci, "Host took too long to start, "  				"waited %u microseconds.\n",  				XHCI_MAX_HALT_USEC); -	if (!ret) +	if (!ret) {  		/* clear state flags. Including dying, halted or removing */  		xhci->xhc_state = 0; +		xhci->run_graceperiod = jiffies + msecs_to_jiffies(500); +	}  	return ret;  } @@ -791,8 +793,6 @@ static void xhci_stop(struct usb_hcd *hcd)  void xhci_shutdown(struct usb_hcd *hcd)  {  	struct xhci_hcd *xhci = hcd_to_xhci(hcd); -	unsigned long flags; -	int i;  	if (xhci->quirks & XHCI_SPURIOUS_REBOOT)  		usb_disable_xhci_ports(to_pci_dev(hcd->self.sysdev)); @@ -808,21 +808,12 @@ void xhci_shutdown(struct usb_hcd *hcd)  		del_timer_sync(&xhci->shared_hcd->rh_timer);  	} -	spin_lock_irqsave(&xhci->lock, flags); +	spin_lock_irq(&xhci->lock);  	xhci_halt(xhci); - -	/* Power off USB2 ports*/ -	for (i = 0; i < xhci->usb2_rhub.num_ports; i++) -		xhci_set_port_power(xhci, xhci->main_hcd, i, false, &flags); - -	/* Power off USB3 ports*/ -	for (i = 0; i < xhci->usb3_rhub.num_ports; i++) -		xhci_set_port_power(xhci, xhci->shared_hcd, i, false, &flags); -  	/* Workaround for spurious wakeups at shutdown with HSW */  	if (xhci->quirks & XHCI_SPURIOUS_WAKEUP)  		xhci_reset(xhci, XHCI_RESET_SHORT_USEC); -	spin_unlock_irqrestore(&xhci->lock, flags); +	spin_unlock_irq(&xhci->lock);  	xhci_cleanup_msix(xhci); diff --git a/drivers/usb/host/xhci.h b/drivers/usb/host/xhci.h index 1960b47acfb2..7caa0db5e826 100644 --- a/drivers/usb/host/xhci.h +++ b/drivers/usb/host/xhci.h @@ -1826,7 +1826,7 @@ struct xhci_hcd {  	/* Host controller watchdog timer structures */  	unsigned int		xhc_state; - +	unsigned long		run_graceperiod;  	u32			command;  	struct s3_save		s3;  /* Host controller is dying - not responding to commands. "I'm not dead yet!" @@ -2196,8 +2196,6 @@ int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, u16 wIndex,  int xhci_hub_status_data(struct usb_hcd *hcd, char *buf);  int xhci_find_raw_port_number(struct usb_hcd *hcd, int port1);  struct xhci_hub *xhci_get_rhub(struct usb_hcd *hcd); -void xhci_set_port_power(struct xhci_hcd *xhci, struct usb_hcd *hcd, u16 index, -			 bool on, unsigned long *flags);  void xhci_hc_died(struct xhci_hcd *xhci); | 
