diff options
| author | Dmitry Torokhov <dmitry.torokhov@gmail.com> | 2022-08-02 10:06:12 -0700 | 
|---|---|---|
| committer | Dmitry Torokhov <dmitry.torokhov@gmail.com> | 2022-08-02 10:06:12 -0700 | 
| commit | 8bb5e7f4dcd9b9ef22a3ea25c9066a8a968f12dd (patch) | |
| tree | 0f1383880607a227142f9388a066959926233ff1 /drivers/usb/dwc3/core.c | |
| parent | 2a96271fb66c499e4a89d76a89d3d01170c10bef (diff) | |
| parent | 7c744d00990ea999d27f306f6db5ccb61b1304b2 (diff) | |
Merge branch 'next' into for-linus
Prepare input updates for 5.20 (or 6.0) merge window.
Diffstat (limited to 'drivers/usb/dwc3/core.c')
| -rw-r--r-- | drivers/usb/dwc3/core.c | 234 | 
1 files changed, 194 insertions, 40 deletions
| diff --git a/drivers/usb/dwc3/core.c b/drivers/usb/dwc3/core.c index f4c09951b517..573421984948 100644 --- a/drivers/usb/dwc3/core.c +++ b/drivers/usb/dwc3/core.c @@ -23,6 +23,7 @@  #include <linux/delay.h>  #include <linux/dma-mapping.h>  #include <linux/of.h> +#include <linux/of_graph.h>  #include <linux/acpi.h>  #include <linux/pinctrl/consumer.h>  #include <linux/reset.h> @@ -85,7 +86,7 @@ static int dwc3_get_dr_mode(struct dwc3 *dwc)  		 * mode. If the controller supports DRD but the dr_mode is not  		 * specified or set to OTG, then set the mode to peripheral.  		 */ -		if (mode == USB_DR_MODE_OTG && +		if (mode == USB_DR_MODE_OTG && !dwc->edev &&  		    (!IS_ENABLED(CONFIG_USB_ROLE_SWITCH) ||  		     !device_property_read_bool(dwc->dev, "usb-role-switch")) &&  		    !DWC3_VER_IS_PRIOR(DWC3, 330A)) @@ -115,8 +116,6 @@ void dwc3_set_prtcap(struct dwc3 *dwc, u32 mode)  	dwc->current_dr_role = mode;  } -static int dwc3_core_soft_reset(struct dwc3 *dwc); -  static void __dwc3_set_mode(struct work_struct *work)  {  	struct dwc3 *dwc = work_to_dwc(work); @@ -261,7 +260,7 @@ u32 dwc3_core_fifo_space(struct dwc3_ep *dep, u8 type)   * dwc3_core_soft_reset - Issues core soft reset and PHY reset   * @dwc: pointer to our context structure   */ -static int dwc3_core_soft_reset(struct dwc3 *dwc) +int dwc3_core_soft_reset(struct dwc3 *dwc)  {  	u32		reg;  	int		retries = 1000; @@ -276,7 +275,8 @@ static int dwc3_core_soft_reset(struct dwc3 *dwc)  	reg = dwc3_readl(dwc->regs, DWC3_DCTL);  	reg |= DWC3_DCTL_CSFTRST; -	dwc3_writel(dwc->regs, DWC3_DCTL, reg); +	reg &= ~DWC3_DCTL_RUN_STOP; +	dwc3_gadget_dctl_write_safe(dwc, reg);  	/*  	 * For DWC_usb31 controller 1.90a and later, the DCTL.CSFRST bit @@ -298,6 +298,7 @@ static int dwc3_core_soft_reset(struct dwc3 *dwc)  			udelay(1);  	} while (--retries); +	dev_warn(dwc->dev, "DWC3 controller soft reset failed.\n");  	return -ETIMEDOUT;  done: @@ -343,21 +344,67 @@ static void dwc3_frame_length_adjustment(struct dwc3 *dwc)   *		from the default, this will set clock period in DWC3_GUCTL   *		register.   * @dwc: Pointer to our controller context structure - * @ref_clk_per: reference clock period in ns   */  static void dwc3_ref_clk_period(struct dwc3 *dwc)  { +	unsigned long period; +	unsigned long fladj; +	unsigned long decr; +	unsigned long rate;  	u32 reg; -	if (dwc->ref_clk_per == 0) +	if (dwc->ref_clk) { +		rate = clk_get_rate(dwc->ref_clk); +		if (!rate) +			return; +		period = NSEC_PER_SEC / rate; +	} else if (dwc->ref_clk_per) { +		period = dwc->ref_clk_per; +		rate = NSEC_PER_SEC / period; +	} else {  		return; +	}  	reg = dwc3_readl(dwc->regs, DWC3_GUCTL);  	reg &= ~DWC3_GUCTL_REFCLKPER_MASK; -	reg |=  FIELD_PREP(DWC3_GUCTL_REFCLKPER_MASK, dwc->ref_clk_per); +	reg |=  FIELD_PREP(DWC3_GUCTL_REFCLKPER_MASK, period);  	dwc3_writel(dwc->regs, DWC3_GUCTL, reg); -} +	if (DWC3_VER_IS_PRIOR(DWC3, 250A)) +		return; + +	/* +	 * The calculation below is +	 * +	 * 125000 * (NSEC_PER_SEC / (rate * period) - 1) +	 * +	 * but rearranged for fixed-point arithmetic. The division must be +	 * 64-bit because 125000 * NSEC_PER_SEC doesn't fit in 32 bits (and +	 * neither does rate * period). +	 * +	 * Note that rate * period ~= NSEC_PER_SECOND, minus the number of +	 * nanoseconds of error caused by the truncation which happened during +	 * the division when calculating rate or period (whichever one was +	 * derived from the other). We first calculate the relative error, then +	 * scale it to units of 8 ppm. +	 */ +	fladj = div64_u64(125000ULL * NSEC_PER_SEC, (u64)rate * period); +	fladj -= 125000; + +	/* +	 * The documented 240MHz constant is scaled by 2 to get PLS1 as well. +	 */ +	decr = 480000000 / rate; + +	reg = dwc3_readl(dwc->regs, DWC3_GFLADJ); +	reg &= ~DWC3_GFLADJ_REFCLK_FLADJ_MASK +	    &  ~DWC3_GFLADJ_240MHZDECR +	    &  ~DWC3_GFLADJ_240MHZDECR_PLS1; +	reg |= FIELD_PREP(DWC3_GFLADJ_REFCLK_FLADJ_MASK, fladj) +	    |  FIELD_PREP(DWC3_GFLADJ_240MHZDECR, decr >> 1) +	    |  FIELD_PREP(DWC3_GFLADJ_240MHZDECR_PLS1, decr & 1); +	dwc3_writel(dwc->regs, DWC3_GFLADJ, reg); +}  /**   * dwc3_free_one_event_buffer - Frees one event buffer @@ -745,6 +792,38 @@ static int dwc3_phy_setup(struct dwc3 *dwc)  	return 0;  } +static int dwc3_clk_enable(struct dwc3 *dwc) +{ +	int ret; + +	ret = clk_prepare_enable(dwc->bus_clk); +	if (ret) +		return ret; + +	ret = clk_prepare_enable(dwc->ref_clk); +	if (ret) +		goto disable_bus_clk; + +	ret = clk_prepare_enable(dwc->susp_clk); +	if (ret) +		goto disable_ref_clk; + +	return 0; + +disable_ref_clk: +	clk_disable_unprepare(dwc->ref_clk); +disable_bus_clk: +	clk_disable_unprepare(dwc->bus_clk); +	return ret; +} + +static void dwc3_clk_disable(struct dwc3 *dwc) +{ +	clk_disable_unprepare(dwc->susp_clk); +	clk_disable_unprepare(dwc->ref_clk); +	clk_disable_unprepare(dwc->bus_clk); +} +  static void dwc3_core_exit(struct dwc3 *dwc)  {  	dwc3_event_buffers_cleanup(dwc); @@ -758,7 +837,7 @@ static void dwc3_core_exit(struct dwc3 *dwc)  	usb_phy_set_suspend(dwc->usb3_phy, 1);  	phy_power_off(dwc->usb2_generic_phy);  	phy_power_off(dwc->usb3_generic_phy); -	clk_bulk_disable_unprepare(dwc->num_clks, dwc->clks); +	dwc3_clk_disable(dwc);  	reset_control_assert(dwc->reset);  } @@ -886,10 +965,8 @@ static void dwc3_set_incr_burst_type(struct dwc3 *dwc)  		return;  	vals = kcalloc(ntype, sizeof(u32), GFP_KERNEL); -	if (!vals) { -		dev_err(dev, "Error to get memory\n"); +	if (!vals)  		return; -	}  	/* Get INCR burst type, and parse it */  	ret = device_property_read_u32_array(dev, @@ -1088,6 +1165,11 @@ static int dwc3_core_init(struct dwc3 *dwc)  		if (dwc->parkmode_disable_ss_quirk)  			reg |= DWC3_GUCTL1_PARKMODE_DISABLE_SS; +		if (DWC3_VER_IS_WITHIN(DWC3, 290A, ANY) && +		    (dwc->maximum_speed == USB_SPEED_HIGH || +		     dwc->maximum_speed == USB_SPEED_FULL)) +			reg |= DWC3_GUCTL1_DEV_FORCE_20_CLK_FOR_30_CLK; +  		dwc3_writel(dwc->regs, DWC3_GUCTL1, reg);  	} @@ -1185,40 +1267,36 @@ static int dwc3_core_get_phy(struct dwc3 *dwc)  	if (IS_ERR(dwc->usb2_phy)) {  		ret = PTR_ERR(dwc->usb2_phy); -		if (ret == -ENXIO || ret == -ENODEV) { +		if (ret == -ENXIO || ret == -ENODEV)  			dwc->usb2_phy = NULL; -		} else { +		else  			return dev_err_probe(dev, ret, "no usb2 phy configured\n"); -		}  	}  	if (IS_ERR(dwc->usb3_phy)) {  		ret = PTR_ERR(dwc->usb3_phy); -		if (ret == -ENXIO || ret == -ENODEV) { +		if (ret == -ENXIO || ret == -ENODEV)  			dwc->usb3_phy = NULL; -		} else { +		else  			return dev_err_probe(dev, ret, "no usb3 phy configured\n"); -		}  	}  	dwc->usb2_generic_phy = devm_phy_get(dev, "usb2-phy");  	if (IS_ERR(dwc->usb2_generic_phy)) {  		ret = PTR_ERR(dwc->usb2_generic_phy); -		if (ret == -ENOSYS || ret == -ENODEV) { +		if (ret == -ENOSYS || ret == -ENODEV)  			dwc->usb2_generic_phy = NULL; -		} else { +		else  			return dev_err_probe(dev, ret, "no usb2 phy configured\n"); -		}  	}  	dwc->usb3_generic_phy = devm_phy_get(dev, "usb3-phy");  	if (IS_ERR(dwc->usb3_generic_phy)) {  		ret = PTR_ERR(dwc->usb3_generic_phy); -		if (ret == -ENOSYS || ret == -ENODEV) { +		if (ret == -ENOSYS || ret == -ENODEV)  			dwc->usb3_generic_phy = NULL; -		} else { +		else  			return dev_err_probe(dev, ret, "no usb3 phy configured\n"); -		}  	}  	return 0; @@ -1295,10 +1373,10 @@ static void dwc3_get_properties(struct dwc3 *dwc)  	u8			lpm_nyet_threshold;  	u8			tx_de_emphasis;  	u8			hird_threshold; -	u8			rx_thr_num_pkt_prd; -	u8			rx_max_burst_prd; -	u8			tx_thr_num_pkt_prd; -	u8			tx_max_burst_prd; +	u8			rx_thr_num_pkt_prd = 0; +	u8			rx_max_burst_prd = 0; +	u8			tx_thr_num_pkt_prd = 0; +	u8			tx_max_burst_prd = 0;  	u8			tx_fifo_resize_max_num;  	const char		*usb_psy_name;  	int			ret; @@ -1550,6 +1628,46 @@ static void dwc3_check_params(struct dwc3 *dwc)  	}  } +static struct extcon_dev *dwc3_get_extcon(struct dwc3 *dwc) +{ +	struct device *dev = dwc->dev; +	struct device_node *np_phy; +	struct extcon_dev *edev = NULL; +	const char *name; + +	if (device_property_read_bool(dev, "extcon")) +		return extcon_get_edev_by_phandle(dev, 0); + +	/* +	 * Device tree platforms should get extcon via phandle. +	 * On ACPI platforms, we get the name from a device property. +	 * This device property is for kernel internal use only and +	 * is expected to be set by the glue code. +	 */ +	if (device_property_read_string(dev, "linux,extcon-name", &name) == 0) +		return extcon_get_extcon_dev(name); + +	/* +	 * Try to get an extcon device from the USB PHY controller's "port" +	 * node. Check if it has the "port" node first, to avoid printing the +	 * error message from underlying code, as it's a valid case: extcon +	 * device (and "port" node) may be missing in case of "usb-role-switch" +	 * or OTG mode. +	 */ +	np_phy = of_parse_phandle(dev->of_node, "phys", 0); +	if (of_graph_is_present(np_phy)) { +		struct device_node *np_conn; + +		np_conn = of_graph_get_remote_node(np_phy, -1, -1); +		if (np_conn) +			edev = extcon_find_edev_by_node(np_conn); +		of_node_put(np_conn); +	} +	of_node_put(np_phy); + +	return edev; +} +  static int dwc3_probe(struct platform_device *pdev)  {  	struct device		*dev = &pdev->dev; @@ -1605,25 +1723,54 @@ static int dwc3_probe(struct platform_device *pdev)  		return PTR_ERR(dwc->reset);  	if (dev->of_node) { -		ret = devm_clk_bulk_get_all(dev, &dwc->clks); -		if (ret == -EPROBE_DEFER) -			return ret;  		/*  		 * Clocks are optional, but new DT platforms should support all  		 * clocks as required by the DT-binding. +		 * Some devices have different clock names in legacy device trees, +		 * check for them to retain backwards compatibility.  		 */ -		if (ret < 0) -			dwc->num_clks = 0; -		else -			dwc->num_clks = ret; +		dwc->bus_clk = devm_clk_get_optional(dev, "bus_early"); +		if (IS_ERR(dwc->bus_clk)) +			return dev_err_probe(dev, PTR_ERR(dwc->bus_clk), +					     "could not get bus clock\n"); + +		if (dwc->bus_clk == NULL) { +			dwc->bus_clk = devm_clk_get_optional(dev, "bus_clk"); +			if (IS_ERR(dwc->bus_clk)) +				return dev_err_probe(dev, PTR_ERR(dwc->bus_clk), +						     "could not get bus clock\n"); +		} + +		dwc->ref_clk = devm_clk_get_optional(dev, "ref"); +		if (IS_ERR(dwc->ref_clk)) +			return dev_err_probe(dev, PTR_ERR(dwc->ref_clk), +					     "could not get ref clock\n"); + +		if (dwc->ref_clk == NULL) { +			dwc->ref_clk = devm_clk_get_optional(dev, "ref_clk"); +			if (IS_ERR(dwc->ref_clk)) +				return dev_err_probe(dev, PTR_ERR(dwc->ref_clk), +						     "could not get ref clock\n"); +		} + +		dwc->susp_clk = devm_clk_get_optional(dev, "suspend"); +		if (IS_ERR(dwc->susp_clk)) +			return dev_err_probe(dev, PTR_ERR(dwc->susp_clk), +					     "could not get suspend clock\n"); +		if (dwc->susp_clk == NULL) { +			dwc->susp_clk = devm_clk_get_optional(dev, "suspend_clk"); +			if (IS_ERR(dwc->susp_clk)) +				return dev_err_probe(dev, PTR_ERR(dwc->susp_clk), +						     "could not get suspend clock\n"); +		}  	}  	ret = reset_control_deassert(dwc->reset);  	if (ret)  		return ret; -	ret = clk_bulk_prepare_enable(dwc->num_clks, dwc->clks); +	ret = dwc3_clk_enable(dwc);  	if (ret)  		goto assert_reset; @@ -1656,6 +1803,13 @@ static int dwc3_probe(struct platform_device *pdev)  		goto err2;  	} +	dwc->edev = dwc3_get_extcon(dwc); +	if (IS_ERR(dwc->edev)) { +		ret = PTR_ERR(dwc->edev); +		dev_err_probe(dwc->dev, ret, "failed to get extcon\n"); +		goto err3; +	} +  	ret = dwc3_get_dr_mode(dwc);  	if (ret)  		goto err3; @@ -1711,7 +1865,7 @@ err1:  	pm_runtime_disable(&pdev->dev);  disable_clks: -	clk_bulk_disable_unprepare(dwc->num_clks, dwc->clks); +	dwc3_clk_disable(dwc);  assert_reset:  	reset_control_assert(dwc->reset); @@ -1755,7 +1909,7 @@ static int dwc3_core_init_for_resume(struct dwc3 *dwc)  	if (ret)  		return ret; -	ret = clk_bulk_prepare_enable(dwc->num_clks, dwc->clks); +	ret = dwc3_clk_enable(dwc);  	if (ret)  		goto assert_reset; @@ -1766,7 +1920,7 @@ static int dwc3_core_init_for_resume(struct dwc3 *dwc)  	return 0;  disable_clks: -	clk_bulk_disable_unprepare(dwc->num_clks, dwc->clks); +	dwc3_clk_disable(dwc);  assert_reset:  	reset_control_assert(dwc->reset); | 
