Hi Dmitry,

On 9/12/2025 7:03 PM, Dmitry Baryshkov wrote:
On Fri, Sep 12, 2025 at 04:58:39PM +0800, Damon Ding wrote:
Apply drm_bridge_connector helper for Analogix DP driver.

The following changes have been made:
- Apply drm_bridge_connector helper to get rid of &drm_connector_funcs
   and &drm_connector_helper_funcs.
- Remove unnecessary parameter struct drm_connector* for callback
   &analogix_dp_plat_data.attach.
- Remove &analogix_dp_device.connector.
- Convert analogix_dp_atomic_check()/analogix_dp_detect() to
   &drm_bridge_funcs.atomic_check()/&drm_bridge_funcs.detect().
- Split analogix_dp_get_modes() into &drm_bridge_funcs.get_modes() and
   &drm_bridge_funcs.edid_read().

Signed-off-by: Damon Ding <[email protected]>

------

Changes in v2:
- For &drm_bridge.ops, remove DRM_BRIDGE_OP_HPD and add
   DRM_BRIDGE_OP_EDID.
- Add analogix_dp_bridge_edid_read().
- Move &analogix_dp_plat_data.skip_connector deletion to the previous
   patches.

Changes in v3:
- Rebase with the new devm_drm_bridge_alloc() related commit
   48f05c3b4b70 ("drm/bridge: analogix_dp: Use devm_drm_bridge_alloc()
   API").
- Expand the commit message.
- Call drm_bridge_get_modes() in analogix_dp_bridge_get_modes() if the
   bridge is available.
- Remove unnecessary parameter struct drm_connector* for callback
   &analogix_dp_plat_data.attach.
- In order to decouple the connector driver and the bridge driver, move
   the bridge connector initilization to the Rockchip and Exynos sides.

Changes in v4:
- Expand analogix_dp_bridge_detect() parameters to &drm_bridge and
   &drm_connector.
- Rename the &analogix_dp_plat_data.bridge to
   &analogix_dp_plat_data.next_bridge.

Changes in v5:
- Set the flag fo drm_bridge_attach() to DRM_BRIDGE_ATTACH_NO_CONNECTOR
   for next bridge attachment of Exynos side.
- Distinguish the &drm_bridge->ops of Analogix bridge based on whether
   the downstream device is a panel, a bridge or neither.
- Remove the calls to &analogix_dp_plat_data.get_modes().
---
  .../drm/bridge/analogix/analogix_dp_core.c    | 151 ++++++++----------
  .../drm/bridge/analogix/analogix_dp_core.h    |   1 -
  drivers/gpu/drm/exynos/exynos_dp.c            |  25 +--
  .../gpu/drm/rockchip/analogix_dp-rockchip.c   |  11 +-
  include/drm/bridge/analogix_dp.h              |   3 +-
  5 files changed, 95 insertions(+), 96 deletions(-)

diff --git a/drivers/gpu/drm/bridge/analogix/analogix_dp_core.c 
b/drivers/gpu/drm/bridge/analogix/analogix_dp_core.c
index 9bf91d7595d6..156114170c4d 100644
--- a/drivers/gpu/drm/bridge/analogix/analogix_dp_core.c
+++ b/drivers/gpu/drm/bridge/analogix/analogix_dp_core.c
@@ -856,44 +856,38 @@ static int analogix_dp_disable_psr(struct 
analogix_dp_device *dp)
        return analogix_dp_send_psr_spd(dp, &psr_vsc, true);
  }
-static int analogix_dp_get_modes(struct drm_connector *connector)
+static int analogix_dp_bridge_get_modes(struct drm_bridge *bridge, struct 
drm_connector *connector)
  {
-       struct analogix_dp_device *dp = to_dp(connector);
-       const struct drm_edid *drm_edid;
+       struct analogix_dp_device *dp = to_dp(bridge);
        int num_modes = 0;
- if (dp->plat_data->panel) {
+       if (dp->plat_data->panel)
                num_modes += drm_panel_get_modes(dp->plat_data->panel, 
connector);
-       } else {
-               drm_edid = drm_edid_read_ddc(connector, &dp->aux.ddc);
- drm_edid_connector_update(&dp->connector, drm_edid);
-
-               if (drm_edid) {
-                       num_modes += 
drm_edid_connector_add_modes(&dp->connector);
-                       drm_edid_free(drm_edid);
-               }
-       }
+       if (dp->plat_data->next_bridge)
+               num_modes += drm_bridge_get_modes(dp->plat_data->next_bridge, 
connector);

This will be already handled by drm_bridge_connector, it will use the
last bridge in chain which implements OP_EDID or OP_MODES (with OP_EDID
having higher priority).


Indeed, I will keep the drm_panel_get_modes() and remove drm_bridge_get_modes() in this commit.

return num_modes;
  }
