Commit 1edc1a1c authored by Laurent Pinchart's avatar Laurent Pinchart Committed by Tomi Valkeinen

drm/omap: hdmi5: Rework EDID read to isolate data read

In preparation of adding DRM bridge support to the hdmi5 encoder code,
rework the EDID read to isolate data read.

The hdmi_read_edid() function is the main entry point. It performs all
initialisation steps required prior to reading the EDID (such as
ensuring the device is powered on), as well as corresponding cleanup
steps afterwards. EDID read itself is handled by hdmi_read_edid_data()
that calls the hdmi5_core_ddc_read() function to read individual blocks.

This new code architecture will allow reusing hdmi_read_edid() and
hdmi5_core_ddc_read() for the drm_bridge EDID read implementation, while
swapping out hdmi_read_edid_data() for the DRM drm_do_get_edid()
function.
Signed-off-by: default avatarLaurent Pinchart <laurent.pinchart@ideasonboard.com>
Reviewed-by: default avatarTomi Valkeinen <tomi.valkeinen@ti.com>
Tested-by: default avatarSebastian Reichel <sebastian.reichel@collabora.com>
Reviewed-by: default avatarSebastian Reichel <sebastian.reichel@collabora.com>
Signed-off-by: default avatarTomi Valkeinen <tomi.valkeinen@ti.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200226112514.12455-31-laurent.pinchart@ideasonboard.com
parent a4659694
...@@ -271,30 +271,6 @@ static int hdmi_dump_regs(struct seq_file *s, void *p) ...@@ -271,30 +271,6 @@ static int hdmi_dump_regs(struct seq_file *s, void *p)
return 0; return 0;
} }
static int read_edid(struct omap_hdmi *hdmi, u8 *buf, int len)
{
int r;
int idlemode;
mutex_lock(&hdmi->lock);
r = hdmi_runtime_get(hdmi);
BUG_ON(r);
idlemode = REG_GET(hdmi->wp.base, HDMI_WP_SYSCONFIG, 3, 2);
/* No-idle mode */
REG_FLD_MOD(hdmi->wp.base, HDMI_WP_SYSCONFIG, 1, 3, 2);
r = hdmi5_read_edid(&hdmi->core, buf, len);
REG_FLD_MOD(hdmi->wp.base, HDMI_WP_SYSCONFIG, idlemode, 3, 2);
hdmi_runtime_put(hdmi);
mutex_unlock(&hdmi->lock);
return r;
}
static void hdmi_start_audio_stream(struct omap_hdmi *hd) static void hdmi_start_audio_stream(struct omap_hdmi *hd)
{ {
REG_FLD_MOD(hd->wp.base, HDMI_WP_SYSCONFIG, 1, 3, 2); REG_FLD_MOD(hd->wp.base, HDMI_WP_SYSCONFIG, 1, 3, 2);
...@@ -412,32 +388,73 @@ static void hdmi_disconnect(struct omap_dss_device *src, ...@@ -412,32 +388,73 @@ static void hdmi_disconnect(struct omap_dss_device *src,
#define MAX_EDID 512 #define MAX_EDID 512
static struct edid *hdmi_read_edid(struct omap_dss_device *dssdev) static struct edid *hdmi_read_edid_data(struct hdmi_core_data *core)
{ {
struct omap_hdmi *hdmi = dssdev_to_hdmi(dssdev); int max_ext_blocks = 3;
bool need_enable; int r, n, i;
u8 *edid; u8 *edid;
int r;
edid = kzalloc(MAX_EDID, GFP_KERNEL); edid = kzalloc(MAX_EDID, GFP_KERNEL);
if (!edid) if (!edid)
return NULL; return NULL;
r = hdmi5_core_ddc_read(core, edid, 0, EDID_LENGTH);
if (r)
goto error;
n = edid[0x7e];
if (n > max_ext_blocks)
n = max_ext_blocks;
for (i = 1; i <= n; i++) {
r = hdmi5_core_ddc_read(core, edid + i * EDID_LENGTH, i,
EDID_LENGTH);
if (r)
goto error;
}
return (struct edid *)edid;
error:
kfree(edid);
return NULL;
}
static struct edid *hdmi_read_edid(struct omap_dss_device *dssdev)
{
struct omap_hdmi *hdmi = dssdev_to_hdmi(dssdev);
struct edid *edid;
bool need_enable;
int idlemode;
int r;
need_enable = hdmi->core_enabled == false; need_enable = hdmi->core_enabled == false;
if (need_enable) { if (need_enable) {
r = hdmi_core_enable(hdmi); r = hdmi_core_enable(hdmi);
if (r) { if (r)
kfree(edid);
return NULL; return NULL;
}
} }
r = read_edid(hdmi, edid, MAX_EDID); mutex_lock(&hdmi->lock);
if (r < 0) { r = hdmi_runtime_get(hdmi);
kfree(edid); BUG_ON(r);
edid = NULL;
} idlemode = REG_GET(hdmi->wp.base, HDMI_WP_SYSCONFIG, 3, 2);
/* No-idle mode */
REG_FLD_MOD(hdmi->wp.base, HDMI_WP_SYSCONFIG, 1, 3, 2);
hdmi5_core_ddc_init(&hdmi->core);
edid = hdmi_read_edid_data(&hdmi->core);
hdmi5_core_ddc_uninit(&hdmi->core);
REG_FLD_MOD(hdmi->wp.base, HDMI_WP_SYSCONFIG, idlemode, 3, 2);
hdmi_runtime_put(hdmi);
mutex_unlock(&hdmi->lock);
if (need_enable) if (need_enable)
hdmi_core_disable(hdmi); hdmi_core_disable(hdmi);
......
...@@ -23,7 +23,7 @@ ...@@ -23,7 +23,7 @@
#include "hdmi5_core.h" #include "hdmi5_core.h"
static void hdmi_core_ddc_init(struct hdmi_core_data *core) void hdmi5_core_ddc_init(struct hdmi_core_data *core)
{ {
void __iomem *base = core->base; void __iomem *base = core->base;
const unsigned long long iclk = 266000000; /* DSS L3 ICLK */ const unsigned long long iclk = 266000000; /* DSS L3 ICLK */
...@@ -102,7 +102,7 @@ static void hdmi_core_ddc_init(struct hdmi_core_data *core) ...@@ -102,7 +102,7 @@ static void hdmi_core_ddc_init(struct hdmi_core_data *core)
REG_FLD_MOD(base, HDMI_CORE_I2CM_INT, 0x0, 2, 2); REG_FLD_MOD(base, HDMI_CORE_I2CM_INT, 0x0, 2, 2);
} }
static void hdmi_core_ddc_uninit(struct hdmi_core_data *core) void hdmi5_core_ddc_uninit(struct hdmi_core_data *core)
{ {
void __iomem *base = core->base; void __iomem *base = core->base;
...@@ -112,14 +112,14 @@ static void hdmi_core_ddc_uninit(struct hdmi_core_data *core) ...@@ -112,14 +112,14 @@ static void hdmi_core_ddc_uninit(struct hdmi_core_data *core)
REG_FLD_MOD(base, HDMI_CORE_I2CM_INT, 0x1, 2, 2); REG_FLD_MOD(base, HDMI_CORE_I2CM_INT, 0x1, 2, 2);
} }
static int hdmi_core_ddc_edid(struct hdmi_core_data *core, u8 *pedid, u8 ext) int hdmi5_core_ddc_read(void *data, u8 *buf, unsigned int block, size_t len)
{ {
struct hdmi_core_data *core = data;
void __iomem *base = core->base; void __iomem *base = core->base;
u8 cur_addr; u8 cur_addr;
char checksum = 0;
const int retries = 1000; const int retries = 1000;
u8 seg_ptr = ext / 2; u8 seg_ptr = block / 2;
u8 edidbase = ((ext % 2) * 0x80); u8 edidbase = ((block % 2) * EDID_LENGTH);
REG_FLD_MOD(base, HDMI_CORE_I2CM_SEGPTR, seg_ptr, 7, 0); REG_FLD_MOD(base, HDMI_CORE_I2CM_SEGPTR, seg_ptr, 7, 0);
...@@ -127,7 +127,7 @@ static int hdmi_core_ddc_edid(struct hdmi_core_data *core, u8 *pedid, u8 ext) ...@@ -127,7 +127,7 @@ static int hdmi_core_ddc_edid(struct hdmi_core_data *core, u8 *pedid, u8 ext)
* TODO: We use polling here, although we probably should use proper * TODO: We use polling here, although we probably should use proper
* interrupts. * interrupts.
*/ */
for (cur_addr = 0; cur_addr < 128; ++cur_addr) { for (cur_addr = 0; cur_addr < len; ++cur_addr) {
int i; int i;
/* clear ERROR and DONE */ /* clear ERROR and DONE */
...@@ -164,45 +164,13 @@ static int hdmi_core_ddc_edid(struct hdmi_core_data *core, u8 *pedid, u8 ext) ...@@ -164,45 +164,13 @@ static int hdmi_core_ddc_edid(struct hdmi_core_data *core, u8 *pedid, u8 ext)
return -EIO; return -EIO;
} }
pedid[cur_addr] = REG_GET(base, HDMI_CORE_I2CM_DATAI, 7, 0); buf[cur_addr] = REG_GET(base, HDMI_CORE_I2CM_DATAI, 7, 0);
checksum += pedid[cur_addr];
} }
return 0; return 0;
} }
int hdmi5_read_edid(struct hdmi_core_data *core, u8 *edid, int len)
{
int r, n, i;
int max_ext_blocks = (len / 128) - 1;
if (len < 128)
return -EINVAL;
hdmi_core_ddc_init(core);
r = hdmi_core_ddc_edid(core, edid, 0);
if (r)
goto out;
n = edid[0x7e];
if (n > max_ext_blocks)
n = max_ext_blocks;
for (i = 1; i <= n; i++) {
r = hdmi_core_ddc_edid(core, edid + i * EDID_LENGTH, i);
if (r)
goto out;
}
out:
hdmi_core_ddc_uninit(core);
return r ? r : len;
}
void hdmi5_core_dump(struct hdmi_core_data *core, struct seq_file *s) void hdmi5_core_dump(struct hdmi_core_data *core, struct seq_file *s)
{ {
......
...@@ -281,7 +281,10 @@ struct csc_table { ...@@ -281,7 +281,10 @@ struct csc_table {
u16 c1, c2, c3, c4; u16 c1, c2, c3, c4;
}; };
int hdmi5_read_edid(struct hdmi_core_data *core, u8 *edid, int len); void hdmi5_core_ddc_init(struct hdmi_core_data *core);
int hdmi5_core_ddc_read(void *data, u8 *buf, unsigned int block, size_t len);
void hdmi5_core_ddc_uninit(struct hdmi_core_data *core);
void hdmi5_core_dump(struct hdmi_core_data *core, struct seq_file *s); void hdmi5_core_dump(struct hdmi_core_data *core, struct seq_file *s);
int hdmi5_core_handle_irqs(struct hdmi_core_data *core); int hdmi5_core_handle_irqs(struct hdmi_core_data *core);
void hdmi5_configure(struct hdmi_core_data *core, struct hdmi_wp_data *wp, void hdmi5_configure(struct hdmi_core_data *core, struct hdmi_wp_data *wp,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment