Commit aec96832 authored by Hans Verkuil's avatar Hans Verkuil Committed by Mauro Carvalho Chehab

[media] vpif_capture: protect dma_queue by a spin_lock

The dma_queue list is accessed by both the interrupt handler and by normal
code. It needs to be protected by a lock to prevent possible list corruption.
Signed-off-by: default avatarHans Verkuil <hans.verkuil@cisco.com>
Acked-by: default avatarLad, Prabhakar <prabhakar.lad@ti.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@redhat.com>
parent 94b76a88
...@@ -201,13 +201,16 @@ static void vpif_buffer_queue(struct vb2_buffer *vb) ...@@ -201,13 +201,16 @@ static void vpif_buffer_queue(struct vb2_buffer *vb)
struct vpif_cap_buffer *buf = container_of(vb, struct vpif_cap_buffer *buf = container_of(vb,
struct vpif_cap_buffer, vb); struct vpif_cap_buffer, vb);
struct common_obj *common; struct common_obj *common;
unsigned long flags;
common = &ch->common[VPIF_VIDEO_INDEX]; common = &ch->common[VPIF_VIDEO_INDEX];
vpif_dbg(2, debug, "vpif_buffer_queue\n"); vpif_dbg(2, debug, "vpif_buffer_queue\n");
spin_lock_irqsave(&common->irqlock, flags);
/* add the buffer to the DMA queue */ /* add the buffer to the DMA queue */
list_add_tail(&buf->list, &common->dma_queue); list_add_tail(&buf->list, &common->dma_queue);
spin_unlock_irqrestore(&common->irqlock, flags);
} }
/** /**
...@@ -278,10 +281,13 @@ static int vpif_start_streaming(struct vb2_queue *vq, unsigned int count) ...@@ -278,10 +281,13 @@ static int vpif_start_streaming(struct vb2_queue *vq, unsigned int count)
struct common_obj *common = &ch->common[VPIF_VIDEO_INDEX]; struct common_obj *common = &ch->common[VPIF_VIDEO_INDEX];
struct vpif_params *vpif = &ch->vpifparams; struct vpif_params *vpif = &ch->vpifparams;
unsigned long addr = 0; unsigned long addr = 0;
unsigned long flags;
int ret; int ret;
/* If buffer queue is empty, return error */ /* If buffer queue is empty, return error */
spin_lock_irqsave(&common->irqlock, flags);
if (list_empty(&common->dma_queue)) { if (list_empty(&common->dma_queue)) {
spin_unlock_irqrestore(&common->irqlock, flags);
vpif_dbg(1, debug, "buffer queue is empty\n"); vpif_dbg(1, debug, "buffer queue is empty\n");
return -EIO; return -EIO;
} }
...@@ -291,6 +297,7 @@ static int vpif_start_streaming(struct vb2_queue *vq, unsigned int count) ...@@ -291,6 +297,7 @@ static int vpif_start_streaming(struct vb2_queue *vq, unsigned int count)
struct vpif_cap_buffer, list); struct vpif_cap_buffer, list);
/* Remove buffer from the buffer queue */ /* Remove buffer from the buffer queue */
list_del(&common->cur_frm->list); list_del(&common->cur_frm->list);
spin_unlock_irqrestore(&common->irqlock, flags);
/* Mark state of the current frame to active */ /* Mark state of the current frame to active */
common->cur_frm->vb.state = VB2_BUF_STATE_ACTIVE; common->cur_frm->vb.state = VB2_BUF_STATE_ACTIVE;
/* Initialize field_id and started member */ /* Initialize field_id and started member */
...@@ -362,6 +369,7 @@ static int vpif_stop_streaming(struct vb2_queue *vq) ...@@ -362,6 +369,7 @@ static int vpif_stop_streaming(struct vb2_queue *vq)
struct vpif_fh *fh = vb2_get_drv_priv(vq); struct vpif_fh *fh = vb2_get_drv_priv(vq);
struct channel_obj *ch = fh->channel; struct channel_obj *ch = fh->channel;
struct common_obj *common; struct common_obj *common;
unsigned long flags;
if (!vb2_is_streaming(vq)) if (!vb2_is_streaming(vq))
return 0; return 0;
...@@ -369,12 +377,14 @@ static int vpif_stop_streaming(struct vb2_queue *vq) ...@@ -369,12 +377,14 @@ static int vpif_stop_streaming(struct vb2_queue *vq)
common = &ch->common[VPIF_VIDEO_INDEX]; common = &ch->common[VPIF_VIDEO_INDEX];
/* release all active buffers */ /* release all active buffers */
spin_lock_irqsave(&common->irqlock, flags);
while (!list_empty(&common->dma_queue)) { while (!list_empty(&common->dma_queue)) {
common->next_frm = list_entry(common->dma_queue.next, common->next_frm = list_entry(common->dma_queue.next,
struct vpif_cap_buffer, list); struct vpif_cap_buffer, list);
list_del(&common->next_frm->list); list_del(&common->next_frm->list);
vb2_buffer_done(&common->next_frm->vb, VB2_BUF_STATE_ERROR); vb2_buffer_done(&common->next_frm->vb, VB2_BUF_STATE_ERROR);
} }
spin_unlock_irqrestore(&common->irqlock, flags);
return 0; return 0;
} }
...@@ -420,10 +430,12 @@ static void vpif_schedule_next_buffer(struct common_obj *common) ...@@ -420,10 +430,12 @@ static void vpif_schedule_next_buffer(struct common_obj *common)
{ {
unsigned long addr = 0; unsigned long addr = 0;
spin_lock(&common->irqlock);
common->next_frm = list_entry(common->dma_queue.next, common->next_frm = list_entry(common->dma_queue.next,
struct vpif_cap_buffer, list); struct vpif_cap_buffer, list);
/* Remove that buffer from the buffer queue */ /* Remove that buffer from the buffer queue */
list_del(&common->next_frm->list); list_del(&common->next_frm->list);
spin_unlock(&common->irqlock);
common->next_frm->vb.state = VB2_BUF_STATE_ACTIVE; common->next_frm->vb.state = VB2_BUF_STATE_ACTIVE;
addr = vb2_dma_contig_plane_dma_addr(&common->next_frm->vb, 0); addr = vb2_dma_contig_plane_dma_addr(&common->next_frm->vb, 0);
...@@ -468,8 +480,12 @@ static irqreturn_t vpif_channel_isr(int irq, void *dev_id) ...@@ -468,8 +480,12 @@ static irqreturn_t vpif_channel_isr(int irq, void *dev_id)
/* Check the field format */ /* Check the field format */
if (1 == ch->vpifparams.std_info.frm_fmt) { if (1 == ch->vpifparams.std_info.frm_fmt) {
/* Progressive mode */ /* Progressive mode */
if (list_empty(&common->dma_queue)) spin_lock(&common->irqlock);
if (list_empty(&common->dma_queue)) {
spin_unlock(&common->irqlock);
continue; continue;
}
spin_unlock(&common->irqlock);
if (!channel_first_int[i][channel_id]) if (!channel_first_int[i][channel_id])
vpif_process_buffer_complete(common); vpif_process_buffer_complete(common);
...@@ -513,9 +529,13 @@ static irqreturn_t vpif_channel_isr(int irq, void *dev_id) ...@@ -513,9 +529,13 @@ static irqreturn_t vpif_channel_isr(int irq, void *dev_id)
vpif_process_buffer_complete(common); vpif_process_buffer_complete(common);
} else if (1 == fid) { } else if (1 == fid) {
/* odd field */ /* odd field */
spin_lock(&common->irqlock);
if (list_empty(&common->dma_queue) || if (list_empty(&common->dma_queue) ||
(common->cur_frm != common->next_frm)) (common->cur_frm != common->next_frm)) {
spin_unlock(&common->irqlock);
continue; continue;
}
spin_unlock(&common->irqlock);
vpif_schedule_next_buffer(common); vpif_schedule_next_buffer(common);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment