[libcamera-devel] [PATCH] imx219: selection compliance fixes
Hans Verkuil
hverkuil at xs4all.nl
Thu Jul 16 11:48:19 CEST 2020
On 15/07/2020 09:19, Jacopo Mondi wrote:
> Hi Laurent,
>
> On Wed, Jul 15, 2020 at 02:49:38AM +0300, Laurent Pinchart wrote:
>> Hi Jacopo,
>>
>> On Tue, Jul 14, 2020 at 02:31:46PM +0200, Jacopo Mondi wrote:
>>> Hi Hans,
>>>
>>> On Thu, Jul 02, 2020 at 03:50:04PM +0200, Hans Verkuil wrote:
>>>> The top/left crop coordinates were 0, 0 instead of 8, 8 in the
>>>> supported_modes array. This was a mismatch with the default values,
>>>> so this is corrected. Found with v4l2-compliance.
>>>>
>>>> Also add V4L2_SEL_TGT_CROP_BOUNDS support: CROP_DEFAULT and CROP_BOUNDS
>>>> always go together. Found with v4l2-compliance.
>>>
>>> I actually introduced this with
>>> e6d4ef7d58aa ("media: i2c: imx219: Implement get_selection")
>>>
>>>>
>>>> Signed-off-by: Hans Verkuil <hverkuil-cisco at xs4all.nl>
>>>> ---
>>>> drivers/media/i2c/imx219.c | 17 +++++++++--------
>>>> 1 file changed, 9 insertions(+), 8 deletions(-)
>>>>
>>>> diff --git a/drivers/media/i2c/imx219.c b/drivers/media/i2c/imx219.c
>>>> index 0a546b8e466c..935e2a258ce5 100644
>>>> --- a/drivers/media/i2c/imx219.c
>>>> +++ b/drivers/media/i2c/imx219.c
>>>> @@ -473,8 +473,8 @@ static const struct imx219_mode supported_modes[] = {
>>>> .width = 3280,
>>>> .height = 2464,
>>>> .crop = {
>>>> - .left = 0,
>>>> - .top = 0,
>>>> + .left = 8,
>>>> + .top = 8,
>>>
>>> Mmmm, why this change ?
>>> This values are used to report V4L2_SEL_TGT_CROP rectangle, which
>>> according to the documentation is defined as
>>> "Crop rectangle. Defines the cropped area."
>>> (not a much extensive description :)
Unless changed by calling S_SELECTION(TGT_CROP) the initial crop is equal
to TGT_CROP_DEFAULT, and but TGT_CROP and TGT_CROP_DEFAULT shall be inside
TGT_CROP_BOUNDS. CROP_BOUNDS may be larger than CROP_DEFAULT and describes
the whole area from which you can crop. I.e. in the case of sensors you can
set the crop rectangle to include optical blanks active pixels.
In this driver the initial TGT_CROP rectangle as specified in supported_modes
(aligned with the top-left pixel) was outside CROP_BOUNDS (centered) and also
a mismatch with CROP_DEFAULT (also centered).
>>>
>>> Clearly this is a faulty definition, and I know from experience how
>>> hard is proving to define pixel array properties and in which extent
>>> the documentation has to go:
>>> https://lists.libcamera.org/pipermail/libcamera-devel/2020-June/009115.html
>>>
>>> My understanding is that target should report the current crop
>>> rectangle, defined from the rectangle retrieved with the
>>> V4L2_SEL_TGT_CROP_DEFAULT target, which, according documentation
>>> reports the:
>>> "Suggested cropping rectangle that covers the “whole picture”.
>>> This includes only active pixels and excludes other non-active pixels such
>>> as black pixels"
>>>
>>> The TGT_CROP_DEFAULT then reports the active pixel array portion, and
>>> needs to be defined in respect to the TGT_NATIVE_SIZE, which reports
>>> the dimensions of the whole pixel matrix, including non-active pixels,
>>> optical blanks active and non-active pixels.
The relationship between NATIVE_SIZE and CROP_BOUNDS is not properly defined,
but I would expect that CROP_BOUNDS is inside the NATIVE_SIZE target rectangle.
If NATIVE_SIZE is larger than BOUNDS, then I would expect that the additional
margins are pixels that are invalid or otherwise useless.
The hard rule though is that you can crop anywhere within the CROP_BOUNDS area.
Historically CROP_BOUNDS originated with analog SDTV video capture where it was
possible to capture more data than just the typical 720x576/480 PAL/NTSC active
video area. Analog video was often overscanned, i.e. there was more video data
outside the 'active' video area. That was how CRTs worked. So you could move the
crop window around within the CROP_BOUNDS area, or just capture the full CROP_BOUNDS
are. Although this was often poorly tested/implemented. The bttv driver is one of
the few that could do this.
This is actually simplified since you could do weird things with the horizontal
sample rate as well, effectively changing the pixel aspect ratio, making things
really complicated. It's analog video so while the video lines were discrete,
horizontally you are just sampling a waveform, so you could sample at different
rates if you wanted to. I doubt anyone ever used it since doing that would give
you a huge headache :-)
With digital video interfaces (HDMI, DVI, SDI, DP, etc.) that no longer applies and
for those receivers the initial CROP/CROP_DEFAULT/CROP_BOUNDS rectangles are all
the same, e.g. 1920x1080 for 1080p HDMI video.
>>>
>>> The TGT_CROP rectangle is hence defined from the CROP_DEFAULT one, and
>>> if the 'whole active area' is selected, its top-left corner is placed
>>> in position (0, 0) (what's the point of defining it in respect to an
>>> area which cannot be read anyway ?)
>>>
>>> Unless TGT_CROP should be defined in respect to the NATIVE_SIZE
>>> rectangle too, but that's not specified anywhere.
>>>
>>> Anyway, those selection targets badly apply to image sensors, are
>>> ill-defined as the definition of active pixels, optical blank (active
>>> and non-active) pixels is not provided anywhere, and it's not specified
>>> anywhere what is the reference area for each of those rectangles, so I
>>> might very well got them wrongly.
>>
>> My understanding is that both TGT_CROP_DEFAULT and TGT_CROP_BOUNDS are
>> relative to TGT_NATIVE_SIZE. BOUNDS defines all the pixels that can be
>
> And what is TGT_CROP reference in your understanding ?
That's the rectangle you are actually cropping. Initially CROP == CROP_DEFAULT
and CROP shall always be inside CROP_BOUNDS. And CROP_BOUNDS shall be equal
or larger than CROP_DEFAULT.
>
>> captured, including optical black and invalid pixels, while DEFAULT
>> defines the active area, excluding optical black and invalida pixels. To
>> put it another way, DEFAULT is what the kernel recommends applications
>> to use if they have no specific requirement and/or no specific knowledge
>> about the sensor.
>>
>> I fully agree this is very under-documented, which also means that my
>> understanding may be wrong :-)
>
> With some consensus on this interpretation I would be happy to update
> the documentation. I already considered that, but the selection API
> does not apply to image sensors only, and giving a description which
> is about the pixel array properties might be not totally opportune as
> it would rule out other devices like bridges or muxers.
And m2m devices like codecs.
Regards,
Hans
>
>>
>>>> .width = 3280,
>>>> .height = 2464
>>>> },
>>>> @@ -489,8 +489,8 @@ static const struct imx219_mode supported_modes[] = {
>>>> .width = 1920,
>>>> .height = 1080,
>>>> .crop = {
>>>> - .left = 680,
>>>> - .top = 692,
>>>> + .left = 8 + 680,
>>>> + .top = 8 + 692,
>>>> .width = 1920,
>>>> .height = 1080
>>>> },
>>>> @@ -505,8 +505,8 @@ static const struct imx219_mode supported_modes[] = {
>>>> .width = 1640,
>>>> .height = 1232,
>>>> .crop = {
>>>> - .left = 0,
>>>> - .top = 0,
>>>> + .left = 8,
>>>> + .top = 8,
>>>> .width = 3280,
>>>> .height = 2464
>>>> },
>>>> @@ -521,8 +521,8 @@ static const struct imx219_mode supported_modes[] = {
>>>> .width = 640,
>>>> .height = 480,
>>>> .crop = {
>>>> - .left = 1000,
>>>> - .top = 752,
>>>> + .left = 8 + 1000,
>>>> + .top = 8 + 752,
>>>> .width = 1280,
>>>> .height = 960
>>>> },
>>>> @@ -1014,6 +1014,7 @@ static int imx219_get_selection(struct v4l2_subdev *sd,
>>>> return 0;
>>>>
>>>> case V4L2_SEL_TGT_CROP_DEFAULT:
>>>> + case V4L2_SEL_TGT_CROP_BOUNDS:
>>>
>>> Still not getting what is the purpose of two targets if the "always
>>> have to go together" :)
>>>
>>>> sel->r.top = IMX219_PIXEL_ARRAY_TOP;
>>>> sel->r.left = IMX219_PIXEL_ARRAY_LEFT;
>>>> sel->r.width = IMX219_PIXEL_ARRAY_WIDTH;
>>
>> --
>> Regards,
>>
>> Laurent Pinchart
More information about the libcamera-devel
mailing list