BACKPORT: media: v4l2-mem2mem: Apply DST_QUEUE_OFF_BASE on MMAP buffers across ioctls
[ Upstream commit 8310ca94075e784bbb06593cd6c068ee6b6e4ca6 ] DST_QUEUE_OFF_BASE is applied to offset/mem_offset on MMAP capture buffers only for the VIDIOC_QUERYBUF ioctl, while the userspace fields (including offset/mem_offset) are filled in for VIDIOC_{QUERY,PREPARE,Q,DQ}BUF ioctls. This leads to differences in the values presented to userspace. If userspace attempts to mmap the capture buffer directly using values from DQBUF, it will fail. Move the code that applies the magic offset into a helper, and call that helper from all four ioctl entry points. [hverkuil: drop unnecessary '= 0' in v4l2_m2m_querybuf() for ret] Bug: 223375145 Fixes:7f98639def
("V4L/DVB: add memory-to-memory device helper framework for videobuf") Fixes:908a0d7c58
("[media] v4l: mem2mem: port to videobuf2") Signed-off-by: Chen-Yu Tsai <wenst@chromium.org> Signed-off-by: Hans Verkuil <hverkuil-cisco@xs4all.nl> Signed-off-by: Mauro Carvalho Chehab <mchehab@kernel.org> Signed-off-by: Sasha Levin <sashal@kernel.org> Signed-off-by: Lee Jones <lee.jones@linaro.org> Change-Id: Ifab9933f85f4ba7e0fadddf52debaf837830a06d
This commit is contained in:
@@ -585,19 +585,14 @@ int v4l2_m2m_reqbufs(struct file *file, struct v4l2_m2m_ctx *m2m_ctx,
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(v4l2_m2m_reqbufs);
|
EXPORT_SYMBOL_GPL(v4l2_m2m_reqbufs);
|
||||||
|
|
||||||
int v4l2_m2m_querybuf(struct file *file, struct v4l2_m2m_ctx *m2m_ctx,
|
static void v4l2_m2m_adjust_mem_offset(struct vb2_queue *vq,
|
||||||
struct v4l2_buffer *buf)
|
struct v4l2_buffer *buf)
|
||||||
{
|
{
|
||||||
struct vb2_queue *vq;
|
|
||||||
int ret = 0;
|
|
||||||
unsigned int i;
|
|
||||||
|
|
||||||
vq = v4l2_m2m_get_vq(m2m_ctx, buf->type);
|
|
||||||
ret = vb2_querybuf(vq, buf);
|
|
||||||
|
|
||||||
/* Adjust MMAP memory offsets for the CAPTURE queue */
|
/* Adjust MMAP memory offsets for the CAPTURE queue */
|
||||||
if (buf->memory == V4L2_MEMORY_MMAP && V4L2_TYPE_IS_CAPTURE(vq->type)) {
|
if (buf->memory == V4L2_MEMORY_MMAP && V4L2_TYPE_IS_CAPTURE(vq->type)) {
|
||||||
if (V4L2_TYPE_IS_MULTIPLANAR(vq->type)) {
|
if (V4L2_TYPE_IS_MULTIPLANAR(vq->type)) {
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
for (i = 0; i < buf->length; ++i)
|
for (i = 0; i < buf->length; ++i)
|
||||||
buf->m.planes[i].m.mem_offset
|
buf->m.planes[i].m.mem_offset
|
||||||
+= DST_QUEUE_OFF_BASE;
|
+= DST_QUEUE_OFF_BASE;
|
||||||
@@ -605,8 +600,23 @@ int v4l2_m2m_querybuf(struct file *file, struct v4l2_m2m_ctx *m2m_ctx,
|
|||||||
buf->m.offset += DST_QUEUE_OFF_BASE;
|
buf->m.offset += DST_QUEUE_OFF_BASE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return ret;
|
int v4l2_m2m_querybuf(struct file *file, struct v4l2_m2m_ctx *m2m_ctx,
|
||||||
|
struct v4l2_buffer *buf)
|
||||||
|
{
|
||||||
|
struct vb2_queue *vq;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
vq = v4l2_m2m_get_vq(m2m_ctx, buf->type);
|
||||||
|
ret = vb2_querybuf(vq, buf);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
/* Adjust MMAP memory offsets for the CAPTURE queue */
|
||||||
|
v4l2_m2m_adjust_mem_offset(vq, buf);
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(v4l2_m2m_querybuf);
|
EXPORT_SYMBOL_GPL(v4l2_m2m_querybuf);
|
||||||
|
|
||||||
@@ -763,6 +773,9 @@ int v4l2_m2m_qbuf(struct file *file, struct v4l2_m2m_ctx *m2m_ctx,
|
|||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
/* Adjust MMAP memory offsets for the CAPTURE queue */
|
||||||
|
v4l2_m2m_adjust_mem_offset(vq, buf);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the capture queue is streaming, but streaming hasn't started
|
* If the capture queue is streaming, but streaming hasn't started
|
||||||
* on the device, but was asked to stop, mark the previously queued
|
* on the device, but was asked to stop, mark the previously queued
|
||||||
@@ -784,9 +797,17 @@ int v4l2_m2m_dqbuf(struct file *file, struct v4l2_m2m_ctx *m2m_ctx,
|
|||||||
struct v4l2_buffer *buf)
|
struct v4l2_buffer *buf)
|
||||||
{
|
{
|
||||||
struct vb2_queue *vq;
|
struct vb2_queue *vq;
|
||||||
|
int ret;
|
||||||
|
|
||||||
vq = v4l2_m2m_get_vq(m2m_ctx, buf->type);
|
vq = v4l2_m2m_get_vq(m2m_ctx, buf->type);
|
||||||
return vb2_dqbuf(vq, buf, file->f_flags & O_NONBLOCK);
|
ret = vb2_dqbuf(vq, buf, file->f_flags & O_NONBLOCK);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
/* Adjust MMAP memory offsets for the CAPTURE queue */
|
||||||
|
v4l2_m2m_adjust_mem_offset(vq, buf);
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(v4l2_m2m_dqbuf);
|
EXPORT_SYMBOL_GPL(v4l2_m2m_dqbuf);
|
||||||
|
|
||||||
@@ -795,9 +816,17 @@ int v4l2_m2m_prepare_buf(struct file *file, struct v4l2_m2m_ctx *m2m_ctx,
|
|||||||
{
|
{
|
||||||
struct video_device *vdev = video_devdata(file);
|
struct video_device *vdev = video_devdata(file);
|
||||||
struct vb2_queue *vq;
|
struct vb2_queue *vq;
|
||||||
|
int ret;
|
||||||
|
|
||||||
vq = v4l2_m2m_get_vq(m2m_ctx, buf->type);
|
vq = v4l2_m2m_get_vq(m2m_ctx, buf->type);
|
||||||
return vb2_prepare_buf(vq, vdev->v4l2_dev->mdev, buf);
|
ret = vb2_prepare_buf(vq, vdev->v4l2_dev->mdev, buf);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
/* Adjust MMAP memory offsets for the CAPTURE queue */
|
||||||
|
v4l2_m2m_adjust_mem_offset(vq, buf);
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(v4l2_m2m_prepare_buf);
|
EXPORT_SYMBOL_GPL(v4l2_m2m_prepare_buf);
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user