media: vb2: drop VB2_BUF_STATE_PREPARED, use bool prepared/synced instead
The PREPARED state becomes a problem with the request API: a buffer could be PREPARED but dequeued, or PREPARED and in state IN_REQUEST. PREPARED is really not a state as such, but more a property of the buffer. So make new 'prepared' and 'synced' bools instead to remember whether the buffer is prepared and/or synced or not. V4L2_BUF_FLAG_PREPARED is only set if the buffer is both synced and prepared and in the DEQUEUED state. Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com> Reviewed-by: Mauro Carvalho Chehab <mchehab+samsung@kernel.org> Signed-off-by: Mauro Carvalho Chehab <mchehab+samsung@kernel.org>
This commit is contained in:

committed by
Mauro Carvalho Chehab

父節點
559cd0a3e4
當前提交
55028695c3
@@ -682,7 +682,7 @@ int vb2_core_reqbufs(struct vb2_queue *q, enum vb2_memory memory,
|
||||
}
|
||||
|
||||
/*
|
||||
* Call queue_cancel to clean up any buffers in the PREPARED or
|
||||
* Call queue_cancel to clean up any buffers in the
|
||||
* QUEUED state which is possible if buffers were prepared or
|
||||
* queued without ever calling STREAMON.
|
||||
*/
|
||||
@@ -921,6 +921,7 @@ void vb2_buffer_done(struct vb2_buffer *vb, enum vb2_buffer_state state)
|
||||
/* sync buffers */
|
||||
for (plane = 0; plane < vb->num_planes; ++plane)
|
||||
call_void_memop(vb, finish, vb->planes[plane].mem_priv);
|
||||
vb->synced = false;
|
||||
}
|
||||
|
||||
spin_lock_irqsave(&q->done_lock, flags);
|
||||
@@ -1239,6 +1240,7 @@ static void __enqueue_in_driver(struct vb2_buffer *vb)
|
||||
static int __buf_prepare(struct vb2_buffer *vb)
|
||||
{
|
||||
struct vb2_queue *q = vb->vb2_queue;
|
||||
enum vb2_buffer_state orig_state = vb->state;
|
||||
unsigned int plane;
|
||||
int ret;
|
||||
|
||||
@@ -1247,6 +1249,10 @@ static int __buf_prepare(struct vb2_buffer *vb)
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
if (vb->prepared)
|
||||
return 0;
|
||||
WARN_ON(vb->synced);
|
||||
|
||||
vb->state = VB2_BUF_STATE_PREPARING;
|
||||
|
||||
switch (q->memory) {
|
||||
@@ -1262,11 +1268,12 @@ static int __buf_prepare(struct vb2_buffer *vb)
|
||||
default:
|
||||
WARN(1, "Invalid queue type\n");
|
||||
ret = -EINVAL;
|
||||
break;
|
||||
}
|
||||
|
||||
if (ret) {
|
||||
dprintk(1, "buffer preparation failed: %d\n", ret);
|
||||
vb->state = VB2_BUF_STATE_DEQUEUED;
|
||||
vb->state = orig_state;
|
||||
return ret;
|
||||
}
|
||||
|
||||
@@ -1274,7 +1281,9 @@ static int __buf_prepare(struct vb2_buffer *vb)
|
||||
for (plane = 0; plane < vb->num_planes; ++plane)
|
||||
call_void_memop(vb, prepare, vb->planes[plane].mem_priv);
|
||||
|
||||
vb->state = VB2_BUF_STATE_PREPARED;
|
||||
vb->synced = true;
|
||||
vb->prepared = true;
|
||||
vb->state = orig_state;
|
||||
|
||||
return 0;
|
||||
}
|
||||
@@ -1290,6 +1299,10 @@ int vb2_core_prepare_buf(struct vb2_queue *q, unsigned int index, void *pb)
|
||||
vb->state);
|
||||
return -EINVAL;
|
||||
}
|
||||
if (vb->prepared) {
|
||||
dprintk(1, "buffer already prepared\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = __buf_prepare(vb);
|
||||
if (ret)
|
||||
@@ -1381,11 +1394,11 @@ int vb2_core_qbuf(struct vb2_queue *q, unsigned int index, void *pb)
|
||||
|
||||
switch (vb->state) {
|
||||
case VB2_BUF_STATE_DEQUEUED:
|
||||
ret = __buf_prepare(vb);
|
||||
if (ret)
|
||||
return ret;
|
||||
break;
|
||||
case VB2_BUF_STATE_PREPARED:
|
||||
if (!vb->prepared) {
|
||||
ret = __buf_prepare(vb);
|
||||
if (ret)
|
||||
return ret;
|
||||
}
|
||||
break;
|
||||
case VB2_BUF_STATE_PREPARING:
|
||||
dprintk(1, "buffer still being prepared\n");
|
||||
@@ -1611,6 +1624,7 @@ int vb2_core_dqbuf(struct vb2_queue *q, unsigned int *pindex, void *pb,
|
||||
}
|
||||
|
||||
call_void_vb_qop(vb, buf_finish, vb);
|
||||
vb->prepared = false;
|
||||
|
||||
if (pindex)
|
||||
*pindex = vb->index;
|
||||
@@ -1699,18 +1713,18 @@ static void __vb2_queue_cancel(struct vb2_queue *q)
|
||||
for (i = 0; i < q->num_buffers; ++i) {
|
||||
struct vb2_buffer *vb = q->bufs[i];
|
||||
|
||||
if (vb->state == VB2_BUF_STATE_PREPARED ||
|
||||
vb->state == VB2_BUF_STATE_QUEUED) {
|
||||
if (vb->synced) {
|
||||
unsigned int plane;
|
||||
|
||||
for (plane = 0; plane < vb->num_planes; ++plane)
|
||||
call_void_memop(vb, finish,
|
||||
vb->planes[plane].mem_priv);
|
||||
vb->synced = false;
|
||||
}
|
||||
|
||||
if (vb->state != VB2_BUF_STATE_DEQUEUED) {
|
||||
vb->state = VB2_BUF_STATE_PREPARED;
|
||||
if (vb->prepared) {
|
||||
call_void_vb_qop(vb, buf_finish, vb);
|
||||
vb->prepared = false;
|
||||
}
|
||||
__vb2_dqbuf(vb);
|
||||
}
|
||||
|
@@ -352,9 +352,13 @@ static int vb2_queue_or_prepare_buf(struct vb2_queue *q, struct v4l2_buffer *b,
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
/* Copy relevant information provided by the userspace */
|
||||
memset(vbuf->planes, 0, sizeof(vbuf->planes[0]) * vb->num_planes);
|
||||
return vb2_fill_vb2_v4l2_buffer(vb, b);
|
||||
if (!vb->prepared) {
|
||||
/* Copy relevant information provided by the userspace */
|
||||
memset(vbuf->planes, 0,
|
||||
sizeof(vbuf->planes[0]) * vb->num_planes);
|
||||
ret = vb2_fill_vb2_v4l2_buffer(vb, b);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
@@ -443,9 +447,6 @@ static void __fill_v4l2_buffer(struct vb2_buffer *vb, void *pb)
|
||||
case VB2_BUF_STATE_DONE:
|
||||
b->flags |= V4L2_BUF_FLAG_DONE;
|
||||
break;
|
||||
case VB2_BUF_STATE_PREPARED:
|
||||
b->flags |= V4L2_BUF_FLAG_PREPARED;
|
||||
break;
|
||||
case VB2_BUF_STATE_PREPARING:
|
||||
case VB2_BUF_STATE_DEQUEUED:
|
||||
case VB2_BUF_STATE_REQUEUEING:
|
||||
@@ -453,6 +454,9 @@ static void __fill_v4l2_buffer(struct vb2_buffer *vb, void *pb)
|
||||
break;
|
||||
}
|
||||
|
||||
if (vb->state == VB2_BUF_STATE_DEQUEUED && vb->synced && vb->prepared)
|
||||
b->flags |= V4L2_BUF_FLAG_PREPARED;
|
||||
|
||||
if (vb2_buffer_in_use(q, vb))
|
||||
b->flags |= V4L2_BUF_FLAG_MAPPED;
|
||||
|
||||
|
Reference in New Issue
Block a user