linux-kernel.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH v10 0/4] Add support for the eDP panel over aux_bus
@ 2022-04-25 11:44 Sankeerth Billakanti
  2022-04-25 11:44 ` [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus Sankeerth Billakanti
                   ` (3 more replies)
  0 siblings, 4 replies; 10+ messages in thread
From: Sankeerth Billakanti @ 2022-04-25 11:44 UTC (permalink / raw)
  To: dri-devel, linux-arm-msm, freedreno, linux-kernel, devicetree
  Cc: Sankeerth Billakanti, robdclark, seanpaul, swboyd, quic_kalyant,
	quic_abhinavk, dianders, quic_khsieh, bjorn.andersson, sean,
	airlied, daniel, dmitry.baryshkov, quic_vproddut, quic_aravindh,
	steev

This series adds support for generic eDP panel over aux_bus.

These changes are dependent on the following patches:
https://patchwork.kernel.org/project/linux-arm-msm/patch/20220211224006.1797846-5-dmitry.baryshkov@linaro.org/
https://patchwork.kernel.org/project/linux-arm-msm/patch/20220211224006.1797846-6-dmitry.baryshkov@linaro.org/

Sankeerth Billakanti (4):
  drm/msm/dp: Add eDP support via aux_bus
  drm/msm/dp: Support only IRQ_HPD and REPLUG interrupts for eDP
  drm/msm/dp: wait for hpd high before aux transaction
  drm/msm/dp: Support the eDP modes given by panel

 drivers/gpu/drm/msm/dp/dp_aux.c     |  21 +++++++-
 drivers/gpu/drm/msm/dp/dp_aux.h     |   3 +-
 drivers/gpu/drm/msm/dp/dp_catalog.c |  29 +++++++---
 drivers/gpu/drm/msm/dp/dp_catalog.h |   1 +
 drivers/gpu/drm/msm/dp/dp_display.c | 104 +++++++++++++++++++++++++++++++++---
 drivers/gpu/drm/msm/dp/dp_display.h |   1 +
 drivers/gpu/drm/msm/dp/dp_drm.c     |  21 ++++++--
 drivers/gpu/drm/msm/dp/dp_parser.c  |  23 +-------
 drivers/gpu/drm/msm/dp/dp_parser.h  |  14 ++++-
 9 files changed, 177 insertions(+), 40 deletions(-)

-- 
2.7.4


^ permalink raw reply	[flat|nested] 10+ messages in thread

