[PATCH v6, 8/8] media: mediatek: vcodec: Return encoding result in asynchronous mode
Allen-KH Cheng (程冠勳)
Allen-KH.Cheng at mediatek.com
Tue Oct 4 05:08:14 PDT 2022
Hi Irui,
On Sat, 2022-10-01 at 11:17 +0800, Irui Wang wrote:
> when enable multi-core encoding, the wait IRQ done synchronous
> function
> should not be called, so the encoding result can't return to client
> in
> device_run. Move the buffer done function in IRQ handler.
>
> Signed-off-by: Irui Wang <irui.wang at mediatek.com>
> ---
> .../platform/mediatek/vcodec/mtk_vcodec_drv.h | 6 ++
> .../platform/mediatek/vcodec/mtk_vcodec_enc.c | 72
> +++++++++++++++++--
> .../platform/mediatek/vcodec/mtk_vcodec_enc.h | 7 +-
> .../mediatek/vcodec/mtk_vcodec_enc_drv.c | 28 +++++++-
> .../mediatek/vcodec/mtk_vcodec_enc_hw.c | 13 +++-
> .../mediatek/vcodec/mtk_vcodec_enc_pm.c | 1 +
> .../mediatek/vcodec/mtk_vcodec_util.h | 1 +
> .../mediatek/vcodec/venc/venc_h264_if.c | 20 ++++--
> .../platform/mediatek/vcodec/venc_drv_if.h | 2 +
> 9 files changed, 137 insertions(+), 13 deletions(-)
>
> diff --git a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_drv.h
> b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_drv.h
> index a75ba675f72b..d9c27ebcf3c3 100644
> --- a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_drv.h
> +++ b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_drv.h
> @@ -291,6 +291,9 @@ struct vdec_pic_info {
> * @msg_queue: msg queue used to store lat buffer information.
> * @q_mutex: vb2_queue mutex.
> * @encoded_frame_cnt: number of encoded frames
> + * @pfrm_buf: used to store current ctx's frame buffer
> + * @pbs_buf: used to store current ctx's bitstream buffer
> + * @hdr_size: used to store prepend header size
> */
> struct mtk_vcodec_ctx {
> enum mtk_instance_type type;
> @@ -340,6 +343,9 @@ struct mtk_vcodec_ctx {
>
> struct mutex q_mutex;
> int encoded_frame_cnt;
> + struct vb2_v4l2_buffer *pfrm_buf[MTK_VENC_HW_MAX];
> + struct vb2_v4l2_buffer *pbs_buf[MTK_VENC_HW_MAX];
> + unsigned int hdr_size;
> };
>
> /*
> diff --git a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc.c
> b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc.c
> index d15e106fb246..3424da4aaa26 100644
> --- a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc.c
> +++ b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc.c
> @@ -958,6 +958,8 @@ static void vb2ops_venc_stop_streaming(struct
> vb2_queue *q)
>
> mtk_v4l2_debug(2, "[%d]-> type=%d", ctx->id, q->type);
>
> + mtk_venc_lock_all(ctx);
> +
> if (q->type == V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE) {
> while ((dst_buf = v4l2_m2m_dst_buf_remove(ctx-
> >m2m_ctx))) {
> vb2_set_plane_payload(&dst_buf->vb2_buf, 0, 0);
> @@ -1193,6 +1195,7 @@ static void mtk_venc_worker(struct work_struct
> *work)
> * is dequeued.
> */
> if (src_buf == &ctx->empty_flush_buf.vb) {
> + mtk_venc_lock_all(ctx);
> vb2_set_plane_payload(&dst_buf->vb2_buf, 0, 0);
> dst_buf->flags |= V4L2_BUF_FLAG_LAST;
> v4l2_m2m_buf_done(dst_buf, VB2_BUF_STATE_DONE);
> @@ -1207,9 +1210,12 @@ static void mtk_venc_worker(struct work_struct
> *work)
> frm_buf.fb_addr[i].size =
> (size_t)src_buf-
> >vb2_buf.planes[i].length;
> }
> + frm_buf.frm_addr = src_buf;
> +
> bs_buf.va = vb2_plane_vaddr(&dst_buf->vb2_buf, 0);
> bs_buf.dma_addr = vb2_dma_contig_plane_dma_addr(&dst_buf-
> >vb2_buf, 0);
> bs_buf.size = (size_t)dst_buf->vb2_buf.planes[0].length;
> + bs_buf.buf = dst_buf;
>
> mtk_v4l2_debug(2,
> "Framebuf PA=%llx Size=0x%zx;PA=0x%llx
> Size=0x%zx;PA=0x%llx Size=%zu",
> @@ -1235,11 +1241,14 @@ static void mtk_venc_worker(struct
> work_struct *work)
> v4l2_m2m_buf_done(dst_buf, VB2_BUF_STATE_ERROR);
> mtk_v4l2_err("venc_if_encode failed=%d", ret);
> } else {
> - v4l2_m2m_buf_done(src_buf, VB2_BUF_STATE_DONE);
> - vb2_set_plane_payload(&dst_buf->vb2_buf, 0,
> enc_result.bs_size);
> - v4l2_m2m_buf_done(dst_buf, VB2_BUF_STATE_DONE);
> - mtk_v4l2_debug(2, "venc_if_encode bs size=%d",
> - enc_result.bs_size);
> + if (!IS_VENC_MULTICORE(ctx->dev->enc_capability)) {
> + v4l2_m2m_buf_done(src_buf, VB2_BUF_STATE_DONE);
> + vb2_set_plane_payload(&dst_buf->vb2_buf, 0,
> + enc_result.bs_size);
> + v4l2_m2m_buf_done(dst_buf, VB2_BUF_STATE_DONE);
> + mtk_v4l2_debug(2, "venc_if_encode bs size=%d",
> + enc_result.bs_size);
> + }
> }
>
> ctx->encoded_frame_cnt++;
> @@ -1453,6 +1462,34 @@ int mtk_vcodec_enc_queue_init(void *priv,
> struct vb2_queue *src_vq,
> return vb2_queue_init(dst_vq);
> }
>
> +void mtk_venc_buf_done(struct mtk_vcodec_ctx *ctx, int hw_id,
> + unsigned int bs_size, bool time_out, bool
> key_frame)
> +{
> + struct vb2_v4l2_buffer *src_vb2_v4l2 = NULL;
> + struct vb2_v4l2_buffer *dst_vb2_v4l2 = NULL;
> +
> + /*
> + * the frm_buf(src_buf) and bs_buf(dst_buf) can be obtained
> from ctx,
> + * then put them to done list, user can get them by dqbuf call
> + */
> + src_vb2_v4l2 = ctx->pfrm_buf[hw_id];
> + dst_vb2_v4l2 = ctx->pbs_buf[hw_id];
> +
Do those buffers require a Null check?
or we can just write
struct vb2_v4l2_buffer *src_vb2_v4l2 = tx->pfrm_buf[hw_id];
struct vb2_v4l2_buffer *dst_vb2_v4l2 = ctx->pbs_buf[hw_id];
BRs,
Allen
> + if (src_vb2_v4l2 && dst_vb2_v4l2) {
> + dst_vb2_v4l2->vb2_buf.timestamp =
> + src_vb2_v4l2->vb2_buf.timestamp;
> + dst_vb2_v4l2->timecode = src_vb2_v4l2->timecode;
> +
> + if (key_frame)
> + dst_vb2_v4l2->flags |= V4L2_BUF_FLAG_KEYFRAME;
> +
> + v4l2_m2m_buf_done(src_vb2_v4l2, VB2_BUF_STATE_DONE);
> + vb2_set_plane_payload(&dst_vb2_v4l2->vb2_buf, 0,
> bs_size);
> + v4l2_m2m_buf_done(dst_vb2_v4l2, VB2_BUF_STATE_DONE);
> + }
> +}
> +EXPORT_SYMBOL_GPL(mtk_venc_buf_done);
> +
> int mtk_venc_unlock(struct mtk_vcodec_ctx *ctx, int hw_id)
> {
> struct mtk_vcodec_dev *dev = ctx->dev;
> @@ -1460,6 +1497,7 @@ int mtk_venc_unlock(struct mtk_vcodec_ctx *ctx,
> int hw_id)
> mutex_unlock(&dev->enc_mutex[hw_id]);
> return 0;
> }
> +EXPORT_SYMBOL_GPL(mtk_venc_unlock);
>
> int mtk_venc_lock(struct mtk_vcodec_ctx *ctx, int hw_id)
> {
> @@ -1468,6 +1506,30 @@ int mtk_venc_lock(struct mtk_vcodec_ctx *ctx,
> int hw_id)
> mutex_lock(&dev->enc_mutex[hw_id]);
> return 0;
> }
> +EXPORT_SYMBOL_GPL(mtk_venc_lock);
> +
> +void mtk_venc_lock_all(struct mtk_vcodec_ctx *ctx)
> +{
> + unsigned int i;
> + struct mtk_vcodec_dev *dev = ctx->dev;
> +
> + /*
> + * For multi-core mode encoding, there are may be bufs being
> encoded
> + * when get the empty flush buffer or stop streaming, for
> example, the
> + * buffer with LAST flag will return to client before the
> encoding
> + * buffers, which will cause frame lost.
> + * The encoder device mutex will be locked during encoding
> process,
> + * when encode done, the mutex unlocked. So if all encoder
> device mutex
> + * can be locked, which means there are no bufs being encoded
> at this
> + * time, then the buffer with LAST flag can return to client
> properly.
> + */
> +
> + for (i = 0; i < MTK_VENC_HW_MAX; i++) {
> + mutex_lock(&dev->enc_mutex[i]);
> + mutex_unlock(&dev->enc_mutex[i]);
> + }
> +}
> +EXPORT_SYMBOL_GPL(mtk_venc_lock_all);
>
> void mtk_vcodec_enc_release(struct mtk_vcodec_ctx *ctx)
> {
> diff --git a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc.h
> b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc.h
> index 29f5c8d1b59f..5ab17381c7ba 100644
> --- a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc.h
> +++ b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc.h
> @@ -20,6 +20,9 @@
>
> #define MTK_VENC_IRQ_STATUS_OFFSET 0x05C
> #define MTK_VENC_IRQ_ACK_OFFSET 0x060
> +#define VENC_PIC_BITSTREAM_BYTE_CNT 0x0098
> +#define VENC_PIC_FRM_TYPE 0x0010
> +#define VENC_PIC_KEY_FRM 0x2
>
> /**
> * struct mtk_video_enc_buf - Private data related to each VB2
> buffer.
> @@ -46,5 +49,7 @@ int mtk_vcodec_enc_queue_init(void *priv, struct
> vb2_queue *src_vq,
> void mtk_vcodec_enc_release(struct mtk_vcodec_ctx *ctx);
> int mtk_vcodec_enc_ctrls_setup(struct mtk_vcodec_ctx *ctx);
> void mtk_vcodec_enc_set_default_params(struct mtk_vcodec_ctx *ctx);
> -
> +void mtk_venc_buf_done(struct mtk_vcodec_ctx *ctx, int hw_id,
> + unsigned int bs_size, bool time_out, bool
> key_frame);
> +void mtk_venc_lock_all(struct mtk_vcodec_ctx *ctx);
> #endif /* _MTK_VCODEC_ENC_H_ */
> diff --git
> a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_drv.c
> b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_drv.c
> index ba9c19a4d2cc..b34d7583fccc 100644
> --- a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_drv.c
> +++ b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_drv.c
> @@ -89,6 +89,9 @@ static irqreturn_t mtk_vcodec_enc_irq_handler(int
> irq, void *priv)
> struct mtk_vcodec_ctx *ctx;
> unsigned long flags;
> void __iomem *addr;
> + unsigned int bs_size;
> + unsigned int frm_type;
> + bool is_key_frame = 0;
>
> spin_lock_irqsave(&dev->irqlock, flags);
> ctx = dev->curr_enc_ctx[MTK_VENC_CORE_0];
> @@ -101,8 +104,32 @@ static irqreturn_t
> mtk_vcodec_enc_irq_handler(int irq, void *priv)
> ctx->irq_status = readl(dev->reg_base[dev->venc_pdata->core_id]
> +
> (MTK_VENC_IRQ_STATUS_OFFSET));
>
> + bs_size = readl(dev->reg_base[dev->venc_pdata->core_id] +
> + (VENC_PIC_BITSTREAM_BYTE_CNT));
> + frm_type = readl(dev->reg_base[dev->venc_pdata->core_id] +
> + (VENC_PIC_FRM_TYPE));
> +
> clean_irq_status(ctx->irq_status, addr);
>
> + if (IS_VENC_MULTICORE(dev->enc_capability)) {
> + if (ctx->irq_status & MTK_VENC_IRQ_STATUS_FRM) {
> + if (ctx->hdr_size != 0) {
> + bs_size += ctx->hdr_size;
> + ctx->hdr_size = 0;
> + }
> +
> + if (frm_type & VENC_PIC_KEY_FRM)
> + is_key_frame = 1;
> +
> + mtk_venc_buf_done(ctx, 0, bs_size, 0,
> is_key_frame);
> + mtk_vcodec_enc_clock_off(dev, 0);
> + mtk_venc_unlock(ctx, 0);
> + } else {
> + wake_up_ctx(ctx, MTK_INST_IRQ_RECEIVED, 0);
> + }
> + return IRQ_HANDLED;
> + }
> +
> wake_up_ctx(ctx, MTK_INST_IRQ_RECEIVED, 0);
> return IRQ_HANDLED;
> }
> @@ -284,7 +311,6 @@ static int mtk_vcodec_probe(struct
> platform_device *pdev)
> goto err_res;
> }
>
> - irq_set_status_flags(dev->enc_irq, IRQ_NOAUTOEN);
> ret = devm_request_irq(&pdev->dev, dev->enc_irq,
> mtk_vcodec_enc_irq_handler,
> 0, pdev->name, dev);
> diff --git
> a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_hw.c
> b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_hw.c
> index 6df5221b742f..0367e59b20a9 100644
> --- a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_hw.c
> +++ b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_hw.c
> @@ -50,6 +50,9 @@ static irqreturn_t mtk_enc_hw_irq_handler(int irq,
> void *priv)
> struct mtk_vcodec_ctx *ctx;
> unsigned long flags;
> void __iomem *addr;
> + unsigned int bs_size;
> + unsigned int frm_type;
> + bool is_key_frame = 0;
>
> spin_lock_irqsave(&main_dev->irqlock, flags);
> ctx = main_dev->curr_enc_ctx[dev->hw_id];
> @@ -61,9 +64,17 @@ static irqreturn_t mtk_enc_hw_irq_handler(int irq,
> void *priv)
>
> addr = dev->reg_base + MTK_VENC_IRQ_ACK_OFFSET;
> ctx->irq_status = readl(dev->reg_base +
> MTK_VENC_IRQ_STATUS_OFFSET);
> + bs_size = readl(dev->reg_base + VENC_PIC_BITSTREAM_BYTE_CNT);
> + frm_type = readl(dev->reg_base + VENC_PIC_FRM_TYPE);
> clean_hw_irq_status(ctx->irq_status, addr);
>
> - wake_up_ctx(ctx, MTK_INST_IRQ_RECEIVED, 0);
> + if (frm_type & VENC_PIC_KEY_FRM)
> + is_key_frame = 1;
> +
> + mtk_venc_buf_done(ctx, dev->hw_id, bs_size, 0, is_key_frame);
> + mtk_vcodec_enc_clock_off(main_dev, dev->hw_id);
> + mtk_venc_unlock(ctx, dev->hw_id);
> +
> return IRQ_HANDLED;
> }
>
> diff --git
> a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_pm.c
> b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_pm.c
> index 2f83aade779a..af2968a8d2e5 100644
> --- a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_pm.c
> +++ b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_enc_pm.c
> @@ -235,3 +235,4 @@ void mtk_vcodec_enc_clock_off(struct
> mtk_vcodec_dev *dev,
> for (i = enc_clk->clk_num - 1; i >= 0; i--)
> clk_disable(enc_clk->clk_info[i].vcodec_clk);
> }
> +EXPORT_SYMBOL_GPL(mtk_vcodec_enc_clock_off);
> diff --git a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_util.h
> b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_util.h
> index 0033c53d5589..cecf78d6d4c6 100644
> --- a/drivers/media/platform/mediatek/vcodec/mtk_vcodec_util.h
> +++ b/drivers/media/platform/mediatek/vcodec/mtk_vcodec_util.h
> @@ -15,6 +15,7 @@ struct mtk_vcodec_mem {
> size_t size;
> void *va;
> dma_addr_t dma_addr;
> + void *buf;
> };
>
> struct mtk_vcodec_fb {
> diff --git
> a/drivers/media/platform/mediatek/vcodec/venc/venc_h264_if.c
> b/drivers/media/platform/mediatek/vcodec/venc/venc_h264_if.c
> index 32497a35afb1..a6990221d845 100644
> --- a/drivers/media/platform/mediatek/vcodec/venc/venc_h264_if.c
> +++ b/drivers/media/platform/mediatek/vcodec/venc/venc_h264_if.c
> @@ -22,7 +22,6 @@
> static const char h264_filler_marker[] = {0x0, 0x0, 0x0, 0x1, 0xc};
>
> #define H264_FILLER_MARKER_SIZE ARRAY_SIZE(h264_filler_marker)
> -#define VENC_PIC_BITSTREAM_BYTE_CNT 0x0098
>
> /*
> * enum venc_h264_frame_type - h264 encoder output bitstream frame
> type
> @@ -620,6 +619,11 @@ static int h264_encode_frame(struct
> venc_h264_inst *inst,
> return ret;
> }
>
> + if (IS_VENC_MULTICORE(ctx->dev->enc_capability)) {
> + ++inst->frm_cnt;
> + return ret;
> + }
> +
> irq_status = h264_enc_wait_venc_done(inst);
> if (irq_status != MTK_VENC_IRQ_STATUS_FRM) {
> mtk_vcodec_err(inst, "irq_status=%d failed",
> irq_status);
> @@ -705,8 +709,6 @@ static int h264_enc_encode(void *handle,
>
> mtk_vcodec_debug(inst, "opt %d ->", opt);
>
> - enable_irq(ctx->dev->enc_irq);
> -
> switch (opt) {
> case VENC_START_OPT_ENCODE_SEQUENCE_HEADER: {
> unsigned int bs_size_hdr;
> @@ -729,6 +731,13 @@ static int h264_enc_encode(void *handle,
> unsigned int bs_size_hdr;
> unsigned int bs_size_frm;
>
> + /*
> + * the frm_buf and bs_buf need to recorded into current
> ctx,
> + * when encoding done, the target buffers can be got
> from ctx.
> + */
> + ctx->pfrm_buf[ctx->hw_id] = frm_buf->frm_addr;
> + ctx->pbs_buf[ctx->hw_id] = bs_buf->buf;
> +
> if (!inst->prepend_hdr) {
> ret = h264_encode_frame(inst, frm_buf, bs_buf,
> &result->bs_size, ctx-
> >hw_id);
> @@ -763,7 +772,9 @@ static int h264_enc_encode(void *handle,
> if (ret)
> goto encode_err;
>
> - result->bs_size = hdr_sz + filler_sz + bs_size_frm;
> + ctx->hdr_size = hdr_sz + filler_sz;
> +
> + result->bs_size = ctx->hdr_size + bs_size_frm;
>
> mtk_vcodec_debug(inst, "hdr %d filler %d frame %d bs
> %d",
> hdr_sz, filler_sz, bs_size_frm,
> @@ -782,7 +793,6 @@ static int h264_enc_encode(void *handle,
>
> encode_err:
>
> - disable_irq(ctx->dev->enc_irq);
> mtk_vcodec_debug(inst, "opt %d <-", opt);
>
> return ret;
> diff --git a/drivers/media/platform/mediatek/vcodec/venc_drv_if.h
> b/drivers/media/platform/mediatek/vcodec/venc_drv_if.h
> index e676ccf6bd25..7e24b7f573d7 100644
> --- a/drivers/media/platform/mediatek/vcodec/venc_drv_if.h
> +++ b/drivers/media/platform/mediatek/vcodec/venc_drv_if.h
> @@ -108,9 +108,11 @@ struct venc_frame_info {
> /*
> * struct venc_frm_buf - frame buffer information used in
> venc_if_encode()
> * @fb_addr: plane frame buffer addresses
> + * @frm_addr: current v4l2 buffer address
> */
> struct venc_frm_buf {
> struct mtk_vcodec_fb fb_addr[MTK_VCODEC_MAX_PLANES];
> + void *frm_addr;
> };
>
> /*
More information about the Linux-mediatek
mailing list