Commit c7405264 authored by Janusz Krzysztofik's avatar Janusz Krzysztofik Committed by Mauro Carvalho Chehab

media: ov6650: Add try support to selection API operations

Try requests are now only supported by format processing pad operations
implemented by the driver.  The driver selection API operations
currently respond to them with -EINVAL.  While that is correct, it
constraints video device drivers to not use subdevice cropping at all
while processing user requested active frame size, otherwise their set
try format results might differ from active.  As a consequence, we
can't fix set format pad operation as not to touch crop rectangle since
that would affect users not being able to set arbitrary frame sizes.
Moreover, without a working set try selection support we are not able
to use pad config crop rectangle as a reference while processing set
try format requests.

Implement missing try selection support.  Moreover, as it will be now
possible to maintain the pad config crop rectangle via selection API,
start using it instead of the active one as a reference while
processing set try format requests.

is_unscaled_ok() helper, now also called from set selection operation,
has been just moved up in the source file to avoid a prototype, with no
functional changes.

[Sakari Ailus: Rebase on subdev state patches]

Fixes: 717fd5b4 ("[media] v4l2: replace try_mbus_fmt by set_fmt")
Signed-off-by: default avatarJanusz Krzysztofik <jmkrzyszt@gmail.com>
Signed-off-by: default avatarSakari Ailus <sakari.ailus@linux.intel.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@kernel.org>
parent 1f6f1e95
...@@ -472,9 +472,16 @@ static int ov6650_get_selection(struct v4l2_subdev *sd, ...@@ -472,9 +472,16 @@ static int ov6650_get_selection(struct v4l2_subdev *sd,
{ {
struct i2c_client *client = v4l2_get_subdevdata(sd); struct i2c_client *client = v4l2_get_subdevdata(sd);
struct ov6650 *priv = to_ov6650(client); struct ov6650 *priv = to_ov6650(client);
struct v4l2_rect *rect;
if (sel->which != V4L2_SUBDEV_FORMAT_ACTIVE) if (sel->which == V4L2_SUBDEV_FORMAT_TRY) {
return -EINVAL; /* pre-select try crop rectangle */
rect = &sd_state->pads->try_crop;
} else {
/* pre-select active crop rectangle */
rect = &priv->rect;
}
switch (sel->target) { switch (sel->target) {
case V4L2_SEL_TGT_CROP_BOUNDS: case V4L2_SEL_TGT_CROP_BOUNDS:
...@@ -483,14 +490,22 @@ static int ov6650_get_selection(struct v4l2_subdev *sd, ...@@ -483,14 +490,22 @@ static int ov6650_get_selection(struct v4l2_subdev *sd,
sel->r.width = W_CIF; sel->r.width = W_CIF;
sel->r.height = H_CIF; sel->r.height = H_CIF;
return 0; return 0;
case V4L2_SEL_TGT_CROP: case V4L2_SEL_TGT_CROP:
sel->r = priv->rect; /* use selected crop rectangle */
sel->r = *rect;
return 0; return 0;
default: default:
return -EINVAL; return -EINVAL;
} }
} }
static bool is_unscaled_ok(int width, int height, struct v4l2_rect *rect)
{
return width > rect->width >> 1 || height > rect->height >> 1;
}
static void ov6650_bind_align_crop_rectangle(struct v4l2_rect *rect) static void ov6650_bind_align_crop_rectangle(struct v4l2_rect *rect)
{ {
v4l_bound_align_image(&rect->width, 2, W_CIF, 1, v4l_bound_align_image(&rect->width, 2, W_CIF, 1,
...@@ -510,12 +525,30 @@ static int ov6650_set_selection(struct v4l2_subdev *sd, ...@@ -510,12 +525,30 @@ static int ov6650_set_selection(struct v4l2_subdev *sd,
struct ov6650 *priv = to_ov6650(client); struct ov6650 *priv = to_ov6650(client);
int ret; int ret;
if (sel->which != V4L2_SUBDEV_FORMAT_ACTIVE || if (sel->target != V4L2_SEL_TGT_CROP)
sel->target != V4L2_SEL_TGT_CROP)
return -EINVAL; return -EINVAL;
ov6650_bind_align_crop_rectangle(&sel->r); ov6650_bind_align_crop_rectangle(&sel->r);
if (sel->which == V4L2_SUBDEV_FORMAT_TRY) {
struct v4l2_rect *crop = &sd_state->pads->try_crop;
struct v4l2_mbus_framefmt *mf = &sd_state->pads->try_fmt;
/* detect current pad config scaling factor */
bool half_scale = !is_unscaled_ok(mf->width, mf->height, crop);
/* store new crop rectangle */
*crop = sel->r;
/* adjust frame size */
mf->width = crop->width >> half_scale;
mf->height = crop->height >> half_scale;
return 0;
}
/* V4L2_SUBDEV_FORMAT_ACTIVE */
/* apply new crop rectangle */
ret = ov6650_reg_write(client, REG_HSTRT, sel->r.left >> 1); ret = ov6650_reg_write(client, REG_HSTRT, sel->r.left >> 1);
if (!ret) { if (!ret) {
priv->rect.width += priv->rect.left - sel->r.left; priv->rect.width += priv->rect.left - sel->r.left;
...@@ -567,11 +600,6 @@ static int ov6650_get_fmt(struct v4l2_subdev *sd, ...@@ -567,11 +600,6 @@ static int ov6650_get_fmt(struct v4l2_subdev *sd,
return 0; return 0;
} }
static bool is_unscaled_ok(int width, int height, struct v4l2_rect *rect)
{
return width > rect->width >> 1 || height > rect->height >> 1;
}
#define to_clkrc(div) ((div) - 1) #define to_clkrc(div) ((div) - 1)
/* set the format we will capture in */ /* set the format we will capture in */
...@@ -692,7 +720,11 @@ static int ov6650_set_fmt(struct v4l2_subdev *sd, ...@@ -692,7 +720,11 @@ static int ov6650_set_fmt(struct v4l2_subdev *sd,
break; break;
} }
*crop = priv->rect; if (format->which == V4L2_SUBDEV_FORMAT_TRY)
*crop = sd_state->pads->try_crop;
else
*crop = priv->rect;
half_scale = !is_unscaled_ok(mf->width, mf->height, crop); half_scale = !is_unscaled_ok(mf->width, mf->height, crop);
/* adjust new crop rectangle position against its current center */ /* adjust new crop rectangle position against its current center */
......
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