* [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus
  2022-04-25 11:44 [PATCH v10 0/4] Add support for the eDP panel over aux_bus Sankeerth Billakanti
@ 2022-04-25 11:44 ` Sankeerth Billakanti
  2022-05-06 20:49   ` Dmitry Baryshkov
  2022-04-25 11:44 ` [PATCH v10 2/4] drm/msm/dp: Support only IRQ_HPD and REPLUG interrupts for eDP Sankeerth Billakanti
                   ` (2 subsequent siblings)
  3 siblings, 1 reply; 10+ messages in thread
From: Sankeerth Billakanti @ 2022-04-25 11:44 UTC (permalink / raw)
  To: dri-devel, linux-arm-msm, freedreno, linux-kernel, devicetree
  Cc: Sankeerth Billakanti, robdclark, seanpaul, swboyd, quic_kalyant,
	quic_abhinavk, dianders, quic_khsieh, bjorn.andersson, sean,
	airlied, daniel, dmitry.baryshkov, quic_vproddut, quic_aravindh,
	steev

This patch adds support for generic eDP sink through aux_bus. The eDP/DP
controller driver should support aux transactions originating from the
panel-edp driver and hence should be initialized and ready.

The panel bridge supporting the panel should be ready before the bridge
connector is initialized. The generic panel probe needs the controller
resources to be enabled to support the aux transactions originating from
the panel probe.

Signed-off-by: Sankeerth Billakanti <quic_sbillaka@quicinc.com>
Reviewed-by: Douglas Anderson <dianders@chromium.org>
Reviewed-by: Stephen Boyd <swboyd@chromium.org>
---
Changes in v10:
  - modify the error handling condition
  - modify the kernel doc

Changes in v9:
  - add comments for panel probe
  - modify the error handling checks

Changes in v8:
  - handle corner cases
  - add comment for the bridge ops

Changes in v7:
  - aux_bus is mandatory for eDP
  - connector type check modified to just check for eDP

Changes in v6:
  - Remove initialization
  - Fix aux_bus node leak
  - Split the patches

 drivers/gpu/drm/msm/dp/dp_display.c | 72 ++++++++++++++++++++++++++++++++++---
 drivers/gpu/drm/msm/dp/dp_display.h |  1 +
 drivers/gpu/drm/msm/dp/dp_drm.c     | 21 ++++++++---
 drivers/gpu/drm/msm/dp/dp_parser.c  | 23 ++----------
 drivers/gpu/drm/msm/dp/dp_parser.h  | 14 +++++++-
 5 files changed, 101 insertions(+), 30 deletions(-)

diff --git a/drivers/gpu/drm/msm/dp/dp_display.c b/drivers/gpu/drm/msm/dp/dp_display.c
index d7a19d6..f772d84 100644
--- a/drivers/gpu/drm/msm/dp/dp_display.c
+++ b/drivers/gpu/drm/msm/dp/dp_display.c
@@ -10,6 +10,7 @@
 #include <linux/component.h>
 #include <linux/of_irq.h>
 #include <linux/delay.h>
+#include <drm/dp/drm_dp_aux_bus.h>
 
 #include "msm_drv.h"
 #include "msm_kms.h"
@@ -259,14 +260,12 @@ static int dp_display_bind(struct device *dev, struct device *master,
 	dp->dp_display.drm_dev = drm;
 	priv->dp[dp->id] = &dp->dp_display;
 
-	rc = dp->parser->parse(dp->parser, dp->dp_display.connector_type);
+	rc = dp->parser->parse(dp->parser);
 	if (rc) {
 		DRM_ERROR("device tree parsing failed\n");
 		goto end;
 	}
 
-	dp->dp_display.next_bridge = dp->parser->next_bridge;
-
 	dp->aux->drm_dev = drm;
 	rc = dp_aux_register(dp->aux);
 	if (rc) {
@@ -1319,6 +1318,8 @@ 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.is_edp =
+		(dp->dp_display.connector_type == DRM_MODE_CONNECTOR_eDP);
 
 	rc = dp_init_sub_modules(dp);
 	if (rc) {
@@ -1508,7 +1509,8 @@ void msm_dp_irq_postinstall(struct msm_dp *dp_display)
 
 	dp_hpd_event_setup(dp);
 
-	dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
+	if (!dp_display->is_edp)
+		dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
 }
 
 void msm_dp_debugfs_init(struct msm_dp *dp_display, struct drm_minor *minor)
@@ -1530,6 +1532,64 @@ void msm_dp_debugfs_init(struct msm_dp *dp_display, struct drm_minor *minor)
 	}
 }
 
+static int dp_display_get_next_bridge(struct msm_dp *dp)
+{
+	int rc;
+	struct dp_display_private *dp_priv;
+	struct device_node *aux_bus;
+	struct device *dev;
+
+	dp_priv = container_of(dp, struct dp_display_private, dp_display);
+	dev = &dp_priv->pdev->dev;
+	aux_bus = of_get_child_by_name(dev->of_node, "aux-bus");
+
+	if (aux_bus && dp->is_edp) {
+		dp_display_host_init(dp_priv);
+		dp_catalog_ctrl_hpd_config(dp_priv->catalog);
+		dp_display_host_phy_init(dp_priv);
+		enable_irq(dp_priv->irq);
+
+		/*
+		 * The code below assumes that the panel will finish probing
+		 * by the time devm_of_dp_aux_populate_ep_devices() returns.
+		 * This isn't a great assumption since it will fail if the
+		 * panel driver is probed asynchronously but is the best we
+		 * can do without a bigger driver reorganization.
+		 */
+		rc = devm_of_dp_aux_populate_ep_devices(dp_priv->aux);
+		of_node_put(aux_bus);
+		if (rc)
+			goto error;
+	} else if (dp->is_edp) {
+		DRM_ERROR("eDP aux_bus not found\n");
+		return -ENODEV;
+	}
+
+	/*
+	 * External bridges are mandatory for eDP interfaces: one has to
+	 * provide at least an eDP panel (which gets wrapped into panel-bridge).
+	 *
+	 * For DisplayPort interfaces external bridges are optional, so
+	 * silently ignore an error if one is not present (-ENODEV).
+	 */
+	rc = dp_parser_find_next_bridge(dp_priv->parser);
+	if (!dp->is_edp && rc == -ENODEV)
+		return 0;
+
+	if (!rc) {
+		dp->next_bridge = dp_priv->parser->next_bridge;
+		return 0;
+	}
+
+error:
+	if (dp->is_edp) {
+		disable_irq(dp_priv->irq);
+		dp_display_host_phy_exit(dp_priv);
+		dp_display_host_deinit(dp_priv);
+	}
+	return rc;
+}
+
 int msm_dp_modeset_init(struct msm_dp *dp_display, struct drm_device *dev,
 			struct drm_encoder *encoder)
 {
@@ -1553,6 +1613,10 @@ int msm_dp_modeset_init(struct msm_dp *dp_display, struct drm_device *dev,
 
 	dp_display->encoder = encoder;
 
+	ret = dp_display_get_next_bridge(dp_display);
+	if (ret)
+		return ret;
+
 	dp_display->bridge = dp_bridge_init(dp_display, dev, encoder);
 	if (IS_ERR(dp_display->bridge)) {
 		ret = PTR_ERR(dp_display->bridge);
diff --git a/drivers/gpu/drm/msm/dp/dp_display.h b/drivers/gpu/drm/msm/dp/dp_display.h
index 49a1d89..1377cc3 100644
--- a/drivers/gpu/drm/msm/dp/dp_display.h
+++ b/drivers/gpu/drm/msm/dp/dp_display.h
@@ -21,6 +21,7 @@ struct msm_dp {
 	bool audio_enabled;
 	bool power_on;
 	unsigned int connector_type;
+	bool is_edp;
 
 	hdmi_codec_plugged_cb plugged_cb;
 
diff --git a/drivers/gpu/drm/msm/dp/dp_drm.c b/drivers/gpu/drm/msm/dp/dp_drm.c
index 7ce1aca..8a75c55 100644
--- a/drivers/gpu/drm/msm/dp/dp_drm.c
+++ b/drivers/gpu/drm/msm/dp/dp_drm.c
@@ -114,10 +114,23 @@ struct drm_bridge *dp_bridge_init(struct msm_dp *dp_display, struct drm_device *
 	bridge->funcs = &dp_bridge_ops;
 	bridge->type = dp_display->connector_type;
 
-	bridge->ops =
-		DRM_BRIDGE_OP_DETECT |
-		DRM_BRIDGE_OP_HPD |
-		DRM_BRIDGE_OP_MODES;
+	/*
+	 * Many ops only make sense for DP. Why?
+	 * - Detect/HPD are used by DRM to know if a display is _physically_
+	 *   there, not whether the display is powered on / finished initting.
+	 *   On eDP we assume the display is always there because you can't
+	 *   know until power is applied. If we don't implement the ops DRM will
+	 *   assume our display is always there.
+	 * - Currently eDP mode reading is driven by the panel driver. This
+	 *   allows the panel driver to properly power itself on to read the
+	 *   modes.
+	 */
+	if (!dp_display->is_edp) {
+		bridge->ops =
+			DRM_BRIDGE_OP_DETECT |
+			DRM_BRIDGE_OP_HPD |
+			DRM_BRIDGE_OP_MODES;
+	}
 
 	rc = drm_bridge_attach(encoder, bridge, NULL, DRM_BRIDGE_ATTACH_NO_CONNECTOR);
 	if (rc) {
diff --git a/drivers/gpu/drm/msm/dp/dp_parser.c b/drivers/gpu/drm/msm/dp/dp_parser.c
index 1056b8d..4bdbf91 100644
--- a/drivers/gpu/drm/msm/dp/dp_parser.c
+++ b/drivers/gpu/drm/msm/dp/dp_parser.c
@@ -265,7 +265,7 @@ static int dp_parser_clock(struct dp_parser *parser)
 	return 0;
 }
 
-static int dp_parser_find_next_bridge(struct dp_parser *parser)
+int dp_parser_find_next_bridge(struct dp_parser *parser)
 {
 	struct device *dev = &parser->pdev->dev;
 	struct drm_bridge *bridge;
@@ -279,7 +279,7 @@ static int dp_parser_find_next_bridge(struct dp_parser *parser)
 	return 0;
 }
 
-static int dp_parser_parse(struct dp_parser *parser, int connector_type)
+static int dp_parser_parse(struct dp_parser *parser)
 {
 	int rc = 0;
 
@@ -300,25 +300,6 @@ static int dp_parser_parse(struct dp_parser *parser, int connector_type)
 	if (rc)
 		return rc;
 
-	/*
-	 * External bridges are mandatory for eDP interfaces: one has to
-	 * provide at least an eDP panel (which gets wrapped into panel-bridge).
-	 *
-	 * For DisplayPort interfaces external bridges are optional, so
-	 * silently ignore an error if one is not present (-ENODEV).
-	 */
-	rc = dp_parser_find_next_bridge(parser);
-	if (rc == -ENODEV) {
-		if (connector_type == DRM_MODE_CONNECTOR_eDP) {
-			DRM_ERROR("eDP: next bridge is not present\n");
-			return rc;
-		}
-	} else if (rc) {
-		if (rc != -EPROBE_DEFER)
-			DRM_ERROR("DP: error parsing next bridge: %d\n", rc);
-		return rc;
-	}
-
 	/* Map the corresponding regulator information according to
 	 * version. Currently, since we only have one supported platform,
 	 * mapping the regulator directly.
diff --git a/drivers/gpu/drm/msm/dp/dp_parser.h b/drivers/gpu/drm/msm/dp/dp_parser.h
index d371bae..3a4d797 100644
--- a/drivers/gpu/drm/msm/dp/dp_parser.h
+++ b/drivers/gpu/drm/msm/dp/dp_parser.h
@@ -125,7 +125,7 @@ struct dp_parser {
 	u32 max_dp_lanes;
 	struct drm_bridge *next_bridge;
 
-	int (*parse)(struct dp_parser *parser, int connector_type);
+	int (*parse)(struct dp_parser *parser);
 };
 
 /**
@@ -141,4 +141,16 @@ struct dp_parser {
  */
 struct dp_parser *dp_parser_get(struct platform_device *pdev);
 
+/**
+ * dp_parser_find_next_bridge() - find an additional bridge to DP
+ *
+ * @parser: dp_parser data from client
+ *
+ * This function is used to find any additional bridge attached to
+ * the DP controller. The eDP interface requires a panel bridge.
+ *
+ * Return: 0 if able to get the bridge, otherwise negative errno for failure.
+ */
+int dp_parser_find_next_bridge(struct dp_parser *parser);
+
 #endif
-- 
2.7.4


^ permalink raw reply related	[flat|nested] 10+ messages in thread

* [PATCH v10 2/4] drm/msm/dp: Support only IRQ_HPD and REPLUG interrupts for eDP
  2022-04-25 11:44 [PATCH v10 0/4] Add support for the eDP panel over aux_bus Sankeerth Billakanti
  2022-04-25 11:44 ` [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus Sankeerth Billakanti
@ 2022-04-25 11:44 ` Sankeerth Billakanti
  2022-04-25 11:44 ` [PATCH v10 3/4] drm/msm/dp: wait for hpd high before aux transaction Sankeerth Billakanti
  2022-04-25 11:44 ` [PATCH v10 4/4] drm/msm/dp: Support the eDP modes given by panel Sankeerth Billakanti
  3 siblings, 0 replies; 10+ messages in thread
From: Sankeerth Billakanti @ 2022-04-25 11:44 UTC (permalink / raw)
  To: dri-devel, linux-arm-msm, freedreno, linux-kernel, devicetree
  Cc: Sankeerth Billakanti, robdclark, seanpaul, swboyd, quic_kalyant,
	quic_abhinavk, dianders, quic_khsieh, bjorn.andersson, sean,
	airlied, daniel, dmitry.baryshkov, quic_vproddut, quic_aravindh,
	steev

The panel-edp enables the eDP panel power during probe, get_modes
and pre-enable. The eDP connect and disconnect interrupts for the eDP/DP
controller are directly dependent on panel power. As eDP display can be
assumed as always connected, the controller driver can skip the eDP
connect and disconnect interrupts. Any disruption in the link status
will be indicated via the IRQ_HPD interrupts.

So, the eDP controller driver can just enable the IRQ_HPD and replug
interrupts. The DP controller driver still needs to enable all the
interrupts.

Signed-off-by: Sankeerth Billakanti <quic_sbillaka@quicinc.com>
Reviewed-by: Douglas Anderson <dianders@chromium.org>
Reviewed-by: Stephen Boyd <swboyd@chromium.org>
---
Changes in v10:
  - none

Changes in v9:
  - add comment explaining the interrupt status register

Changes in v8:
  - add comment explaining the interrupt status return

Changes in v7:
  - reordered the patch in the series
  - modified the return statement for isr
  - connector check modified to just check for eDP

 drivers/gpu/drm/msm/dp/dp_catalog.c | 16 ++++++++++------
 drivers/gpu/drm/msm/dp/dp_display.c | 22 +++++++++++++++++++++-
 2 files changed, 31 insertions(+), 7 deletions(-)

diff --git a/drivers/gpu/drm/msm/dp/dp_catalog.c b/drivers/gpu/drm/msm/dp/dp_catalog.c
index fac815f..df9670d 100644
--- a/drivers/gpu/drm/msm/dp/dp_catalog.c
+++ b/drivers/gpu/drm/msm/dp/dp_catalog.c
@@ -569,10 +569,6 @@ void dp_catalog_ctrl_hpd_config(struct dp_catalog *dp_catalog)
 
 	u32 reftimer = dp_read_aux(catalog, REG_DP_DP_HPD_REFTIMER);
 
-	/* enable HPD plug and unplug interrupts */
-	dp_catalog_hpd_config_intr(dp_catalog,
-		DP_DP_HPD_PLUG_INT_MASK | DP_DP_HPD_UNPLUG_INT_MASK, true);
-
 	/* Configure REFTIMER and enable it */
 	reftimer |= DP_DP_HPD_REFTIMER_ENABLE;
 	dp_write_aux(catalog, REG_DP_DP_HPD_REFTIMER, reftimer);
@@ -599,13 +595,21 @@ u32 dp_catalog_hpd_get_intr_status(struct dp_catalog *dp_catalog)
 {
 	struct dp_catalog_private *catalog = container_of(dp_catalog,
 				struct dp_catalog_private, dp_catalog);
-	int isr = 0;
+	int isr, mask;
 
 	isr = dp_read_aux(catalog, REG_DP_DP_HPD_INT_STATUS);
 	dp_write_aux(catalog, REG_DP_DP_HPD_INT_ACK,
 				 (isr & DP_DP_HPD_INT_MASK));
+	mask = dp_read_aux(catalog, REG_DP_DP_HPD_INT_MASK);
 
-	return isr;
+	/*
+	 * We only want to return interrupts that are unmasked to the caller.
+	 * However, the interrupt status field also contains other
+	 * informational bits about the HPD state status, so we only mask
+	 * out the part of the register that tells us about which interrupts
+	 * are pending.
+	 */
+	return isr & (mask | ~DP_DP_HPD_INT_MASK);
 }
 
 int dp_catalog_ctrl_get_interrupt(struct dp_catalog *dp_catalog)
diff --git a/drivers/gpu/drm/msm/dp/dp_display.c b/drivers/gpu/drm/msm/dp/dp_display.c
index f772d84..d25fa1f 100644
--- a/drivers/gpu/drm/msm/dp/dp_display.c
+++ b/drivers/gpu/drm/msm/dp/dp_display.c
@@ -683,7 +683,8 @@ static int dp_hpd_unplug_handle(struct dp_display_private *dp, u32 data)
 	dp_display_handle_plugged_change(&dp->dp_display, false);
 
 	/* enable HDP plug interrupt to prepare for next plugin */
-	dp_catalog_hpd_config_intr(dp->catalog, DP_DP_HPD_PLUG_INT_MASK, true);
+	if (!dp->dp_display.is_edp)
+		dp_catalog_hpd_config_intr(dp->catalog, DP_DP_HPD_PLUG_INT_MASK, true);
 
 	DRM_DEBUG_DP("After, type=%d hpd_state=%d\n",
 			dp->dp_display.connector_type, state);
@@ -1096,6 +1097,13 @@ static void dp_display_config_hpd(struct dp_display_private *dp)
 	dp_display_host_init(dp);
 	dp_catalog_ctrl_hpd_config(dp->catalog);
 
+	/* Enable plug and unplug interrupts only for external DisplayPort */
+	if (!dp->dp_display.is_edp)
+		dp_catalog_hpd_config_intr(dp->catalog,
+				DP_DP_HPD_PLUG_INT_MASK |
+				DP_DP_HPD_UNPLUG_INT_MASK,
+				true);
+
 	/* Enable interrupt first time
 	 * we are leaving dp clocks on during disconnect
 	 * and never disable interrupt
@@ -1381,6 +1389,12 @@ static int dp_pm_resume(struct device *dev)
 	dp_catalog_ctrl_hpd_config(dp->catalog);
 
 
+	if (!dp->dp_display.is_edp)
+		dp_catalog_hpd_config_intr(dp->catalog,
+				DP_DP_HPD_PLUG_INT_MASK |
+				DP_DP_HPD_UNPLUG_INT_MASK,
+				true);
+
 	if (dp_catalog_link_is_connected(dp->catalog)) {
 		/*
 		 * set sink to normal operation mode -- D0
@@ -1658,6 +1672,9 @@ void dp_bridge_enable(struct drm_bridge *drm_bridge)
 		return;
 	}
 
+	if (dp->is_edp)
+		dp_hpd_plug_handle(dp_display, 0);
+
 	mutex_lock(&dp_display->event_mutex);
 
 	/* stop sentinel checking */
@@ -1722,6 +1739,9 @@ void dp_bridge_post_disable(struct drm_bridge *drm_bridge)
 
 	dp_display = container_of(dp, struct dp_display_private, dp_display);
 
+	if (dp->is_edp)
+		dp_hpd_unplug_handle(dp_display, 0);
+
 	mutex_lock(&dp_display->event_mutex);
 
 	/* stop sentinel checking */
-- 
2.7.4


^ permalink raw reply related	[flat|nested] 10+ messages in thread

* [PATCH v10 3/4] drm/msm/dp: wait for hpd high before aux transaction
  2022-04-25 11:44 [PATCH v10 0/4] Add support for the eDP panel over aux_bus Sankeerth Billakanti
  2022-04-25 11:44 ` [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus Sankeerth Billakanti
  2022-04-25 11:44 ` [PATCH v10 2/4] drm/msm/dp: Support only IRQ_HPD and REPLUG interrupts for eDP Sankeerth Billakanti
@ 2022-04-25 11:44 ` Sankeerth Billakanti
  2022-04-25 11:44 ` [PATCH v10 4/4] drm/msm/dp: Support the eDP modes given by panel Sankeerth Billakanti
  3 siblings, 0 replies; 10+ messages in thread
From: Sankeerth Billakanti @ 2022-04-25 11:44 UTC (permalink / raw)
  To: dri-devel, linux-arm-msm, freedreno, linux-kernel, devicetree
  Cc: Sankeerth Billakanti, robdclark, seanpaul, swboyd, quic_kalyant,
	quic_abhinavk, dianders, quic_khsieh, bjorn.andersson, sean,
	airlied, daniel, dmitry.baryshkov, quic_vproddut, quic_aravindh,
	steev

The source device should ensure the sink is ready before proceeding to
read the sink capability or perform any aux transactions. The sink
will indicate its readiness by asserting the HPD line. The controller
driver needs to wait for the hpd line to be asserted by the sink before
it performs any aux transactions.

The eDP sink is assumed to be always connected. It needs power from the
source and its HPD line will be asserted only after the panel is powered
on. The panel power will be enabled from the panel-edp driver and only
after that, the hpd line will be asserted.

Whereas for DP, the sink can be hotplugged and unplugged anytime. The hpd
line gets asserted to indicate the sink is connected and ready. Hence
there is no need to wait for the hpd line to be asserted for a DP sink.

Signed-off-by: Sankeerth Billakanti <quic_sbillaka@quicinc.com>
Reviewed-by: Douglas Anderson <dianders@chromium.org>
Reviewed-by: Stephen Boyd <swboyd@chromium.org>
---
These changes may be handled in is_hpd_asserted when
https://lore.kernel.org/r/20220408193536.RFC.3.Icf57bb12233a47727013c6ab69eebf803e22ebc1@changeid/
is accepted upstream.

Changes in v10:
  - none

Changes in v9:
  - none

Changes in v8:
  - correct the indentation

Changes in v7:
  - add a comment to say why the wait is done for eDP
  - correct the commit text

Changes in v6:
  - Wait for hpd high only for eDP
  - Split into smaller patches

 drivers/gpu/drm/msm/dp/dp_aux.c     | 21 ++++++++++++++++++++-
 drivers/gpu/drm/msm/dp/dp_aux.h     |  3 ++-
 drivers/gpu/drm/msm/dp/dp_catalog.c | 13 +++++++++++++
 drivers/gpu/drm/msm/dp/dp_catalog.h |  1 +
 drivers/gpu/drm/msm/dp/dp_display.c |  2 +-
 5 files changed, 37 insertions(+), 3 deletions(-)

diff --git a/drivers/gpu/drm/msm/dp/dp_aux.c b/drivers/gpu/drm/msm/dp/dp_aux.c
index 6d36f63..d030a93 100644
--- a/drivers/gpu/drm/msm/dp/dp_aux.c
+++ b/drivers/gpu/drm/msm/dp/dp_aux.c
@@ -34,6 +34,7 @@ struct dp_aux_private {
 	bool no_send_addr;
 	bool no_send_stop;
 	bool initted;
+	bool is_edp;
 	u32 offset;
 	u32 segment;
 
@@ -337,6 +338,22 @@ static ssize_t dp_aux_transfer(struct drm_dp_aux *dp_aux,
 		goto exit;
 	}
 
+	/*
+	 * For eDP it's important to give a reasonably long wait here for HPD
+	 * to be asserted. This is because the panel driver may have _just_
+	 * turned on the panel and then tried to do an AUX transfer. The panel
+	 * driver has no way of knowing when the panel is ready, so it's up
+	 * to us to wait. For DP we never get into this situation so let's
+	 * avoid ever doing the extra long wait for DP.
+	 */
+	if (aux->is_edp) {
+		ret = dp_catalog_aux_wait_for_hpd_connect_state(aux->catalog);
+		if (ret) {
+			DRM_DEBUG_DP("Panel not ready for aux transactions\n");
+			goto exit;
+		}
+	}
+
 	dp_aux_update_offset_and_segment(aux, msg);
 	dp_aux_transfer_helper(aux, msg, true);
 
@@ -491,7 +508,8 @@ void dp_aux_unregister(struct drm_dp_aux *dp_aux)
 	drm_dp_aux_unregister(dp_aux);
 }
 
-struct drm_dp_aux *dp_aux_get(struct device *dev, struct dp_catalog *catalog)
+struct drm_dp_aux *dp_aux_get(struct device *dev, struct dp_catalog *catalog,
+			      bool is_edp)
 {
 	struct dp_aux_private *aux;
 
@@ -506,6 +524,7 @@ struct drm_dp_aux *dp_aux_get(struct device *dev, struct dp_catalog *catalog)
 
 	init_completion(&aux->comp);
 	aux->cmd_busy = false;
+	aux->is_edp = is_edp;
 	mutex_init(&aux->mutex);
 
 	aux->dev = dev;
diff --git a/drivers/gpu/drm/msm/dp/dp_aux.h b/drivers/gpu/drm/msm/dp/dp_aux.h
index 82afc8d..5a50c08 100644
--- a/drivers/gpu/drm/msm/dp/dp_aux.h
+++ b/drivers/gpu/drm/msm/dp/dp_aux.h
@@ -16,7 +16,8 @@ void dp_aux_init(struct drm_dp_aux *dp_aux);
 void dp_aux_deinit(struct drm_dp_aux *dp_aux);
 void dp_aux_reconfig(struct drm_dp_aux *dp_aux);
 
-struct drm_dp_aux *dp_aux_get(struct device *dev, struct dp_catalog *catalog);
+struct drm_dp_aux *dp_aux_get(struct device *dev, struct dp_catalog *catalog,
+			      bool is_edp);
 void dp_aux_put(struct drm_dp_aux *aux);
 
 #endif /*__DP_AUX_H_*/
diff --git a/drivers/gpu/drm/msm/dp/dp_catalog.c b/drivers/gpu/drm/msm/dp/dp_catalog.c
index df9670d..0c6a96e 100644
--- a/drivers/gpu/drm/msm/dp/dp_catalog.c
+++ b/drivers/gpu/drm/msm/dp/dp_catalog.c
@@ -242,6 +242,19 @@ void dp_catalog_aux_update_cfg(struct dp_catalog *dp_catalog)
 	phy_calibrate(phy);
 }
 
+int dp_catalog_aux_wait_for_hpd_connect_state(struct dp_catalog *dp_catalog)
+{
+	u32 state;
+	struct dp_catalog_private *catalog = container_of(dp_catalog,
+				struct dp_catalog_private, dp_catalog);
+
+	/* poll for hpd connected status every 2ms and timeout after 500ms */
+	return readl_poll_timeout(catalog->io->dp_controller.aux.base +
+				REG_DP_DP_HPD_INT_STATUS,
+				state, state & DP_DP_HPD_STATE_STATUS_CONNECTED,
+				2000, 500000);
+}
+
 static void dump_regs(void __iomem *base, int len)
 {
 	int i;
diff --git a/drivers/gpu/drm/msm/dp/dp_catalog.h b/drivers/gpu/drm/msm/dp/dp_catalog.h
index 7dea101..45140a3 100644
--- a/drivers/gpu/drm/msm/dp/dp_catalog.h
+++ b/drivers/gpu/drm/msm/dp/dp_catalog.h
@@ -84,6 +84,7 @@ int dp_catalog_aux_clear_hw_interrupts(struct dp_catalog *dp_catalog);
 void dp_catalog_aux_reset(struct dp_catalog *dp_catalog);
 void dp_catalog_aux_enable(struct dp_catalog *dp_catalog, bool enable);
 void dp_catalog_aux_update_cfg(struct dp_catalog *dp_catalog);
+int dp_catalog_aux_wait_for_hpd_connect_state(struct dp_catalog *dp_catalog);
 u32 dp_catalog_aux_get_irq(struct dp_catalog *dp_catalog);
 
 /* DP Controller APIs */
diff --git a/drivers/gpu/drm/msm/dp/dp_display.c b/drivers/gpu/drm/msm/dp/dp_display.c
index d25fa1f..fd1dddb9 100644
--- a/drivers/gpu/drm/msm/dp/dp_display.c
+++ b/drivers/gpu/drm/msm/dp/dp_display.c
@@ -805,7 +805,7 @@ static int dp_init_sub_modules(struct dp_display_private *dp)
 		goto error;
 	}
 
-	dp->aux = dp_aux_get(dev, dp->catalog);
+	dp->aux = dp_aux_get(dev, dp->catalog, dp->dp_display.is_edp);
 	if (IS_ERR(dp->aux)) {
 		rc = PTR_ERR(dp->aux);
 		DRM_ERROR("failed to initialize aux, rc = %d\n", rc);
-- 
2.7.4


^ permalink raw reply related	[flat|nested] 10+ messages in thread

* [PATCH v10 4/4] drm/msm/dp: Support the eDP modes given by panel
  2022-04-25 11:44 [PATCH v10 0/4] Add support for the eDP panel over aux_bus Sankeerth Billakanti
                   ` (2 preceding siblings ...)
  2022-04-25 11:44 ` [PATCH v10 3/4] drm/msm/dp: wait for hpd high before aux transaction Sankeerth Billakanti
@ 2022-04-25 11:44 ` Sankeerth Billakanti
  3 siblings, 0 replies; 10+ messages in thread
From: Sankeerth Billakanti @ 2022-04-25 11:44 UTC (permalink / raw)
  To: dri-devel, linux-arm-msm, freedreno, linux-kernel, devicetree
  Cc: Sankeerth Billakanti, robdclark, seanpaul, swboyd, quic_kalyant,
	quic_abhinavk, dianders, quic_khsieh, bjorn.andersson, sean,
	airlied, daniel, dmitry.baryshkov, quic_vproddut, quic_aravindh,
	steev

The eDP controller does not have a reliable way keep panel
powered on to read the sink capabilities. So, the controller
driver cannot validate if a mode can be supported by the
source. We will rely on the panel driver to populate only
the supported modes for now.

Signed-off-by: Sankeerth Billakanti <quic_sbillaka@quicinc.com>
Reviewed-by: Douglas Anderson <dianders@chromium.org>
Reviewed-by: Stephen Boyd <swboyd@chromium.org>
---
Changes in v10:
  - none 

Changes in v9:
  - none

Changes in v8:
  - add the drm/msm/dp tag in the commit title

 drivers/gpu/drm/msm/dp/dp_display.c | 8 ++++++++
 1 file changed, 8 insertions(+)

diff --git a/drivers/gpu/drm/msm/dp/dp_display.c b/drivers/gpu/drm/msm/dp/dp_display.c
index fd1dddb9..637fb63 100644
--- a/drivers/gpu/drm/msm/dp/dp_display.c
+++ b/drivers/gpu/drm/msm/dp/dp_display.c
@@ -998,6 +998,14 @@ enum drm_mode_status dp_bridge_mode_valid(struct drm_bridge *bridge,
 		return -EINVAL;
 	}
 
+	/*
+	 * The eDP controller currently does not have a reliable way of
+	 * enabling panel power to read sink capabilities. So, we rely
+	 * on the panel driver to populate only supported modes for now.
+	 */
+	if (dp->is_edp)
+		return MODE_OK;
+
 	if ((dp->max_pclk_khz <= 0) ||
 			(dp->max_pclk_khz > DP_MAX_PIXEL_CLK_KHZ) ||
 			(mode->clock > dp->max_pclk_khz))
-- 
2.7.4


^ permalink raw reply related	[flat|nested] 10+ messages in thread

* Re: [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus
  2022-04-25 11:44 ` [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus Sankeerth Billakanti
@ 2022-05-06 20:49   ` Dmitry Baryshkov
  2022-05-06 21:03     ` Abhinav Kumar
  0 siblings, 1 reply; 10+ messages in thread
From: Dmitry Baryshkov @ 2022-05-06 20:49 UTC (permalink / raw)
  To: Sankeerth Billakanti, dri-devel, linux-arm-msm, freedreno,
	linux-kernel, devicetree
  Cc: robdclark, seanpaul, swboyd, quic_kalyant, quic_abhinavk,
	dianders, quic_khsieh, bjorn.andersson, sean, airlied, daniel,
	quic_vproddut, quic_aravindh, steev

On 25/04/2022 14:44, Sankeerth Billakanti wrote:
> This patch adds support for generic eDP sink through aux_bus. The eDP/DP
> controller driver should support aux transactions originating from the
> panel-edp driver and hence should be initialized and ready.
> 
> The panel bridge supporting the panel should be ready before the bridge
> connector is initialized. The generic panel probe needs the controller
> resources to be enabled to support the aux transactions originating from
> the panel probe.
> 
> Signed-off-by: Sankeerth Billakanti <quic_sbillaka@quicinc.com>
> Reviewed-by: Douglas Anderson <dianders@chromium.org>
> Reviewed-by: Stephen Boyd <swboyd@chromium.org>

An additional side effect from this patch. Previously missing panel 
would have caused the bind error. Now it is the dp_modeset_init error, 
which translates to kms_hw_init returning -517. I kind ask to move the 
next_bridge acquisition back to the dp_bind in one of the followup patches.

> ---
> Changes in v10:
>    - modify the error handling condition
>    - modify the kernel doc
> 
> Changes in v9:
>    - add comments for panel probe
>    - modify the error handling checks
> 
> Changes in v8:
>    - handle corner cases
>    - add comment for the bridge ops
> 
> Changes in v7:
>    - aux_bus is mandatory for eDP
>    - connector type check modified to just check for eDP
> 
> Changes in v6:
>    - Remove initialization
>    - Fix aux_bus node leak
>    - Split the patches
> 
>   drivers/gpu/drm/msm/dp/dp_display.c | 72 ++++++++++++++++++++++++++++++++++---
>   drivers/gpu/drm/msm/dp/dp_display.h |  1 +
>   drivers/gpu/drm/msm/dp/dp_drm.c     | 21 ++++++++---
>   drivers/gpu/drm/msm/dp/dp_parser.c  | 23 ++----------
>   drivers/gpu/drm/msm/dp/dp_parser.h  | 14 +++++++-
>   5 files changed, 101 insertions(+), 30 deletions(-)
> 
> diff --git a/drivers/gpu/drm/msm/dp/dp_display.c b/drivers/gpu/drm/msm/dp/dp_display.c
> index d7a19d6..f772d84 100644
> --- a/drivers/gpu/drm/msm/dp/dp_display.c
> +++ b/drivers/gpu/drm/msm/dp/dp_display.c
> @@ -10,6 +10,7 @@
>   #include <linux/component.h>
>   #include <linux/of_irq.h>
>   #include <linux/delay.h>
> +#include <drm/dp/drm_dp_aux_bus.h>
>   
>   #include "msm_drv.h"
>   #include "msm_kms.h"
> @@ -259,14 +260,12 @@ static int dp_display_bind(struct device *dev, struct device *master,
>   	dp->dp_display.drm_dev = drm;
>   	priv->dp[dp->id] = &dp->dp_display;
>   
> -	rc = dp->parser->parse(dp->parser, dp->dp_display.connector_type);
> +	rc = dp->parser->parse(dp->parser);
>   	if (rc) {
>   		DRM_ERROR("device tree parsing failed\n");
>   		goto end;
>   	}
>   
> -	dp->dp_display.next_bridge = dp->parser->next_bridge;
> -
>   	dp->aux->drm_dev = drm;
>   	rc = dp_aux_register(dp->aux);
>   	if (rc) {
> @@ -1319,6 +1318,8 @@ 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.is_edp =
> +		(dp->dp_display.connector_type == DRM_MODE_CONNECTOR_eDP);
>   
>   	rc = dp_init_sub_modules(dp);
>   	if (rc) {
> @@ -1508,7 +1509,8 @@ void msm_dp_irq_postinstall(struct msm_dp *dp_display)
>   
>   	dp_hpd_event_setup(dp);
>   
> -	dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
> +	if (!dp_display->is_edp)
> +		dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
>   }
>   
>   void msm_dp_debugfs_init(struct msm_dp *dp_display, struct drm_minor *minor)
> @@ -1530,6 +1532,64 @@ void msm_dp_debugfs_init(struct msm_dp *dp_display, struct drm_minor *minor)
>   	}
>   }
>   
> +static int dp_display_get_next_bridge(struct msm_dp *dp)
> +{
> +	int rc;
> +	struct dp_display_private *dp_priv;
> +	struct device_node *aux_bus;
> +	struct device *dev;
> +
> +	dp_priv = container_of(dp, struct dp_display_private, dp_display);
> +	dev = &dp_priv->pdev->dev;
> +	aux_bus = of_get_child_by_name(dev->of_node, "aux-bus");
> +
> +	if (aux_bus && dp->is_edp) {
> +		dp_display_host_init(dp_priv);
> +		dp_catalog_ctrl_hpd_config(dp_priv->catalog);
> +		dp_display_host_phy_init(dp_priv);
> +		enable_irq(dp_priv->irq);
> +
> +		/*
> +		 * The code below assumes that the panel will finish probing
> +		 * by the time devm_of_dp_aux_populate_ep_devices() returns.
> +		 * This isn't a great assumption since it will fail if the
> +		 * panel driver is probed asynchronously but is the best we
> +		 * can do without a bigger driver reorganization.
> +		 */
> +		rc = devm_of_dp_aux_populate_ep_devices(dp_priv->aux);
> +		of_node_put(aux_bus);
> +		if (rc)
> +			goto error;
> +	} else if (dp->is_edp) {
> +		DRM_ERROR("eDP aux_bus not found\n");
> +		return -ENODEV;
> +	}
> +
> +	/*
> +	 * External bridges are mandatory for eDP interfaces: one has to
> +	 * provide at least an eDP panel (which gets wrapped into panel-bridge).
> +	 *
> +	 * For DisplayPort interfaces external bridges are optional, so
> +	 * silently ignore an error if one is not present (-ENODEV).
> +	 */
> +	rc = dp_parser_find_next_bridge(dp_priv->parser);
> +	if (!dp->is_edp && rc == -ENODEV)
> +		return 0;
> +
> +	if (!rc) {
> +		dp->next_bridge = dp_priv->parser->next_bridge;
> +		return 0;
> +	}
> +
> +error:
> +	if (dp->is_edp) {
> +		disable_irq(dp_priv->irq);
> +		dp_display_host_phy_exit(dp_priv);
> +		dp_display_host_deinit(dp_priv);
> +	}
> +	return rc;
> +}
> +
>   int msm_dp_modeset_init(struct msm_dp *dp_display, struct drm_device *dev,
>   			struct drm_encoder *encoder)
>   {
> @@ -1553,6 +1613,10 @@ int msm_dp_modeset_init(struct msm_dp *dp_display, struct drm_device *dev,
>   
>   	dp_display->encoder = encoder;
>   
> +	ret = dp_display_get_next_bridge(dp_display);
> +	if (ret)
> +		return ret;
> +
>   	dp_display->bridge = dp_bridge_init(dp_display, dev, encoder);
>   	if (IS_ERR(dp_display->bridge)) {
>   		ret = PTR_ERR(dp_display->bridge);
> diff --git a/drivers/gpu/drm/msm/dp/dp_display.h b/drivers/gpu/drm/msm/dp/dp_display.h
> index 49a1d89..1377cc3 100644
> --- a/drivers/gpu/drm/msm/dp/dp_display.h
> +++ b/drivers/gpu/drm/msm/dp/dp_display.h
> @@ -21,6 +21,7 @@ struct msm_dp {
>   	bool audio_enabled;
>   	bool power_on;
>   	unsigned int connector_type;
> +	bool is_edp;
>   
>   	hdmi_codec_plugged_cb plugged_cb;
>   
> diff --git a/drivers/gpu/drm/msm/dp/dp_drm.c b/drivers/gpu/drm/msm/dp/dp_drm.c
> index 7ce1aca..8a75c55 100644
> --- a/drivers/gpu/drm/msm/dp/dp_drm.c
> +++ b/drivers/gpu/drm/msm/dp/dp_drm.c
> @@ -114,10 +114,23 @@ struct drm_bridge *dp_bridge_init(struct msm_dp *dp_display, struct drm_device *
>   	bridge->funcs = &dp_bridge_ops;
>   	bridge->type = dp_display->connector_type;
>   
> -	bridge->ops =
> -		DRM_BRIDGE_OP_DETECT |
> -		DRM_BRIDGE_OP_HPD |
> -		DRM_BRIDGE_OP_MODES;
> +	/*
> +	 * Many ops only make sense for DP. Why?
> +	 * - Detect/HPD are used by DRM to know if a display is _physically_
> +	 *   there, not whether the display is powered on / finished initting.
> +	 *   On eDP we assume the display is always there because you can't
> +	 *   know until power is applied. If we don't implement the ops DRM will
> +	 *   assume our display is always there.
> +	 * - Currently eDP mode reading is driven by the panel driver. This
> +	 *   allows the panel driver to properly power itself on to read the
> +	 *   modes.
> +	 */
> +	if (!dp_display->is_edp) {
> +		bridge->ops =
> +			DRM_BRIDGE_OP_DETECT |
> +			DRM_BRIDGE_OP_HPD |
> +			DRM_BRIDGE_OP_MODES;
> +	}
>   
>   	rc = drm_bridge_attach(encoder, bridge, NULL, DRM_BRIDGE_ATTACH_NO_CONNECTOR);
>   	if (rc) {
> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.c b/drivers/gpu/drm/msm/dp/dp_parser.c
> index 1056b8d..4bdbf91 100644
> --- a/drivers/gpu/drm/msm/dp/dp_parser.c
> +++ b/drivers/gpu/drm/msm/dp/dp_parser.c
> @@ -265,7 +265,7 @@ static int dp_parser_clock(struct dp_parser *parser)
>   	return 0;
>   }
>   
> -static int dp_parser_find_next_bridge(struct dp_parser *parser)
> +int dp_parser_find_next_bridge(struct dp_parser *parser)
>   {
>   	struct device *dev = &parser->pdev->dev;
>   	struct drm_bridge *bridge;
> @@ -279,7 +279,7 @@ static int dp_parser_find_next_bridge(struct dp_parser *parser)
>   	return 0;
>   }
>   
> -static int dp_parser_parse(struct dp_parser *parser, int connector_type)
> +static int dp_parser_parse(struct dp_parser *parser)
>   {
>   	int rc = 0;
>   
> @@ -300,25 +300,6 @@ static int dp_parser_parse(struct dp_parser *parser, int connector_type)
>   	if (rc)
>   		return rc;
>   
> -	/*
> -	 * External bridges are mandatory for eDP interfaces: one has to
> -	 * provide at least an eDP panel (which gets wrapped into panel-bridge).
> -	 *
> -	 * For DisplayPort interfaces external bridges are optional, so
> -	 * silently ignore an error if one is not present (-ENODEV).
> -	 */
> -	rc = dp_parser_find_next_bridge(parser);
> -	if (rc == -ENODEV) {
> -		if (connector_type == DRM_MODE_CONNECTOR_eDP) {
> -			DRM_ERROR("eDP: next bridge is not present\n");
> -			return rc;
> -		}
> -	} else if (rc) {
> -		if (rc != -EPROBE_DEFER)
> -			DRM_ERROR("DP: error parsing next bridge: %d\n", rc);
> -		return rc;
> -	}
> -
>   	/* Map the corresponding regulator information according to
>   	 * version. Currently, since we only have one supported platform,
>   	 * mapping the regulator directly.
> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.h b/drivers/gpu/drm/msm/dp/dp_parser.h
> index d371bae..3a4d797 100644
> --- a/drivers/gpu/drm/msm/dp/dp_parser.h
> +++ b/drivers/gpu/drm/msm/dp/dp_parser.h
> @@ -125,7 +125,7 @@ struct dp_parser {
>   	u32 max_dp_lanes;
>   	struct drm_bridge *next_bridge;
>   
> -	int (*parse)(struct dp_parser *parser, int connector_type);
> +	int (*parse)(struct dp_parser *parser);
>   };
>   
>   /**
> @@ -141,4 +141,16 @@ struct dp_parser {
>    */
>   struct dp_parser *dp_parser_get(struct platform_device *pdev);
>   
> +/**
> + * dp_parser_find_next_bridge() - find an additional bridge to DP
> + *
> + * @parser: dp_parser data from client
> + *
> + * This function is used to find any additional bridge attached to
> + * the DP controller. The eDP interface requires a panel bridge.
> + *
> + * Return: 0 if able to get the bridge, otherwise negative errno for failure.
> + */
> +int dp_parser_find_next_bridge(struct dp_parser *parser);
> +
>   #endif


-- 
With best wishes
Dmitry

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus
  2022-05-06 20:49   ` Dmitry Baryshkov
@ 2022-05-06 21:03     ` Abhinav Kumar
  2022-05-06 21:05       ` Dmitry Baryshkov
  0 siblings, 1 reply; 10+ messages in thread
From: Abhinav Kumar @ 2022-05-06 21:03 UTC (permalink / raw)
  To: Dmitry Baryshkov, Sankeerth Billakanti, dri-devel, linux-arm-msm,
	freedreno, linux-kernel, devicetree
  Cc: robdclark, seanpaul, swboyd, quic_kalyant, dianders, quic_khsieh,
	bjorn.andersson, sean, airlied, daniel, quic_vproddut,
	quic_aravindh, steev



On 5/6/2022 1:49 PM, Dmitry Baryshkov wrote:
> On 25/04/2022 14:44, Sankeerth Billakanti wrote:
>> This patch adds support for generic eDP sink through aux_bus. The eDP/DP
>> controller driver should support aux transactions originating from the
>> panel-edp driver and hence should be initialized and ready.
>>
>> The panel bridge supporting the panel should be ready before the bridge
>> connector is initialized. The generic panel probe needs the controller
>> resources to be enabled to support the aux transactions originating from
>> the panel probe.
>>
>> Signed-off-by: Sankeerth Billakanti <quic_sbillaka@quicinc.com>
>> Reviewed-by: Douglas Anderson <dianders@chromium.org>
>> Reviewed-by: Stephen Boyd <swboyd@chromium.org>
> 
> An additional side effect from this patch. Previously missing panel 
> would have caused the bind error. Now it is the dp_modeset_init error, 
> which translates to kms_hw_init returning -517. I kind ask to move the 
> next_bridge acquisition back to the dp_bind in one of the followup patches.
> 

This is true. But the end result would be same isnt it?

When dp_display_bind() failed earlier, it would cause master bind 
failure too due to component model.

Even now, it causes the same result?

>> ---
>> Changes in v10:
>>    - modify the error handling condition
>>    - modify the kernel doc
>>
>> Changes in v9:
>>    - add comments for panel probe
>>    - modify the error handling checks
>>
>> Changes in v8:
>>    - handle corner cases
>>    - add comment for the bridge ops
>>
>> Changes in v7:
>>    - aux_bus is mandatory for eDP
>>    - connector type check modified to just check for eDP
>>
>> Changes in v6:
>>    - Remove initialization
>>    - Fix aux_bus node leak
>>    - Split the patches
>>
>>   drivers/gpu/drm/msm/dp/dp_display.c | 72 
>> ++++++++++++++++++++++++++++++++++---
>>   drivers/gpu/drm/msm/dp/dp_display.h |  1 +
>>   drivers/gpu/drm/msm/dp/dp_drm.c     | 21 ++++++++---
>>   drivers/gpu/drm/msm/dp/dp_parser.c  | 23 ++----------
>>   drivers/gpu/drm/msm/dp/dp_parser.h  | 14 +++++++-
>>   5 files changed, 101 insertions(+), 30 deletions(-)
>>
>> diff --git a/drivers/gpu/drm/msm/dp/dp_display.c 
>> b/drivers/gpu/drm/msm/dp/dp_display.c
>> index d7a19d6..f772d84 100644
>> --- a/drivers/gpu/drm/msm/dp/dp_display.c
>> +++ b/drivers/gpu/drm/msm/dp/dp_display.c
>> @@ -10,6 +10,7 @@
>>   #include <linux/component.h>
>>   #include <linux/of_irq.h>
>>   #include <linux/delay.h>
>> +#include <drm/dp/drm_dp_aux_bus.h>
>>   #include "msm_drv.h"
>>   #include "msm_kms.h"
>> @@ -259,14 +260,12 @@ static int dp_display_bind(struct device *dev, 
>> struct device *master,
>>       dp->dp_display.drm_dev = drm;
>>       priv->dp[dp->id] = &dp->dp_display;
>> -    rc = dp->parser->parse(dp->parser, dp->dp_display.connector_type);
>> +    rc = dp->parser->parse(dp->parser);
>>       if (rc) {
>>           DRM_ERROR("device tree parsing failed\n");
>>           goto end;
>>       }
>> -    dp->dp_display.next_bridge = dp->parser->next_bridge;
>> -
>>       dp->aux->drm_dev = drm;
>>       rc = dp_aux_register(dp->aux);
>>       if (rc) {
>> @@ -1319,6 +1318,8 @@ 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.is_edp =
>> +        (dp->dp_display.connector_type == DRM_MODE_CONNECTOR_eDP);
>>       rc = dp_init_sub_modules(dp);
>>       if (rc) {
>> @@ -1508,7 +1509,8 @@ void msm_dp_irq_postinstall(struct msm_dp 
>> *dp_display)
>>       dp_hpd_event_setup(dp);
>> -    dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
>> +    if (!dp_display->is_edp)
>> +        dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
>>   }
>>   void msm_dp_debugfs_init(struct msm_dp *dp_display, struct drm_minor 
>> *minor)
>> @@ -1530,6 +1532,64 @@ void msm_dp_debugfs_init(struct msm_dp 
>> *dp_display, struct drm_minor *minor)
>>       }
>>   }
>> +static int dp_display_get_next_bridge(struct msm_dp *dp)
>> +{
>> +    int rc;
>> +    struct dp_display_private *dp_priv;
>> +    struct device_node *aux_bus;
>> +    struct device *dev;
>> +
>> +    dp_priv = container_of(dp, struct dp_display_private, dp_display);
>> +    dev = &dp_priv->pdev->dev;
>> +    aux_bus = of_get_child_by_name(dev->of_node, "aux-bus");
>> +
>> +    if (aux_bus && dp->is_edp) {
>> +        dp_display_host_init(dp_priv);
>> +        dp_catalog_ctrl_hpd_config(dp_priv->catalog);
>> +        dp_display_host_phy_init(dp_priv);
>> +        enable_irq(dp_priv->irq);
>> +
>> +        /*
>> +         * The code below assumes that the panel will finish probing
>> +         * by the time devm_of_dp_aux_populate_ep_devices() returns.
>> +         * This isn't a great assumption since it will fail if the
>> +         * panel driver is probed asynchronously but is the best we
>> +         * can do without a bigger driver reorganization.
>> +         */
>> +        rc = devm_of_dp_aux_populate_ep_devices(dp_priv->aux);
>> +        of_node_put(aux_bus);
>> +        if (rc)
>> +            goto error;
>> +    } else if (dp->is_edp) {
>> +        DRM_ERROR("eDP aux_bus not found\n");
>> +        return -ENODEV;
>> +    }
>> +
>> +    /*
>> +     * External bridges are mandatory for eDP interfaces: one has to
>> +     * provide at least an eDP panel (which gets wrapped into 
>> panel-bridge).
>> +     *
>> +     * For DisplayPort interfaces external bridges are optional, so
>> +     * silently ignore an error if one is not present (-ENODEV).
>> +     */
>> +    rc = dp_parser_find_next_bridge(dp_priv->parser);
>> +    if (!dp->is_edp && rc == -ENODEV)
>> +        return 0;
>> +
>> +    if (!rc) {
>> +        dp->next_bridge = dp_priv->parser->next_bridge;
>> +        return 0;
>> +    }
>> +
>> +error:
>> +    if (dp->is_edp) {
>> +        disable_irq(dp_priv->irq);
>> +        dp_display_host_phy_exit(dp_priv);
>> +        dp_display_host_deinit(dp_priv);
>> +    }
>> +    return rc;
>> +}
>> +
>>   int msm_dp_modeset_init(struct msm_dp *dp_display, struct drm_device 
>> *dev,
>>               struct drm_encoder *encoder)
>>   {
>> @@ -1553,6 +1613,10 @@ int msm_dp_modeset_init(struct msm_dp 
>> *dp_display, struct drm_device *dev,
>>       dp_display->encoder = encoder;
>> +    ret = dp_display_get_next_bridge(dp_display);
>> +    if (ret)
>> +        return ret;
>> +
>>       dp_display->bridge = dp_bridge_init(dp_display, dev, encoder);
>>       if (IS_ERR(dp_display->bridge)) {
>>           ret = PTR_ERR(dp_display->bridge);
>> diff --git a/drivers/gpu/drm/msm/dp/dp_display.h 
>> b/drivers/gpu/drm/msm/dp/dp_display.h
>> index 49a1d89..1377cc3 100644
>> --- a/drivers/gpu/drm/msm/dp/dp_display.h
>> +++ b/drivers/gpu/drm/msm/dp/dp_display.h
>> @@ -21,6 +21,7 @@ struct msm_dp {
>>       bool audio_enabled;
>>       bool power_on;
>>       unsigned int connector_type;
>> +    bool is_edp;
>>       hdmi_codec_plugged_cb plugged_cb;
>> diff --git a/drivers/gpu/drm/msm/dp/dp_drm.c 
>> b/drivers/gpu/drm/msm/dp/dp_drm.c
>> index 7ce1aca..8a75c55 100644
>> --- a/drivers/gpu/drm/msm/dp/dp_drm.c
>> +++ b/drivers/gpu/drm/msm/dp/dp_drm.c
>> @@ -114,10 +114,23 @@ struct drm_bridge *dp_bridge_init(struct msm_dp 
>> *dp_display, struct drm_device *
>>       bridge->funcs = &dp_bridge_ops;
>>       bridge->type = dp_display->connector_type;
>> -    bridge->ops =
>> -        DRM_BRIDGE_OP_DETECT |
>> -        DRM_BRIDGE_OP_HPD |
>> -        DRM_BRIDGE_OP_MODES;
>> +    /*
>> +     * Many ops only make sense for DP. Why?
>> +     * - Detect/HPD are used by DRM to know if a display is _physically_
>> +     *   there, not whether the display is powered on / finished 
>> initting.
>> +     *   On eDP we assume the display is always there because you can't
>> +     *   know until power is applied. If we don't implement the ops 
>> DRM will
>> +     *   assume our display is always there.
>> +     * - Currently eDP mode reading is driven by the panel driver. This
>> +     *   allows the panel driver to properly power itself on to read the
>> +     *   modes.
>> +     */
>> +    if (!dp_display->is_edp) {
>> +        bridge->ops =
>> +            DRM_BRIDGE_OP_DETECT |
>> +            DRM_BRIDGE_OP_HPD |
>> +            DRM_BRIDGE_OP_MODES;
>> +    }
>>       rc = drm_bridge_attach(encoder, bridge, NULL, 
>> DRM_BRIDGE_ATTACH_NO_CONNECTOR);
>>       if (rc) {
>> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.c 
>> b/drivers/gpu/drm/msm/dp/dp_parser.c
>> index 1056b8d..4bdbf91 100644
>> --- a/drivers/gpu/drm/msm/dp/dp_parser.c
>> +++ b/drivers/gpu/drm/msm/dp/dp_parser.c
>> @@ -265,7 +265,7 @@ static int dp_parser_clock(struct dp_parser *parser)
>>       return 0;
>>   }
>> -static int dp_parser_find_next_bridge(struct dp_parser *parser)
>> +int dp_parser_find_next_bridge(struct dp_parser *parser)
>>   {
>>       struct device *dev = &parser->pdev->dev;
>>       struct drm_bridge *bridge;
>> @@ -279,7 +279,7 @@ static int dp_parser_find_next_bridge(struct 
>> dp_parser *parser)
>>       return 0;
>>   }
>> -static int dp_parser_parse(struct dp_parser *parser, int connector_type)
>> +static int dp_parser_parse(struct dp_parser *parser)
>>   {
>>       int rc = 0;
>> @@ -300,25 +300,6 @@ static int dp_parser_parse(struct dp_parser 
>> *parser, int connector_type)
>>       if (rc)
>>           return rc;
>> -    /*
>> -     * External bridges are mandatory for eDP interfaces: one has to
>> -     * provide at least an eDP panel (which gets wrapped into 
>> panel-bridge).
>> -     *
>> -     * For DisplayPort interfaces external bridges are optional, so
>> -     * silently ignore an error if one is not present (-ENODEV).
>> -     */
>> -    rc = dp_parser_find_next_bridge(parser);
>> -    if (rc == -ENODEV) {
>> -        if (connector_type == DRM_MODE_CONNECTOR_eDP) {
>> -            DRM_ERROR("eDP: next bridge is not present\n");
>> -            return rc;
>> -        }
>> -    } else if (rc) {
>> -        if (rc != -EPROBE_DEFER)
>> -            DRM_ERROR("DP: error parsing next bridge: %d\n", rc);
>> -        return rc;
>> -    }
>> -
>>       /* Map the corresponding regulator information according to
>>        * version. Currently, since we only have one supported platform,
>>        * mapping the regulator directly.
>> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.h 
>> b/drivers/gpu/drm/msm/dp/dp_parser.h
>> index d371bae..3a4d797 100644
>> --- a/drivers/gpu/drm/msm/dp/dp_parser.h
>> +++ b/drivers/gpu/drm/msm/dp/dp_parser.h
>> @@ -125,7 +125,7 @@ struct dp_parser {
>>       u32 max_dp_lanes;
>>       struct drm_bridge *next_bridge;
>> -    int (*parse)(struct dp_parser *parser, int connector_type);
>> +    int (*parse)(struct dp_parser *parser);
>>   };
>>   /**
>> @@ -141,4 +141,16 @@ struct dp_parser {
>>    */
>>   struct dp_parser *dp_parser_get(struct platform_device *pdev);
>> +/**
>> + * dp_parser_find_next_bridge() - find an additional bridge to DP
>> + *
>> + * @parser: dp_parser data from client
>> + *
>> + * This function is used to find any additional bridge attached to
>> + * the DP controller. The eDP interface requires a panel bridge.
>> + *
>> + * Return: 0 if able to get the bridge, otherwise negative errno for 
>> failure.
>> + */
>> +int dp_parser_find_next_bridge(struct dp_parser *parser);
>> +
>>   #endif
> 
> 

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus
  2022-05-06 21:03     ` Abhinav Kumar
@ 2022-05-06 21:05       ` Dmitry Baryshkov
  2022-05-06 21:17         ` [Freedreno] " Abhinav Kumar
  0 siblings, 1 reply; 10+ messages in thread
From: Dmitry Baryshkov @ 2022-05-06 21:05 UTC (permalink / raw)
  To: Abhinav Kumar, Sankeerth Billakanti, dri-devel, linux-arm-msm,
	freedreno, linux-kernel, devicetree
  Cc: robdclark, seanpaul, swboyd, quic_kalyant, dianders, quic_khsieh,
	bjorn.andersson, sean, airlied, daniel, quic_vproddut,
	quic_aravindh, steev

On 07/05/2022 00:03, Abhinav Kumar wrote:
> 
> 
> On 5/6/2022 1:49 PM, Dmitry Baryshkov wrote:
>> On 25/04/2022 14:44, Sankeerth Billakanti wrote:
>>> This patch adds support for generic eDP sink through aux_bus. The eDP/DP
>>> controller driver should support aux transactions originating from the
>>> panel-edp driver and hence should be initialized and ready.
>>>
>>> The panel bridge supporting the panel should be ready before the bridge
>>> connector is initialized. The generic panel probe needs the controller
>>> resources to be enabled to support the aux transactions originating from
>>> the panel probe.
>>>
>>> Signed-off-by: Sankeerth Billakanti <quic_sbillaka@quicinc.com>
>>> Reviewed-by: Douglas Anderson <dianders@chromium.org>
>>> Reviewed-by: Stephen Boyd <swboyd@chromium.org>
>>
>> An additional side effect from this patch. Previously missing panel 
>> would have caused the bind error. Now it is the dp_modeset_init error, 
>> which translates to kms_hw_init returning -517. I kind ask to move the 
>> next_bridge acquisition back to the dp_bind in one of the followup 
>> patches.
>>
> 
> This is true. But the end result would be same isnt it?
> 
> When dp_display_bind() failed earlier, it would cause master bind 
> failure too due to component model.
> 
> Even now, it causes the same result?

Yes, it helped us to uncover several error. But I'd still prefer to have 
EPROBE_DEFER being returned earlier rather than later.

> 
>>> ---
>>> Changes in v10:
>>>    - modify the error handling condition
>>>    - modify the kernel doc
>>>
>>> Changes in v9:
>>>    - add comments for panel probe
>>>    - modify the error handling checks
>>>
>>> Changes in v8:
>>>    - handle corner cases
>>>    - add comment for the bridge ops
>>>
>>> Changes in v7:
>>>    - aux_bus is mandatory for eDP
>>>    - connector type check modified to just check for eDP
>>>
>>> Changes in v6:
>>>    - Remove initialization
>>>    - Fix aux_bus node leak
>>>    - Split the patches
>>>
>>>   drivers/gpu/drm/msm/dp/dp_display.c | 72 
>>> ++++++++++++++++++++++++++++++++++---
>>>   drivers/gpu/drm/msm/dp/dp_display.h |  1 +
>>>   drivers/gpu/drm/msm/dp/dp_drm.c     | 21 ++++++++---
>>>   drivers/gpu/drm/msm/dp/dp_parser.c  | 23 ++----------
>>>   drivers/gpu/drm/msm/dp/dp_parser.h  | 14 +++++++-
>>>   5 files changed, 101 insertions(+), 30 deletions(-)
>>>
>>> diff --git a/drivers/gpu/drm/msm/dp/dp_display.c 
>>> b/drivers/gpu/drm/msm/dp/dp_display.c
>>> index d7a19d6..f772d84 100644
>>> --- a/drivers/gpu/drm/msm/dp/dp_display.c
>>> +++ b/drivers/gpu/drm/msm/dp/dp_display.c
>>> @@ -10,6 +10,7 @@
>>>   #include <linux/component.h>
>>>   #include <linux/of_irq.h>
>>>   #include <linux/delay.h>
>>> +#include <drm/dp/drm_dp_aux_bus.h>
>>>   #include "msm_drv.h"
>>>   #include "msm_kms.h"
>>> @@ -259,14 +260,12 @@ static int dp_display_bind(struct device *dev, 
>>> struct device *master,
>>>       dp->dp_display.drm_dev = drm;
>>>       priv->dp[dp->id] = &dp->dp_display;
>>> -    rc = dp->parser->parse(dp->parser, dp->dp_display.connector_type);
>>> +    rc = dp->parser->parse(dp->parser);
>>>       if (rc) {
>>>           DRM_ERROR("device tree parsing failed\n");
>>>           goto end;
>>>       }
>>> -    dp->dp_display.next_bridge = dp->parser->next_bridge;
>>> -
>>>       dp->aux->drm_dev = drm;
>>>       rc = dp_aux_register(dp->aux);
>>>       if (rc) {
>>> @@ -1319,6 +1318,8 @@ 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.is_edp =
>>> +        (dp->dp_display.connector_type == DRM_MODE_CONNECTOR_eDP);
>>>       rc = dp_init_sub_modules(dp);
>>>       if (rc) {
>>> @@ -1508,7 +1509,8 @@ void msm_dp_irq_postinstall(struct msm_dp 
>>> *dp_display)
>>>       dp_hpd_event_setup(dp);
>>> -    dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
>>> +    if (!dp_display->is_edp)
>>> +        dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
>>>   }
>>>   void msm_dp_debugfs_init(struct msm_dp *dp_display, struct 
>>> drm_minor *minor)
>>> @@ -1530,6 +1532,64 @@ void msm_dp_debugfs_init(struct msm_dp 
>>> *dp_display, struct drm_minor *minor)
>>>       }
>>>   }
>>> +static int dp_display_get_next_bridge(struct msm_dp *dp)
>>> +{
>>> +    int rc;
>>> +    struct dp_display_private *dp_priv;
>>> +    struct device_node *aux_bus;
>>> +    struct device *dev;
>>> +
>>> +    dp_priv = container_of(dp, struct dp_display_private, dp_display);
>>> +    dev = &dp_priv->pdev->dev;
>>> +    aux_bus = of_get_child_by_name(dev->of_node, "aux-bus");
>>> +
>>> +    if (aux_bus && dp->is_edp) {
>>> +        dp_display_host_init(dp_priv);
>>> +        dp_catalog_ctrl_hpd_config(dp_priv->catalog);
>>> +        dp_display_host_phy_init(dp_priv);
>>> +        enable_irq(dp_priv->irq);
>>> +
>>> +        /*
>>> +         * The code below assumes that the panel will finish probing
>>> +         * by the time devm_of_dp_aux_populate_ep_devices() returns.
>>> +         * This isn't a great assumption since it will fail if the
>>> +         * panel driver is probed asynchronously but is the best we
>>> +         * can do without a bigger driver reorganization.
>>> +         */
>>> +        rc = devm_of_dp_aux_populate_ep_devices(dp_priv->aux);
>>> +        of_node_put(aux_bus);
>>> +        if (rc)
>>> +            goto error;
>>> +    } else if (dp->is_edp) {
>>> +        DRM_ERROR("eDP aux_bus not found\n");
>>> +        return -ENODEV;
>>> +    }
>>> +
>>> +    /*
>>> +     * External bridges are mandatory for eDP interfaces: one has to
>>> +     * provide at least an eDP panel (which gets wrapped into 
>>> panel-bridge).
>>> +     *
>>> +     * For DisplayPort interfaces external bridges are optional, so
>>> +     * silently ignore an error if one is not present (-ENODEV).
>>> +     */
>>> +    rc = dp_parser_find_next_bridge(dp_priv->parser);
>>> +    if (!dp->is_edp && rc == -ENODEV)
>>> +        return 0;
>>> +
>>> +    if (!rc) {
>>> +        dp->next_bridge = dp_priv->parser->next_bridge;
>>> +        return 0;
>>> +    }
>>> +
>>> +error:
>>> +    if (dp->is_edp) {
>>> +        disable_irq(dp_priv->irq);
>>> +        dp_display_host_phy_exit(dp_priv);
>>> +        dp_display_host_deinit(dp_priv);
>>> +    }
>>> +    return rc;
>>> +}
>>> +
>>>   int msm_dp_modeset_init(struct msm_dp *dp_display, struct 
>>> drm_device *dev,
>>>               struct drm_encoder *encoder)
>>>   {
>>> @@ -1553,6 +1613,10 @@ int msm_dp_modeset_init(struct msm_dp 
>>> *dp_display, struct drm_device *dev,
>>>       dp_display->encoder = encoder;
>>> +    ret = dp_display_get_next_bridge(dp_display);
>>> +    if (ret)
>>> +        return ret;
>>> +
>>>       dp_display->bridge = dp_bridge_init(dp_display, dev, encoder);
>>>       if (IS_ERR(dp_display->bridge)) {
>>>           ret = PTR_ERR(dp_display->bridge);
>>> diff --git a/drivers/gpu/drm/msm/dp/dp_display.h 
>>> b/drivers/gpu/drm/msm/dp/dp_display.h
>>> index 49a1d89..1377cc3 100644
>>> --- a/drivers/gpu/drm/msm/dp/dp_display.h
>>> +++ b/drivers/gpu/drm/msm/dp/dp_display.h
>>> @@ -21,6 +21,7 @@ struct msm_dp {
>>>       bool audio_enabled;
>>>       bool power_on;
>>>       unsigned int connector_type;
>>> +    bool is_edp;
>>>       hdmi_codec_plugged_cb plugged_cb;
>>> diff --git a/drivers/gpu/drm/msm/dp/dp_drm.c 
>>> b/drivers/gpu/drm/msm/dp/dp_drm.c
>>> index 7ce1aca..8a75c55 100644
>>> --- a/drivers/gpu/drm/msm/dp/dp_drm.c
>>> +++ b/drivers/gpu/drm/msm/dp/dp_drm.c
>>> @@ -114,10 +114,23 @@ struct drm_bridge *dp_bridge_init(struct msm_dp 
>>> *dp_display, struct drm_device *
>>>       bridge->funcs = &dp_bridge_ops;
>>>       bridge->type = dp_display->connector_type;
>>> -    bridge->ops =
>>> -        DRM_BRIDGE_OP_DETECT |
>>> -        DRM_BRIDGE_OP_HPD |
>>> -        DRM_BRIDGE_OP_MODES;
>>> +    /*
>>> +     * Many ops only make sense for DP. Why?
>>> +     * - Detect/HPD are used by DRM to know if a display is 
>>> _physically_
>>> +     *   there, not whether the display is powered on / finished 
>>> initting.
>>> +     *   On eDP we assume the display is always there because you can't
>>> +     *   know until power is applied. If we don't implement the ops 
>>> DRM will
>>> +     *   assume our display is always there.
>>> +     * - Currently eDP mode reading is driven by the panel driver. This
>>> +     *   allows the panel driver to properly power itself on to read 
>>> the
>>> +     *   modes.
>>> +     */
>>> +    if (!dp_display->is_edp) {
>>> +        bridge->ops =
>>> +            DRM_BRIDGE_OP_DETECT |
>>> +            DRM_BRIDGE_OP_HPD |
>>> +            DRM_BRIDGE_OP_MODES;
>>> +    }
>>>       rc = drm_bridge_attach(encoder, bridge, NULL, 
>>> DRM_BRIDGE_ATTACH_NO_CONNECTOR);
>>>       if (rc) {
>>> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.c 
>>> b/drivers/gpu/drm/msm/dp/dp_parser.c
>>> index 1056b8d..4bdbf91 100644
>>> --- a/drivers/gpu/drm/msm/dp/dp_parser.c
>>> +++ b/drivers/gpu/drm/msm/dp/dp_parser.c
>>> @@ -265,7 +265,7 @@ static int dp_parser_clock(struct dp_parser *parser)
>>>       return 0;
>>>   }
>>> -static int dp_parser_find_next_bridge(struct dp_parser *parser)
>>> +int dp_parser_find_next_bridge(struct dp_parser *parser)
>>>   {
>>>       struct device *dev = &parser->pdev->dev;
>>>       struct drm_bridge *bridge;
>>> @@ -279,7 +279,7 @@ static int dp_parser_find_next_bridge(struct 
>>> dp_parser *parser)
>>>       return 0;
>>>   }
>>> -static int dp_parser_parse(struct dp_parser *parser, int 
>>> connector_type)
>>> +static int dp_parser_parse(struct dp_parser *parser)
>>>   {
>>>       int rc = 0;
>>> @@ -300,25 +300,6 @@ static int dp_parser_parse(struct dp_parser 
>>> *parser, int connector_type)
>>>       if (rc)
>>>           return rc;
>>> -    /*
>>> -     * External bridges are mandatory for eDP interfaces: one has to
>>> -     * provide at least an eDP panel (which gets wrapped into 
>>> panel-bridge).
>>> -     *
>>> -     * For DisplayPort interfaces external bridges are optional, so
>>> -     * silently ignore an error if one is not present (-ENODEV).
>>> -     */
>>> -    rc = dp_parser_find_next_bridge(parser);
>>> -    if (rc == -ENODEV) {
>>> -        if (connector_type == DRM_MODE_CONNECTOR_eDP) {
>>> -            DRM_ERROR("eDP: next bridge is not present\n");
>>> -            return rc;
>>> -        }
>>> -    } else if (rc) {
>>> -        if (rc != -EPROBE_DEFER)
>>> -            DRM_ERROR("DP: error parsing next bridge: %d\n", rc);
>>> -        return rc;
>>> -    }
>>> -
>>>       /* Map the corresponding regulator information according to
>>>        * version. Currently, since we only have one supported platform,
>>>        * mapping the regulator directly.
>>> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.h 
>>> b/drivers/gpu/drm/msm/dp/dp_parser.h
>>> index d371bae..3a4d797 100644
>>> --- a/drivers/gpu/drm/msm/dp/dp_parser.h
>>> +++ b/drivers/gpu/drm/msm/dp/dp_parser.h
>>> @@ -125,7 +125,7 @@ struct dp_parser {
>>>       u32 max_dp_lanes;
>>>       struct drm_bridge *next_bridge;
>>> -    int (*parse)(struct dp_parser *parser, int connector_type);
>>> +    int (*parse)(struct dp_parser *parser);
>>>   };
>>>   /**
>>> @@ -141,4 +141,16 @@ struct dp_parser {
>>>    */
>>>   struct dp_parser *dp_parser_get(struct platform_device *pdev);
>>> +/**
>>> + * dp_parser_find_next_bridge() - find an additional bridge to DP
>>> + *
>>> + * @parser: dp_parser data from client
>>> + *
>>> + * This function is used to find any additional bridge attached to
>>> + * the DP controller. The eDP interface requires a panel bridge.
>>> + *
>>> + * Return: 0 if able to get the bridge, otherwise negative errno for 
>>> failure.
>>> + */
>>> +int dp_parser_find_next_bridge(struct dp_parser *parser);
>>> +
>>>   #endif
>>
>>


-- 
With best wishes
Dmitry

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [Freedreno] [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus
  2022-05-06 21:05       ` Dmitry Baryshkov
@ 2022-05-06 21:17         ` Abhinav Kumar
  2022-05-06 21:17           ` Dmitry Baryshkov
  0 siblings, 1 reply; 10+ messages in thread
From: Abhinav Kumar @ 2022-05-06 21:17 UTC (permalink / raw)
  To: Dmitry Baryshkov, Sankeerth Billakanti, dri-devel, linux-arm-msm,
	freedreno, linux-kernel, devicetree
  Cc: quic_kalyant, bjorn.andersson, quic_vproddut, airlied, dianders,
	steev, quic_khsieh, robdclark, seanpaul, daniel, quic_aravindh,
	swboyd, sean



On 5/6/2022 2:05 PM, Dmitry Baryshkov wrote:
> On 07/05/2022 00:03, Abhinav Kumar wrote:
>>
>>
>> On 5/6/2022 1:49 PM, Dmitry Baryshkov wrote:
>>> On 25/04/2022 14:44, Sankeerth Billakanti wrote:
>>>> This patch adds support for generic eDP sink through aux_bus. The 
>>>> eDP/DP
>>>> controller driver should support aux transactions originating from the
>>>> panel-edp driver and hence should be initialized and ready.
>>>>
>>>> The panel bridge supporting the panel should be ready before the bridge
>>>> connector is initialized. The generic panel probe needs the controller
>>>> resources to be enabled to support the aux transactions originating 
>>>> from
>>>> the panel probe.
>>>>
>>>> Signed-off-by: Sankeerth Billakanti <quic_sbillaka@quicinc.com>
>>>> Reviewed-by: Douglas Anderson <dianders@chromium.org>
>>>> Reviewed-by: Stephen Boyd <swboyd@chromium.org>
>>>
>>> An additional side effect from this patch. Previously missing panel 
>>> would have caused the bind error. Now it is the dp_modeset_init 
>>> error, which translates to kms_hw_init returning -517. I kind ask to 
>>> move the next_bridge acquisition back to the dp_bind in one of the 
>>> followup patches.
>>>
>>
>> This is true. But the end result would be same isnt it?
>>
>> When dp_display_bind() failed earlier, it would cause master bind 
>> failure too due to component model.
>>
>> Even now, it causes the same result?
> 
> Yes, it helped us to uncover several error. But I'd still prefer to have 
> EPROBE_DEFER being returned earlier rather than later.

Alright, point noted to try moving this earlier.

We will be following this up with rounds of cleanups to implement the 
suggestions given by you and Doug earlier.

This point shall be noted too.

> 
>>
>>>> ---
>>>> Changes in v10:
>>>>    - modify the error handling condition
>>>>    - modify the kernel doc
>>>>
>>>> Changes in v9:
>>>>    - add comments for panel probe
>>>>    - modify the error handling checks
>>>>
>>>> Changes in v8:
>>>>    - handle corner cases
>>>>    - add comment for the bridge ops
>>>>
>>>> Changes in v7:
>>>>    - aux_bus is mandatory for eDP
>>>>    - connector type check modified to just check for eDP
>>>>
>>>> Changes in v6:
>>>>    - Remove initialization
>>>>    - Fix aux_bus node leak
>>>>    - Split the patches
>>>>
>>>>   drivers/gpu/drm/msm/dp/dp_display.c | 72 
>>>> ++++++++++++++++++++++++++++++++++---
>>>>   drivers/gpu/drm/msm/dp/dp_display.h |  1 +
>>>>   drivers/gpu/drm/msm/dp/dp_drm.c     | 21 ++++++++---
>>>>   drivers/gpu/drm/msm/dp/dp_parser.c  | 23 ++----------
>>>>   drivers/gpu/drm/msm/dp/dp_parser.h  | 14 +++++++-
>>>>   5 files changed, 101 insertions(+), 30 deletions(-)
>>>>
>>>> diff --git a/drivers/gpu/drm/msm/dp/dp_display.c 
>>>> b/drivers/gpu/drm/msm/dp/dp_display.c
>>>> index d7a19d6..f772d84 100644
>>>> --- a/drivers/gpu/drm/msm/dp/dp_display.c
>>>> +++ b/drivers/gpu/drm/msm/dp/dp_display.c
>>>> @@ -10,6 +10,7 @@
>>>>   #include <linux/component.h>
>>>>   #include <linux/of_irq.h>
>>>>   #include <linux/delay.h>
>>>> +#include <drm/dp/drm_dp_aux_bus.h>
>>>>   #include "msm_drv.h"
>>>>   #include "msm_kms.h"
>>>> @@ -259,14 +260,12 @@ static int dp_display_bind(struct device *dev, 
>>>> struct device *master,
>>>>       dp->dp_display.drm_dev = drm;
>>>>       priv->dp[dp->id] = &dp->dp_display;
>>>> -    rc = dp->parser->parse(dp->parser, dp->dp_display.connector_type);
>>>> +    rc = dp->parser->parse(dp->parser);
>>>>       if (rc) {
>>>>           DRM_ERROR("device tree parsing failed\n");
>>>>           goto end;
>>>>       }
>>>> -    dp->dp_display.next_bridge = dp->parser->next_bridge;
>>>> -
>>>>       dp->aux->drm_dev = drm;
>>>>       rc = dp_aux_register(dp->aux);
>>>>       if (rc) {
>>>> @@ -1319,6 +1318,8 @@ 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.is_edp =
>>>> +        (dp->dp_display.connector_type == DRM_MODE_CONNECTOR_eDP);
>>>>       rc = dp_init_sub_modules(dp);
>>>>       if (rc) {
>>>> @@ -1508,7 +1509,8 @@ void msm_dp_irq_postinstall(struct msm_dp 
>>>> *dp_display)
>>>>       dp_hpd_event_setup(dp);
>>>> -    dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
>>>> +    if (!dp_display->is_edp)
>>>> +        dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
>>>>   }
>>>>   void msm_dp_debugfs_init(struct msm_dp *dp_display, struct 
>>>> drm_minor *minor)
>>>> @@ -1530,6 +1532,64 @@ void msm_dp_debugfs_init(struct msm_dp 
>>>> *dp_display, struct drm_minor *minor)
>>>>       }
>>>>   }
>>>> +static int dp_display_get_next_bridge(struct msm_dp *dp)
>>>> +{
>>>> +    int rc;
>>>> +    struct dp_display_private *dp_priv;
>>>> +    struct device_node *aux_bus;
>>>> +    struct device *dev;
>>>> +
>>>> +    dp_priv = container_of(dp, struct dp_display_private, dp_display);
>>>> +    dev = &dp_priv->pdev->dev;
>>>> +    aux_bus = of_get_child_by_name(dev->of_node, "aux-bus");
>>>> +
>>>> +    if (aux_bus && dp->is_edp) {
>>>> +        dp_display_host_init(dp_priv);
>>>> +        dp_catalog_ctrl_hpd_config(dp_priv->catalog);
>>>> +        dp_display_host_phy_init(dp_priv);
>>>> +        enable_irq(dp_priv->irq);
>>>> +
>>>> +        /*
>>>> +         * The code below assumes that the panel will finish probing
>>>> +         * by the time devm_of_dp_aux_populate_ep_devices() returns.
>>>> +         * This isn't a great assumption since it will fail if the
>>>> +         * panel driver is probed asynchronously but is the best we
>>>> +         * can do without a bigger driver reorganization.
>>>> +         */
>>>> +        rc = devm_of_dp_aux_populate_ep_devices(dp_priv->aux);
>>>> +        of_node_put(aux_bus);
>>>> +        if (rc)
>>>> +            goto error;
>>>> +    } else if (dp->is_edp) {
>>>> +        DRM_ERROR("eDP aux_bus not found\n");
>>>> +        return -ENODEV;
>>>> +    }
>>>> +
>>>> +    /*
>>>> +     * External bridges are mandatory for eDP interfaces: one has to
>>>> +     * provide at least an eDP panel (which gets wrapped into 
>>>> panel-bridge).
>>>> +     *
>>>> +     * For DisplayPort interfaces external bridges are optional, so
>>>> +     * silently ignore an error if one is not present (-ENODEV).
>>>> +     */
>>>> +    rc = dp_parser_find_next_bridge(dp_priv->parser);
>>>> +    if (!dp->is_edp && rc == -ENODEV)
>>>> +        return 0;
>>>> +
>>>> +    if (!rc) {
>>>> +        dp->next_bridge = dp_priv->parser->next_bridge;
>>>> +        return 0;
>>>> +    }
>>>> +
>>>> +error:
>>>> +    if (dp->is_edp) {
>>>> +        disable_irq(dp_priv->irq);
>>>> +        dp_display_host_phy_exit(dp_priv);
>>>> +        dp_display_host_deinit(dp_priv);
>>>> +    }
>>>> +    return rc;
>>>> +}
>>>> +
>>>>   int msm_dp_modeset_init(struct msm_dp *dp_display, struct 
>>>> drm_device *dev,
>>>>               struct drm_encoder *encoder)
>>>>   {
>>>> @@ -1553,6 +1613,10 @@ int msm_dp_modeset_init(struct msm_dp 
>>>> *dp_display, struct drm_device *dev,
>>>>       dp_display->encoder = encoder;
>>>> +    ret = dp_display_get_next_bridge(dp_display);
>>>> +    if (ret)
>>>> +        return ret;
>>>> +
>>>>       dp_display->bridge = dp_bridge_init(dp_display, dev, encoder);
>>>>       if (IS_ERR(dp_display->bridge)) {
>>>>           ret = PTR_ERR(dp_display->bridge);
>>>> diff --git a/drivers/gpu/drm/msm/dp/dp_display.h 
>>>> b/drivers/gpu/drm/msm/dp/dp_display.h
>>>> index 49a1d89..1377cc3 100644
>>>> --- a/drivers/gpu/drm/msm/dp/dp_display.h
>>>> +++ b/drivers/gpu/drm/msm/dp/dp_display.h
>>>> @@ -21,6 +21,7 @@ struct msm_dp {
>>>>       bool audio_enabled;
>>>>       bool power_on;
>>>>       unsigned int connector_type;
>>>> +    bool is_edp;
>>>>       hdmi_codec_plugged_cb plugged_cb;
>>>> diff --git a/drivers/gpu/drm/msm/dp/dp_drm.c 
>>>> b/drivers/gpu/drm/msm/dp/dp_drm.c
>>>> index 7ce1aca..8a75c55 100644
>>>> --- a/drivers/gpu/drm/msm/dp/dp_drm.c
>>>> +++ b/drivers/gpu/drm/msm/dp/dp_drm.c
>>>> @@ -114,10 +114,23 @@ struct drm_bridge *dp_bridge_init(struct 
>>>> msm_dp *dp_display, struct drm_device *
>>>>       bridge->funcs = &dp_bridge_ops;
>>>>       bridge->type = dp_display->connector_type;
>>>> -    bridge->ops =
>>>> -        DRM_BRIDGE_OP_DETECT |
>>>> -        DRM_BRIDGE_OP_HPD |
>>>> -        DRM_BRIDGE_OP_MODES;
>>>> +    /*
>>>> +     * Many ops only make sense for DP. Why?
>>>> +     * - Detect/HPD are used by DRM to know if a display is 
>>>> _physically_
>>>> +     *   there, not whether the display is powered on / finished 
>>>> initting.
>>>> +     *   On eDP we assume the display is always there because you 
>>>> can't
>>>> +     *   know until power is applied. If we don't implement the ops 
>>>> DRM will
>>>> +     *   assume our display is always there.
>>>> +     * - Currently eDP mode reading is driven by the panel driver. 
>>>> This
>>>> +     *   allows the panel driver to properly power itself on to 
>>>> read the
>>>> +     *   modes.
>>>> +     */
>>>> +    if (!dp_display->is_edp) {
>>>> +        bridge->ops =
>>>> +            DRM_BRIDGE_OP_DETECT |
>>>> +            DRM_BRIDGE_OP_HPD |
>>>> +            DRM_BRIDGE_OP_MODES;
>>>> +    }
>>>>       rc = drm_bridge_attach(encoder, bridge, NULL, 
>>>> DRM_BRIDGE_ATTACH_NO_CONNECTOR);
>>>>       if (rc) {
>>>> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.c 
>>>> b/drivers/gpu/drm/msm/dp/dp_parser.c
>>>> index 1056b8d..4bdbf91 100644
>>>> --- a/drivers/gpu/drm/msm/dp/dp_parser.c
>>>> +++ b/drivers/gpu/drm/msm/dp/dp_parser.c
>>>> @@ -265,7 +265,7 @@ static int dp_parser_clock(struct dp_parser 
>>>> *parser)
>>>>       return 0;
>>>>   }
>>>> -static int dp_parser_find_next_bridge(struct dp_parser *parser)
>>>> +int dp_parser_find_next_bridge(struct dp_parser *parser)
>>>>   {
>>>>       struct device *dev = &parser->pdev->dev;
>>>>       struct drm_bridge *bridge;
>>>> @@ -279,7 +279,7 @@ static int dp_parser_find_next_bridge(struct 
>>>> dp_parser *parser)
>>>>       return 0;
>>>>   }
>>>> -static int dp_parser_parse(struct dp_parser *parser, int 
>>>> connector_type)
>>>> +static int dp_parser_parse(struct dp_parser *parser)
>>>>   {
>>>>       int rc = 0;
>>>> @@ -300,25 +300,6 @@ static int dp_parser_parse(struct dp_parser 
>>>> *parser, int connector_type)
>>>>       if (rc)
>>>>           return rc;
>>>> -    /*
>>>> -     * External bridges are mandatory for eDP interfaces: one has to
>>>> -     * provide at least an eDP panel (which gets wrapped into 
>>>> panel-bridge).
>>>> -     *
>>>> -     * For DisplayPort interfaces external bridges are optional, so
>>>> -     * silently ignore an error if one is not present (-ENODEV).
>>>> -     */
>>>> -    rc = dp_parser_find_next_bridge(parser);
>>>> -    if (rc == -ENODEV) {
>>>> -        if (connector_type == DRM_MODE_CONNECTOR_eDP) {
>>>> -            DRM_ERROR("eDP: next bridge is not present\n");
>>>> -            return rc;
>>>> -        }
>>>> -    } else if (rc) {
>>>> -        if (rc != -EPROBE_DEFER)
>>>> -            DRM_ERROR("DP: error parsing next bridge: %d\n", rc);
>>>> -        return rc;
>>>> -    }
>>>> -
>>>>       /* Map the corresponding regulator information according to
>>>>        * version. Currently, since we only have one supported platform,
>>>>        * mapping the regulator directly.
>>>> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.h 
>>>> b/drivers/gpu/drm/msm/dp/dp_parser.h
>>>> index d371bae..3a4d797 100644
>>>> --- a/drivers/gpu/drm/msm/dp/dp_parser.h
>>>> +++ b/drivers/gpu/drm/msm/dp/dp_parser.h
>>>> @@ -125,7 +125,7 @@ struct dp_parser {
>>>>       u32 max_dp_lanes;
>>>>       struct drm_bridge *next_bridge;
>>>> -    int (*parse)(struct dp_parser *parser, int connector_type);
>>>> +    int (*parse)(struct dp_parser *parser);
>>>>   };
>>>>   /**
>>>> @@ -141,4 +141,16 @@ struct dp_parser {
>>>>    */
>>>>   struct dp_parser *dp_parser_get(struct platform_device *pdev);
>>>> +/**
>>>> + * dp_parser_find_next_bridge() - find an additional bridge to DP
>>>> + *
>>>> + * @parser: dp_parser data from client
>>>> + *
>>>> + * This function is used to find any additional bridge attached to
>>>> + * the DP controller. The eDP interface requires a panel bridge.
>>>> + *
>>>> + * Return: 0 if able to get the bridge, otherwise negative errno 
>>>> for failure.
>>>> + */
>>>> +int dp_parser_find_next_bridge(struct dp_parser *parser);
>>>> +
>>>>   #endif
>>>
>>>
> 
> 

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [Freedreno] [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus
  2022-05-06 21:17         ` [Freedreno] " Abhinav Kumar
@ 2022-05-06 21:17           ` Dmitry Baryshkov
  0 siblings, 0 replies; 10+ messages in thread
From: Dmitry Baryshkov @ 2022-05-06 21:17 UTC (permalink / raw)
  To: Abhinav Kumar, Sankeerth Billakanti, dri-devel, linux-arm-msm,
	freedreno, linux-kernel, devicetree
  Cc: quic_kalyant, bjorn.andersson, quic_vproddut, airlied, dianders,
	steev, quic_khsieh, robdclark, seanpaul, daniel, quic_aravindh,
	swboyd, sean

On 07/05/2022 00:17, Abhinav Kumar wrote:
> 
> 
> On 5/6/2022 2:05 PM, Dmitry Baryshkov wrote:
>> On 07/05/2022 00:03, Abhinav Kumar wrote:
>>>
>>>
>>> On 5/6/2022 1:49 PM, Dmitry Baryshkov wrote:
>>>> On 25/04/2022 14:44, Sankeerth Billakanti wrote:
>>>>> This patch adds support for generic eDP sink through aux_bus. The 
>>>>> eDP/DP
>>>>> controller driver should support aux transactions originating from the
>>>>> panel-edp driver and hence should be initialized and ready.
>>>>>
>>>>> The panel bridge supporting the panel should be ready before the 
>>>>> bridge
>>>>> connector is initialized. The generic panel probe needs the controller
>>>>> resources to be enabled to support the aux transactions originating 
>>>>> from
>>>>> the panel probe.
>>>>>
>>>>> Signed-off-by: Sankeerth Billakanti <quic_sbillaka@quicinc.com>
>>>>> Reviewed-by: Douglas Anderson <dianders@chromium.org>
>>>>> Reviewed-by: Stephen Boyd <swboyd@chromium.org>
>>>>
>>>> An additional side effect from this patch. Previously missing panel 
>>>> would have caused the bind error. Now it is the dp_modeset_init 
>>>> error, which translates to kms_hw_init returning -517. I kind ask to 
>>>> move the next_bridge acquisition back to the dp_bind in one of the 
>>>> followup patches.
>>>>
>>>
>>> This is true. But the end result would be same isnt it?
>>>
>>> When dp_display_bind() failed earlier, it would cause master bind 
>>> failure too due to component model.
>>>
>>> Even now, it causes the same result?
>>
>> Yes, it helped us to uncover several error. But I'd still prefer to 
>> have EPROBE_DEFER being returned earlier rather than later.
> 
> Alright, point noted to try moving this earlier.
> 
> We will be following this up with rounds of cleanups to implement the 
> suggestions given by you and Doug earlier.
> 
> This point shall be noted too.

Thanks a lot.

> 
>>
>>>
>>>>> ---
>>>>> Changes in v10:
>>>>>    - modify the error handling condition
>>>>>    - modify the kernel doc
>>>>>
>>>>> Changes in v9:
>>>>>    - add comments for panel probe
>>>>>    - modify the error handling checks
>>>>>
>>>>> Changes in v8:
>>>>>    - handle corner cases
>>>>>    - add comment for the bridge ops
>>>>>
>>>>> Changes in v7:
>>>>>    - aux_bus is mandatory for eDP
>>>>>    - connector type check modified to just check for eDP
>>>>>
>>>>> Changes in v6:
>>>>>    - Remove initialization
>>>>>    - Fix aux_bus node leak
>>>>>    - Split the patches
>>>>>
>>>>>   drivers/gpu/drm/msm/dp/dp_display.c | 72 
>>>>> ++++++++++++++++++++++++++++++++++---
>>>>>   drivers/gpu/drm/msm/dp/dp_display.h |  1 +
>>>>>   drivers/gpu/drm/msm/dp/dp_drm.c     | 21 ++++++++---
>>>>>   drivers/gpu/drm/msm/dp/dp_parser.c  | 23 ++----------
>>>>>   drivers/gpu/drm/msm/dp/dp_parser.h  | 14 +++++++-
>>>>>   5 files changed, 101 insertions(+), 30 deletions(-)
>>>>>
>>>>> diff --git a/drivers/gpu/drm/msm/dp/dp_display.c 
>>>>> b/drivers/gpu/drm/msm/dp/dp_display.c
>>>>> index d7a19d6..f772d84 100644
>>>>> --- a/drivers/gpu/drm/msm/dp/dp_display.c
>>>>> +++ b/drivers/gpu/drm/msm/dp/dp_display.c
>>>>> @@ -10,6 +10,7 @@
>>>>>   #include <linux/component.h>
>>>>>   #include <linux/of_irq.h>
>>>>>   #include <linux/delay.h>
>>>>> +#include <drm/dp/drm_dp_aux_bus.h>
>>>>>   #include "msm_drv.h"
>>>>>   #include "msm_kms.h"
>>>>> @@ -259,14 +260,12 @@ static int dp_display_bind(struct device 
>>>>> *dev, struct device *master,
>>>>>       dp->dp_display.drm_dev = drm;
>>>>>       priv->dp[dp->id] = &dp->dp_display;
>>>>> -    rc = dp->parser->parse(dp->parser, 
>>>>> dp->dp_display.connector_type);
>>>>> +    rc = dp->parser->parse(dp->parser);
>>>>>       if (rc) {
>>>>>           DRM_ERROR("device tree parsing failed\n");
>>>>>           goto end;
>>>>>       }
>>>>> -    dp->dp_display.next_bridge = dp->parser->next_bridge;
>>>>> -
>>>>>       dp->aux->drm_dev = drm;
>>>>>       rc = dp_aux_register(dp->aux);
>>>>>       if (rc) {
>>>>> @@ -1319,6 +1318,8 @@ 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.is_edp =
>>>>> +        (dp->dp_display.connector_type == DRM_MODE_CONNECTOR_eDP);
>>>>>       rc = dp_init_sub_modules(dp);
>>>>>       if (rc) {
>>>>> @@ -1508,7 +1509,8 @@ void msm_dp_irq_postinstall(struct msm_dp 
>>>>> *dp_display)
>>>>>       dp_hpd_event_setup(dp);
>>>>> -    dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
>>>>> +    if (!dp_display->is_edp)
>>>>> +        dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100);
>>>>>   }
>>>>>   void msm_dp_debugfs_init(struct msm_dp *dp_display, struct 
>>>>> drm_minor *minor)
>>>>> @@ -1530,6 +1532,64 @@ void msm_dp_debugfs_init(struct msm_dp 
>>>>> *dp_display, struct drm_minor *minor)
>>>>>       }
>>>>>   }
>>>>> +static int dp_display_get_next_bridge(struct msm_dp *dp)
>>>>> +{
>>>>> +    int rc;
>>>>> +    struct dp_display_private *dp_priv;
>>>>> +    struct device_node *aux_bus;
>>>>> +    struct device *dev;
>>>>> +
>>>>> +    dp_priv = container_of(dp, struct dp_display_private, 
>>>>> dp_display);
>>>>> +    dev = &dp_priv->pdev->dev;
>>>>> +    aux_bus = of_get_child_by_name(dev->of_node, "aux-bus");
>>>>> +
>>>>> +    if (aux_bus && dp->is_edp) {
>>>>> +        dp_display_host_init(dp_priv);
>>>>> +        dp_catalog_ctrl_hpd_config(dp_priv->catalog);
>>>>> +        dp_display_host_phy_init(dp_priv);
>>>>> +        enable_irq(dp_priv->irq);
>>>>> +
>>>>> +        /*
>>>>> +         * The code below assumes that the panel will finish probing
>>>>> +         * by the time devm_of_dp_aux_populate_ep_devices() returns.
>>>>> +         * This isn't a great assumption since it will fail if the
>>>>> +         * panel driver is probed asynchronously but is the best we
>>>>> +         * can do without a bigger driver reorganization.
>>>>> +         */
>>>>> +        rc = devm_of_dp_aux_populate_ep_devices(dp_priv->aux);
>>>>> +        of_node_put(aux_bus);
>>>>> +        if (rc)
>>>>> +            goto error;
>>>>> +    } else if (dp->is_edp) {
>>>>> +        DRM_ERROR("eDP aux_bus not found\n");
>>>>> +        return -ENODEV;
>>>>> +    }
>>>>> +
>>>>> +    /*
>>>>> +     * External bridges are mandatory for eDP interfaces: one has to
>>>>> +     * provide at least an eDP panel (which gets wrapped into 
>>>>> panel-bridge).
>>>>> +     *
>>>>> +     * For DisplayPort interfaces external bridges are optional, so
>>>>> +     * silently ignore an error if one is not present (-ENODEV).
>>>>> +     */
>>>>> +    rc = dp_parser_find_next_bridge(dp_priv->parser);
>>>>> +    if (!dp->is_edp && rc == -ENODEV)
>>>>> +        return 0;
>>>>> +
>>>>> +    if (!rc) {
>>>>> +        dp->next_bridge = dp_priv->parser->next_bridge;
>>>>> +        return 0;
>>>>> +    }
>>>>> +
>>>>> +error:
>>>>> +    if (dp->is_edp) {
>>>>> +        disable_irq(dp_priv->irq);
>>>>> +        dp_display_host_phy_exit(dp_priv);
>>>>> +        dp_display_host_deinit(dp_priv);
>>>>> +    }
>>>>> +    return rc;
>>>>> +}
>>>>> +
>>>>>   int msm_dp_modeset_init(struct msm_dp *dp_display, struct 
>>>>> drm_device *dev,
>>>>>               struct drm_encoder *encoder)
>>>>>   {
>>>>> @@ -1553,6 +1613,10 @@ int msm_dp_modeset_init(struct msm_dp 
>>>>> *dp_display, struct drm_device *dev,
>>>>>       dp_display->encoder = encoder;
>>>>> +    ret = dp_display_get_next_bridge(dp_display);
>>>>> +    if (ret)
>>>>> +        return ret;
>>>>> +
>>>>>       dp_display->bridge = dp_bridge_init(dp_display, dev, encoder);
>>>>>       if (IS_ERR(dp_display->bridge)) {
>>>>>           ret = PTR_ERR(dp_display->bridge);
>>>>> diff --git a/drivers/gpu/drm/msm/dp/dp_display.h 
>>>>> b/drivers/gpu/drm/msm/dp/dp_display.h
>>>>> index 49a1d89..1377cc3 100644
>>>>> --- a/drivers/gpu/drm/msm/dp/dp_display.h
>>>>> +++ b/drivers/gpu/drm/msm/dp/dp_display.h
>>>>> @@ -21,6 +21,7 @@ struct msm_dp {
>>>>>       bool audio_enabled;
>>>>>       bool power_on;
>>>>>       unsigned int connector_type;
>>>>> +    bool is_edp;
>>>>>       hdmi_codec_plugged_cb plugged_cb;
>>>>> diff --git a/drivers/gpu/drm/msm/dp/dp_drm.c 
>>>>> b/drivers/gpu/drm/msm/dp/dp_drm.c
>>>>> index 7ce1aca..8a75c55 100644
>>>>> --- a/drivers/gpu/drm/msm/dp/dp_drm.c
>>>>> +++ b/drivers/gpu/drm/msm/dp/dp_drm.c
>>>>> @@ -114,10 +114,23 @@ struct drm_bridge *dp_bridge_init(struct 
>>>>> msm_dp *dp_display, struct drm_device *
>>>>>       bridge->funcs = &dp_bridge_ops;
>>>>>       bridge->type = dp_display->connector_type;
>>>>> -    bridge->ops =
>>>>> -        DRM_BRIDGE_OP_DETECT |
>>>>> -        DRM_BRIDGE_OP_HPD |
>>>>> -        DRM_BRIDGE_OP_MODES;
>>>>> +    /*
>>>>> +     * Many ops only make sense for DP. Why?
>>>>> +     * - Detect/HPD are used by DRM to know if a display is 
>>>>> _physically_
>>>>> +     *   there, not whether the display is powered on / finished 
>>>>> initting.
>>>>> +     *   On eDP we assume the display is always there because you 
>>>>> can't
>>>>> +     *   know until power is applied. If we don't implement the 
>>>>> ops DRM will
>>>>> +     *   assume our display is always there.
>>>>> +     * - Currently eDP mode reading is driven by the panel driver. 
>>>>> This
>>>>> +     *   allows the panel driver to properly power itself on to 
>>>>> read the
>>>>> +     *   modes.
>>>>> +     */
>>>>> +    if (!dp_display->is_edp) {
>>>>> +        bridge->ops =
>>>>> +            DRM_BRIDGE_OP_DETECT |
>>>>> +            DRM_BRIDGE_OP_HPD |
>>>>> +            DRM_BRIDGE_OP_MODES;
>>>>> +    }
>>>>>       rc = drm_bridge_attach(encoder, bridge, NULL, 
>>>>> DRM_BRIDGE_ATTACH_NO_CONNECTOR);
>>>>>       if (rc) {
>>>>> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.c 
>>>>> b/drivers/gpu/drm/msm/dp/dp_parser.c
>>>>> index 1056b8d..4bdbf91 100644
>>>>> --- a/drivers/gpu/drm/msm/dp/dp_parser.c
>>>>> +++ b/drivers/gpu/drm/msm/dp/dp_parser.c
>>>>> @@ -265,7 +265,7 @@ static int dp_parser_clock(struct dp_parser 
>>>>> *parser)
>>>>>       return 0;
>>>>>   }
>>>>> -static int dp_parser_find_next_bridge(struct dp_parser *parser)
>>>>> +int dp_parser_find_next_bridge(struct dp_parser *parser)
>>>>>   {
>>>>>       struct device *dev = &parser->pdev->dev;
>>>>>       struct drm_bridge *bridge;
>>>>> @@ -279,7 +279,7 @@ static int dp_parser_find_next_bridge(struct 
>>>>> dp_parser *parser)
>>>>>       return 0;
>>>>>   }
>>>>> -static int dp_parser_parse(struct dp_parser *parser, int 
>>>>> connector_type)
>>>>> +static int dp_parser_parse(struct dp_parser *parser)
>>>>>   {
>>>>>       int rc = 0;
>>>>> @@ -300,25 +300,6 @@ static int dp_parser_parse(struct dp_parser 
>>>>> *parser, int connector_type)
>>>>>       if (rc)
>>>>>           return rc;
>>>>> -    /*
>>>>> -     * External bridges are mandatory for eDP interfaces: one has to
>>>>> -     * provide at least an eDP panel (which gets wrapped into 
>>>>> panel-bridge).
>>>>> -     *
>>>>> -     * For DisplayPort interfaces external bridges are optional, so
>>>>> -     * silently ignore an error if one is not present (-ENODEV).
>>>>> -     */
>>>>> -    rc = dp_parser_find_next_bridge(parser);
>>>>> -    if (rc == -ENODEV) {
>>>>> -        if (connector_type == DRM_MODE_CONNECTOR_eDP) {
>>>>> -            DRM_ERROR("eDP: next bridge is not present\n");
>>>>> -            return rc;
>>>>> -        }
>>>>> -    } else if (rc) {
>>>>> -        if (rc != -EPROBE_DEFER)
>>>>> -            DRM_ERROR("DP: error parsing next bridge: %d\n", rc);
>>>>> -        return rc;
>>>>> -    }
>>>>> -
>>>>>       /* Map the corresponding regulator information according to
>>>>>        * version. Currently, since we only have one supported 
>>>>> platform,
>>>>>        * mapping the regulator directly.
>>>>> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.h 
>>>>> b/drivers/gpu/drm/msm/dp/dp_parser.h
>>>>> index d371bae..3a4d797 100644
>>>>> --- a/drivers/gpu/drm/msm/dp/dp_parser.h
>>>>> +++ b/drivers/gpu/drm/msm/dp/dp_parser.h
>>>>> @@ -125,7 +125,7 @@ struct dp_parser {
>>>>>       u32 max_dp_lanes;
>>>>>       struct drm_bridge *next_bridge;
>>>>> -    int (*parse)(struct dp_parser *parser, int connector_type);
>>>>> +    int (*parse)(struct dp_parser *parser);
>>>>>   };
>>>>>   /**
>>>>> @@ -141,4 +141,16 @@ struct dp_parser {
>>>>>    */
>>>>>   struct dp_parser *dp_parser_get(struct platform_device *pdev);
>>>>> +/**
>>>>> + * dp_parser_find_next_bridge() - find an additional bridge to DP
>>>>> + *
>>>>> + * @parser: dp_parser data from client
>>>>> + *
>>>>> + * This function is used to find any additional bridge attached to
>>>>> + * the DP controller. The eDP interface requires a panel bridge.
>>>>> + *
>>>>> + * Return: 0 if able to get the bridge, otherwise negative errno 
>>>>> for failure.
>>>>> + */
>>>>> +int dp_parser_find_next_bridge(struct dp_parser *parser);
>>>>> +
>>>>>   #endif
>>>>
>>>>
>>
>>


-- 
With best wishes
Dmitry

^ permalink raw reply	[flat|nested] 10+ messages in thread

end of thread, other threads:[~2022-05-06 21:18 UTC | newest]

Thread overview: 10+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2022-04-25 11:44 [PATCH v10 0/4] Add support for the eDP panel over aux_bus Sankeerth Billakanti
2022-04-25 11:44 ` [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus Sankeerth Billakanti
2022-05-06 20:49   ` Dmitry Baryshkov
2022-05-06 21:03     ` Abhinav Kumar
2022-05-06 21:05       ` Dmitry Baryshkov
2022-05-06 21:17         ` [Freedreno] " Abhinav Kumar
2022-05-06 21:17           ` Dmitry Baryshkov
2022-04-25 11:44 ` [PATCH v10 2/4] drm/msm/dp: Support only IRQ_HPD and REPLUG interrupts for eDP Sankeerth Billakanti
2022-04-25 11:44 ` [PATCH v10 3/4] drm/msm/dp: wait for hpd high before aux transaction Sankeerth Billakanti
2022-04-25 11:44 ` [PATCH v10 4/4] drm/msm/dp: Support the eDP modes given by panel Sankeerth Billakanti

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).