[PATCH v3 2/2] drm/msm/dp: Implement oob_hotplug_event()

Dmitry Baryshkov dmitry.baryshkov at linaro.org
Fri Apr 22 23:07:16 UTC 2022


On 23/04/2022 01:32, Bjorn Andersson wrote:
> The Qualcomm DisplayPort driver contains traces of the necessary
> plumbing to hook up USB HPD, in the form of the dp_hpd module and the
> dp_usbpd_cb struct. Use this as basis for implementing the
> oob_hotplug_event() callback, by amending the dp_hpd module with the
> missing logic.
> 
> Overall the solution is similar to what's done downstream, but upstream
> all the code to disect the HPD notification lives on the calling side of
> drm_connector_oob_hotplug_event().
> 
> drm_connector_oob_hotplug_event() performs the lookup of the
> drm_connector based on fwnode, hence the need to assign the fwnode in
> dp_drm_connector_init().
> 
> Signed-off-by: Bjorn Andersson <bjorn.andersson at linaro.org>
> ---
> 
> Changes since v2:
> - Rebased patch
> 
>   drivers/gpu/drm/msm/dp/dp_display.c |  9 +++++++++
>   drivers/gpu/drm/msm/dp/dp_display.h |  3 +++
>   drivers/gpu/drm/msm/dp/dp_drm.c     | 11 +++++++++++
>   drivers/gpu/drm/msm/dp/dp_hpd.c     | 21 +++++++++++++++++++++
>   drivers/gpu/drm/msm/dp/dp_hpd.h     |  5 +++++
>   5 files changed, 49 insertions(+)
> 
> diff --git a/drivers/gpu/drm/msm/dp/dp_display.c b/drivers/gpu/drm/msm/dp/dp_display.c
> index a42732b67349..1019f6d8fd03 100644
> --- a/drivers/gpu/drm/msm/dp/dp_display.c
> +++ b/drivers/gpu/drm/msm/dp/dp_display.c
> @@ -449,6 +449,14 @@ static int dp_display_usbpd_configure_cb(struct device *dev)
>   	return dp_display_process_hpd_high(dp);
>   }
>   
> +void dp_display_oob_hotplug_event(struct msm_dp *dp_display,
> +				  enum drm_connector_hpd_state hpd_state)
> +{
> +	struct dp_display_private *dp = container_of(dp_display, struct dp_display_private, dp_display);
> +
> +	dp->usbpd->oob_event(dp->usbpd, hpd_state);
> +}
> +
>   static int dp_display_usbpd_disconnect_cb(struct device *dev)
>   {
>   	struct dp_display_private *dp = dev_get_dp_display_private(dev);
> @@ -1302,6 +1310,7 @@ static int dp_display_probe(struct platform_device *pdev)
>   	dp->pdev = pdev;
>   	dp->name = "drm_dp";
>   	dp->dp_display.connector_type = desc->connector_type;
> +	dp->dp_display.dev = &pdev->dev;
>   
>   	rc = dp_init_sub_modules(dp);
>   	if (rc) {
> diff --git a/drivers/gpu/drm/msm/dp/dp_display.h b/drivers/gpu/drm/msm/dp/dp_display.h
> index 7af2b186d2d9..16658270df2c 100644
> --- a/drivers/gpu/drm/msm/dp/dp_display.h
> +++ b/drivers/gpu/drm/msm/dp/dp_display.h
> @@ -11,6 +11,7 @@
>   #include "disp/msm_disp_snapshot.h"
>   
>   struct msm_dp {
> +	struct device *dev;
>   	struct drm_device *drm_dev;
>   	struct device *codec_dev;
>   	struct drm_bridge *bridge;
> @@ -40,5 +41,7 @@ bool dp_display_check_video_test(struct msm_dp *dp_display);
>   int dp_display_get_test_bpp(struct msm_dp *dp_display);
>   void dp_display_signal_audio_start(struct msm_dp *dp_display);
>   void dp_display_signal_audio_complete(struct msm_dp *dp_display);
> +void dp_display_oob_hotplug_event(struct msm_dp *dp_display,
> +				  enum drm_connector_hpd_state hpd_state);
>   
>   #endif /* _DP_DISPLAY_H_ */
> diff --git a/drivers/gpu/drm/msm/dp/dp_drm.c b/drivers/gpu/drm/msm/dp/dp_drm.c
> index 80f59cf99089..76904b1601b1 100644
> --- a/drivers/gpu/drm/msm/dp/dp_drm.c
> +++ b/drivers/gpu/drm/msm/dp/dp_drm.c
> @@ -123,6 +123,14 @@ static enum drm_mode_status dp_connector_mode_valid(
>   	return dp_display_validate_mode(dp_disp, mode->clock);
>   }
>   
> +static void dp_oob_hotplug_event(struct drm_connector *connector,
> +				 enum drm_connector_hpd_state hpd_state)
> +{
> +	struct msm_dp *dp_disp = to_dp_connector(connector)->dp_display;
> +
> +	dp_display_oob_hotplug_event(dp_disp, hpd_state);
> +}
> +
>   static const struct drm_connector_funcs dp_connector_funcs = {
>   	.detect = dp_connector_detect,
>   	.fill_modes = drm_helper_probe_single_connector_modes,
> @@ -130,6 +138,7 @@ static const struct drm_connector_funcs dp_connector_funcs = {
>   	.reset = drm_atomic_helper_connector_reset,
>   	.atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
>   	.atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
> +	.oob_hotplug_event = dp_oob_hotplug_event,

We were (are) going to switch dp driver to use drm_bridge_connector (to 
fix support for bridge chains, eDP panels, etc.

So these changes must be ported to drm_bridge_connector (or we must 
abandon/defer the idea of using the bridge_connector).

For the oob_hotplug_event() callback proper support might be as simple 
as calling drm_bridge_connector_hpd_cb().

>   };
>   
>   static const struct drm_connector_helper_funcs dp_connector_helper_funcs = {
> @@ -160,6 +169,8 @@ struct drm_connector *dp_drm_connector_init(struct msm_dp *dp_display)
>   	if (ret)
>   		return ERR_PTR(ret);
>   
> +	connector->fwnode = fwnode_handle_get(dev_fwnode(dp_display->dev));
> +

This would be much more interesting. Supporting this in a generic way 
might be tricky. But we can still set the fwnode manually from the dp code.

>   	drm_connector_helper_add(connector, &dp_connector_helper_funcs);
>   
>   	/*
> diff --git a/drivers/gpu/drm/msm/dp/dp_hpd.c b/drivers/gpu/drm/msm/dp/dp_hpd.c
> index db98a1d431eb..cdb1feea5ebf 100644
> --- a/drivers/gpu/drm/msm/dp/dp_hpd.c
> +++ b/drivers/gpu/drm/msm/dp/dp_hpd.c
> @@ -7,6 +7,8 @@
>   
>   #include <linux/slab.h>
>   #include <linux/device.h>
> +#include <drm/drm_connector.h>
> +#include <drm/drm_print.h>
>   
>   #include "dp_hpd.h"
>   
> @@ -45,6 +47,24 @@ int dp_hpd_connect(struct dp_usbpd *dp_usbpd, bool hpd)
>   	return rc;
>   }
>   
> +static void dp_hpd_oob_event(struct dp_usbpd *dp_usbpd,
> +			     enum drm_connector_hpd_state hpd_state)
> +{
> +	struct dp_hpd_private *hpd_priv = container_of(dp_usbpd, struct dp_hpd_private, dp_usbpd);
> +
> +	DRM_DEBUG_DP("hpd_state: %d connected: %d\n", hpd_state, dp_usbpd->connected);
> +
> +	if (!dp_usbpd->connected && hpd_state == DRM_CONNECTOR_HPD_HIGH) {
> +		dp_usbpd->connected = true;
> +		hpd_priv->dp_cb->configure(hpd_priv->dev);
> +	} else if (hpd_state == DRM_CONNECTOR_HPD_LOW) {
> +		dp_usbpd->connected = false;
> +		hpd_priv->dp_cb->disconnect(hpd_priv->dev);
> +	} else {
> +		hpd_priv->dp_cb->attention(hpd_priv->dev);
> +	}
> +}
> +
>   struct dp_usbpd *dp_hpd_get(struct device *dev, struct dp_usbpd_cb *cb)
>   {
>   	struct dp_hpd_private *dp_hpd;
> @@ -62,6 +82,7 @@ struct dp_usbpd *dp_hpd_get(struct device *dev, struct dp_usbpd_cb *cb)
>   	dp_hpd->dp_cb = cb;
>   
>   	dp_hpd->dp_usbpd.connect = dp_hpd_connect;
> +	dp_hpd->dp_usbpd.oob_event = dp_hpd_oob_event;
>   
>   	return &dp_hpd->dp_usbpd;
>   }
> diff --git a/drivers/gpu/drm/msm/dp/dp_hpd.h b/drivers/gpu/drm/msm/dp/dp_hpd.h
> index 8feec5aa5027..4166e5fd3156 100644
> --- a/drivers/gpu/drm/msm/dp/dp_hpd.h
> +++ b/drivers/gpu/drm/msm/dp/dp_hpd.h
> @@ -29,7 +29,9 @@ enum plug_orientation {
>    * @hpd_irq: Change in the status since last message
>    * @alt_mode_cfg_done: bool to specify alt mode status
>    * @debug_en: bool to specify debug mode
> + * @connected: cable currently connected
>    * @connect: simulate disconnect or connect for debug mode
> + * @oob_event: deliver oob event to the usbpd code
>    */
>   struct dp_usbpd {
>   	enum plug_orientation orientation;
> @@ -41,8 +43,11 @@ struct dp_usbpd {
>   	bool hpd_irq;
>   	bool alt_mode_cfg_done;
>   	bool debug_en;
> +	bool connected;
>   
>   	int (*connect)(struct dp_usbpd *dp_usbpd, bool hpd);
> +	void (*oob_event)(struct dp_usbpd *dp_usbpd,
> +			  enum drm_connector_hpd_state hpd_state);
>   };
>   
>   /**


-- 
With best wishes
Dmitry


More information about the dri-devel mailing list