diff options
Diffstat (limited to 'drivers/usb/dwc3/core.c')
| -rw-r--r-- | drivers/usb/dwc3/core.c | 56 | 
1 files changed, 46 insertions, 10 deletions
| diff --git a/drivers/usb/dwc3/core.c b/drivers/usb/dwc3/core.c index 573421984948..c5c238ab3083 100644 --- a/drivers/usb/dwc3/core.c +++ b/drivers/usb/dwc3/core.c @@ -158,8 +158,13 @@ static void __dwc3_set_mode(struct work_struct *work)  		break;  	} -	/* For DRD host or device mode only */ -	if (dwc->desired_dr_role != DWC3_GCTL_PRTCAP_OTG) { +	/* +	 * When current_dr_role is not set, there's no role switching. +	 * Only perform GCTL.CoreSoftReset when there's DRD role switching. +	 */ +	if (dwc->current_dr_role && ((DWC3_IP_IS(DWC3) || +			DWC3_VER_IS_PRIOR(DWC31, 190A)) && +			dwc->desired_dr_role != DWC3_GCTL_PRTCAP_OTG)) {  		reg = dwc3_readl(dwc->regs, DWC3_GCTL);  		reg |= DWC3_GCTL_CORESOFTRESET;  		dwc3_writel(dwc->regs, DWC3_GCTL, reg); @@ -426,7 +431,7 @@ static void dwc3_free_one_event_buffer(struct dwc3 *dwc,   * otherwise ERR_PTR(errno).   */  static struct dwc3_event_buffer *dwc3_alloc_one_event_buffer(struct dwc3 *dwc, -		unsigned length) +		unsigned int length)  {  	struct dwc3_event_buffer	*evt; @@ -469,7 +474,7 @@ static void dwc3_free_event_buffers(struct dwc3 *dwc)   * Returns 0 on success otherwise negative errno. In the error case, dwc   * may contain some buffers allocated but not all which were requested.   */ -static int dwc3_alloc_event_buffers(struct dwc3 *dwc, unsigned length) +static int dwc3_alloc_event_buffers(struct dwc3 *dwc, unsigned int length)  {  	struct dwc3_event_buffer *evt; @@ -1029,6 +1034,37 @@ static void dwc3_set_incr_burst_type(struct dwc3 *dwc)  	dwc3_writel(dwc->regs, DWC3_GSBUSCFG0, cfg);  } +static void dwc3_set_power_down_clk_scale(struct dwc3 *dwc) +{ +	u32 scale; +	u32 reg; + +	if (!dwc->susp_clk) +		return; + +	/* +	 * The power down scale field specifies how many suspend_clk +	 * periods fit into a 16KHz clock period. When performing +	 * the division, round up the remainder. +	 * +	 * The power down scale value is calculated using the fastest +	 * frequency of the suspend_clk. If it isn't fixed (but within +	 * the accuracy requirement), the driver may not know the max +	 * rate of the suspend_clk, so only update the power down scale +	 * if the default is less than the calculated value from +	 * clk_get_rate() or if the default is questionably high +	 * (3x or more) to be within the requirement. +	 */ +	scale = DIV_ROUND_UP(clk_get_rate(dwc->susp_clk), 16000); +	reg = dwc3_readl(dwc->regs, DWC3_GCTL); +	if ((reg & DWC3_GCTL_PWRDNSCALE_MASK) < DWC3_GCTL_PWRDNSCALE(scale) || +	    (reg & DWC3_GCTL_PWRDNSCALE_MASK) > DWC3_GCTL_PWRDNSCALE(scale*3)) { +		reg &= ~(DWC3_GCTL_PWRDNSCALE_MASK); +		reg |= DWC3_GCTL_PWRDNSCALE(scale); +		dwc3_writel(dwc->regs, DWC3_GCTL, reg); +	} +} +  /**   * dwc3_core_init - Low-level initialization of DWC3 Core   * @dwc: Pointer to our controller context structure @@ -1105,6 +1141,9 @@ static int dwc3_core_init(struct dwc3 *dwc)  	if (ret)  		goto err1; +	/* Set power down scale of suspend_clk */ +	dwc3_set_power_down_clk_scale(dwc); +  	/* Adjust Frame Length */  	dwc3_frame_length_adjustment(dwc); @@ -1782,6 +1821,7 @@ static int dwc3_probe(struct platform_device *pdev)  	platform_set_drvdata(pdev, dwc);  	dwc3_cache_hwparams(dwc); +	device_init_wakeup(&pdev->dev, of_property_read_bool(dev->of_node, "wakeup-source"));  	spin_lock_init(&dwc->lock);  	mutex_init(&dwc->mutex); @@ -1943,7 +1983,7 @@ static int dwc3_suspend_common(struct dwc3 *dwc, pm_message_t msg)  		dwc3_core_exit(dwc);  		break;  	case DWC3_GCTL_PRTCAP_HOST: -		if (!PMSG_IS_AUTO(msg)) { +		if (!PMSG_IS_AUTO(msg) && !device_can_wakeup(dwc->dev)) {  			dwc3_core_exit(dwc);  			break;  		} @@ -2004,7 +2044,7 @@ static int dwc3_resume_common(struct dwc3 *dwc, pm_message_t msg)  		spin_unlock_irqrestore(&dwc->lock, flags);  		break;  	case DWC3_GCTL_PRTCAP_HOST: -		if (!PMSG_IS_AUTO(msg)) { +		if (!PMSG_IS_AUTO(msg) && !device_can_wakeup(dwc->dev)) {  			ret = dwc3_core_init_for_resume(dwc);  			if (ret)  				return ret; @@ -2081,8 +2121,6 @@ static int dwc3_runtime_suspend(struct device *dev)  	if (ret)  		return ret; -	device_init_wakeup(dev, true); -  	return 0;  } @@ -2091,8 +2129,6 @@ static int dwc3_runtime_resume(struct device *dev)  	struct dwc3     *dwc = dev_get_drvdata(dev);  	int		ret; -	device_init_wakeup(dev, false); -  	ret = dwc3_resume_common(dwc, PMSG_AUTO_RESUME);  	if (ret)  		return ret; | 
