[RFC] drm/imx: parallel-display: add ddc support

linux-kernel-dev at beckhoff.com linux-kernel-dev at beckhoff.com
Fri Nov 27 03:42:51 PST 2015


From: Patrick Brünn <p.bruenn at beckhoff.com>

Add ddc-i2c-bus as an additional source to read edid.

If you convert the 24-bit parallel-display signal to DVI and combine it
with I2C, you can connect DVI-D displays to your imx powered devices. An
example for this setup is the CX9020 Embedded PC.

Signed-off-by: Patrick Brünn <p.bruenn at beckhoff.com>
---
 drivers/gpu/drm/imx/parallel-display.c | 30 ++++++++++++++++++++++++++++++
 1 file changed, 30 insertions(+)

diff --git a/drivers/gpu/drm/imx/parallel-display.c b/drivers/gpu/drm/imx/parallel-display.c
index b4deb9c..32b745a 100644
--- a/drivers/gpu/drm/imx/parallel-display.c
+++ b/drivers/gpu/drm/imx/parallel-display.c
@@ -15,6 +15,7 @@
 
 #include <linux/component.h>
 #include <linux/module.h>
+#include <linux/i2c.h>
 #include <drm/drmP.h>
 #include <drm/drm_fb_helper.h>
 #include <drm/drm_crtc_helper.h>
@@ -38,6 +39,7 @@ struct imx_parallel_display {
 	int mode_valid;
 	struct drm_display_mode mode;
 	struct drm_panel *panel;
+	struct i2c_adapter *ddc;
 };
 
 static enum drm_connector_status imx_pd_connector_detect(
@@ -52,6 +54,16 @@ static int imx_pd_connector_get_modes(struct drm_connector *connector)
 	struct device_node *np = imxpd->dev->of_node;
 	int num_modes = 0;
 
+	if (imxpd->ddc) {
+		kfree(imxpd->edid);
+		imxpd->edid = drm_get_edid(connector, imxpd->ddc);
+		drm_mode_connector_update_edid_property(connector, imxpd->edid);
+		num_modes = drm_add_edid_modes(connector, imxpd->edid);
+		if (num_modes > 0)
+			return num_modes;
+	}
+
+
 	if (imxpd->panel && imxpd->panel->funcs &&
 	    imxpd->panel->funcs->get_modes) {
 		num_modes = imxpd->panel->funcs->get_modes(imxpd->panel);
@@ -90,6 +102,16 @@ static int imx_pd_connector_get_modes(struct drm_connector *connector)
 	return num_modes;
 }
 
+static enum drm_mode_status imx_pd_connector_mode_valid(struct drm_connector *connector,
+							struct drm_display_mode *mode)
+{
+	static const int IMX53_MAX_PIXEL_CLOCK_KHZ = 170000;
+
+	if (mode->clock > IMX53_MAX_PIXEL_CLOCK_KHZ)
+		return MODE_CLOCK_HIGH;
+	return MODE_OK;
+}
+
 static struct drm_encoder *imx_pd_connector_best_encoder(
 		struct drm_connector *connector)
 {
@@ -153,6 +175,7 @@ static struct drm_connector_funcs imx_pd_connector_funcs = {
 
 static struct drm_connector_helper_funcs imx_pd_connector_helper_funcs = {
 	.get_modes = imx_pd_connector_get_modes,
+	.mode_valid = imx_pd_connector_mode_valid,
 	.best_encoder = imx_pd_connector_best_encoder,
 };
 
@@ -209,6 +232,7 @@ static int imx_pd_bind(struct device *dev, struct device *master, void *data)
 {
 	struct drm_device *drm = data;
 	struct device_node *np = dev->of_node;
+	struct device_node *ddc_node;
 	struct device_node *port;
 	const u8 *edidp;
 	struct imx_parallel_display *imxpd;
@@ -219,6 +243,12 @@ static int imx_pd_bind(struct device *dev, struct device *master, void *data)
 	if (!imxpd)
 		return -ENOMEM;
 
+	ddc_node = of_parse_phandle(np, "ddc-i2c-bus", 0);
+	if (ddc_node) {
+		imxpd->ddc = of_find_i2c_adapter_by_node(ddc_node);
+		of_node_put(ddc_node);
+	}
+
 	edidp = of_get_property(np, "edid", &imxpd->edid_len);
 	if (edidp)
 		imxpd->edid = kmemdup(edidp, imxpd->edid_len, GFP_KERNEL);
-- 
1.9.1



More information about the dri-devel mailing list