-static struct drm_encoder *
-analogix_dp_best_encoder(struct drm_connector *connector)
+static const struct drm_edid *analogix_dp_bridge_edid_read(struct drm_bridge 
*bridge,
+                                                          struct drm_connector 
*connector)
  {
-       struct analogix_dp_device *dp = to_dp(connector);
+       struct analogix_dp_device *dp = to_dp(bridge);
+       const struct drm_edid *drm_edid = NULL;
- return dp->encoder;
-}
+       drm_edid = drm_edid_read_ddc(connector, &dp->aux.ddc);

return drm_edid_read_ddc(...);


Will do in v6.

+ return drm_edid;
+}
-static int analogix_dp_atomic_check(struct drm_connector *connector,
-                                   struct drm_atomic_state *state)
+static int analogix_dp_bridge_atomic_check(struct drm_bridge *bridge,
+                                          struct drm_bridge_state 
*bridge_state,
+                                          struct drm_crtc_state *crtc_state,
+                                          struct drm_connector_state 
*conn_state)
  {
-       struct analogix_dp_device *dp = to_dp(connector);
-       struct drm_display_info *di = &connector->display_info;
-       struct drm_connector_state *conn_state;
-       struct drm_crtc_state *crtc_state;
+       struct analogix_dp_device *dp = to_dp(bridge);
+       struct drm_display_info *di = &conn_state->connector->display_info;
        u32 mask = DRM_COLOR_FORMAT_YCBCR444 | DRM_COLOR_FORMAT_YCBCR422;
if (is_rockchip(dp->plat_data->dev_type)) {
@@ -905,35 +899,18 @@ static int analogix_dp_atomic_check(struct drm_connector 
*connector,
                }
        }
- conn_state = drm_atomic_get_new_connector_state(state, connector);
-       if (WARN_ON(!conn_state))
-               return -ENODEV;
-
        conn_state->self_refresh_aware = true;
- if (!conn_state->crtc)
-               return 0;
-
-       crtc_state = drm_atomic_get_new_crtc_state(state, conn_state->crtc);
-       if (!crtc_state)
-               return 0;
-
        if (crtc_state->self_refresh_active && !dp->psr_supported)
                return -EINVAL;
return 0;
  }
-static const struct drm_connector_helper_funcs analogix_dp_connector_helper_funcs = {
-       .get_modes = analogix_dp_get_modes,
-       .best_encoder = analogix_dp_best_encoder,
-       .atomic_check = analogix_dp_atomic_check,
-};
-
  static enum drm_connector_status
-analogix_dp_detect(struct drm_connector *connector, bool force)
+analogix_dp_bridge_detect(struct drm_bridge *bridge, struct drm_connector 
*connector)
  {
-       struct analogix_dp_device *dp = to_dp(connector);
+       struct analogix_dp_device *dp = to_dp(bridge);
        enum drm_connector_status status = connector_status_disconnected;
if (dp->plat_data->panel)
@@ -945,21 +922,11 @@ analogix_dp_detect(struct drm_connector *connector, bool 
force)
        return status;
  }
