Commit dd08beb9 authored by Mats Randgaard's avatar Mats Randgaard Committed by Mauro Carvalho Chehab

[media] adv7604: improve EDID handling

- split edid_write_block()
- do not use edid->edid before the validity check
- Return -EINVAL if edid->pad is invalid
- Save both registers for SPA port A
- Set SPA location to default value if it is not found
Signed-off-by: default avatarMats Randgaard <matrandg@cisco.com>
Signed-off-by: default avatarHans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: default avatarMauro Carvalho Chehab <m.chehab@samsung.com>
parent 3e86aa85
...@@ -72,7 +72,7 @@ struct adv7604_state { ...@@ -72,7 +72,7 @@ struct adv7604_state {
u32 present; u32 present;
unsigned blocks; unsigned blocks;
} edid; } edid;
u16 spa_port_a; u16 spa_port_a[2];
struct v4l2_fract aspect_ratio; struct v4l2_fract aspect_ratio;
u32 rgb_quantization_range; u32 rgb_quantization_range;
struct workqueue_struct *work_queues; struct workqueue_struct *work_queues;
...@@ -510,22 +510,9 @@ static inline int edid_read_block(struct v4l2_subdev *sd, unsigned len, u8 *val) ...@@ -510,22 +510,9 @@ static inline int edid_read_block(struct v4l2_subdev *sd, unsigned len, u8 *val)
return 0; return 0;
} }
static void adv7604_delayed_work_enable_hotplug(struct work_struct *work)
{
struct delayed_work *dwork = to_delayed_work(work);
struct adv7604_state *state = container_of(dwork, struct adv7604_state,
delayed_work_enable_hotplug);
struct v4l2_subdev *sd = &state->sd;
v4l2_dbg(2, debug, sd, "%s: enable hotplug\n", __func__);
v4l2_subdev_notify(sd, ADV7604_HOTPLUG, (void *)&state->edid.present);
}
static inline int edid_write_block(struct v4l2_subdev *sd, static inline int edid_write_block(struct v4l2_subdev *sd,
unsigned len, const u8 *val) unsigned len, const u8 *val)
{ {
struct i2c_client *client = v4l2_get_subdevdata(sd);
struct adv7604_state *state = to_state(sd); struct adv7604_state *state = to_state(sd);
int err = 0; int err = 0;
int i; int i;
...@@ -535,24 +522,19 @@ static inline int edid_write_block(struct v4l2_subdev *sd, ...@@ -535,24 +522,19 @@ static inline int edid_write_block(struct v4l2_subdev *sd,
for (i = 0; !err && i < len; i += I2C_SMBUS_BLOCK_MAX) for (i = 0; !err && i < len; i += I2C_SMBUS_BLOCK_MAX)
err = adv_smbus_write_i2c_block_data(state->i2c_edid, i, err = adv_smbus_write_i2c_block_data(state->i2c_edid, i,
I2C_SMBUS_BLOCK_MAX, val + i); I2C_SMBUS_BLOCK_MAX, val + i);
if (err) return err;
return err; }
/* adv7604 calculates the checksums and enables I2C access to internal static void adv7604_delayed_work_enable_hotplug(struct work_struct *work)
EDID RAM from DDC port. */ {
rep_write_and_or(sd, 0x77, 0xf0, state->edid.present); struct delayed_work *dwork = to_delayed_work(work);
struct adv7604_state *state = container_of(dwork, struct adv7604_state,
delayed_work_enable_hotplug);
struct v4l2_subdev *sd = &state->sd;
for (i = 0; i < 1000; i++) { v4l2_dbg(2, debug, sd, "%s: enable hotplug\n", __func__);
if (rep_read(sd, 0x7d) & state->edid.present)
break;
mdelay(1);
}
if (i == 1000) {
v4l_err(client, "error enabling edid (0x%x)\n", state->edid.present);
return -EIO;
}
return 0; v4l2_subdev_notify(sd, ADV7604_HOTPLUG, (void *)&state->edid.present);
} }
static inline int hdmi_read(struct v4l2_subdev *sd, u8 reg) static inline int hdmi_read(struct v4l2_subdev *sd, u8 reg)
...@@ -1621,7 +1603,7 @@ static int adv7604_get_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edi ...@@ -1621,7 +1603,7 @@ static int adv7604_get_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edi
return 0; return 0;
} }
static int get_edid_spa_location(struct v4l2_subdev *sd, const u8 *edid) static int get_edid_spa_location(const u8 *edid)
{ {
u8 d; u8 d;
...@@ -1652,9 +1634,10 @@ static int get_edid_spa_location(struct v4l2_subdev *sd, const u8 *edid) ...@@ -1652,9 +1634,10 @@ static int get_edid_spa_location(struct v4l2_subdev *sd, const u8 *edid)
static int adv7604_set_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edid) static int adv7604_set_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edid)
{ {
struct adv7604_state *state = to_state(sd); struct adv7604_state *state = to_state(sd);
int spa_loc = get_edid_spa_location(sd, edid->edid); int spa_loc;
int tmp = 0; int tmp = 0;
int err; int err;
int i;
if (edid->pad > ADV7604_EDID_PORT_D) if (edid->pad > ADV7604_EDID_PORT_D)
return -EINVAL; return -EINVAL;
...@@ -1684,35 +1667,43 @@ static int adv7604_set_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edi ...@@ -1684,35 +1667,43 @@ static int adv7604_set_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edi
if (!edid->edid) if (!edid->edid)
return -EINVAL; return -EINVAL;
v4l2_dbg(2, debug, sd, "%s: write EDID pad %d, edid.present = 0x%x\n",
__func__, edid->pad, state->edid.present);
/* Disable hotplug and I2C access to EDID RAM from DDC port */ /* Disable hotplug and I2C access to EDID RAM from DDC port */
cancel_delayed_work_sync(&state->delayed_work_enable_hotplug); cancel_delayed_work_sync(&state->delayed_work_enable_hotplug);
v4l2_subdev_notify(sd, ADV7604_HOTPLUG, (void *)&tmp); v4l2_subdev_notify(sd, ADV7604_HOTPLUG, (void *)&tmp);
rep_write_and_or(sd, 0x77, 0xf0, 0x00); rep_write_and_or(sd, 0x77, 0xf0, 0x00);
v4l2_dbg(2, debug, sd, "%s: write EDID pad %d, edid.present = 0x%x\n", spa_loc = get_edid_spa_location(edid->edid);
__func__, edid->pad, state->edid.present); if (spa_loc < 0)
spa_loc = 0xc0; /* Default value [REF_02, p. 116] */
switch (edid->pad) { switch (edid->pad) {
case ADV7604_EDID_PORT_A: case ADV7604_EDID_PORT_A:
state->spa_port_a = edid->edid[spa_loc]; state->spa_port_a[0] = edid->edid[spa_loc];
state->spa_port_a[1] = edid->edid[spa_loc + 1];
break; break;
case ADV7604_EDID_PORT_B: case ADV7604_EDID_PORT_B:
rep_write(sd, 0x70, (spa_loc < 0) ? 0 : edid->edid[spa_loc]); rep_write(sd, 0x70, edid->edid[spa_loc]);
rep_write(sd, 0x71, (spa_loc < 0) ? 0 : edid->edid[spa_loc + 1]); rep_write(sd, 0x71, edid->edid[spa_loc + 1]);
break; break;
case ADV7604_EDID_PORT_C: case ADV7604_EDID_PORT_C:
rep_write(sd, 0x72, (spa_loc < 0) ? 0 : edid->edid[spa_loc]); rep_write(sd, 0x72, edid->edid[spa_loc]);
rep_write(sd, 0x73, (spa_loc < 0) ? 0 : edid->edid[spa_loc + 1]); rep_write(sd, 0x73, edid->edid[spa_loc + 1]);
break; break;
case ADV7604_EDID_PORT_D: case ADV7604_EDID_PORT_D:
rep_write(sd, 0x74, (spa_loc < 0) ? 0 : edid->edid[spa_loc]); rep_write(sd, 0x74, edid->edid[spa_loc]);
rep_write(sd, 0x75, (spa_loc < 0) ? 0 : edid->edid[spa_loc + 1]); rep_write(sd, 0x75, edid->edid[spa_loc + 1]);
break; break;
default:
return -EINVAL;
} }
rep_write(sd, 0x76, (spa_loc < 0) ? 0x00 : spa_loc); rep_write(sd, 0x76, spa_loc & 0xff);
rep_write_and_or(sd, 0x77, 0xbf, (spa_loc < 0) ? 0x00 : (spa_loc >> 2) & 0x40); rep_write_and_or(sd, 0x77, 0xbf, (spa_loc >> 2) & 0x40);
if (spa_loc > 0) edid->edid[spa_loc] = state->spa_port_a[0];
edid->edid[spa_loc] = state->spa_port_a; edid->edid[spa_loc + 1] = state->spa_port_a[1];
memcpy(state->edid.edid, edid->edid, 128 * edid->blocks); memcpy(state->edid.edid, edid->edid, 128 * edid->blocks);
state->edid.blocks = edid->blocks; state->edid.blocks = edid->blocks;
...@@ -1726,6 +1717,21 @@ static int adv7604_set_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edi ...@@ -1726,6 +1717,21 @@ static int adv7604_set_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edi
return err; return err;
} }
/* adv7604 calculates the checksums and enables I2C access to internal
EDID RAM from DDC port. */
rep_write_and_or(sd, 0x77, 0xf0, state->edid.present);
for (i = 0; i < 1000; i++) {
if (rep_read(sd, 0x7d) & state->edid.present)
break;
mdelay(1);
}
if (i == 1000) {
v4l2_err(sd, "error enabling edid (0x%x)\n", state->edid.present);
return -EIO;
}
/* enable hotplug after 100 ms */ /* enable hotplug after 100 ms */
queue_delayed_work(state->work_queues, queue_delayed_work(state->work_queues,
&state->delayed_work_enable_hotplug, HZ / 10); &state->delayed_work_enable_hotplug, HZ / 10);
......
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