Hi Sakari and all, I'm sorry to up this thread from an year ago but I'm currently thinking about a problem which is related to this so I decided to ask here. On 20.07.2017 18:17, Sakari Ailus wrote: > Hi Todor, > > On Mon, Jul 17, 2017 at 01:33:42PM +0300, Todor Tomov wrote: >> Add support for horizontal and vertical frame padding. >> >> Signed-off-by: Todor Tomov <todor.tomov@xxxxxxxxxx> >> --- >> drivers/media/platform/qcom/camss-8x16/camss-vfe.c | 86 +++++++++++++++++----- >> .../media/platform/qcom/camss-8x16/camss-video.c | 69 ++++++++++++----- >> .../media/platform/qcom/camss-8x16/camss-video.h | 2 + >> 3 files changed, 121 insertions(+), 36 deletions(-) >> ... >> diff --git a/drivers/media/platform/qcom/camss-8x16/camss-video.c b/drivers/media/platform/qcom/camss-8x16/camss-video.c >> index c5ebf5c..5a2bf18 100644 >> --- a/drivers/media/platform/qcom/camss-8x16/camss-video.c >> +++ b/drivers/media/platform/qcom/camss-8x16/camss-video.c ... >> @@ -542,28 +537,68 @@ static int video_g_fmt(struct file *file, void *fh, struct v4l2_format *f) >> return 0; >> } >> >> -static int video_s_fmt(struct file *file, void *fh, struct v4l2_format *f) >> +static int video_try_fmt(struct file *file, void *fh, struct v4l2_format *f) >> { >> struct camss_video *video = video_drvdata(file); >> + struct v4l2_plane_pix_format *p; >> + u32 bytesperline[3] = { 0 }; >> + u32 sizeimage[3] = { 0 }; >> + u32 lines; >> int ret; >> + int i; >> >> - if (vb2_is_busy(&video->vb2_q)) >> - return -EBUSY; >> + if (video->line_based) >> + for (i = 0; i < f->fmt.pix_mp.num_planes && i < 3; i++) { >> + p = &f->fmt.pix_mp.plane_fmt[i]; >> + bytesperline[i] = clamp_t(u32, p->bytesperline, >> + 1, 65528); >> + sizeimage[i] = clamp_t(u32, p->sizeimage, >> + bytesperline[i], >> + bytesperline[i] * 4096); >> + } >> >> ret = video_get_subdev_format(video, f); >> if (ret < 0) >> return ret; > > If you take the width and height from the sub-device format, then for the > user to figure out how big a buffer is needed for a particular format it > takes to change the sub-device format. > > I wouldn't do this but keep the image dimensions on the video node > independent of what's configured on the sub-device. So the question is whether the video device node should: a) keep its format and framesize always in sync with what is set on the subdev node with active link to it. This means all s_fmt and enum_fmt will return only the value which is in sink with the subdev node; b) allow to set all possible allowed formats and framesizes. This however allows the userspace to try to start the streaming with format and framesizes not in sync (on video and subdev node) in which case the start streaming will fail. Currently the driver is implemented as in b) and I hit this problem (in b)) when I try to use it from opencv. I wonder how this can be overcome, the userspace cannot be blamed that it tried to start streaming for format and framesize which were allowed to be set. Are there any new insights on this lately - what can be done to avoid the problems in a) and b)? Thank you. > > This patch doesn't really change the behaviour, but a patch before this > one. That's where the fix should be (as well). > >> >> - video->active_fmt = *f; >> + if (video->line_based) >> + for (i = 0; i < f->fmt.pix_mp.num_planes; i++) { >> + p = &f->fmt.pix_mp.plane_fmt[i]; >> + p->bytesperline = clamp_t(u32, p->bytesperline, >> + 1, 65528); >> + p->sizeimage = clamp_t(u32, p->sizeimage, >> + p->bytesperline, >> + p->bytesperline * 4096); >> + lines = p->sizeimage / p->bytesperline; >> + >> + if (p->bytesperline < bytesperline[i]) >> + p->bytesperline = ALIGN(bytesperline[i], 8); >> + >> + if (p->sizeimage < p->bytesperline * lines) >> + p->sizeimage = p->bytesperline * lines; >> + >> + if (p->sizeimage < sizeimage[i]) >> + p->sizeimage = sizeimage[i]; >> + } >> >> return 0; >> } >> >> -static int video_try_fmt(struct file *file, void *fh, struct v4l2_format *f) >> +static int video_s_fmt(struct file *file, void *fh, struct v4l2_format *f) >> { >> struct camss_video *video = video_drvdata(file); >> + int ret; >> + >> + if (vb2_is_busy(&video->vb2_q)) >> + return -EBUSY; >> >> - return video_get_subdev_format(video, f); >> + ret = video_try_fmt(file, fh, f); >> + if (ret < 0) >> + return ret; >> + >> + video->active_fmt = *f; >> + >> + return 0; >> } >> >> static int video_enum_input(struct file *file, void *fh, -- Best regards, Todor Tomov