提交 0cf4daee 编写于 作者: B Brandon Philips 提交者: Mauro Carvalho Chehab

V4L/DVB (7562): videobuf: Require spinlocks for all videobuf users

A spinlock is necessary for queue_cancel to work with every driver in the tree.
Otherwise a race exists between IRQ handlers removing buffers from the queue
and queue_cancel invalidating the queue.
Signed-off-by: NBrandon Philips <bphilips@suse.de>
Signed-off-by: NMauro Carvalho Chehab <mchehab@infradead.org>
上级 aa9479ed
...@@ -123,6 +123,9 @@ void videobuf_queue_core_init(struct videobuf_queue *q, ...@@ -123,6 +123,9 @@ void videobuf_queue_core_init(struct videobuf_queue *q,
BUG_ON(!q->ops->buf_queue); BUG_ON(!q->ops->buf_queue);
BUG_ON(!q->ops->buf_release); BUG_ON(!q->ops->buf_release);
/* Lock is mandatory for queue_cancel to work */
BUG_ON(!irqlock);
/* Having implementations for abstract methods are mandatory */ /* Having implementations for abstract methods are mandatory */
BUG_ON(!q->int_ops); BUG_ON(!q->int_ops);
...@@ -180,7 +183,6 @@ void videobuf_queue_cancel(struct videobuf_queue *q) ...@@ -180,7 +183,6 @@ void videobuf_queue_cancel(struct videobuf_queue *q)
wake_up_interruptible_sync(&q->wait); wake_up_interruptible_sync(&q->wait);
/* remove queued buffers from list */ /* remove queued buffers from list */
if (q->irqlock)
spin_lock_irqsave(q->irqlock, flags); spin_lock_irqsave(q->irqlock, flags);
for (i = 0; i < VIDEO_MAX_FRAME; i++) { for (i = 0; i < VIDEO_MAX_FRAME; i++) {
if (NULL == q->bufs[i]) if (NULL == q->bufs[i])
...@@ -191,7 +193,6 @@ void videobuf_queue_cancel(struct videobuf_queue *q) ...@@ -191,7 +193,6 @@ void videobuf_queue_cancel(struct videobuf_queue *q)
wake_up_all(&q->bufs[i]->done); wake_up_all(&q->bufs[i]->done);
} }
} }
if (q->irqlock)
spin_unlock_irqrestore(q->irqlock, flags); spin_unlock_irqrestore(q->irqlock, flags);
/* free all buffers + clear queue */ /* free all buffers + clear queue */
...@@ -548,10 +549,8 @@ int videobuf_qbuf(struct videobuf_queue *q, ...@@ -548,10 +549,8 @@ int videobuf_qbuf(struct videobuf_queue *q,
list_add_tail(&buf->stream, &q->stream); list_add_tail(&buf->stream, &q->stream);
if (q->streaming) { if (q->streaming) {
if (q->irqlock)
spin_lock_irqsave(q->irqlock, flags); spin_lock_irqsave(q->irqlock, flags);
q->ops->buf_queue(q, buf); q->ops->buf_queue(q, buf);
if (q->irqlock)
spin_unlock_irqrestore(q->irqlock, flags); spin_unlock_irqrestore(q->irqlock, flags);
} }
dprintk(1, "qbuf: succeded\n"); dprintk(1, "qbuf: succeded\n");
...@@ -689,12 +688,10 @@ int videobuf_streamon(struct videobuf_queue *q) ...@@ -689,12 +688,10 @@ int videobuf_streamon(struct videobuf_queue *q)
if (q->streaming) if (q->streaming)
goto done; goto done;
q->streaming = 1; q->streaming = 1;
if (q->irqlock)
spin_lock_irqsave(q->irqlock, flags); spin_lock_irqsave(q->irqlock, flags);
list_for_each_entry(buf, &q->stream, stream) list_for_each_entry(buf, &q->stream, stream)
if (buf->state == VIDEOBUF_PREPARED) if (buf->state == VIDEOBUF_PREPARED)
q->ops->buf_queue(q, buf); q->ops->buf_queue(q, buf);
if (q->irqlock)
spin_unlock_irqrestore(q->irqlock, flags); spin_unlock_irqrestore(q->irqlock, flags);
wake_up_interruptible_sync(&q->wait); wake_up_interruptible_sync(&q->wait);
...@@ -751,10 +748,8 @@ static ssize_t videobuf_read_zerocopy(struct videobuf_queue *q, ...@@ -751,10 +748,8 @@ static ssize_t videobuf_read_zerocopy(struct videobuf_queue *q,
goto done; goto done;
/* start capture & wait */ /* start capture & wait */
if (q->irqlock)
spin_lock_irqsave(q->irqlock, flags); spin_lock_irqsave(q->irqlock, flags);
q->ops->buf_queue(q, q->read_buf); q->ops->buf_queue(q, q->read_buf);
if (q->irqlock)
spin_unlock_irqrestore(q->irqlock, flags); spin_unlock_irqrestore(q->irqlock, flags);
retval = videobuf_waiton(q->read_buf, 0, 0); retval = videobuf_waiton(q->read_buf, 0, 0);
if (0 == retval) { if (0 == retval) {
...@@ -816,12 +811,11 @@ ssize_t videobuf_read_one(struct videobuf_queue *q, ...@@ -816,12 +811,11 @@ ssize_t videobuf_read_one(struct videobuf_queue *q,
q->read_buf = NULL; q->read_buf = NULL;
goto done; goto done;
} }
if (q->irqlock)
spin_lock_irqsave(q->irqlock, flags);
spin_lock_irqsave(q->irqlock, flags);
q->ops->buf_queue(q, q->read_buf); q->ops->buf_queue(q, q->read_buf);
if (q->irqlock)
spin_unlock_irqrestore(q->irqlock, flags); spin_unlock_irqrestore(q->irqlock, flags);
q->read_off = 0; q->read_off = 0;
} }
...@@ -887,11 +881,9 @@ static int __videobuf_read_start(struct videobuf_queue *q) ...@@ -887,11 +881,9 @@ static int __videobuf_read_start(struct videobuf_queue *q)
return err; return err;
list_add_tail(&q->bufs[i]->stream, &q->stream); list_add_tail(&q->bufs[i]->stream, &q->stream);
} }
if (q->irqlock)
spin_lock_irqsave(q->irqlock, flags); spin_lock_irqsave(q->irqlock, flags);
for (i = 0; i < count; i++) for (i = 0; i < count; i++)
q->ops->buf_queue(q, q->bufs[i]); q->ops->buf_queue(q, q->bufs[i]);
if (q->irqlock)
spin_unlock_irqrestore(q->irqlock, flags); spin_unlock_irqrestore(q->irqlock, flags);
q->reading = 1; q->reading = 1;
return 0; return 0;
...@@ -1004,10 +996,8 @@ ssize_t videobuf_read_stream(struct videobuf_queue *q, ...@@ -1004,10 +996,8 @@ ssize_t videobuf_read_stream(struct videobuf_queue *q,
if (q->read_off == q->read_buf->size) { if (q->read_off == q->read_buf->size) {
list_add_tail(&q->read_buf->stream, list_add_tail(&q->read_buf->stream,
&q->stream); &q->stream);
if (q->irqlock)
spin_lock_irqsave(q->irqlock, flags); spin_lock_irqsave(q->irqlock, flags);
q->ops->buf_queue(q, q->read_buf); q->ops->buf_queue(q, q->read_buf);
if (q->irqlock)
spin_unlock_irqrestore(q->irqlock, flags); spin_unlock_irqrestore(q->irqlock, flags);
q->read_buf = NULL; q->read_buf = NULL;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册