]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
media: chips-media: wave5: add missing spinlock protection for send_eos_event()
authorZiyi Guo <n7l8m4@u.northwestern.edu>
Sat, 31 Jan 2026 22:03:23 +0000 (22:03 +0000)
committerHans Verkuil <hverkuil+cisco@kernel.org>
Mon, 23 Mar 2026 10:18:33 +0000 (11:18 +0100)
Add spin_lock_irqsave()/spin_unlock_irqrestore() around send_eos_event()
calls in the VB2 buffer queue and streamoff callbacks to fix the missing
lock protection.

wave5_vpu_dec_buf_queue_dst() and streamoff_output() call send_eos_event()
without holding inst->state_spinlock. However, send_eos_event() has
lockdep_assert_held(&inst->state_spinlock) indicating that callers must
hold this lock.

Other callers of send_eos_event() properly acquire the spinlock:
- wave5_vpu_dec_finish_decode() acquires lock at line 431
- wave5_vpu_dec_encoder_cmd() acquires lock at line 821
- wave5_vpu_dec_device_run() acquires lock at line 1592

Signed-off-by: Ziyi Guo <n7l8m4@u.northwestern.edu>
Reviewed-by: Nicolas Dufresne <nicolas.dufresne@collabora.com>
Fixes: 9707a6254a8a6b ("media: chips-media: wave5: Add the v4l2 layer")
Cc: stable@vger.kernel.org
Signed-off-by: Nicolas Dufresne <nicolas.dufresne@collabora.com>
Signed-off-by: Hans Verkuil <hverkuil+cisco@kernel.org>
drivers/media/platform/chips-media/wave5/wave5-vpu-dec.c

index 80e1831a42e02ffddc6fdc43e1c68d114be9e7d4..18cd1a6a7e1d71933c3637b22da97132ea4c4c47 100644 (file)
@@ -1303,13 +1303,17 @@ static void wave5_vpu_dec_buf_queue_dst(struct vb2_buffer *vb)
 
        if (vb2_is_streaming(vb->vb2_queue) && v4l2_m2m_dst_buf_is_last(m2m_ctx)) {
                unsigned int i;
+               unsigned long flags;
 
                for (i = 0; i < vb->num_planes; i++)
                        vb2_set_plane_payload(vb, i, 0);
 
                vbuf->field = V4L2_FIELD_NONE;
 
+               spin_lock_irqsave(&inst->state_spinlock, flags);
                send_eos_event(inst);
+               spin_unlock_irqrestore(&inst->state_spinlock, flags);
+
                v4l2_m2m_last_buffer_done(m2m_ctx, vbuf);
        } else {
                v4l2_m2m_buf_queue(m2m_ctx, vbuf);
@@ -1462,8 +1466,13 @@ static int streamoff_output(struct vb2_queue *q)
        inst->codec_info->dec_info.stream_rd_ptr = new_rd_ptr;
        inst->codec_info->dec_info.stream_wr_ptr = new_rd_ptr;
 
-       if (v4l2_m2m_has_stopped(m2m_ctx))
+       if (v4l2_m2m_has_stopped(m2m_ctx)) {
+               unsigned long flags;
+
+               spin_lock_irqsave(&inst->state_spinlock, flags);
                send_eos_event(inst);
+               spin_unlock_irqrestore(&inst->state_spinlock, flags);
+       }
 
        /* streamoff on output cancels any draining operation */
        inst->eos = false;