[Freedreno] [PATCH] drm/msm/dp: do not initialize combo phy until plugin interrupt

Bjorn Andersson bjorn.andersson at linaro.org
Tue Oct 12 02:46:57 UTC 2021


On Wed 06 Oct 10:59 CDT 2021, Kuogee Hsieh wrote:

> Combo phy support both USB3 and DP simultaneously. USB3 is the
> master of combo phy so that USB3 should initialize and power on
> its phy before DP initialize its phy. At current implementation,
> DP driver initialize its phy happen earlier than  USB3 initialize
> its phy which cause timeout error at procedure of power up USB3 phy
> which prevent USB3 from working.

To me this is a problem with the QMP phy driver and shouldn't be hacked
around in the DP driver.

In particular if we're taking this route, this should clearly state what
the actual problem in the QMP phy is and why it's impossible to solve
this without hacking around in the DP driver.

> To avoid confliction of phy
> initialization between USB3 and DP, this patch have DP driver postpone
> phy initialization until plugin interrupt handler. DP driver only enable
> regulator, configure its HPD controller and enable interrupt so that it
> is able to receive HPD interrupts after completion of the initialization
> phase. DP driver will initialize and power up phy at plugin interrupt
> handler during normal operation so that both USB3 and DP are work
> simultaneously.
> 

What happens if I get a HPD before the USB driver probes?

Regards,
Bjorn