-static const struct drm_connector_funcs analogix_dp_connector_funcs = {
-       .fill_modes = drm_helper_probe_single_connector_modes,
-       .detect = analogix_dp_detect,
-       .destroy = drm_connector_cleanup,
-       .reset = drm_atomic_helper_connector_reset,
-       .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
-       .atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
-};
-
  static int analogix_dp_bridge_attach(struct drm_bridge *bridge,
                                     struct drm_encoder *encoder,
                                     enum drm_bridge_attach_flags flags)
  {
        struct analogix_dp_device *dp = to_dp(bridge);
-       struct drm_connector *connector = NULL;
        int ret = 0;
if (flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR) {
@@ -967,31 +934,8 @@ static int analogix_dp_bridge_attach(struct drm_bridge 
*bridge,
                return -EINVAL;
        }
- if (!dp->plat_data->next_bridge) {
-               connector = &dp->connector;
-               connector->polled = DRM_CONNECTOR_POLL_HPD;
-
-               ret = drm_connector_init(dp->drm_dev, connector,
-                                        &analogix_dp_connector_funcs,
-                                        DRM_MODE_CONNECTOR_eDP);
-               if (ret) {
-                       DRM_ERROR("Failed to initialize connector with drm\n");
-                       return ret;
-               }
-
-               drm_connector_helper_add(connector,
-                                        &analogix_dp_connector_helper_funcs);
-               drm_connector_attach_encoder(connector, encoder);
-       }
-
-       /*
-        * NOTE: the connector registration is implemented in analogix
-        * platform driver, that to say connector would be exist after
-        * plat_data->attch return, that's why we record the connector
-        * point after plat attached.
-        */
        if (dp->plat_data->attach) {
-               ret = dp->plat_data->attach(dp->plat_data, bridge, connector);
+               ret = dp->plat_data->attach(dp->plat_data, bridge);
                if (ret) {
                        DRM_ERROR("Failed at platform attach func\n");
                        return ret;
@@ -1095,14 +1039,21 @@ static int analogix_dp_set_bridge(struct 
analogix_dp_device *dp)
  }
static void analogix_dp_bridge_mode_set(struct drm_bridge *bridge,
+                                       struct drm_atomic_state *state,
                                        const struct drm_display_mode *mode)
  {
        struct analogix_dp_device *dp = to_dp(bridge);
-       struct drm_display_info *display_info = &dp->connector.display_info;
        struct video_info *video = &dp->video_info;
        struct device_node *dp_node = dp->dev->of_node;
+       struct drm_connector *connector;
+       struct drm_display_info *display_info;
        int vic;
+ connector = drm_atomic_get_new_connector_for_encoder(state, bridge->encoder);
+       if (!connector)
+               return;
+       display_info = &connector->display_info;
+
        /* Input video interlaces & hsync pol & vsync pol */
        video->interlaced = !!(mode->flags & DRM_MODE_FLAG_INTERLACE);
        video->v_sync_polarity = !!(mode->flags & DRM_MODE_FLAG_NVSYNC);
@@ -1186,7 +1137,7 @@ static void analogix_dp_bridge_atomic_enable(struct 
drm_bridge *bridge,
        new_crtc_state = drm_atomic_get_new_crtc_state(old_state, crtc);
        if (!new_crtc_state)
                return;
-       analogix_dp_bridge_mode_set(bridge, &new_crtc_state->adjusted_mode);
+       analogix_dp_bridge_mode_set(bridge, old_state, 
&new_crtc_state->adjusted_mode);
old_crtc_state = drm_atomic_get_old_crtc_state(old_state, crtc);
        /* Not a full enable, just disable PSR and continue */
@@ -1302,7 +1253,11 @@ static const struct drm_bridge_funcs 
analogix_dp_bridge_funcs = {
        .atomic_enable = analogix_dp_bridge_atomic_enable,
        .atomic_disable = analogix_dp_bridge_atomic_disable,
        .atomic_post_disable = analogix_dp_bridge_atomic_post_disable,
+       .atomic_check = analogix_dp_bridge_atomic_check,
        .attach = analogix_dp_bridge_attach,
+       .get_modes = analogix_dp_bridge_get_modes,
+       .edid_read = analogix_dp_bridge_edid_read,
+       .detect = analogix_dp_bridge_detect,
  };
static int analogix_dp_dt_parse_pdata(struct analogix_dp_device *dp)
@@ -1532,6 +1487,7 @@ EXPORT_SYMBOL_GPL(analogix_dp_resume);
int analogix_dp_bind(struct analogix_dp_device *dp, struct drm_device *drm_dev)
  {
+       struct drm_bridge *bridge = &dp->bridge;
        int ret;
dp->drm_dev = drm_dev;
@@ -1545,7 +1501,23 @@ int analogix_dp_bind(struct analogix_dp_device *dp, 
struct drm_device *drm_dev)
                return ret;
        }
- ret = drm_bridge_attach(dp->encoder, &dp->bridge, NULL, 0);
+       if (dp->plat_data->panel)
+               /* If the next is a panel, the EDID parsing is checked by the 
panel driver */
+               bridge->ops = DRM_BRIDGE_OP_MODES | DRM_BRIDGE_OP_DETECT;
+       else if (dp->plat_data->next_bridge)
+               /* If the next is a bridge, the supported operations depend on 
the next bridge */
+               bridge->ops = 0;

And what if the next bridge is dp_connector without a separate HPD pin?

This case was indeed not taken into account.

If the next is a bridge, it should be better to set DRM_BRIDGE_OP_DETECT and return connector_status_connected in analogix_dp_bridge_detect(). This ensures the connection status remains connected for both the dp-connector and the bridges without DRM_BRIDGE_OP_DETECT.


+       else
+               /* In DP mode, the EDID parsing and HPD detection should be 
supported */
+               bridge->ops = DRM_BRIDGE_OP_EDID | DRM_BRIDGE_OP_DETECT;
+
+       bridge->of_node = dp->dev->of_node;
+       bridge->type = DRM_MODE_CONNECTOR_eDP;
+       ret = devm_drm_bridge_add(dp->dev, &dp->bridge);
+       if (ret)
+               goto err_unregister_aux;
+
+       ret = drm_bridge_attach(dp->encoder, bridge, NULL, 0);
        if (ret) {
                DRM_ERROR("failed to create bridge (%d)\n", ret);
                goto err_unregister_aux;


Best regards,
Damon

Reply via email to