On 03/11/2014 04:09 PM, Laurent Pinchart wrote: > The subdev EDID ioctls receive a pad field that must reference an > existing pad and an EDID field that must point to a buffer. Validate > both fields in the core code instead of duplicating validation in all > drivers. > > Signed-off-by: Laurent Pinchart <laurent.pinchart@xxxxxxxxxxxxxxxx> > Acked-by: Sakari Ailus <sakari.ailus@xxxxxxxxxxxxxxx> Here is my: Reviewed-by: Hans Verkuil <hans.verkuil@xxxxxxxxx> But take note: the adv7604 driver does not handle a get_edid with edid->blocks == 0 correctly: it should fill in the blocks field with the real number of blocks and return 0 instead of returning EINVAL. I also read through the spec again and it does not actually explicitly say that you can use G_EDID with edid->blocks == 0, but I think it makes a lot of sense to do that. All existing drivers that use get_edid all return -EINVAL if blocks == 0, so this patch does not change anything with that. I plan on making a patch to clarify the spec and update the drivers, but you might want to make a patch for adv7604 yourself instead of waiting for me. I leave that up to you. Anyway, this patch is fine. Regards, Hans > --- > drivers/media/i2c/ad9389b.c | 2 -- > drivers/media/i2c/adv7511.c | 2 -- > drivers/media/i2c/adv7604.c | 4 ---- > drivers/media/i2c/adv7842.c | 4 ---- > drivers/media/v4l2-core/v4l2-subdev.c | 24 ++++++++++++++++++++---- > 5 files changed, 20 insertions(+), 16 deletions(-) > > diff --git a/drivers/media/i2c/ad9389b.c b/drivers/media/i2c/ad9389b.c > index 4cdff9e..5b78828 100644 > --- a/drivers/media/i2c/ad9389b.c > +++ b/drivers/media/i2c/ad9389b.c > @@ -683,8 +683,6 @@ static int ad9389b_get_edid(struct v4l2_subdev *sd, > return -EINVAL; > if (edid->blocks == 0 || edid->blocks > 256) > return -EINVAL; > - if (!edid->edid) > - return -EINVAL; > if (!state->edid.segments) { > v4l2_dbg(1, debug, sd, "EDID segment 0 not found\n"); > return -ENODATA; > diff --git a/drivers/media/i2c/adv7511.c b/drivers/media/i2c/adv7511.c > index de7ddf5..ff1c2cd 100644 > --- a/drivers/media/i2c/adv7511.c > +++ b/drivers/media/i2c/adv7511.c > @@ -784,8 +784,6 @@ static int adv7511_get_edid(struct v4l2_subdev *sd, > return -EINVAL; > if ((edid->blocks == 0) || (edid->blocks > 256)) > return -EINVAL; > - if (!edid->edid) > - return -EINVAL; > if (!state->edid.segments) { > v4l2_dbg(1, debug, sd, "EDID segment 0 not found\n"); > return -ENODATA; > diff --git a/drivers/media/i2c/adv7604.c b/drivers/media/i2c/adv7604.c > index 71c8570..de3db42 100644 > --- a/drivers/media/i2c/adv7604.c > +++ b/drivers/media/i2c/adv7604.c > @@ -1673,8 +1673,6 @@ static int adv7604_get_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edi > return -EINVAL; > if (edid->start_block == 1) > edid->blocks = 1; > - if (!edid->edid) > - return -EINVAL; > > if (edid->blocks > state->edid.blocks) > edid->blocks = state->edid.blocks; > @@ -1761,8 +1759,6 @@ static int adv7604_set_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edi > edid->blocks = 2; > return -E2BIG; > } > - if (!edid->edid) > - return -EINVAL; > > v4l2_dbg(2, debug, sd, "%s: write EDID pad %d, edid.present = 0x%x\n", > __func__, edid->pad, state->edid.present); > diff --git a/drivers/media/i2c/adv7842.c b/drivers/media/i2c/adv7842.c > index a319275..3c5a7d9 100644 > --- a/drivers/media/i2c/adv7842.c > +++ b/drivers/media/i2c/adv7842.c > @@ -2035,8 +2035,6 @@ static int adv7842_get_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *edi > return -EINVAL; > if (edid->start_block == 1) > edid->blocks = 1; > - if (!edid->edid) > - return -EINVAL; > > switch (edid->pad) { > case ADV7842_EDID_PORT_A: > @@ -2071,8 +2069,6 @@ static int adv7842_set_edid(struct v4l2_subdev *sd, struct v4l2_subdev_edid *e) > return -EINVAL; > if (e->blocks > 2) > return -E2BIG; > - if (!e->edid) > - return -EINVAL; > > /* todo, per edid */ > state->aspect_ratio = v4l2_calc_aspect_ratio(e->edid[0x15], > diff --git a/drivers/media/v4l2-core/v4l2-subdev.c b/drivers/media/v4l2-core/v4l2-subdev.c > index 853fb84..f6185f9 100644 > --- a/drivers/media/v4l2-core/v4l2-subdev.c > +++ b/drivers/media/v4l2-core/v4l2-subdev.c > @@ -349,11 +349,27 @@ static long subdev_do_ioctl(struct file *file, unsigned int cmd, void *arg) > sd, pad, set_selection, subdev_fh, sel); > } > > - case VIDIOC_SUBDEV_G_EDID: > - return v4l2_subdev_call(sd, pad, get_edid, arg); > + case VIDIOC_SUBDEV_G_EDID: { > + struct v4l2_subdev_edid *edid = arg; > > - case VIDIOC_SUBDEV_S_EDID: > - return v4l2_subdev_call(sd, pad, set_edid, arg); > + if (edid->pad >= sd->entity.num_pads) > + return -EINVAL; > + if (edid->blocks && edid->edid == NULL) > + return -EINVAL; > + > + return v4l2_subdev_call(sd, pad, get_edid, edid); > + } > + > + case VIDIOC_SUBDEV_S_EDID: { > + struct v4l2_subdev_edid *edid = arg; > + > + if (edid->pad >= sd->entity.num_pads) > + return -EINVAL; > + if (edid->blocks && edid->edid == NULL) > + return -EINVAL; > + > + return v4l2_subdev_call(sd, pad, set_edid, edid); > + } > > case VIDIOC_SUBDEV_DV_TIMINGS_CAP: { > struct v4l2_dv_timings_cap *cap = arg; > -- To unsubscribe from this list: send the line "unsubscribe linux-media" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html