> Signed-off-by: Kuogee Hsieh <khsieh at codeaurora.org>
> ---
>  drivers/gpu/drm/msm/dp/dp_ctrl.c    | 68 ++++++++++++++++++++++---------------
>  drivers/gpu/drm/msm/dp/dp_ctrl.h    |  9 +++--
>  drivers/gpu/drm/msm/dp/dp_display.c | 51 +++++++++++++++++++++-------
>  3 files changed, 84 insertions(+), 44 deletions(-)
> 
> diff --git a/drivers/gpu/drm/msm/dp/dp_ctrl.c b/drivers/gpu/drm/msm/dp/dp_ctrl.c
> index 5551a8d..4c5d507 100644
> --- a/drivers/gpu/drm/msm/dp/dp_ctrl.c
> +++ b/drivers/gpu/drm/msm/dp/dp_ctrl.c
> @@ -1378,7 +1378,25 @@ static int dp_ctrl_enable_stream_clocks(struct dp_ctrl_private *ctrl)
>  	return ret;
>  }
>  
> -int dp_ctrl_host_init(struct dp_ctrl *dp_ctrl, bool flip, bool reset)
> +void dp_ctrl_irq_enable(struct dp_ctrl *dp_ctrl, bool flip)
> +{
> +	struct dp_ctrl_private *ctrl;
> +
> +	if (!dp_ctrl) {
> +		DRM_ERROR("Invalid input data\n");
> +		return;
> +	}
> +
> +	ctrl = container_of(dp_ctrl, struct dp_ctrl_private, dp_ctrl);
> +
> +	ctrl->dp_ctrl.orientation = flip;
> +
> +	dp_catalog_ctrl_reset(ctrl->catalog);
> +
> +	dp_catalog_ctrl_enable_irq(ctrl->catalog, true);
> +}
> +
> +void dp_ctrl_phy_init(struct dp_ctrl *dp_ctrl)
>  {
>  	struct dp_ctrl_private *ctrl;
>  	struct dp_io *dp_io;
> @@ -1386,33 +1404,44 @@ int dp_ctrl_host_init(struct dp_ctrl *dp_ctrl, bool flip, bool reset)
>  
>  	if (!dp_ctrl) {
>  		DRM_ERROR("Invalid input data\n");
> -		return -EINVAL;
> +		return;
>  	}
>  
>  	ctrl = container_of(dp_ctrl, struct dp_ctrl_private, dp_ctrl);
>  	dp_io = &ctrl->parser->io;
>  	phy = dp_io->phy;
>  
> -	ctrl->dp_ctrl.orientation = flip;
> -
> -	if (reset)
> -		dp_catalog_ctrl_reset(ctrl->catalog);
> -
>  	dp_catalog_ctrl_phy_reset(ctrl->catalog);
>  	phy_init(phy);
> -	dp_catalog_ctrl_enable_irq(ctrl->catalog, true);
> +}
>  
> -	return 0;
> +void dp_ctrl_phy_exit(struct dp_ctrl *dp_ctrl)
> +{
> +	struct dp_ctrl_private *ctrl;
> +	struct dp_io *dp_io;
> +	struct phy *phy;
> +
> +	if (!dp_ctrl) {
> +		DRM_ERROR("Invalid input data\n");
> +		return;
> +	}
> +
> +	ctrl = container_of(dp_ctrl, struct dp_ctrl_private, dp_ctrl);
> +	dp_io = &ctrl->parser->io;
> +	phy = dp_io->phy;
> +
> +	dp_catalog_ctrl_phy_reset(ctrl->catalog);
> +	phy_exit(phy);
>  }
>  
>  /**
> - * dp_ctrl_host_deinit() - Uninitialize DP controller
> + * dp_ctrl_irq_phy_exit() - disable dp irq and exit phy
>   * @dp_ctrl: Display Port Driver data
>   *
>   * Perform required steps to uninitialize DP controller
>   * and its resources.
>   */
> -void dp_ctrl_host_deinit(struct dp_ctrl *dp_ctrl)
> +void dp_ctrl_irq_phy_exit(struct dp_ctrl *dp_ctrl)
>  {
>  	struct dp_ctrl_private *ctrl;
>  	struct dp_io *dp_io;
> @@ -1866,23 +1895,6 @@ int dp_ctrl_off_link_stream(struct dp_ctrl *dp_ctrl)
>  	return ret;
>  }
>  
> -void dp_ctrl_off_phy(struct dp_ctrl *dp_ctrl)
> -{
> -	struct dp_ctrl_private *ctrl;
> -	struct dp_io *dp_io;
> -	struct phy *phy;
> -
> -	ctrl = container_of(dp_ctrl, struct dp_ctrl_private, dp_ctrl);
> -	dp_io = &ctrl->parser->io;
> -	phy = dp_io->phy;
> -
> -	dp_catalog_ctrl_reset(ctrl->catalog);
> -
> -	phy_exit(phy);
> -
> -	DRM_DEBUG_DP("DP off phy done\n");
> -}
> -
>  int dp_ctrl_off(struct dp_ctrl *dp_ctrl)
>  {
>  	struct dp_ctrl_private *ctrl;
> diff --git a/drivers/gpu/drm/msm/dp/dp_ctrl.h b/drivers/gpu/drm/msm/dp/dp_ctrl.h
> index 2363a2d..c1e4b1b 100644
> --- a/drivers/gpu/drm/msm/dp/dp_ctrl.h
> +++ b/drivers/gpu/drm/msm/dp/dp_ctrl.h
> @@ -19,12 +19,9 @@ struct dp_ctrl {
>  	u32 pixel_rate;
>  };
>  
> -int dp_ctrl_host_init(struct dp_ctrl *dp_ctrl, bool flip, bool reset);
> -void dp_ctrl_host_deinit(struct dp_ctrl *dp_ctrl);
>  int dp_ctrl_on_link(struct dp_ctrl *dp_ctrl);
>  int dp_ctrl_on_stream(struct dp_ctrl *dp_ctrl);
>  int dp_ctrl_off_link_stream(struct dp_ctrl *dp_ctrl);
> -void dp_ctrl_off_phy(struct dp_ctrl *dp_ctrl);
>  int dp_ctrl_off(struct dp_ctrl *dp_ctrl);
>  void dp_ctrl_push_idle(struct dp_ctrl *dp_ctrl);
>  void dp_ctrl_isr(struct dp_ctrl *dp_ctrl);
> @@ -34,4 +31,10 @@ struct dp_ctrl *dp_ctrl_get(struct device *dev, struct dp_link *link,
>  			struct dp_power *power, struct dp_catalog *catalog,
>  			struct dp_parser *parser);
>  
> +void dp_ctrl_irq_enable(struct dp_ctrl *dp_ctrl, bool flip);
> +void dp_ctrl_irq_disable(struct dp_ctrl *dp_ctrl);
> +void dp_ctrl_phy_init(struct dp_ctrl *dp_ctrl);
> +void dp_ctrl_phy_exit(struct dp_ctrl *dp_ctrl);
> +void dp_ctrl_irq_phy_exit(struct dp_ctrl *dp_ctrl);
> +
>  #endif /* _DP_CTRL_H_ */
> diff --git a/drivers/gpu/drm/msm/dp/dp_display.c b/drivers/gpu/drm/msm/dp/dp_display.c
> index cad25dd..44032ae 100644
> --- a/drivers/gpu/drm/msm/dp/dp_display.c
> +++ b/drivers/gpu/drm/msm/dp/dp_display.c
> @@ -62,6 +62,11 @@ enum {
>  	EV_DISCONNECT_PENDING_TIMEOUT,
>  };
>  
> +enum {
> +	TYPE_eDP = 1,
> +	TYPE_DP,
> +};
> +
>  #define EVENT_TIMEOUT	(HZ/10)	/* 100ms */
>  #define DP_EVENT_Q_MAX	8
>  
> @@ -81,6 +86,7 @@ struct dp_display_private {
>  	int irq;
>  
>  	int id;
> +	int type;
>  
>  	/* state variables */
>  	bool core_initialized;
> @@ -121,31 +127,37 @@ struct dp_display_private {
>  
>  struct msm_dp_config { 
>  	phys_addr_t io_start[3];
> +	int type;
>  	size_t num_dp;
>  };
>  
>  static const struct msm_dp_config sc7180_dp_cfg = {
>  	.io_start = { 0x0ae90000 },
> +	.type = TYPE_DP,
>  	.num_dp = 1,
>  };
>  
>  static const struct msm_dp_config sc8180x_dp_cfg = {
>  	.io_start = { 0xae90000, 0xae98000, 0 },
> +	.type = TYPE_DP,
>  	.num_dp = 3,
>  };
>  
>  static const struct msm_dp_config sc8180x_edp_cfg = {
>  	.io_start = { 0, 0, 0xae9a000 },
>  	.num_dp = 3,
> +	.type = TYPE_eDP,
>  };
>  
>  static const struct msm_dp_config sc7280_edp_cfg = {
>  	.io_start = { 0xaea0000, 0 },
> +	.type = TYPE_eDP,
>  	.num_dp = 2,
>  };
>  
>  static const struct msm_dp_config sc7280_dp_cfg = { 
>  	.io_start = { 0, 0xae90000 },
> +	.type = TYPE_DP,
>  	.num_dp = 2,
>  };
>  
> @@ -392,7 +404,7 @@ static int dp_display_process_hpd_high(struct dp_display_private *dp)
>  	return rc;
>  }
>  
> -static void dp_display_host_init(struct dp_display_private *dp, int reset)
> +static void dp_display_host_init(struct dp_display_private *dp)
>  {
>  	bool flip = false;
>  
> @@ -404,12 +416,21 @@ static void dp_display_host_init(struct dp_display_private *dp, int reset)
>  	if (dp->usbpd->orientation == ORIENTATION_CC2)
>  		flip = true;
>  
> -	dp_power_init(dp->power, flip);
> -	dp_ctrl_host_init(dp->ctrl, flip, reset);
> +	dp_power_init(dp->power, false);
> +	dp_ctrl_irq_enable(dp->ctrl, flip);
> +
> +	if (dp->type == TYPE_eDP)
> +		dp_ctrl_phy_init(dp->ctrl);
> +
>  	dp_aux_init(dp->aux);
>  	dp->core_initialized = true;
>  }
>  
> +static void dp_display_host_phy_init(struct dp_display_private *dp)
> +{
> +	dp_ctrl_phy_init(dp->ctrl);
> +}
> +
>  static void dp_display_host_deinit(struct dp_display_private *dp)
>  {
>  	if (!dp->core_initialized) {
> @@ -417,7 +438,7 @@ static void dp_display_host_deinit(struct dp_display_private *dp)
>  		return;
>  	}
>  
> -	dp_ctrl_host_deinit(dp->ctrl);
> +	dp_ctrl_irq_phy_exit(dp->ctrl);
>  	dp_aux_deinit(dp->aux);
>  	dp_power_deinit(dp->power);
>  
> @@ -435,7 +456,7 @@ static int dp_display_usbpd_configure_cb(struct device *dev)
>  		goto end;
>  	}
>  
> -	dp_display_host_init(dp, false);
> +	dp_display_host_phy_init(dp);
>  
>  	rc = dp_display_process_hpd_high(dp);
>  end:
> @@ -646,9 +667,8 @@ static int dp_hpd_unplug_handle(struct dp_display_private *dp, u32 data)
>  	if (state == ST_DISCONNECTED) {
>  		/* triggered by irq_hdp with sink_count = 0 */
>  		if (dp->link->sink_count == 0) {
> -			dp_ctrl_off_phy(dp->ctrl);
> +			dp_ctrl_phy_exit(dp->ctrl);
>  			hpd->hpd_high = 0;
> -			dp->core_initialized = false;
>  		}
>  		mutex_unlock(&dp->event_mutex);
>  		return 0;
> @@ -1040,7 +1060,7 @@ int dp_display_get_test_bpp(struct msm_dp *dp)
>  static void dp_display_config_hpd(struct dp_display_private *dp)
>  {
>  
> -	dp_display_host_init(dp, true);
> +	dp_display_host_init(dp);
>  	dp_catalog_ctrl_hpd_config(dp->catalog);
>  
>  	/* Enable interrupt first time
> @@ -1222,7 +1242,8 @@ int dp_display_request_irq(struct msm_dp *dp_display)
>  	return 0;
>  }
>  
> -static int dp_display_get_id(struct platform_device *pdev)
> +static int dp_display_get_id(struct platform_device *pdev,
> +				struct msm_dp_config *dp_cfg)
>  {
>  	const struct msm_dp_config *cfg = of_device_get_match_data(&pdev->dev);
>  	struct resource *res;
> @@ -1234,8 +1255,10 @@ static int dp_display_get_id(struct platform_device *pdev)
>  		return -EINVAL;
>  
>  	for (i = 0; i < cfg->num_dp; i++) {
> -		if (cfg->io_start[i] == res->start)
> +		if (cfg->io_start[i] == res->start) {
> +			*dp_cfg = *cfg;
>  			return i;
> +		}
>  	}
>  
>  	dev_err(&pdev->dev, "unknown displayport instance\n");
> @@ -1246,6 +1269,7 @@ static int dp_display_probe(struct platform_device *pdev)
>  {
>  	int rc = 0;
>  	struct dp_display_private *dp;
> +	struct msm_dp_config dp_cfg;
>  
>  	if (!pdev || !pdev->dev.of_node) {
>  		DRM_ERROR("pdev not found\n");
> @@ -1256,12 +1280,13 @@ static int dp_display_probe(struct platform_device *pdev)
>  	if (!dp)
>  		return -ENOMEM;
>  
> -	dp->id = dp_display_get_id(pdev);
> +	dp->id = dp_display_get_id(pdev, &dp_cfg);
>  	if (dp->id < 0)
>  		return -EINVAL;
>  
>  	dp->pdev = pdev;
>  	dp->name = "drm_dp";
> +	dp->type = dp_cfg.type;
>  
>  	rc = dp_init_sub_modules(dp);
>  	if (rc) {
> @@ -1314,7 +1339,7 @@ static int dp_pm_resume(struct device *dev)
>  	dp->hpd_state = ST_DISCONNECTED;
>  
>  	/* turn on dp ctrl/phy */
> -	dp_display_host_init(dp, true);
> +	dp_display_host_init(dp);
>  
>  	dp_catalog_ctrl_hpd_config(dp->catalog);
>  
> @@ -1531,7 +1556,7 @@ int msm_dp_display_enable(struct msm_drm_private *priv, struct drm_encoder *enco
>  	state =  dp_display->hpd_state;
>  
>  	if (state == ST_DISPLAY_OFF)
> -		dp_display_host_init(dp_display, true);
> +		dp_display_host_phy_init(dp_display);
>  
>  	dp_display_enable(dp_display, 0);
>  
> -- 
> The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum,
> a Linux Foundation Collaborative Project
> 


More information about the Freedreno mailing list