]> asedeno.scripts.mit.edu Git - linux.git/commitdiff
media: v4l2-mem2mem: add stateless_(try_)decoder_cmd ioctl helpers
authorJernej Skrabec <jernej.skrabec@siol.net>
Fri, 11 Oct 2019 09:32:43 +0000 (06:32 -0300)
committerMauro Carvalho Chehab <mchehab+samsung@kernel.org>
Mon, 21 Oct 2019 10:41:35 +0000 (07:41 -0300)
These helpers are used by stateless codecs when they support multiple
slices per frame and hold capture buffer flag is set. It's expected that
all such codecs will use this code.

Signed-off-by: Jernej Skrabec <jernej.skrabec@siol.net>
Co-developed-by: Hans Verkuil <hverkuil-cisco@xs4all.nl>
Signed-off-by: Hans Verkuil <hverkuil-cisco@xs4all.nl>
Signed-off-by: Mauro Carvalho Chehab <mchehab+samsung@kernel.org>
drivers/media/v4l2-core/v4l2-mem2mem.c
include/media/v4l2-mem2mem.h

index 79c3656f24f7cf4857987b8846a0572d9d588f9f..b46d2c38834996edb707ce43ccd75f3f08e41037 100644 (file)
@@ -1218,6 +1218,59 @@ int v4l2_m2m_ioctl_try_decoder_cmd(struct file *file, void *fh,
 }
 EXPORT_SYMBOL_GPL(v4l2_m2m_ioctl_try_decoder_cmd);
 
+int v4l2_m2m_ioctl_stateless_try_decoder_cmd(struct file *file, void *fh,
+                                            struct v4l2_decoder_cmd *dc)
+{
+       if (dc->cmd != V4L2_DEC_CMD_FLUSH)
+               return -EINVAL;
+
+       dc->flags = 0;
+
+       return 0;
+}
+EXPORT_SYMBOL_GPL(v4l2_m2m_ioctl_stateless_try_decoder_cmd);
+
+int v4l2_m2m_ioctl_stateless_decoder_cmd(struct file *file, void *priv,
+                                        struct v4l2_decoder_cmd *dc)
+{
+       struct v4l2_fh *fh = file->private_data;
+       struct vb2_v4l2_buffer *out_vb, *cap_vb;
+       struct v4l2_m2m_dev *m2m_dev = fh->m2m_ctx->m2m_dev;
+       unsigned long flags;
+       int ret;
+
+       ret = v4l2_m2m_ioctl_stateless_try_decoder_cmd(file, priv, dc);
+       if (ret < 0)
+               return ret;
+
+       spin_lock_irqsave(&m2m_dev->job_spinlock, flags);
+       out_vb = v4l2_m2m_last_src_buf(fh->m2m_ctx);
+       cap_vb = v4l2_m2m_last_dst_buf(fh->m2m_ctx);
+
+       /*
+        * If there is an out buffer pending, then clear any HOLD flag.
+        *
+        * By clearing this flag we ensure that when this output
+        * buffer is processed any held capture buffer will be released.
+        */
+       if (out_vb) {
+               out_vb->flags &= ~V4L2_BUF_FLAG_M2M_HOLD_CAPTURE_BUF;
+       } else if (cap_vb && cap_vb->is_held) {
+               /*
+                * If there were no output buffers, but there is a
+                * capture buffer that is held, then release that
+                * buffer.
+                */
+               cap_vb->is_held = false;
+               v4l2_m2m_dst_buf_remove(fh->m2m_ctx);
+               v4l2_m2m_buf_done(cap_vb, VB2_BUF_STATE_DONE);
+       }
+       spin_unlock_irqrestore(&m2m_dev->job_spinlock, flags);
+
+       return 0;
+}
+EXPORT_SYMBOL_GPL(v4l2_m2m_ioctl_stateless_decoder_cmd);
+
 /*
  * v4l2_file_operations helpers. It is assumed here same lock is used
  * for the output and the capture buffer queue.
index 229d9f5d43708417e2b8dd603bb071fa8a388592..3d9e48ed8817041a489096650b45bd40526dbe08 100644 (file)
@@ -701,6 +701,10 @@ int v4l2_m2m_ioctl_try_encoder_cmd(struct file *file, void *fh,
                                   struct v4l2_encoder_cmd *ec);
 int v4l2_m2m_ioctl_try_decoder_cmd(struct file *file, void *fh,
                                   struct v4l2_decoder_cmd *dc);
+int v4l2_m2m_ioctl_stateless_try_decoder_cmd(struct file *file, void *fh,
+                                            struct v4l2_decoder_cmd *dc);
+int v4l2_m2m_ioctl_stateless_decoder_cmd(struct file *file, void *priv,
+                                        struct v4l2_decoder_cmd *dc);
 int v4l2_m2m_fop_mmap(struct file *file, struct vm_area_struct *vma);
 __poll_t v4l2_m2m_fop_poll(struct file *file, poll_table *wait);