media: vivid: fix vid_out_buf_prepare()
The wrong size check was performed for output formats like NV24 which set vfmt->buffers to 1, but vfmt->planes is 2. It was incorrectly checking the payload size for plane 1, which doesn't exist. Note: vfmt->buffers refers to the number of per-plane-buffers that should be allocated. vfmt->planes refers to the number of planes that make up an image. vfmt->planes may be > vfmt->buffers. Signed-off-by: Hans Verkuil <hverkuil-cisco@xs4all.nl> Signed-off-by: Mauro Carvalho Chehab <mchehab+samsung@kernel.org>
This commit is contained in:
parent
1284ed59a1
commit
560c053deb
@ -102,17 +102,20 @@ static int vid_out_buf_out_validate(struct vb2_buffer *vb)
|
|||||||
static int vid_out_buf_prepare(struct vb2_buffer *vb)
|
static int vid_out_buf_prepare(struct vb2_buffer *vb)
|
||||||
{
|
{
|
||||||
struct vivid_dev *dev = vb2_get_drv_priv(vb->vb2_queue);
|
struct vivid_dev *dev = vb2_get_drv_priv(vb->vb2_queue);
|
||||||
unsigned long size;
|
const struct vivid_fmt *vfmt = dev->fmt_out;
|
||||||
unsigned planes;
|
unsigned int planes = vfmt->buffers;
|
||||||
|
unsigned int h = dev->fmt_out_rect.height;
|
||||||
|
unsigned int size = dev->bytesperline_out[0] * h;
|
||||||
unsigned p;
|
unsigned p;
|
||||||
|
|
||||||
|
for (p = vfmt->buffers; p < vfmt->planes; p++)
|
||||||
|
size += dev->bytesperline_out[p] * h / vfmt->vdownsampling[p];
|
||||||
|
|
||||||
dprintk(dev, 1, "%s\n", __func__);
|
dprintk(dev, 1, "%s\n", __func__);
|
||||||
|
|
||||||
if (WARN_ON(NULL == dev->fmt_out))
|
if (WARN_ON(NULL == dev->fmt_out))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
planes = dev->fmt_out->planes;
|
|
||||||
|
|
||||||
if (dev->buf_prepare_error) {
|
if (dev->buf_prepare_error) {
|
||||||
/*
|
/*
|
||||||
* Error injection: test what happens if buf_prepare() returns
|
* Error injection: test what happens if buf_prepare() returns
|
||||||
@ -123,11 +126,12 @@ static int vid_out_buf_prepare(struct vb2_buffer *vb)
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (p = 0; p < planes; p++) {
|
for (p = 0; p < planes; p++) {
|
||||||
size = dev->bytesperline_out[p] * dev->fmt_out_rect.height +
|
if (p)
|
||||||
vb->planes[p].data_offset;
|
size = dev->bytesperline_out[p] * h;
|
||||||
|
size += vb->planes[p].data_offset;
|
||||||
|
|
||||||
if (vb2_get_plane_payload(vb, p) < size) {
|
if (vb2_get_plane_payload(vb, p) < size) {
|
||||||
dprintk(dev, 1, "%s the payload is too small for plane %u (%lu < %lu)\n",
|
dprintk(dev, 1, "%s the payload is too small for plane %u (%lu < %u)\n",
|
||||||
__func__, p, vb2_get_plane_payload(vb, p), size);
|
__func__, p, vb2_get_plane_payload(vb, p), size);
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user