All of lore.kernel.org
 help / color / mirror / Atom feed
From: "yunfei.dong@mediatek.com" <yunfei.dong@mediatek.com>
To: Steve Cho <stevecho@chromium.org>
Cc: Alexandre Courbot <acourbot@chromium.org>,
	Hans Verkuil <hverkuil-cisco@xs4all.nl>,
	Tzung-Bi Shih <tzungbi@chromium.org>,
	"Tiffany Lin" <tiffany.lin@mediatek.com>,
	Andrew-CT Chen <andrew-ct.chen@mediatek.com>,
	Mauro Carvalho Chehab <mchehab@kernel.org>,
	Rob Herring <robh+dt@kernel.org>,
	Matthias Brugger <matthias.bgg@gmail.com>,
	Tomasz Figa <tfiga@google.com>,
	Hsin-Yi Wang <hsinyi@chromium.org>,
	"Fritz Koenig" <frkoenig@chromium.org>,
	Dafna Hirschfeld <dafna.hirschfeld@collabora.com>,
	Benjamin Gaignard <benjamin.gaignard@collabora.com>,
	Daniel Vetter <daniel@ffwll.ch>,
	dri-devel <dri-devel@lists.freedesktop.org>,
	Irui Wang <irui.wang@mediatek.com>,
	AngeloGioacchino Del Regno 
	<angelogioacchino.delregno@collabora.com>,
	<linux-media@vger.kernel.org>, <devicetree@vger.kernel.org>,
	<linux-kernel@vger.kernel.org>,
	<linux-arm-kernel@lists.infradead.org>,
	<srv_heupstream@mediatek.com>,
	<linux-mediatek@lists.infradead.org>,
	<Project_Global_Chrome_Upstream_Group@mediatek.com>
Subject: Re: [PATCH v12, 06/19] media: mtk-vcodec: Add to support multi hardware decode
Date: Mon, 13 Dec 2021 16:40:47 +0800	[thread overview]
Message-ID: <87c70b080162b63454609d647ff1e2b39d06d406.camel@mediatek.com> (raw)
In-Reply-To: <CAC-pXoOCa+13u--9sCKDorh+RbuhL89Ajur1t8bSx6w=MgoyYg@mail.gmail.com>

Hi Steve,

Thanks for your suggestion,
On Thu, 2021-12-09 at 15:29 -0800, Steve Cho wrote:
> Few comments and questions.
> 
> On Wed, Dec 1, 2021 at 7:46 PM Yunfei Dong <yunfei.dong@mediatek.com>
> wrote:
> > 
> > There are more than two hardwares for decoder: LAT0, LAT1 and CORE.
> > In order to
> > manage these hardwares, register each hardware as independent
> > platform device
> > for the larbs are different.
> 
> basic question: what is "larbs"?
> 
Can regard larbs as some hardware which decode depends on.
> > 
> > Each hardware module controls its own information which includes
> > interrupt/power/
> > clocks/registers.
> > 
> > Calling of_platform_populate in parent device, and use
> > subdev_bitmap to record
> > whether the hardwares are registered done.
> 
> nit: s/registered done/registered/ ?
> 
Fix in v13.
> > 
> > Signed-off-by: Yunfei Dong <yunfei.dong@mediatek.com>
> > ---
> >  drivers/media/platform/mtk-vcodec/Makefile    |   5 +-
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_drv.c  | 111 +++++++----
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_hw.c   | 172
> > ++++++++++++++++++
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_hw.h   |  51 ++++++
> >  .../mtk-vcodec/mtk_vcodec_dec_stateful.c      |   1 +
> >  .../mtk-vcodec/mtk_vcodec_dec_stateless.c     |   2 +
> >  .../platform/mtk-vcodec/mtk_vcodec_drv.h      |  19 ++
> >  7 files changed, 329 insertions(+), 32 deletions(-)
> >  create mode 100644 drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_hw.c
> >  create mode 100644 drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_hw.h
> > 
> > diff --git a/drivers/media/platform/mtk-vcodec/Makefile
> > b/drivers/media/platform/mtk-vcodec/Makefile
> > index ca8e9e7a9c4e..c61bfb179bcc 100644
> > --- a/drivers/media/platform/mtk-vcodec/Makefile
> > +++ b/drivers/media/platform/mtk-vcodec/Makefile
> > @@ -2,7 +2,8 @@
> > 
> >  obj-$(CONFIG_VIDEO_MEDIATEK_VCODEC) += mtk-vcodec-dec.o \
> >                                        mtk-vcodec-enc.o \
> > -                                      mtk-vcodec-common.o
> > +                                      mtk-vcodec-common.o \
> > +                                      mtk-vcodec-dec-hw.o
> > 
> >  mtk-vcodec-dec-y := vdec/vdec_h264_if.o \
> >                 vdec/vdec_vp8_if.o \
> > @@ -16,6 +17,8 @@ mtk-vcodec-dec-y := vdec/vdec_h264_if.o \
> >                 mtk_vcodec_dec_stateless.o \
> >                 mtk_vcodec_dec_pm.o \
> > 
> > +mtk-vcodec-dec-hw-y := mtk_vcodec_dec_hw.o
> > +
> >  mtk-vcodec-enc-y := venc/venc_vp8_if.o \
> >                 venc/venc_h264_if.o \
> >                 mtk_vcodec_enc.o \
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c 
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > index b7a51e96d4ba..95fbe9be3f6d 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > @@ -18,19 +18,40 @@
> > 
> >  #include "mtk_vcodec_drv.h"
> >  #include "mtk_vcodec_dec.h"
> > +#include "mtk_vcodec_dec_hw.h"
> >  #include "mtk_vcodec_dec_pm.h"
> >  #include "mtk_vcodec_intr.h"
> >  #include "mtk_vcodec_util.h"
> >  #include "mtk_vcodec_fw.h"
> > 
> > -#define VDEC_HW_ACTIVE 0x10
> > -#define VDEC_IRQ_CFG   0x11
> > -#define VDEC_IRQ_CLR   0x10
> > -#define VDEC_IRQ_CFG_REG       0xa4
> > -
> >  module_param(mtk_v4l2_dbg_level, int, 0644);
> >  module_param(mtk_vcodec_dbg, bool, 0644);
> > 
> > +static int mtk_vcodec_subdev_device_check(struct mtk_vcodec_dev
> > *vdec_dev)
> > +{
> > +       struct platform_device *pdev = vdec_dev->plat_dev;
> > +       struct device_node *subdev_node;
> > +       enum mtk_vdec_hw_id hw_idx;
> > +       const struct of_device_id *of_id;
> > +       int i;
> > +
> > +       for (i = 0; i < ARRAY_SIZE(mtk_vdec_hw_match); i++) {
> > +               of_id = &mtk_vdec_hw_match[i];
> > +               subdev_node = of_find_compatible_node(NULL, NULL,
> > +                       of_id->compatible);
> > +               if (!subdev_node)
> > +                       continue;
> > +
> > +               hw_idx = (enum mtk_vdec_hw_id)(uintptr_t)of_id-
> > >data;
> > +               if (!test_bit(hw_idx, vdec_dev->subdev_bitmap)) {
> > +                       dev_err(&pdev->dev, "Vdec %d is not ready",
> > hw_idx);
> > +                       return -EAGAIN;
> > +               }
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> >  static irqreturn_t mtk_vcodec_dec_irq_handler(int irq, void *priv)
> >  {
> >         struct mtk_vcodec_dev *dev = priv;
> > @@ -95,6 +116,42 @@ static int mtk_vcodec_get_reg_bases(struct
> > mtk_vcodec_dev *dev)
> >         return 0;
> >  }
> > 
> > +static int mtk_vcodec_init_dec_resources(struct mtk_vcodec_dev
> > *dev)
> > +{
> > +       struct platform_device *pdev = dev->plat_dev;
> > +       int ret;
> > +
> > +       ret = mtk_vcodec_get_reg_bases(dev);
> > +       if (ret)
> > +               return ret;
> > +
> > +       if (dev->vdec_pdata->is_subdev_supported)
> > +               return 0;
> > +
> > +       dev->dec_irq = platform_get_irq(pdev, 0);
> > +       if (dev->dec_irq < 0) {
> > +               dev_err(&pdev->dev, "failed to get irq number");
> > +               return dev->dec_irq;
> > +       }
> > +
> > +       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > +       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > +               mtk_vcodec_dec_irq_handler, 0, pdev->name, dev);
> > +       if (ret) {
> > +               dev_err(&pdev->dev, "failed to install dev->dec_irq 
> > %d (%d)",
> > +                       dev->dec_irq, ret);
> > +               return ret;
> > +       }
> > +
> > +       ret = mtk_vcodec_init_dec_pm(pdev, &dev->pm);
> > +       if (ret < 0) {
> > +               dev_err(&pdev->dev, "failed to get mt vcodec clock
> > source");
> > +               return ret;
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> >  static int fops_vcodec_open(struct file *file)
> >  {
> >         struct mtk_vcodec_dev *dev = video_drvdata(file);
> > @@ -116,6 +173,12 @@ static int fops_vcodec_open(struct file *file)
> >         init_waitqueue_head(&ctx->queue);
> >         mutex_init(&ctx->lock);
> > 
> > +       ret = mtk_vcodec_subdev_device_check(dev);
> > +       if (ret) {
> > +               mtk_v4l2_err("Failed to check vdec comp device.");
> 
> basic question: what is "comp" device?
> 
This line can be removed, "comp" is subdev.
> > +               goto err_ctrls_setup;
> > +       }
> > +
> >         ctx->type = MTK_INST_DECODER;
> >         ret = dev->vdec_pdata->ctrls_setup(ctx);
> >         if (ret) {
> > @@ -220,7 +283,6 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >  {
> >         struct mtk_vcodec_dev *dev;
> >         struct video_device *vfd_dec;
> > -       struct resource *res;
> >         phandle rproc_phandle;
> >         enum mtk_vcodec_fw_type fw_type;
> >         int ret;
> > @@ -249,32 +311,10 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >         if (IS_ERR(dev->fw_handler))
> >                 return PTR_ERR(dev->fw_handler);
> > 
> > -       ret = mtk_vcodec_init_dec_pm(dev->plat_dev, &dev->pm);
> > -       if (ret < 0) {
> > -               dev_err(&pdev->dev, "Failed to get mt vcodec clock
> > source");
> > -               goto err_dec_pm;
> > -       }
> > -
> > -       ret = mtk_vcodec_get_reg_bases(dev);
> > -       if (ret)
> > -               goto err_res;
> > -
> > -       res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
> > -       if (res == NULL) {
> > -               dev_err(&pdev->dev, "failed to get irq resource");
> > -               ret = -ENOENT;
> > -               goto err_res;
> > -       }
> > -
> > -       dev->dec_irq = platform_get_irq(pdev, 0);
> > -       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > -       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > -                       mtk_vcodec_dec_irq_handler, 0, pdev->name,
> > dev);
> > +       ret = mtk_vcodec_init_dec_resources(dev);
> >         if (ret) {
> > -               dev_err(&pdev->dev, "Failed to install dev->dec_irq 
> > %d (%d)",
> > -                       dev->dec_irq,
> > -                       ret);
> > -               goto err_res;
> > +               dev_err(&pdev->dev, "Failed to init dec
> > resources");
> > +               goto err_dec_pm;
> >         }
> > 
> >         mutex_init(&dev->dec_mutex);
> > @@ -329,6 +369,15 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >                 goto err_event_workq;
> >         }
> > 
> > +       if (dev->vdec_pdata->is_subdev_supported) {
> > +               ret = of_platform_populate(pdev->dev.of_node, NULL,
> > NULL,
> > +                       &pdev->dev);
> > +               if (ret) {
> > +                       mtk_v4l2_err("Main device
> > of_platform_populate failed.");
> > +                       goto err_event_workq;
> > +               }
> > +       }
> > +
> >         if (dev->vdec_pdata->uses_stateless_api) {
> >                 dev->mdev_dec.dev = &pdev->dev;
> >                 strscpy(dev->mdev_dec.model, MTK_VCODEC_DEC_NAME,
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > new file mode 100644
> > index 000000000000..025a103008ba
> > --- /dev/null
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > @@ -0,0 +1,172 @@
> > +// SPDX-License-Identifier: GPL-2.0
> > +/*
> > + * Copyright (c) 2021 MediaTek Inc.
> > + * Author: Yunfei Dong <yunfei.dong@mediatek.com>
> > + */
> > +
> > +#include <linux/interrupt.h>
> > +#include <linux/irq.h>
> > +#include <linux/module.h>
> > +#include <linux/of.h>
> > +#include <linux/of_device.h>
> > +#include <linux/slab.h>
> > +
> > +#include "mtk_vcodec_drv.h"
> > +#include "mtk_vcodec_dec.h"
> > +#include "mtk_vcodec_dec_hw.h"
> > +#include "mtk_vcodec_dec_pm.h"
> > +#include "mtk_vcodec_intr.h"
> > +#include "mtk_vcodec_util.h"
> > +
> > +const struct of_device_id mtk_vdec_hw_match[] = {
> > +       {
> > +               .compatible = "mediatek,mtk-vcodec-lat",
> > +               .data = (void *)MTK_VDEC_LAT0,
> > +       },
> 
> Are we supposed to have LAT1 here as well?
At current period, need not to add it. LAT1 may be used in the future.
> 
> > +       {
> > +               .compatible = "mediatek,mtk-vcodec-core",
> > +               .data = (void *)MTK_VDEC_CORE,
> > +       },
> > +       {},
> > +};
> > +EXPORT_SYMBOL_GPL(mtk_vdec_hw_match);
> > +
> > +MODULE_DEVICE_TABLE(of, mtk_vdec_hw_match);
> > +
> > +static irqreturn_t mtk_vdec_hw_irq_handler(int irq, void *priv)
> > +{
> > +       struct mtk_vdec_hw_dev *dev = priv;
> > +       struct mtk_vcodec_ctx *ctx;
> > +       u32 cg_status;
> > +       unsigned int dec_done_status;
> > +       void __iomem *vdec_misc_addr = dev->reg_base[VDEC_HW_MISC]
> > +
> > +                                       VDEC_IRQ_CFG_REG;
> > +
> > +       ctx = mtk_vcodec_get_curr_ctx(dev->main_dev);
> > +
> > +       /* check if HW active or not */
> > +       cg_status = readl(dev->reg_base[VDEC_HW_SYS]);
> > +       if (cg_status & VDEC_HW_ACTIVE) {
> > +               mtk_v4l2_err("vdec active is not 0x0 (0x%08x)",
> > +                       cg_status);
> > +               return IRQ_HANDLED;
> > +       }
> > +
> > +       dec_done_status = readl(vdec_misc_addr);
> > +       if ((dec_done_status & MTK_VDEC_IRQ_STATUS_DEC_SUCCESS) !=
> > +           MTK_VDEC_IRQ_STATUS_DEC_SUCCESS)
> > +               return IRQ_HANDLED;
> > +
> > +       /* clear interrupt */
> > +       writel(dec_done_status | VDEC_IRQ_CFG, vdec_misc_addr);
> 
> Is VDEC_IRQ_CFG also intended for clear interrupt?
> 
VDEC_IRQ_CFG and VDEC_IRQ_CLR are different bits to clear interrupt.
> > +       writel(dec_done_status & ~VDEC_IRQ_CLR, vdec_misc_addr);
> > +
> > +       wake_up_ctx(ctx, MTK_INST_IRQ_RECEIVED);
> > +
> > +       mtk_v4l2_debug(3, "wake up ctx %d, dec_done_status=%x",
> > +               ctx->id, dec_done_status);
> > +
> > +       return IRQ_HANDLED;
> > +}
> > +
> > +static int mtk_vdec_hw_init_irq(struct mtk_vdec_hw_dev *dev)
> > +{
> > +       struct platform_device *pdev = dev->plat_dev;
> > +       int ret;
> > +
> > +       dev->dec_irq = platform_get_irq(pdev, 0);
> > +       if (dev->dec_irq < 0) {
> > +               dev_err(&pdev->dev, "Failed to get irq resource");
> > +               return dev->dec_irq;
> > +       }
> > +
> > +       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > +       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > +                               mtk_vdec_hw_irq_handler, 0, pdev-
> > >name, dev);
> > +       if (ret) {
> > +               dev_err(&pdev->dev, "Failed to install dev->dec_irq 
> > %d (%d)",
> > +                       dev->dec_irq, ret);
> > +               return ret;
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> > +static int mtk_vdec_hw_probe(struct platform_device *pdev)
> > +{
> > +       struct device *dev = &pdev->dev;
> > +       struct mtk_vdec_hw_dev *subdev_dev;
> > +       struct mtk_vcodec_dev *main_dev;
> > +       const struct of_device_id *of_id;
> > +       int hw_idx;
> > +       int ret;
> > +
> > +       if (!dev->parent) {
> > +               dev_err(dev, "no parent for hardware devices.\n");
> > +               return -ENODEV;
> > +       }
> > +
> > +       main_dev = dev_get_drvdata(dev->parent);
> > +       if (!main_dev) {
> > +               dev_err(dev, "failed to get parent driver data");
> > +               return -EINVAL;
> > +       }
> > +
> > +       subdev_dev = devm_kzalloc(dev, sizeof(*subdev_dev),
> > GFP_KERNEL);
> > +       if (!subdev_dev)
> > +               return -ENOMEM;
> > +
> > +       subdev_dev->plat_dev = pdev;
> > +       ret = mtk_vcodec_init_dec_pm(pdev, &subdev_dev->pm);
> > +       if (ret)
> > +               return ret;
> > +
> > +       subdev_dev->reg_base[VDEC_HW_MISC] =
> > +               devm_platform_ioremap_resource(pdev, 0);
> > +       if (IS_ERR((__force void *)subdev_dev-
> > >reg_base[VDEC_HW_MISC])) {
> > +               ret = PTR_ERR((__force void *)subdev_dev-
> > >reg_base[VDEC_HW_MISC]);
> > +               goto err;
> > +       }
> > +
> > +       ret = mtk_vdec_hw_init_irq(subdev_dev);
> > +       if (ret)
> > +               goto err;
> > +
> > +       of_id = of_match_device(mtk_vdec_hw_match, dev);
> > +       if (!of_id) {
> > +               dev_err(dev, "Can't get vdec subdev id.\n");
> > +               ret = -EINVAL;
> > +               goto err;
> > +       }
> > +
> > +       hw_idx = (enum mtk_vdec_hw_id)(uintptr_t)of_id->data;
> > +       if (hw_idx >= MTK_VDEC_HW_MAX) {
> > +               dev_err(dev, "Hardware index %d not correct.\n",
> > hw_idx);
> > +               ret = -EINVAL;
> > +               goto err;
> > +       }
> > +
> > +       main_dev->subdev_dev[hw_idx] = subdev_dev;
> > +       subdev_dev->hw_idx = hw_idx;
> > +       subdev_dev->main_dev = main_dev;
> > +       subdev_dev->reg_base[VDEC_HW_SYS] = main_dev-
> > >reg_base[VDEC_HW_SYS];
> > +       set_bit(subdev_dev->hw_idx, main_dev->subdev_bitmap);
> > +
> > +       platform_set_drvdata(pdev, subdev_dev);
> > +       return 0;
> > +err:
> > +       mtk_vcodec_release_dec_pm(&subdev_dev->pm);
> > +       return ret;
> > +}
> > +
> > +static struct platform_driver mtk_vdec_driver = {
> > +       .probe  = mtk_vdec_hw_probe,
> > +       .driver = {
> > +               .name   = "mtk-vdec-comp",
> > +               .of_match_table = mtk_vdec_hw_match,
> > +       },
> > +};
> > +module_platform_driver(mtk_vdec_driver);
> > +
> > +MODULE_LICENSE("GPL v2");
> > +MODULE_DESCRIPTION("Mediatek video decoder hardware driver");
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > new file mode 100644
> > index 000000000000..f7f36790629d
> > --- /dev/null
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > @@ -0,0 +1,51 @@
> > +// SPDX-License-Identifier: GPL-2.0
> > +/*
> > + * Copyright (c) 2021 MediaTek Inc.
> > + * Author: Yunfei Dong <yunfei.dong@mediatek.com>
> > + */
> > +
> > +#ifndef _MTK_VCODEC_DEC_HW_H_
> > +#define _MTK_VCODEC_DEC_HW_H_
> > +
> > +#include <linux/io.h>
> > +#include <linux/platform_device.h>
> > +
> > +#include "mtk_vcodec_drv.h"
> > +
> > +#define VDEC_HW_ACTIVE 0x10
> > +#define VDEC_IRQ_CFG 0x11
> > +#define VDEC_IRQ_CLR 0x10
> > +#define VDEC_IRQ_CFG_REG 0xa4
> > +
> > +extern const struct of_device_id
> > mtk_vdec_hw_match[MTK_VDEC_HW_MAX];
> > +
> > +/**
> > + * enum mtk_vdec_hw_reg_idx - subdev hardware register base index
> > + */
> > +enum mtk_vdec_hw_reg_idx {
> > +       VDEC_HW_SYS,
> > +       VDEC_HW_MISC,
> > +       VDEC_HW_MAX
> > +};
> > +
> > +/**
> > + * struct mtk_vdec_hw_dev - vdec hardware driver data
> > + * @plat_dev: platform device
> > + * @main_dev: main device
> > + * @reg_base: Mapped address of MTK Vcodec registers.
> > + *
> > + * @dec_irq: decoder irq resource
> > + * @pm: power management control
> > + * @hw_idx: each hardware index
> > + */
> > +struct mtk_vdec_hw_dev {
> > +       struct platform_device *plat_dev;
> > +       struct mtk_vcodec_dev *main_dev;
> > +       void __iomem *reg_base[VDEC_HW_MAX];
> > +
> > +       int dec_irq;
> > +       struct mtk_vcodec_pm pm;
> > +       int hw_idx;
> > +};
> > +
> > +#endif /* _MTK_VCODEC_DEC_HW_H_ */
> > diff --git a/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateful.c b/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateful.c
> > index bef49244e61b..c7f9259ad094 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateful.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateful.c
> > @@ -625,4 +625,5 @@ const struct mtk_vcodec_dec_pdata
> > mtk_vdec_8173_pdata = {
> >         .num_framesizes = NUM_SUPPORTED_FRAMESIZE,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = false,
> >  };
> > diff --git a/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateless.c b/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateless.c
> > index 26e4d6f4ec04..2d285515b625 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateless.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateless.c
> > @@ -357,6 +357,7 @@ const struct mtk_vcodec_dec_pdata
> > mtk_vdec_8183_pdata = {
> 
> 8173 might be only one for stateful. But we have multiple platforms
> for 8183. Shouldn't this name reflect this?
> 
This platform is used for mt8183, all mt8183 need to used it.
And 8173 is stateful, 8183 is stateless.
> >         .uses_stateless_api = true,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = false,
> >  };
> > 
> >  const struct mtk_vcodec_dec_pdata mtk_lat_sig_core_pdata = {
> > @@ -373,4 +374,5 @@ const struct mtk_vcodec_dec_pdata
> > mtk_lat_sig_core_pdata = {
> 
> Which platforms is this supposed to be used? Some comments on this
> might be helpful.
> 
Add in v13.

Thanks,
Yunfei Dong
> >         .uses_stateless_api = true,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = true,
> >  };
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > index 0fa9d85114b9..2160f26263ef 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > @@ -93,6 +93,17 @@ enum mtk_fmt_type {
> >         MTK_FMT_FRAME = 2,
> >  };
> > 
> > +/**
> > + * struct mtk_vdec_hw_id - Hardware index used to separate
> > + *                         different hardware
> > + */
> > +enum mtk_vdec_hw_id {
> > +       MTK_VDEC_CORE,
> > +       MTK_VDEC_LAT0,
> > +       MTK_VDEC_LAT1,
> > +       MTK_VDEC_HW_MAX,
> > +};
> > +
> >  /*
> >   * struct mtk_video_fmt - Structure used to store information
> > about pixelformats
> >   */
> > @@ -332,6 +343,7 @@ enum mtk_chip {
> >   *
> >   * @chip: chip this decoder is compatible with
> >   *
> > + * @is_subdev_supported: whether support parent-node
> > architecture(subdev)
> >   * @uses_stateless_api: whether the decoder uses the stateless API
> > with requests
> >   */
> > 
> > @@ -353,6 +365,7 @@ struct mtk_vcodec_dec_pdata {
> > 
> >         enum mtk_chip chip;
> > 
> > +       bool is_subdev_supported;
> >         bool uses_stateless_api;
> >  };
> > 
> > @@ -423,6 +436,9 @@ struct mtk_vcodec_enc_pdata {
> >   * @pm: power management control
> >   * @dec_capability: used to identify decode capability, ex: 4k
> >   * @enc_capability: used to identify encode capability
> > + *
> > + * @subdev_dev: subdev hardware device
> > + * @subdev_bitmap: used to record hardware is ready or not
> >   */
> >  struct mtk_vcodec_dev {
> >         struct v4l2_device v4l2_dev;
> > @@ -460,6 +476,9 @@ struct mtk_vcodec_dev {
> >         struct mtk_vcodec_pm pm;
> >         unsigned int dec_capability;
> >         unsigned int enc_capability;
> > +
> > +       void *subdev_dev[MTK_VDEC_HW_MAX];
> > +       DECLARE_BITMAP(subdev_bitmap, MTK_VDEC_HW_MAX);
> >  };
> > 
> >  static inline struct mtk_vcodec_ctx *fh_to_ctx(struct v4l2_fh *fh)
> > --
> > 2.25.1
> > 

WARNING: multiple messages have this Message-ID (diff)
From: "yunfei.dong@mediatek.com" <yunfei.dong@mediatek.com>
To: Steve Cho <stevecho@chromium.org>
Cc: Andrew-CT Chen <andrew-ct.chen@mediatek.com>,
	Dafna Hirschfeld <dafna.hirschfeld@collabora.com>,
	dri-devel <dri-devel@lists.freedesktop.org>,
	Irui Wang <irui.wang@mediatek.com>,
	Mauro Carvalho Chehab <mchehab@kernel.org>,
	Benjamin Gaignard <benjamin.gaignard@collabora.com>,
	Project_Global_Chrome_Upstream_Group@mediatek.com,
	Fritz Koenig <frkoenig@chromium.org>,
	linux-media@vger.kernel.org, devicetree@vger.kernel.org,
	Tzung-Bi Shih <tzungbi@chromium.org>,
	Tomasz Figa <tfiga@google.com>, Rob Herring <robh+dt@kernel.org>,
	linux-mediatek@lists.infradead.org,
	Hsin-Yi Wang <hsinyi@chromium.org>,
	Matthias Brugger <matthias.bgg@gmail.com>,
	Tiffany Lin <tiffany.lin@mediatek.com>,
	linux-arm-kernel@lists.infradead.org,
	AngeloGioacchino Del Regno
	<angelogioacchino.delregno@collabora.com>,
	Alexandre Courbot <acourbot@chromium.org>,
	srv_heupstream@mediatek.com, linux-kernel@vger.kernel.org,
	Hans Verkuil <hverkuil-cisco@xs4all.nl>
Subject: Re: [PATCH v12, 06/19] media: mtk-vcodec: Add to support multi hardware decode
Date: Mon, 13 Dec 2021 16:40:47 +0800	[thread overview]
Message-ID: <87c70b080162b63454609d647ff1e2b39d06d406.camel@mediatek.com> (raw)
In-Reply-To: <CAC-pXoOCa+13u--9sCKDorh+RbuhL89Ajur1t8bSx6w=MgoyYg@mail.gmail.com>

Hi Steve,

Thanks for your suggestion,
On Thu, 2021-12-09 at 15:29 -0800, Steve Cho wrote:
> Few comments and questions.
> 
> On Wed, Dec 1, 2021 at 7:46 PM Yunfei Dong <yunfei.dong@mediatek.com>
> wrote:
> > 
> > There are more than two hardwares for decoder: LAT0, LAT1 and CORE.
> > In order to
> > manage these hardwares, register each hardware as independent
> > platform device
> > for the larbs are different.
> 
> basic question: what is "larbs"?
> 
Can regard larbs as some hardware which decode depends on.
> > 
> > Each hardware module controls its own information which includes
> > interrupt/power/
> > clocks/registers.
> > 
> > Calling of_platform_populate in parent device, and use
> > subdev_bitmap to record
> > whether the hardwares are registered done.
> 
> nit: s/registered done/registered/ ?
> 
Fix in v13.
> > 
> > Signed-off-by: Yunfei Dong <yunfei.dong@mediatek.com>
> > ---
> >  drivers/media/platform/mtk-vcodec/Makefile    |   5 +-
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_drv.c  | 111 +++++++----
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_hw.c   | 172
> > ++++++++++++++++++
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_hw.h   |  51 ++++++
> >  .../mtk-vcodec/mtk_vcodec_dec_stateful.c      |   1 +
> >  .../mtk-vcodec/mtk_vcodec_dec_stateless.c     |   2 +
> >  .../platform/mtk-vcodec/mtk_vcodec_drv.h      |  19 ++
> >  7 files changed, 329 insertions(+), 32 deletions(-)
> >  create mode 100644 drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_hw.c
> >  create mode 100644 drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_hw.h
> > 
> > diff --git a/drivers/media/platform/mtk-vcodec/Makefile
> > b/drivers/media/platform/mtk-vcodec/Makefile
> > index ca8e9e7a9c4e..c61bfb179bcc 100644
> > --- a/drivers/media/platform/mtk-vcodec/Makefile
> > +++ b/drivers/media/platform/mtk-vcodec/Makefile
> > @@ -2,7 +2,8 @@
> > 
> >  obj-$(CONFIG_VIDEO_MEDIATEK_VCODEC) += mtk-vcodec-dec.o \
> >                                        mtk-vcodec-enc.o \
> > -                                      mtk-vcodec-common.o
> > +                                      mtk-vcodec-common.o \
> > +                                      mtk-vcodec-dec-hw.o
> > 
> >  mtk-vcodec-dec-y := vdec/vdec_h264_if.o \
> >                 vdec/vdec_vp8_if.o \
> > @@ -16,6 +17,8 @@ mtk-vcodec-dec-y := vdec/vdec_h264_if.o \
> >                 mtk_vcodec_dec_stateless.o \
> >                 mtk_vcodec_dec_pm.o \
> > 
> > +mtk-vcodec-dec-hw-y := mtk_vcodec_dec_hw.o
> > +
> >  mtk-vcodec-enc-y := venc/venc_vp8_if.o \
> >                 venc/venc_h264_if.o \
> >                 mtk_vcodec_enc.o \
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c 
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > index b7a51e96d4ba..95fbe9be3f6d 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > @@ -18,19 +18,40 @@
> > 
> >  #include "mtk_vcodec_drv.h"
> >  #include "mtk_vcodec_dec.h"
> > +#include "mtk_vcodec_dec_hw.h"
> >  #include "mtk_vcodec_dec_pm.h"
> >  #include "mtk_vcodec_intr.h"
> >  #include "mtk_vcodec_util.h"
> >  #include "mtk_vcodec_fw.h"
> > 
> > -#define VDEC_HW_ACTIVE 0x10
> > -#define VDEC_IRQ_CFG   0x11
> > -#define VDEC_IRQ_CLR   0x10
> > -#define VDEC_IRQ_CFG_REG       0xa4
> > -
> >  module_param(mtk_v4l2_dbg_level, int, 0644);
> >  module_param(mtk_vcodec_dbg, bool, 0644);
> > 
> > +static int mtk_vcodec_subdev_device_check(struct mtk_vcodec_dev
> > *vdec_dev)
> > +{
> > +       struct platform_device *pdev = vdec_dev->plat_dev;
> > +       struct device_node *subdev_node;
> > +       enum mtk_vdec_hw_id hw_idx;
> > +       const struct of_device_id *of_id;
> > +       int i;
> > +
> > +       for (i = 0; i < ARRAY_SIZE(mtk_vdec_hw_match); i++) {
> > +               of_id = &mtk_vdec_hw_match[i];
> > +               subdev_node = of_find_compatible_node(NULL, NULL,
> > +                       of_id->compatible);
> > +               if (!subdev_node)
> > +                       continue;
> > +
> > +               hw_idx = (enum mtk_vdec_hw_id)(uintptr_t)of_id-
> > >data;
> > +               if (!test_bit(hw_idx, vdec_dev->subdev_bitmap)) {
> > +                       dev_err(&pdev->dev, "Vdec %d is not ready",
> > hw_idx);
> > +                       return -EAGAIN;
> > +               }
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> >  static irqreturn_t mtk_vcodec_dec_irq_handler(int irq, void *priv)
> >  {
> >         struct mtk_vcodec_dev *dev = priv;
> > @@ -95,6 +116,42 @@ static int mtk_vcodec_get_reg_bases(struct
> > mtk_vcodec_dev *dev)
> >         return 0;
> >  }
> > 
> > +static int mtk_vcodec_init_dec_resources(struct mtk_vcodec_dev
> > *dev)
> > +{
> > +       struct platform_device *pdev = dev->plat_dev;
> > +       int ret;
> > +
> > +       ret = mtk_vcodec_get_reg_bases(dev);
> > +       if (ret)
> > +               return ret;
> > +
> > +       if (dev->vdec_pdata->is_subdev_supported)
> > +               return 0;
> > +
> > +       dev->dec_irq = platform_get_irq(pdev, 0);
> > +       if (dev->dec_irq < 0) {
> > +               dev_err(&pdev->dev, "failed to get irq number");
> > +               return dev->dec_irq;
> > +       }
> > +
> > +       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > +       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > +               mtk_vcodec_dec_irq_handler, 0, pdev->name, dev);
> > +       if (ret) {
> > +               dev_err(&pdev->dev, "failed to install dev->dec_irq 
> > %d (%d)",
> > +                       dev->dec_irq, ret);
> > +               return ret;
> > +       }
> > +
> > +       ret = mtk_vcodec_init_dec_pm(pdev, &dev->pm);
> > +       if (ret < 0) {
> > +               dev_err(&pdev->dev, "failed to get mt vcodec clock
> > source");
> > +               return ret;
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> >  static int fops_vcodec_open(struct file *file)
> >  {
> >         struct mtk_vcodec_dev *dev = video_drvdata(file);
> > @@ -116,6 +173,12 @@ static int fops_vcodec_open(struct file *file)
> >         init_waitqueue_head(&ctx->queue);
> >         mutex_init(&ctx->lock);
> > 
> > +       ret = mtk_vcodec_subdev_device_check(dev);
> > +       if (ret) {
> > +               mtk_v4l2_err("Failed to check vdec comp device.");
> 
> basic question: what is "comp" device?
> 
This line can be removed, "comp" is subdev.
> > +               goto err_ctrls_setup;
> > +       }
> > +
> >         ctx->type = MTK_INST_DECODER;
> >         ret = dev->vdec_pdata->ctrls_setup(ctx);
> >         if (ret) {
> > @@ -220,7 +283,6 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >  {
> >         struct mtk_vcodec_dev *dev;
> >         struct video_device *vfd_dec;
> > -       struct resource *res;
> >         phandle rproc_phandle;
> >         enum mtk_vcodec_fw_type fw_type;
> >         int ret;
> > @@ -249,32 +311,10 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >         if (IS_ERR(dev->fw_handler))
> >                 return PTR_ERR(dev->fw_handler);
> > 
> > -       ret = mtk_vcodec_init_dec_pm(dev->plat_dev, &dev->pm);
> > -       if (ret < 0) {
> > -               dev_err(&pdev->dev, "Failed to get mt vcodec clock
> > source");
> > -               goto err_dec_pm;
> > -       }
> > -
> > -       ret = mtk_vcodec_get_reg_bases(dev);
> > -       if (ret)
> > -               goto err_res;
> > -
> > -       res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
> > -       if (res == NULL) {
> > -               dev_err(&pdev->dev, "failed to get irq resource");
> > -               ret = -ENOENT;
> > -               goto err_res;
> > -       }
> > -
> > -       dev->dec_irq = platform_get_irq(pdev, 0);
> > -       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > -       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > -                       mtk_vcodec_dec_irq_handler, 0, pdev->name,
> > dev);
> > +       ret = mtk_vcodec_init_dec_resources(dev);
> >         if (ret) {
> > -               dev_err(&pdev->dev, "Failed to install dev->dec_irq 
> > %d (%d)",
> > -                       dev->dec_irq,
> > -                       ret);
> > -               goto err_res;
> > +               dev_err(&pdev->dev, "Failed to init dec
> > resources");
> > +               goto err_dec_pm;
> >         }
> > 
> >         mutex_init(&dev->dec_mutex);
> > @@ -329,6 +369,15 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >                 goto err_event_workq;
> >         }
> > 
> > +       if (dev->vdec_pdata->is_subdev_supported) {
> > +               ret = of_platform_populate(pdev->dev.of_node, NULL,
> > NULL,
> > +                       &pdev->dev);
> > +               if (ret) {
> > +                       mtk_v4l2_err("Main device
> > of_platform_populate failed.");
> > +                       goto err_event_workq;
> > +               }
> > +       }
> > +
> >         if (dev->vdec_pdata->uses_stateless_api) {
> >                 dev->mdev_dec.dev = &pdev->dev;
> >                 strscpy(dev->mdev_dec.model, MTK_VCODEC_DEC_NAME,
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > new file mode 100644
> > index 000000000000..025a103008ba
> > --- /dev/null
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > @@ -0,0 +1,172 @@
> > +// SPDX-License-Identifier: GPL-2.0
> > +/*
> > + * Copyright (c) 2021 MediaTek Inc.
> > + * Author: Yunfei Dong <yunfei.dong@mediatek.com>
> > + */
> > +
> > +#include <linux/interrupt.h>
> > +#include <linux/irq.h>
> > +#include <linux/module.h>
> > +#include <linux/of.h>
> > +#include <linux/of_device.h>
> > +#include <linux/slab.h>
> > +
> > +#include "mtk_vcodec_drv.h"
> > +#include "mtk_vcodec_dec.h"
> > +#include "mtk_vcodec_dec_hw.h"
> > +#include "mtk_vcodec_dec_pm.h"
> > +#include "mtk_vcodec_intr.h"
> > +#include "mtk_vcodec_util.h"
> > +
> > +const struct of_device_id mtk_vdec_hw_match[] = {
> > +       {
> > +               .compatible = "mediatek,mtk-vcodec-lat",
> > +               .data = (void *)MTK_VDEC_LAT0,
> > +       },
> 
> Are we supposed to have LAT1 here as well?
At current period, need not to add it. LAT1 may be used in the future.
> 
> > +       {
> > +               .compatible = "mediatek,mtk-vcodec-core",
> > +               .data = (void *)MTK_VDEC_CORE,
> > +       },
> > +       {},
> > +};
> > +EXPORT_SYMBOL_GPL(mtk_vdec_hw_match);
> > +
> > +MODULE_DEVICE_TABLE(of, mtk_vdec_hw_match);
> > +
> > +static irqreturn_t mtk_vdec_hw_irq_handler(int irq, void *priv)
> > +{
> > +       struct mtk_vdec_hw_dev *dev = priv;
> > +       struct mtk_vcodec_ctx *ctx;
> > +       u32 cg_status;
> > +       unsigned int dec_done_status;
> > +       void __iomem *vdec_misc_addr = dev->reg_base[VDEC_HW_MISC]
> > +
> > +                                       VDEC_IRQ_CFG_REG;
> > +
> > +       ctx = mtk_vcodec_get_curr_ctx(dev->main_dev);
> > +
> > +       /* check if HW active or not */
> > +       cg_status = readl(dev->reg_base[VDEC_HW_SYS]);
> > +       if (cg_status & VDEC_HW_ACTIVE) {
> > +               mtk_v4l2_err("vdec active is not 0x0 (0x%08x)",
> > +                       cg_status);
> > +               return IRQ_HANDLED;
> > +       }
> > +
> > +       dec_done_status = readl(vdec_misc_addr);
> > +       if ((dec_done_status & MTK_VDEC_IRQ_STATUS_DEC_SUCCESS) !=
> > +           MTK_VDEC_IRQ_STATUS_DEC_SUCCESS)
> > +               return IRQ_HANDLED;
> > +
> > +       /* clear interrupt */
> > +       writel(dec_done_status | VDEC_IRQ_CFG, vdec_misc_addr);
> 
> Is VDEC_IRQ_CFG also intended for clear interrupt?
> 
VDEC_IRQ_CFG and VDEC_IRQ_CLR are different bits to clear interrupt.
> > +       writel(dec_done_status & ~VDEC_IRQ_CLR, vdec_misc_addr);
> > +
> > +       wake_up_ctx(ctx, MTK_INST_IRQ_RECEIVED);
> > +
> > +       mtk_v4l2_debug(3, "wake up ctx %d, dec_done_status=%x",
> > +               ctx->id, dec_done_status);
> > +
> > +       return IRQ_HANDLED;
> > +}
> > +
> > +static int mtk_vdec_hw_init_irq(struct mtk_vdec_hw_dev *dev)
> > +{
> > +       struct platform_device *pdev = dev->plat_dev;
> > +       int ret;
> > +
> > +       dev->dec_irq = platform_get_irq(pdev, 0);
> > +       if (dev->dec_irq < 0) {
> > +               dev_err(&pdev->dev, "Failed to get irq resource");
> > +               return dev->dec_irq;
> > +       }
> > +
> > +       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > +       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > +                               mtk_vdec_hw_irq_handler, 0, pdev-
> > >name, dev);
> > +       if (ret) {
> > +               dev_err(&pdev->dev, "Failed to install dev->dec_irq 
> > %d (%d)",
> > +                       dev->dec_irq, ret);
> > +               return ret;
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> > +static int mtk_vdec_hw_probe(struct platform_device *pdev)
> > +{
> > +       struct device *dev = &pdev->dev;
> > +       struct mtk_vdec_hw_dev *subdev_dev;
> > +       struct mtk_vcodec_dev *main_dev;
> > +       const struct of_device_id *of_id;
> > +       int hw_idx;
> > +       int ret;
> > +
> > +       if (!dev->parent) {
> > +               dev_err(dev, "no parent for hardware devices.\n");
> > +               return -ENODEV;
> > +       }
> > +
> > +       main_dev = dev_get_drvdata(dev->parent);
> > +       if (!main_dev) {
> > +               dev_err(dev, "failed to get parent driver data");
> > +               return -EINVAL;
> > +       }
> > +
> > +       subdev_dev = devm_kzalloc(dev, sizeof(*subdev_dev),
> > GFP_KERNEL);
> > +       if (!subdev_dev)
> > +               return -ENOMEM;
> > +
> > +       subdev_dev->plat_dev = pdev;
> > +       ret = mtk_vcodec_init_dec_pm(pdev, &subdev_dev->pm);
> > +       if (ret)
> > +               return ret;
> > +
> > +       subdev_dev->reg_base[VDEC_HW_MISC] =
> > +               devm_platform_ioremap_resource(pdev, 0);
> > +       if (IS_ERR((__force void *)subdev_dev-
> > >reg_base[VDEC_HW_MISC])) {
> > +               ret = PTR_ERR((__force void *)subdev_dev-
> > >reg_base[VDEC_HW_MISC]);
> > +               goto err;
> > +       }
> > +
> > +       ret = mtk_vdec_hw_init_irq(subdev_dev);
> > +       if (ret)
> > +               goto err;
> > +
> > +       of_id = of_match_device(mtk_vdec_hw_match, dev);
> > +       if (!of_id) {
> > +               dev_err(dev, "Can't get vdec subdev id.\n");
> > +               ret = -EINVAL;
> > +               goto err;
> > +       }
> > +
> > +       hw_idx = (enum mtk_vdec_hw_id)(uintptr_t)of_id->data;
> > +       if (hw_idx >= MTK_VDEC_HW_MAX) {
> > +               dev_err(dev, "Hardware index %d not correct.\n",
> > hw_idx);
> > +               ret = -EINVAL;
> > +               goto err;
> > +       }
> > +
> > +       main_dev->subdev_dev[hw_idx] = subdev_dev;
> > +       subdev_dev->hw_idx = hw_idx;
> > +       subdev_dev->main_dev = main_dev;
> > +       subdev_dev->reg_base[VDEC_HW_SYS] = main_dev-
> > >reg_base[VDEC_HW_SYS];
> > +       set_bit(subdev_dev->hw_idx, main_dev->subdev_bitmap);
> > +
> > +       platform_set_drvdata(pdev, subdev_dev);
> > +       return 0;
> > +err:
> > +       mtk_vcodec_release_dec_pm(&subdev_dev->pm);
> > +       return ret;
> > +}
> > +
> > +static struct platform_driver mtk_vdec_driver = {
> > +       .probe  = mtk_vdec_hw_probe,
> > +       .driver = {
> > +               .name   = "mtk-vdec-comp",
> > +               .of_match_table = mtk_vdec_hw_match,
> > +       },
> > +};
> > +module_platform_driver(mtk_vdec_driver);
> > +
> > +MODULE_LICENSE("GPL v2");
> > +MODULE_DESCRIPTION("Mediatek video decoder hardware driver");
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > new file mode 100644
> > index 000000000000..f7f36790629d
> > --- /dev/null
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > @@ -0,0 +1,51 @@
> > +// SPDX-License-Identifier: GPL-2.0
> > +/*
> > + * Copyright (c) 2021 MediaTek Inc.
> > + * Author: Yunfei Dong <yunfei.dong@mediatek.com>
> > + */
> > +
> > +#ifndef _MTK_VCODEC_DEC_HW_H_
> > +#define _MTK_VCODEC_DEC_HW_H_
> > +
> > +#include <linux/io.h>
> > +#include <linux/platform_device.h>
> > +
> > +#include "mtk_vcodec_drv.h"
> > +
> > +#define VDEC_HW_ACTIVE 0x10
> > +#define VDEC_IRQ_CFG 0x11
> > +#define VDEC_IRQ_CLR 0x10
> > +#define VDEC_IRQ_CFG_REG 0xa4
> > +
> > +extern const struct of_device_id
> > mtk_vdec_hw_match[MTK_VDEC_HW_MAX];
> > +
> > +/**
> > + * enum mtk_vdec_hw_reg_idx - subdev hardware register base index
> > + */
> > +enum mtk_vdec_hw_reg_idx {
> > +       VDEC_HW_SYS,
> > +       VDEC_HW_MISC,
> > +       VDEC_HW_MAX
> > +};
> > +
> > +/**
> > + * struct mtk_vdec_hw_dev - vdec hardware driver data
> > + * @plat_dev: platform device
> > + * @main_dev: main device
> > + * @reg_base: Mapped address of MTK Vcodec registers.
> > + *
> > + * @dec_irq: decoder irq resource
> > + * @pm: power management control
> > + * @hw_idx: each hardware index
> > + */
> > +struct mtk_vdec_hw_dev {
> > +       struct platform_device *plat_dev;
> > +       struct mtk_vcodec_dev *main_dev;
> > +       void __iomem *reg_base[VDEC_HW_MAX];
> > +
> > +       int dec_irq;
> > +       struct mtk_vcodec_pm pm;
> > +       int hw_idx;
> > +};
> > +
> > +#endif /* _MTK_VCODEC_DEC_HW_H_ */
> > diff --git a/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateful.c b/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateful.c
> > index bef49244e61b..c7f9259ad094 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateful.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateful.c
> > @@ -625,4 +625,5 @@ const struct mtk_vcodec_dec_pdata
> > mtk_vdec_8173_pdata = {
> >         .num_framesizes = NUM_SUPPORTED_FRAMESIZE,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = false,
> >  };
> > diff --git a/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateless.c b/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateless.c
> > index 26e4d6f4ec04..2d285515b625 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateless.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateless.c
> > @@ -357,6 +357,7 @@ const struct mtk_vcodec_dec_pdata
> > mtk_vdec_8183_pdata = {
> 
> 8173 might be only one for stateful. But we have multiple platforms
> for 8183. Shouldn't this name reflect this?
> 
This platform is used for mt8183, all mt8183 need to used it.
And 8173 is stateful, 8183 is stateless.
> >         .uses_stateless_api = true,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = false,
> >  };
> > 
> >  const struct mtk_vcodec_dec_pdata mtk_lat_sig_core_pdata = {
> > @@ -373,4 +374,5 @@ const struct mtk_vcodec_dec_pdata
> > mtk_lat_sig_core_pdata = {
> 
> Which platforms is this supposed to be used? Some comments on this
> might be helpful.
> 
Add in v13.

Thanks,
Yunfei Dong
> >         .uses_stateless_api = true,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = true,
> >  };
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > index 0fa9d85114b9..2160f26263ef 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > @@ -93,6 +93,17 @@ enum mtk_fmt_type {
> >         MTK_FMT_FRAME = 2,
> >  };
> > 
> > +/**
> > + * struct mtk_vdec_hw_id - Hardware index used to separate
> > + *                         different hardware
> > + */
> > +enum mtk_vdec_hw_id {
> > +       MTK_VDEC_CORE,
> > +       MTK_VDEC_LAT0,
> > +       MTK_VDEC_LAT1,
> > +       MTK_VDEC_HW_MAX,
> > +};
> > +
> >  /*
> >   * struct mtk_video_fmt - Structure used to store information
> > about pixelformats
> >   */
> > @@ -332,6 +343,7 @@ enum mtk_chip {
> >   *
> >   * @chip: chip this decoder is compatible with
> >   *
> > + * @is_subdev_supported: whether support parent-node
> > architecture(subdev)
> >   * @uses_stateless_api: whether the decoder uses the stateless API
> > with requests
> >   */
> > 
> > @@ -353,6 +365,7 @@ struct mtk_vcodec_dec_pdata {
> > 
> >         enum mtk_chip chip;
> > 
> > +       bool is_subdev_supported;
> >         bool uses_stateless_api;
> >  };
> > 
> > @@ -423,6 +436,9 @@ struct mtk_vcodec_enc_pdata {
> >   * @pm: power management control
> >   * @dec_capability: used to identify decode capability, ex: 4k
> >   * @enc_capability: used to identify encode capability
> > + *
> > + * @subdev_dev: subdev hardware device
> > + * @subdev_bitmap: used to record hardware is ready or not
> >   */
> >  struct mtk_vcodec_dev {
> >         struct v4l2_device v4l2_dev;
> > @@ -460,6 +476,9 @@ struct mtk_vcodec_dev {
> >         struct mtk_vcodec_pm pm;
> >         unsigned int dec_capability;
> >         unsigned int enc_capability;
> > +
> > +       void *subdev_dev[MTK_VDEC_HW_MAX];
> > +       DECLARE_BITMAP(subdev_bitmap, MTK_VDEC_HW_MAX);
> >  };
> > 
> >  static inline struct mtk_vcodec_ctx *fh_to_ctx(struct v4l2_fh *fh)
> > --
> > 2.25.1
> > 

WARNING: multiple messages have this Message-ID (diff)
From: "yunfei.dong@mediatek.com" <yunfei.dong@mediatek.com>
To: Steve Cho <stevecho@chromium.org>
Cc: Alexandre Courbot <acourbot@chromium.org>,
	Hans Verkuil <hverkuil-cisco@xs4all.nl>,
	Tzung-Bi Shih <tzungbi@chromium.org>,
	"Tiffany Lin" <tiffany.lin@mediatek.com>,
	Andrew-CT Chen <andrew-ct.chen@mediatek.com>,
	Mauro Carvalho Chehab <mchehab@kernel.org>,
	Rob Herring <robh+dt@kernel.org>,
	Matthias Brugger <matthias.bgg@gmail.com>,
	Tomasz Figa <tfiga@google.com>,
	Hsin-Yi Wang <hsinyi@chromium.org>,
	"Fritz Koenig" <frkoenig@chromium.org>,
	Dafna Hirschfeld <dafna.hirschfeld@collabora.com>,
	Benjamin Gaignard <benjamin.gaignard@collabora.com>,
	Daniel Vetter <daniel@ffwll.ch>,
	dri-devel <dri-devel@lists.freedesktop.org>,
	Irui Wang <irui.wang@mediatek.com>,
	AngeloGioacchino Del Regno
	<angelogioacchino.delregno@collabora.com>,
	<linux-media@vger.kernel.org>, <devicetree@vger.kernel.org>,
	<linux-kernel@vger.kernel.org>,
	<linux-arm-kernel@lists.infradead.org>,
	<srv_heupstream@mediatek.com>,
	<linux-mediatek@lists.infradead.org>,
	<Project_Global_Chrome_Upstream_Group@mediatek.com>
Subject: Re: [PATCH v12, 06/19] media: mtk-vcodec: Add to support multi hardware decode
Date: Mon, 13 Dec 2021 16:40:47 +0800	[thread overview]
Message-ID: <87c70b080162b63454609d647ff1e2b39d06d406.camel@mediatek.com> (raw)
In-Reply-To: <CAC-pXoOCa+13u--9sCKDorh+RbuhL89Ajur1t8bSx6w=MgoyYg@mail.gmail.com>

Hi Steve,

Thanks for your suggestion,
On Thu, 2021-12-09 at 15:29 -0800, Steve Cho wrote:
> Few comments and questions.
> 
> On Wed, Dec 1, 2021 at 7:46 PM Yunfei Dong <yunfei.dong@mediatek.com>
> wrote:
> > 
> > There are more than two hardwares for decoder: LAT0, LAT1 and CORE.
> > In order to
> > manage these hardwares, register each hardware as independent
> > platform device
> > for the larbs are different.
> 
> basic question: what is "larbs"?
> 
Can regard larbs as some hardware which decode depends on.
> > 
> > Each hardware module controls its own information which includes
> > interrupt/power/
> > clocks/registers.
> > 
> > Calling of_platform_populate in parent device, and use
> > subdev_bitmap to record
> > whether the hardwares are registered done.
> 
> nit: s/registered done/registered/ ?
> 
Fix in v13.
> > 
> > Signed-off-by: Yunfei Dong <yunfei.dong@mediatek.com>
> > ---
> >  drivers/media/platform/mtk-vcodec/Makefile    |   5 +-
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_drv.c  | 111 +++++++----
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_hw.c   | 172
> > ++++++++++++++++++
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_hw.h   |  51 ++++++
> >  .../mtk-vcodec/mtk_vcodec_dec_stateful.c      |   1 +
> >  .../mtk-vcodec/mtk_vcodec_dec_stateless.c     |   2 +
> >  .../platform/mtk-vcodec/mtk_vcodec_drv.h      |  19 ++
> >  7 files changed, 329 insertions(+), 32 deletions(-)
> >  create mode 100644 drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_hw.c
> >  create mode 100644 drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_hw.h
> > 
> > diff --git a/drivers/media/platform/mtk-vcodec/Makefile
> > b/drivers/media/platform/mtk-vcodec/Makefile
> > index ca8e9e7a9c4e..c61bfb179bcc 100644
> > --- a/drivers/media/platform/mtk-vcodec/Makefile
> > +++ b/drivers/media/platform/mtk-vcodec/Makefile
> > @@ -2,7 +2,8 @@
> > 
> >  obj-$(CONFIG_VIDEO_MEDIATEK_VCODEC) += mtk-vcodec-dec.o \
> >                                        mtk-vcodec-enc.o \
> > -                                      mtk-vcodec-common.o
> > +                                      mtk-vcodec-common.o \
> > +                                      mtk-vcodec-dec-hw.o
> > 
> >  mtk-vcodec-dec-y := vdec/vdec_h264_if.o \
> >                 vdec/vdec_vp8_if.o \
> > @@ -16,6 +17,8 @@ mtk-vcodec-dec-y := vdec/vdec_h264_if.o \
> >                 mtk_vcodec_dec_stateless.o \
> >                 mtk_vcodec_dec_pm.o \
> > 
> > +mtk-vcodec-dec-hw-y := mtk_vcodec_dec_hw.o
> > +
> >  mtk-vcodec-enc-y := venc/venc_vp8_if.o \
> >                 venc/venc_h264_if.o \
> >                 mtk_vcodec_enc.o \
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c 
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > index b7a51e96d4ba..95fbe9be3f6d 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > @@ -18,19 +18,40 @@
> > 
> >  #include "mtk_vcodec_drv.h"
> >  #include "mtk_vcodec_dec.h"
> > +#include "mtk_vcodec_dec_hw.h"
> >  #include "mtk_vcodec_dec_pm.h"
> >  #include "mtk_vcodec_intr.h"
> >  #include "mtk_vcodec_util.h"
> >  #include "mtk_vcodec_fw.h"
> > 
> > -#define VDEC_HW_ACTIVE 0x10
> > -#define VDEC_IRQ_CFG   0x11
> > -#define VDEC_IRQ_CLR   0x10
> > -#define VDEC_IRQ_CFG_REG       0xa4
> > -
> >  module_param(mtk_v4l2_dbg_level, int, 0644);
> >  module_param(mtk_vcodec_dbg, bool, 0644);
> > 
> > +static int mtk_vcodec_subdev_device_check(struct mtk_vcodec_dev
> > *vdec_dev)
> > +{
> > +       struct platform_device *pdev = vdec_dev->plat_dev;
> > +       struct device_node *subdev_node;
> > +       enum mtk_vdec_hw_id hw_idx;
> > +       const struct of_device_id *of_id;
> > +       int i;
> > +
> > +       for (i = 0; i < ARRAY_SIZE(mtk_vdec_hw_match); i++) {
> > +               of_id = &mtk_vdec_hw_match[i];
> > +               subdev_node = of_find_compatible_node(NULL, NULL,
> > +                       of_id->compatible);
> > +               if (!subdev_node)
> > +                       continue;
> > +
> > +               hw_idx = (enum mtk_vdec_hw_id)(uintptr_t)of_id-
> > >data;
> > +               if (!test_bit(hw_idx, vdec_dev->subdev_bitmap)) {
> > +                       dev_err(&pdev->dev, "Vdec %d is not ready",
> > hw_idx);
> > +                       return -EAGAIN;
> > +               }
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> >  static irqreturn_t mtk_vcodec_dec_irq_handler(int irq, void *priv)
> >  {
> >         struct mtk_vcodec_dev *dev = priv;
> > @@ -95,6 +116,42 @@ static int mtk_vcodec_get_reg_bases(struct
> > mtk_vcodec_dev *dev)
> >         return 0;
> >  }
> > 
> > +static int mtk_vcodec_init_dec_resources(struct mtk_vcodec_dev
> > *dev)
> > +{
> > +       struct platform_device *pdev = dev->plat_dev;
> > +       int ret;
> > +
> > +       ret = mtk_vcodec_get_reg_bases(dev);
> > +       if (ret)
> > +               return ret;
> > +
> > +       if (dev->vdec_pdata->is_subdev_supported)
> > +               return 0;
> > +
> > +       dev->dec_irq = platform_get_irq(pdev, 0);
> > +       if (dev->dec_irq < 0) {
> > +               dev_err(&pdev->dev, "failed to get irq number");
> > +               return dev->dec_irq;
> > +       }
> > +
> > +       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > +       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > +               mtk_vcodec_dec_irq_handler, 0, pdev->name, dev);
> > +       if (ret) {
> > +               dev_err(&pdev->dev, "failed to install dev->dec_irq 
> > %d (%d)",
> > +                       dev->dec_irq, ret);
> > +               return ret;
> > +       }
> > +
> > +       ret = mtk_vcodec_init_dec_pm(pdev, &dev->pm);
> > +       if (ret < 0) {
> > +               dev_err(&pdev->dev, "failed to get mt vcodec clock
> > source");
> > +               return ret;
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> >  static int fops_vcodec_open(struct file *file)
> >  {
> >         struct mtk_vcodec_dev *dev = video_drvdata(file);
> > @@ -116,6 +173,12 @@ static int fops_vcodec_open(struct file *file)
> >         init_waitqueue_head(&ctx->queue);
> >         mutex_init(&ctx->lock);
> > 
> > +       ret = mtk_vcodec_subdev_device_check(dev);
> > +       if (ret) {
> > +               mtk_v4l2_err("Failed to check vdec comp device.");
> 
> basic question: what is "comp" device?
> 
This line can be removed, "comp" is subdev.
> > +               goto err_ctrls_setup;
> > +       }
> > +
> >         ctx->type = MTK_INST_DECODER;
> >         ret = dev->vdec_pdata->ctrls_setup(ctx);
> >         if (ret) {
> > @@ -220,7 +283,6 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >  {
> >         struct mtk_vcodec_dev *dev;
> >         struct video_device *vfd_dec;
> > -       struct resource *res;
> >         phandle rproc_phandle;
> >         enum mtk_vcodec_fw_type fw_type;
> >         int ret;
> > @@ -249,32 +311,10 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >         if (IS_ERR(dev->fw_handler))
> >                 return PTR_ERR(dev->fw_handler);
> > 
> > -       ret = mtk_vcodec_init_dec_pm(dev->plat_dev, &dev->pm);
> > -       if (ret < 0) {
> > -               dev_err(&pdev->dev, "Failed to get mt vcodec clock
> > source");
> > -               goto err_dec_pm;
> > -       }
> > -
> > -       ret = mtk_vcodec_get_reg_bases(dev);
> > -       if (ret)
> > -               goto err_res;
> > -
> > -       res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
> > -       if (res == NULL) {
> > -               dev_err(&pdev->dev, "failed to get irq resource");
> > -               ret = -ENOENT;
> > -               goto err_res;
> > -       }
> > -
> > -       dev->dec_irq = platform_get_irq(pdev, 0);
> > -       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > -       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > -                       mtk_vcodec_dec_irq_handler, 0, pdev->name,
> > dev);
> > +       ret = mtk_vcodec_init_dec_resources(dev);
> >         if (ret) {
> > -               dev_err(&pdev->dev, "Failed to install dev->dec_irq 
> > %d (%d)",
> > -                       dev->dec_irq,
> > -                       ret);
> > -               goto err_res;
> > +               dev_err(&pdev->dev, "Failed to init dec
> > resources");
> > +               goto err_dec_pm;
> >         }
> > 
> >         mutex_init(&dev->dec_mutex);
> > @@ -329,6 +369,15 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >                 goto err_event_workq;
> >         }
> > 
> > +       if (dev->vdec_pdata->is_subdev_supported) {
> > +               ret = of_platform_populate(pdev->dev.of_node, NULL,
> > NULL,
> > +                       &pdev->dev);
> > +               if (ret) {
> > +                       mtk_v4l2_err("Main device
> > of_platform_populate failed.");
> > +                       goto err_event_workq;
> > +               }
> > +       }
> > +
> >         if (dev->vdec_pdata->uses_stateless_api) {
> >                 dev->mdev_dec.dev = &pdev->dev;
> >                 strscpy(dev->mdev_dec.model, MTK_VCODEC_DEC_NAME,
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > new file mode 100644
> > index 000000000000..025a103008ba
> > --- /dev/null
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > @@ -0,0 +1,172 @@
> > +// SPDX-License-Identifier: GPL-2.0
> > +/*
> > + * Copyright (c) 2021 MediaTek Inc.
> > + * Author: Yunfei Dong <yunfei.dong@mediatek.com>
> > + */
> > +
> > +#include <linux/interrupt.h>
> > +#include <linux/irq.h>
> > +#include <linux/module.h>
> > +#include <linux/of.h>
> > +#include <linux/of_device.h>
> > +#include <linux/slab.h>
> > +
> > +#include "mtk_vcodec_drv.h"
> > +#include "mtk_vcodec_dec.h"
> > +#include "mtk_vcodec_dec_hw.h"
> > +#include "mtk_vcodec_dec_pm.h"
> > +#include "mtk_vcodec_intr.h"
> > +#include "mtk_vcodec_util.h"
> > +
> > +const struct of_device_id mtk_vdec_hw_match[] = {
> > +       {
> > +               .compatible = "mediatek,mtk-vcodec-lat",
> > +               .data = (void *)MTK_VDEC_LAT0,
> > +       },
> 
> Are we supposed to have LAT1 here as well?
At current period, need not to add it. LAT1 may be used in the future.
> 
> > +       {
> > +               .compatible = "mediatek,mtk-vcodec-core",
> > +               .data = (void *)MTK_VDEC_CORE,
> > +       },
> > +       {},
> > +};
> > +EXPORT_SYMBOL_GPL(mtk_vdec_hw_match);
> > +
> > +MODULE_DEVICE_TABLE(of, mtk_vdec_hw_match);
> > +
> > +static irqreturn_t mtk_vdec_hw_irq_handler(int irq, void *priv)
> > +{
> > +       struct mtk_vdec_hw_dev *dev = priv;
> > +       struct mtk_vcodec_ctx *ctx;
> > +       u32 cg_status;
> > +       unsigned int dec_done_status;
> > +       void __iomem *vdec_misc_addr = dev->reg_base[VDEC_HW_MISC]
> > +
> > +                                       VDEC_IRQ_CFG_REG;
> > +
> > +       ctx = mtk_vcodec_get_curr_ctx(dev->main_dev);
> > +
> > +       /* check if HW active or not */
> > +       cg_status = readl(dev->reg_base[VDEC_HW_SYS]);
> > +       if (cg_status & VDEC_HW_ACTIVE) {
> > +               mtk_v4l2_err("vdec active is not 0x0 (0x%08x)",
> > +                       cg_status);
> > +               return IRQ_HANDLED;
> > +       }
> > +
> > +       dec_done_status = readl(vdec_misc_addr);
> > +       if ((dec_done_status & MTK_VDEC_IRQ_STATUS_DEC_SUCCESS) !=
> > +           MTK_VDEC_IRQ_STATUS_DEC_SUCCESS)
> > +               return IRQ_HANDLED;
> > +
> > +       /* clear interrupt */
> > +       writel(dec_done_status | VDEC_IRQ_CFG, vdec_misc_addr);
> 
> Is VDEC_IRQ_CFG also intended for clear interrupt?
> 
VDEC_IRQ_CFG and VDEC_IRQ_CLR are different bits to clear interrupt.
> > +       writel(dec_done_status & ~VDEC_IRQ_CLR, vdec_misc_addr);
> > +
> > +       wake_up_ctx(ctx, MTK_INST_IRQ_RECEIVED);
> > +
> > +       mtk_v4l2_debug(3, "wake up ctx %d, dec_done_status=%x",
> > +               ctx->id, dec_done_status);
> > +
> > +       return IRQ_HANDLED;
> > +}
> > +
> > +static int mtk_vdec_hw_init_irq(struct mtk_vdec_hw_dev *dev)
> > +{
> > +       struct platform_device *pdev = dev->plat_dev;
> > +       int ret;
> > +
> > +       dev->dec_irq = platform_get_irq(pdev, 0);
> > +       if (dev->dec_irq < 0) {
> > +               dev_err(&pdev->dev, "Failed to get irq resource");
> > +               return dev->dec_irq;
> > +       }
> > +
> > +       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > +       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > +                               mtk_vdec_hw_irq_handler, 0, pdev-
> > >name, dev);
> > +       if (ret) {
> > +               dev_err(&pdev->dev, "Failed to install dev->dec_irq 
> > %d (%d)",
> > +                       dev->dec_irq, ret);
> > +               return ret;
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> > +static int mtk_vdec_hw_probe(struct platform_device *pdev)
> > +{
> > +       struct device *dev = &pdev->dev;
> > +       struct mtk_vdec_hw_dev *subdev_dev;
> > +       struct mtk_vcodec_dev *main_dev;
> > +       const struct of_device_id *of_id;
> > +       int hw_idx;
> > +       int ret;
> > +
> > +       if (!dev->parent) {
> > +               dev_err(dev, "no parent for hardware devices.\n");
> > +               return -ENODEV;
> > +       }
> > +
> > +       main_dev = dev_get_drvdata(dev->parent);
> > +       if (!main_dev) {
> > +               dev_err(dev, "failed to get parent driver data");
> > +               return -EINVAL;
> > +       }
> > +
> > +       subdev_dev = devm_kzalloc(dev, sizeof(*subdev_dev),
> > GFP_KERNEL);
> > +       if (!subdev_dev)
> > +               return -ENOMEM;
> > +
> > +       subdev_dev->plat_dev = pdev;
> > +       ret = mtk_vcodec_init_dec_pm(pdev, &subdev_dev->pm);
> > +       if (ret)
> > +               return ret;
> > +
> > +       subdev_dev->reg_base[VDEC_HW_MISC] =
> > +               devm_platform_ioremap_resource(pdev, 0);
> > +       if (IS_ERR((__force void *)subdev_dev-
> > >reg_base[VDEC_HW_MISC])) {
> > +               ret = PTR_ERR((__force void *)subdev_dev-
> > >reg_base[VDEC_HW_MISC]);
> > +               goto err;
> > +       }
> > +
> > +       ret = mtk_vdec_hw_init_irq(subdev_dev);
> > +       if (ret)
> > +               goto err;
> > +
> > +       of_id = of_match_device(mtk_vdec_hw_match, dev);
> > +       if (!of_id) {
> > +               dev_err(dev, "Can't get vdec subdev id.\n");
> > +               ret = -EINVAL;
> > +               goto err;
> > +       }
> > +
> > +       hw_idx = (enum mtk_vdec_hw_id)(uintptr_t)of_id->data;
> > +       if (hw_idx >= MTK_VDEC_HW_MAX) {
> > +               dev_err(dev, "Hardware index %d not correct.\n",
> > hw_idx);
> > +               ret = -EINVAL;
> > +               goto err;
> > +       }
> > +
> > +       main_dev->subdev_dev[hw_idx] = subdev_dev;
> > +       subdev_dev->hw_idx = hw_idx;
> > +       subdev_dev->main_dev = main_dev;
> > +       subdev_dev->reg_base[VDEC_HW_SYS] = main_dev-
> > >reg_base[VDEC_HW_SYS];
> > +       set_bit(subdev_dev->hw_idx, main_dev->subdev_bitmap);
> > +
> > +       platform_set_drvdata(pdev, subdev_dev);
> > +       return 0;
> > +err:
> > +       mtk_vcodec_release_dec_pm(&subdev_dev->pm);
> > +       return ret;
> > +}
> > +
> > +static struct platform_driver mtk_vdec_driver = {
> > +       .probe  = mtk_vdec_hw_probe,
> > +       .driver = {
> > +               .name   = "mtk-vdec-comp",
> > +               .of_match_table = mtk_vdec_hw_match,
> > +       },
> > +};
> > +module_platform_driver(mtk_vdec_driver);
> > +
> > +MODULE_LICENSE("GPL v2");
> > +MODULE_DESCRIPTION("Mediatek video decoder hardware driver");
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > new file mode 100644
> > index 000000000000..f7f36790629d
> > --- /dev/null
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > @@ -0,0 +1,51 @@
> > +// SPDX-License-Identifier: GPL-2.0
> > +/*
> > + * Copyright (c) 2021 MediaTek Inc.
> > + * Author: Yunfei Dong <yunfei.dong@mediatek.com>
> > + */
> > +
> > +#ifndef _MTK_VCODEC_DEC_HW_H_
> > +#define _MTK_VCODEC_DEC_HW_H_
> > +
> > +#include <linux/io.h>
> > +#include <linux/platform_device.h>
> > +
> > +#include "mtk_vcodec_drv.h"
> > +
> > +#define VDEC_HW_ACTIVE 0x10
> > +#define VDEC_IRQ_CFG 0x11
> > +#define VDEC_IRQ_CLR 0x10
> > +#define VDEC_IRQ_CFG_REG 0xa4
> > +
> > +extern const struct of_device_id
> > mtk_vdec_hw_match[MTK_VDEC_HW_MAX];
> > +
> > +/**
> > + * enum mtk_vdec_hw_reg_idx - subdev hardware register base index
> > + */
> > +enum mtk_vdec_hw_reg_idx {
> > +       VDEC_HW_SYS,
> > +       VDEC_HW_MISC,
> > +       VDEC_HW_MAX
> > +};
> > +
> > +/**
> > + * struct mtk_vdec_hw_dev - vdec hardware driver data
> > + * @plat_dev: platform device
> > + * @main_dev: main device
> > + * @reg_base: Mapped address of MTK Vcodec registers.
> > + *
> > + * @dec_irq: decoder irq resource
> > + * @pm: power management control
> > + * @hw_idx: each hardware index
> > + */
> > +struct mtk_vdec_hw_dev {
> > +       struct platform_device *plat_dev;
> > +       struct mtk_vcodec_dev *main_dev;
> > +       void __iomem *reg_base[VDEC_HW_MAX];
> > +
> > +       int dec_irq;
> > +       struct mtk_vcodec_pm pm;
> > +       int hw_idx;
> > +};
> > +
> > +#endif /* _MTK_VCODEC_DEC_HW_H_ */
> > diff --git a/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateful.c b/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateful.c
> > index bef49244e61b..c7f9259ad094 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateful.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateful.c
> > @@ -625,4 +625,5 @@ const struct mtk_vcodec_dec_pdata
> > mtk_vdec_8173_pdata = {
> >         .num_framesizes = NUM_SUPPORTED_FRAMESIZE,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = false,
> >  };
> > diff --git a/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateless.c b/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateless.c
> > index 26e4d6f4ec04..2d285515b625 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateless.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateless.c
> > @@ -357,6 +357,7 @@ const struct mtk_vcodec_dec_pdata
> > mtk_vdec_8183_pdata = {
> 
> 8173 might be only one for stateful. But we have multiple platforms
> for 8183. Shouldn't this name reflect this?
> 
This platform is used for mt8183, all mt8183 need to used it.
And 8173 is stateful, 8183 is stateless.
> >         .uses_stateless_api = true,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = false,
> >  };
> > 
> >  const struct mtk_vcodec_dec_pdata mtk_lat_sig_core_pdata = {
> > @@ -373,4 +374,5 @@ const struct mtk_vcodec_dec_pdata
> > mtk_lat_sig_core_pdata = {
> 
> Which platforms is this supposed to be used? Some comments on this
> might be helpful.
> 
Add in v13.

Thanks,
Yunfei Dong
> >         .uses_stateless_api = true,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = true,
> >  };
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > index 0fa9d85114b9..2160f26263ef 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > @@ -93,6 +93,17 @@ enum mtk_fmt_type {
> >         MTK_FMT_FRAME = 2,
> >  };
> > 
> > +/**
> > + * struct mtk_vdec_hw_id - Hardware index used to separate
> > + *                         different hardware
> > + */
> > +enum mtk_vdec_hw_id {
> > +       MTK_VDEC_CORE,
> > +       MTK_VDEC_LAT0,
> > +       MTK_VDEC_LAT1,
> > +       MTK_VDEC_HW_MAX,
> > +};
> > +
> >  /*
> >   * struct mtk_video_fmt - Structure used to store information
> > about pixelformats
> >   */
> > @@ -332,6 +343,7 @@ enum mtk_chip {
> >   *
> >   * @chip: chip this decoder is compatible with
> >   *
> > + * @is_subdev_supported: whether support parent-node
> > architecture(subdev)
> >   * @uses_stateless_api: whether the decoder uses the stateless API
> > with requests
> >   */
> > 
> > @@ -353,6 +365,7 @@ struct mtk_vcodec_dec_pdata {
> > 
> >         enum mtk_chip chip;
> > 
> > +       bool is_subdev_supported;
> >         bool uses_stateless_api;
> >  };
> > 
> > @@ -423,6 +436,9 @@ struct mtk_vcodec_enc_pdata {
> >   * @pm: power management control
> >   * @dec_capability: used to identify decode capability, ex: 4k
> >   * @enc_capability: used to identify encode capability
> > + *
> > + * @subdev_dev: subdev hardware device
> > + * @subdev_bitmap: used to record hardware is ready or not
> >   */
> >  struct mtk_vcodec_dev {
> >         struct v4l2_device v4l2_dev;
> > @@ -460,6 +476,9 @@ struct mtk_vcodec_dev {
> >         struct mtk_vcodec_pm pm;
> >         unsigned int dec_capability;
> >         unsigned int enc_capability;
> > +
> > +       void *subdev_dev[MTK_VDEC_HW_MAX];
> > +       DECLARE_BITMAP(subdev_bitmap, MTK_VDEC_HW_MAX);
> >  };
> > 
> >  static inline struct mtk_vcodec_ctx *fh_to_ctx(struct v4l2_fh *fh)
> > --
> > 2.25.1
> > 
_______________________________________________
Linux-mediatek mailing list
Linux-mediatek@lists.infradead.org
http://lists.infradead.org/mailman/listinfo/linux-mediatek

WARNING: multiple messages have this Message-ID (diff)
From: "yunfei.dong@mediatek.com" <yunfei.dong@mediatek.com>
To: Steve Cho <stevecho@chromium.org>
Cc: Alexandre Courbot <acourbot@chromium.org>,
	Hans Verkuil <hverkuil-cisco@xs4all.nl>,
	Tzung-Bi Shih <tzungbi@chromium.org>,
	"Tiffany Lin" <tiffany.lin@mediatek.com>,
	Andrew-CT Chen <andrew-ct.chen@mediatek.com>,
	Mauro Carvalho Chehab <mchehab@kernel.org>,
	Rob Herring <robh+dt@kernel.org>,
	Matthias Brugger <matthias.bgg@gmail.com>,
	Tomasz Figa <tfiga@google.com>,
	Hsin-Yi Wang <hsinyi@chromium.org>,
	"Fritz Koenig" <frkoenig@chromium.org>,
	Dafna Hirschfeld <dafna.hirschfeld@collabora.com>,
	Benjamin Gaignard <benjamin.gaignard@collabora.com>,
	Daniel Vetter <daniel@ffwll.ch>,
	dri-devel <dri-devel@lists.freedesktop.org>,
	Irui Wang <irui.wang@mediatek.com>,
	AngeloGioacchino Del Regno
	<angelogioacchino.delregno@collabora.com>,
	<linux-media@vger.kernel.org>, <devicetree@vger.kernel.org>,
	<linux-kernel@vger.kernel.org>,
	<linux-arm-kernel@lists.infradead.org>,
	<srv_heupstream@mediatek.com>,
	<linux-mediatek@lists.infradead.org>,
	<Project_Global_Chrome_Upstream_Group@mediatek.com>
Subject: Re: [PATCH v12, 06/19] media: mtk-vcodec: Add to support multi hardware decode
Date: Mon, 13 Dec 2021 16:40:47 +0800	[thread overview]
Message-ID: <87c70b080162b63454609d647ff1e2b39d06d406.camel@mediatek.com> (raw)
In-Reply-To: <CAC-pXoOCa+13u--9sCKDorh+RbuhL89Ajur1t8bSx6w=MgoyYg@mail.gmail.com>

Hi Steve,

Thanks for your suggestion,
On Thu, 2021-12-09 at 15:29 -0800, Steve Cho wrote:
> Few comments and questions.
> 
> On Wed, Dec 1, 2021 at 7:46 PM Yunfei Dong <yunfei.dong@mediatek.com>
> wrote:
> > 
> > There are more than two hardwares for decoder: LAT0, LAT1 and CORE.
> > In order to
> > manage these hardwares, register each hardware as independent
> > platform device
> > for the larbs are different.
> 
> basic question: what is "larbs"?
> 
Can regard larbs as some hardware which decode depends on.
> > 
> > Each hardware module controls its own information which includes
> > interrupt/power/
> > clocks/registers.
> > 
> > Calling of_platform_populate in parent device, and use
> > subdev_bitmap to record
> > whether the hardwares are registered done.
> 
> nit: s/registered done/registered/ ?
> 
Fix in v13.
> > 
> > Signed-off-by: Yunfei Dong <yunfei.dong@mediatek.com>
> > ---
> >  drivers/media/platform/mtk-vcodec/Makefile    |   5 +-
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_drv.c  | 111 +++++++----
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_hw.c   | 172
> > ++++++++++++++++++
> >  .../platform/mtk-vcodec/mtk_vcodec_dec_hw.h   |  51 ++++++
> >  .../mtk-vcodec/mtk_vcodec_dec_stateful.c      |   1 +
> >  .../mtk-vcodec/mtk_vcodec_dec_stateless.c     |   2 +
> >  .../platform/mtk-vcodec/mtk_vcodec_drv.h      |  19 ++
> >  7 files changed, 329 insertions(+), 32 deletions(-)
> >  create mode 100644 drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_hw.c
> >  create mode 100644 drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_hw.h
> > 
> > diff --git a/drivers/media/platform/mtk-vcodec/Makefile
> > b/drivers/media/platform/mtk-vcodec/Makefile
> > index ca8e9e7a9c4e..c61bfb179bcc 100644
> > --- a/drivers/media/platform/mtk-vcodec/Makefile
> > +++ b/drivers/media/platform/mtk-vcodec/Makefile
> > @@ -2,7 +2,8 @@
> > 
> >  obj-$(CONFIG_VIDEO_MEDIATEK_VCODEC) += mtk-vcodec-dec.o \
> >                                        mtk-vcodec-enc.o \
> > -                                      mtk-vcodec-common.o
> > +                                      mtk-vcodec-common.o \
> > +                                      mtk-vcodec-dec-hw.o
> > 
> >  mtk-vcodec-dec-y := vdec/vdec_h264_if.o \
> >                 vdec/vdec_vp8_if.o \
> > @@ -16,6 +17,8 @@ mtk-vcodec-dec-y := vdec/vdec_h264_if.o \
> >                 mtk_vcodec_dec_stateless.o \
> >                 mtk_vcodec_dec_pm.o \
> > 
> > +mtk-vcodec-dec-hw-y := mtk_vcodec_dec_hw.o
> > +
> >  mtk-vcodec-enc-y := venc/venc_vp8_if.o \
> >                 venc/venc_h264_if.o \
> >                 mtk_vcodec_enc.o \
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c 
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > index b7a51e96d4ba..95fbe9be3f6d 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_drv.c
> > @@ -18,19 +18,40 @@
> > 
> >  #include "mtk_vcodec_drv.h"
> >  #include "mtk_vcodec_dec.h"
> > +#include "mtk_vcodec_dec_hw.h"
> >  #include "mtk_vcodec_dec_pm.h"
> >  #include "mtk_vcodec_intr.h"
> >  #include "mtk_vcodec_util.h"
> >  #include "mtk_vcodec_fw.h"
> > 
> > -#define VDEC_HW_ACTIVE 0x10
> > -#define VDEC_IRQ_CFG   0x11
> > -#define VDEC_IRQ_CLR   0x10
> > -#define VDEC_IRQ_CFG_REG       0xa4
> > -
> >  module_param(mtk_v4l2_dbg_level, int, 0644);
> >  module_param(mtk_vcodec_dbg, bool, 0644);
> > 
> > +static int mtk_vcodec_subdev_device_check(struct mtk_vcodec_dev
> > *vdec_dev)
> > +{
> > +       struct platform_device *pdev = vdec_dev->plat_dev;
> > +       struct device_node *subdev_node;
> > +       enum mtk_vdec_hw_id hw_idx;
> > +       const struct of_device_id *of_id;
> > +       int i;
> > +
> > +       for (i = 0; i < ARRAY_SIZE(mtk_vdec_hw_match); i++) {
> > +               of_id = &mtk_vdec_hw_match[i];
> > +               subdev_node = of_find_compatible_node(NULL, NULL,
> > +                       of_id->compatible);
> > +               if (!subdev_node)
> > +                       continue;
> > +
> > +               hw_idx = (enum mtk_vdec_hw_id)(uintptr_t)of_id-
> > >data;
> > +               if (!test_bit(hw_idx, vdec_dev->subdev_bitmap)) {
> > +                       dev_err(&pdev->dev, "Vdec %d is not ready",
> > hw_idx);
> > +                       return -EAGAIN;
> > +               }
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> >  static irqreturn_t mtk_vcodec_dec_irq_handler(int irq, void *priv)
> >  {
> >         struct mtk_vcodec_dev *dev = priv;
> > @@ -95,6 +116,42 @@ static int mtk_vcodec_get_reg_bases(struct
> > mtk_vcodec_dev *dev)
> >         return 0;
> >  }
> > 
> > +static int mtk_vcodec_init_dec_resources(struct mtk_vcodec_dev
> > *dev)
> > +{
> > +       struct platform_device *pdev = dev->plat_dev;
> > +       int ret;
> > +
> > +       ret = mtk_vcodec_get_reg_bases(dev);
> > +       if (ret)
> > +               return ret;
> > +
> > +       if (dev->vdec_pdata->is_subdev_supported)
> > +               return 0;
> > +
> > +       dev->dec_irq = platform_get_irq(pdev, 0);
> > +       if (dev->dec_irq < 0) {
> > +               dev_err(&pdev->dev, "failed to get irq number");
> > +               return dev->dec_irq;
> > +       }
> > +
> > +       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > +       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > +               mtk_vcodec_dec_irq_handler, 0, pdev->name, dev);
> > +       if (ret) {
> > +               dev_err(&pdev->dev, "failed to install dev->dec_irq 
> > %d (%d)",
> > +                       dev->dec_irq, ret);
> > +               return ret;
> > +       }
> > +
> > +       ret = mtk_vcodec_init_dec_pm(pdev, &dev->pm);
> > +       if (ret < 0) {
> > +               dev_err(&pdev->dev, "failed to get mt vcodec clock
> > source");
> > +               return ret;
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> >  static int fops_vcodec_open(struct file *file)
> >  {
> >         struct mtk_vcodec_dev *dev = video_drvdata(file);
> > @@ -116,6 +173,12 @@ static int fops_vcodec_open(struct file *file)
> >         init_waitqueue_head(&ctx->queue);
> >         mutex_init(&ctx->lock);
> > 
> > +       ret = mtk_vcodec_subdev_device_check(dev);
> > +       if (ret) {
> > +               mtk_v4l2_err("Failed to check vdec comp device.");
> 
> basic question: what is "comp" device?
> 
This line can be removed, "comp" is subdev.
> > +               goto err_ctrls_setup;
> > +       }
> > +
> >         ctx->type = MTK_INST_DECODER;
> >         ret = dev->vdec_pdata->ctrls_setup(ctx);
> >         if (ret) {
> > @@ -220,7 +283,6 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >  {
> >         struct mtk_vcodec_dev *dev;
> >         struct video_device *vfd_dec;
> > -       struct resource *res;
> >         phandle rproc_phandle;
> >         enum mtk_vcodec_fw_type fw_type;
> >         int ret;
> > @@ -249,32 +311,10 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >         if (IS_ERR(dev->fw_handler))
> >                 return PTR_ERR(dev->fw_handler);
> > 
> > -       ret = mtk_vcodec_init_dec_pm(dev->plat_dev, &dev->pm);
> > -       if (ret < 0) {
> > -               dev_err(&pdev->dev, "Failed to get mt vcodec clock
> > source");
> > -               goto err_dec_pm;
> > -       }
> > -
> > -       ret = mtk_vcodec_get_reg_bases(dev);
> > -       if (ret)
> > -               goto err_res;
> > -
> > -       res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
> > -       if (res == NULL) {
> > -               dev_err(&pdev->dev, "failed to get irq resource");
> > -               ret = -ENOENT;
> > -               goto err_res;
> > -       }
> > -
> > -       dev->dec_irq = platform_get_irq(pdev, 0);
> > -       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > -       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > -                       mtk_vcodec_dec_irq_handler, 0, pdev->name,
> > dev);
> > +       ret = mtk_vcodec_init_dec_resources(dev);
> >         if (ret) {
> > -               dev_err(&pdev->dev, "Failed to install dev->dec_irq 
> > %d (%d)",
> > -                       dev->dec_irq,
> > -                       ret);
> > -               goto err_res;
> > +               dev_err(&pdev->dev, "Failed to init dec
> > resources");
> > +               goto err_dec_pm;
> >         }
> > 
> >         mutex_init(&dev->dec_mutex);
> > @@ -329,6 +369,15 @@ static int mtk_vcodec_probe(struct
> > platform_device *pdev)
> >                 goto err_event_workq;
> >         }
> > 
> > +       if (dev->vdec_pdata->is_subdev_supported) {
> > +               ret = of_platform_populate(pdev->dev.of_node, NULL,
> > NULL,
> > +                       &pdev->dev);
> > +               if (ret) {
> > +                       mtk_v4l2_err("Main device
> > of_platform_populate failed.");
> > +                       goto err_event_workq;
> > +               }
> > +       }
> > +
> >         if (dev->vdec_pdata->uses_stateless_api) {
> >                 dev->mdev_dec.dev = &pdev->dev;
> >                 strscpy(dev->mdev_dec.model, MTK_VCODEC_DEC_NAME,
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > new file mode 100644
> > index 000000000000..025a103008ba
> > --- /dev/null
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.c
> > @@ -0,0 +1,172 @@
> > +// SPDX-License-Identifier: GPL-2.0
> > +/*
> > + * Copyright (c) 2021 MediaTek Inc.
> > + * Author: Yunfei Dong <yunfei.dong@mediatek.com>
> > + */
> > +
> > +#include <linux/interrupt.h>
> > +#include <linux/irq.h>
> > +#include <linux/module.h>
> > +#include <linux/of.h>
> > +#include <linux/of_device.h>
> > +#include <linux/slab.h>
> > +
> > +#include "mtk_vcodec_drv.h"
> > +#include "mtk_vcodec_dec.h"
> > +#include "mtk_vcodec_dec_hw.h"
> > +#include "mtk_vcodec_dec_pm.h"
> > +#include "mtk_vcodec_intr.h"
> > +#include "mtk_vcodec_util.h"
> > +
> > +const struct of_device_id mtk_vdec_hw_match[] = {
> > +       {
> > +               .compatible = "mediatek,mtk-vcodec-lat",
> > +               .data = (void *)MTK_VDEC_LAT0,
> > +       },
> 
> Are we supposed to have LAT1 here as well?
At current period, need not to add it. LAT1 may be used in the future.
> 
> > +       {
> > +               .compatible = "mediatek,mtk-vcodec-core",
> > +               .data = (void *)MTK_VDEC_CORE,
> > +       },
> > +       {},
> > +};
> > +EXPORT_SYMBOL_GPL(mtk_vdec_hw_match);
> > +
> > +MODULE_DEVICE_TABLE(of, mtk_vdec_hw_match);
> > +
> > +static irqreturn_t mtk_vdec_hw_irq_handler(int irq, void *priv)
> > +{
> > +       struct mtk_vdec_hw_dev *dev = priv;
> > +       struct mtk_vcodec_ctx *ctx;
> > +       u32 cg_status;
> > +       unsigned int dec_done_status;
> > +       void __iomem *vdec_misc_addr = dev->reg_base[VDEC_HW_MISC]
> > +
> > +                                       VDEC_IRQ_CFG_REG;
> > +
> > +       ctx = mtk_vcodec_get_curr_ctx(dev->main_dev);
> > +
> > +       /* check if HW active or not */
> > +       cg_status = readl(dev->reg_base[VDEC_HW_SYS]);
> > +       if (cg_status & VDEC_HW_ACTIVE) {
> > +               mtk_v4l2_err("vdec active is not 0x0 (0x%08x)",
> > +                       cg_status);
> > +               return IRQ_HANDLED;
> > +       }
> > +
> > +       dec_done_status = readl(vdec_misc_addr);
> > +       if ((dec_done_status & MTK_VDEC_IRQ_STATUS_DEC_SUCCESS) !=
> > +           MTK_VDEC_IRQ_STATUS_DEC_SUCCESS)
> > +               return IRQ_HANDLED;
> > +
> > +       /* clear interrupt */
> > +       writel(dec_done_status | VDEC_IRQ_CFG, vdec_misc_addr);
> 
> Is VDEC_IRQ_CFG also intended for clear interrupt?
> 
VDEC_IRQ_CFG and VDEC_IRQ_CLR are different bits to clear interrupt.
> > +       writel(dec_done_status & ~VDEC_IRQ_CLR, vdec_misc_addr);
> > +
> > +       wake_up_ctx(ctx, MTK_INST_IRQ_RECEIVED);
> > +
> > +       mtk_v4l2_debug(3, "wake up ctx %d, dec_done_status=%x",
> > +               ctx->id, dec_done_status);
> > +
> > +       return IRQ_HANDLED;
> > +}
> > +
> > +static int mtk_vdec_hw_init_irq(struct mtk_vdec_hw_dev *dev)
> > +{
> > +       struct platform_device *pdev = dev->plat_dev;
> > +       int ret;
> > +
> > +       dev->dec_irq = platform_get_irq(pdev, 0);
> > +       if (dev->dec_irq < 0) {
> > +               dev_err(&pdev->dev, "Failed to get irq resource");
> > +               return dev->dec_irq;
> > +       }
> > +
> > +       irq_set_status_flags(dev->dec_irq, IRQ_NOAUTOEN);
> > +       ret = devm_request_irq(&pdev->dev, dev->dec_irq,
> > +                               mtk_vdec_hw_irq_handler, 0, pdev-
> > >name, dev);
> > +       if (ret) {
> > +               dev_err(&pdev->dev, "Failed to install dev->dec_irq 
> > %d (%d)",
> > +                       dev->dec_irq, ret);
> > +               return ret;
> > +       }
> > +
> > +       return 0;
> > +}
> > +
> > +static int mtk_vdec_hw_probe(struct platform_device *pdev)
> > +{
> > +       struct device *dev = &pdev->dev;
> > +       struct mtk_vdec_hw_dev *subdev_dev;
> > +       struct mtk_vcodec_dev *main_dev;
> > +       const struct of_device_id *of_id;
> > +       int hw_idx;
> > +       int ret;
> > +
> > +       if (!dev->parent) {
> > +               dev_err(dev, "no parent for hardware devices.\n");
> > +               return -ENODEV;
> > +       }
> > +
> > +       main_dev = dev_get_drvdata(dev->parent);
> > +       if (!main_dev) {
> > +               dev_err(dev, "failed to get parent driver data");
> > +               return -EINVAL;
> > +       }
> > +
> > +       subdev_dev = devm_kzalloc(dev, sizeof(*subdev_dev),
> > GFP_KERNEL);
> > +       if (!subdev_dev)
> > +               return -ENOMEM;
> > +
> > +       subdev_dev->plat_dev = pdev;
> > +       ret = mtk_vcodec_init_dec_pm(pdev, &subdev_dev->pm);
> > +       if (ret)
> > +               return ret;
> > +
> > +       subdev_dev->reg_base[VDEC_HW_MISC] =
> > +               devm_platform_ioremap_resource(pdev, 0);
> > +       if (IS_ERR((__force void *)subdev_dev-
> > >reg_base[VDEC_HW_MISC])) {
> > +               ret = PTR_ERR((__force void *)subdev_dev-
> > >reg_base[VDEC_HW_MISC]);
> > +               goto err;
> > +       }
> > +
> > +       ret = mtk_vdec_hw_init_irq(subdev_dev);
> > +       if (ret)
> > +               goto err;
> > +
> > +       of_id = of_match_device(mtk_vdec_hw_match, dev);
> > +       if (!of_id) {
> > +               dev_err(dev, "Can't get vdec subdev id.\n");
> > +               ret = -EINVAL;
> > +               goto err;
> > +       }
> > +
> > +       hw_idx = (enum mtk_vdec_hw_id)(uintptr_t)of_id->data;
> > +       if (hw_idx >= MTK_VDEC_HW_MAX) {
> > +               dev_err(dev, "Hardware index %d not correct.\n",
> > hw_idx);
> > +               ret = -EINVAL;
> > +               goto err;
> > +       }
> > +
> > +       main_dev->subdev_dev[hw_idx] = subdev_dev;
> > +       subdev_dev->hw_idx = hw_idx;
> > +       subdev_dev->main_dev = main_dev;
> > +       subdev_dev->reg_base[VDEC_HW_SYS] = main_dev-
> > >reg_base[VDEC_HW_SYS];
> > +       set_bit(subdev_dev->hw_idx, main_dev->subdev_bitmap);
> > +
> > +       platform_set_drvdata(pdev, subdev_dev);
> > +       return 0;
> > +err:
> > +       mtk_vcodec_release_dec_pm(&subdev_dev->pm);
> > +       return ret;
> > +}
> > +
> > +static struct platform_driver mtk_vdec_driver = {
> > +       .probe  = mtk_vdec_hw_probe,
> > +       .driver = {
> > +               .name   = "mtk-vdec-comp",
> > +               .of_match_table = mtk_vdec_hw_match,
> > +       },
> > +};
> > +module_platform_driver(mtk_vdec_driver);
> > +
> > +MODULE_LICENSE("GPL v2");
> > +MODULE_DESCRIPTION("Mediatek video decoder hardware driver");
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > new file mode 100644
> > index 000000000000..f7f36790629d
> > --- /dev/null
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_hw.h
> > @@ -0,0 +1,51 @@
> > +// SPDX-License-Identifier: GPL-2.0
> > +/*
> > + * Copyright (c) 2021 MediaTek Inc.
> > + * Author: Yunfei Dong <yunfei.dong@mediatek.com>
> > + */
> > +
> > +#ifndef _MTK_VCODEC_DEC_HW_H_
> > +#define _MTK_VCODEC_DEC_HW_H_
> > +
> > +#include <linux/io.h>
> > +#include <linux/platform_device.h>
> > +
> > +#include "mtk_vcodec_drv.h"
> > +
> > +#define VDEC_HW_ACTIVE 0x10
> > +#define VDEC_IRQ_CFG 0x11
> > +#define VDEC_IRQ_CLR 0x10
> > +#define VDEC_IRQ_CFG_REG 0xa4
> > +
> > +extern const struct of_device_id
> > mtk_vdec_hw_match[MTK_VDEC_HW_MAX];
> > +
> > +/**
> > + * enum mtk_vdec_hw_reg_idx - subdev hardware register base index
> > + */
> > +enum mtk_vdec_hw_reg_idx {
> > +       VDEC_HW_SYS,
> > +       VDEC_HW_MISC,
> > +       VDEC_HW_MAX
> > +};
> > +
> > +/**
> > + * struct mtk_vdec_hw_dev - vdec hardware driver data
> > + * @plat_dev: platform device
> > + * @main_dev: main device
> > + * @reg_base: Mapped address of MTK Vcodec registers.
> > + *
> > + * @dec_irq: decoder irq resource
> > + * @pm: power management control
> > + * @hw_idx: each hardware index
> > + */
> > +struct mtk_vdec_hw_dev {
> > +       struct platform_device *plat_dev;
> > +       struct mtk_vcodec_dev *main_dev;
> > +       void __iomem *reg_base[VDEC_HW_MAX];
> > +
> > +       int dec_irq;
> > +       struct mtk_vcodec_pm pm;
> > +       int hw_idx;
> > +};
> > +
> > +#endif /* _MTK_VCODEC_DEC_HW_H_ */
> > diff --git a/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateful.c b/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateful.c
> > index bef49244e61b..c7f9259ad094 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateful.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateful.c
> > @@ -625,4 +625,5 @@ const struct mtk_vcodec_dec_pdata
> > mtk_vdec_8173_pdata = {
> >         .num_framesizes = NUM_SUPPORTED_FRAMESIZE,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = false,
> >  };
> > diff --git a/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateless.c b/drivers/media/platform/mtk-
> > vcodec/mtk_vcodec_dec_stateless.c
> > index 26e4d6f4ec04..2d285515b625 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateless.c
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_dec_stateless.c
> > @@ -357,6 +357,7 @@ const struct mtk_vcodec_dec_pdata
> > mtk_vdec_8183_pdata = {
> 
> 8173 might be only one for stateful. But we have multiple platforms
> for 8183. Shouldn't this name reflect this?
> 
This platform is used for mt8183, all mt8183 need to used it.
And 8173 is stateful, 8183 is stateless.
> >         .uses_stateless_api = true,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = false,
> >  };
> > 
> >  const struct mtk_vcodec_dec_pdata mtk_lat_sig_core_pdata = {
> > @@ -373,4 +374,5 @@ const struct mtk_vcodec_dec_pdata
> > mtk_lat_sig_core_pdata = {
> 
> Which platforms is this supposed to be used? Some comments on this
> might be helpful.
> 
Add in v13.

Thanks,
Yunfei Dong
> >         .uses_stateless_api = true,
> >         .worker = mtk_vdec_worker,
> >         .flush_decoder = mtk_vdec_flush_decoder,
> > +       .is_subdev_supported = true,
> >  };
> > diff --git a/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > b/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > index 0fa9d85114b9..2160f26263ef 100644
> > --- a/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > +++ b/drivers/media/platform/mtk-vcodec/mtk_vcodec_drv.h
> > @@ -93,6 +93,17 @@ enum mtk_fmt_type {
> >         MTK_FMT_FRAME = 2,
> >  };
> > 
> > +/**
> > + * struct mtk_vdec_hw_id - Hardware index used to separate
> > + *                         different hardware
> > + */
> > +enum mtk_vdec_hw_id {
> > +       MTK_VDEC_CORE,
> > +       MTK_VDEC_LAT0,
> > +       MTK_VDEC_LAT1,
> > +       MTK_VDEC_HW_MAX,
> > +};
> > +
> >  /*
> >   * struct mtk_video_fmt - Structure used to store information
> > about pixelformats
> >   */
> > @@ -332,6 +343,7 @@ enum mtk_chip {
> >   *
> >   * @chip: chip this decoder is compatible with
> >   *
> > + * @is_subdev_supported: whether support parent-node
> > architecture(subdev)
> >   * @uses_stateless_api: whether the decoder uses the stateless API
> > with requests
> >   */
> > 
> > @@ -353,6 +365,7 @@ struct mtk_vcodec_dec_pdata {
> > 
> >         enum mtk_chip chip;
> > 
> > +       bool is_subdev_supported;
> >         bool uses_stateless_api;
> >  };
> > 
> > @@ -423,6 +436,9 @@ struct mtk_vcodec_enc_pdata {
> >   * @pm: power management control
> >   * @dec_capability: used to identify decode capability, ex: 4k
> >   * @enc_capability: used to identify encode capability
> > + *
> > + * @subdev_dev: subdev hardware device
> > + * @subdev_bitmap: used to record hardware is ready or not
> >   */
> >  struct mtk_vcodec_dev {
> >         struct v4l2_device v4l2_dev;
> > @@ -460,6 +476,9 @@ struct mtk_vcodec_dev {
> >         struct mtk_vcodec_pm pm;
> >         unsigned int dec_capability;
> >         unsigned int enc_capability;
> > +
> > +       void *subdev_dev[MTK_VDEC_HW_MAX];
> > +       DECLARE_BITMAP(subdev_bitmap, MTK_VDEC_HW_MAX);
> >  };
> > 
> >  static inline struct mtk_vcodec_ctx *fh_to_ctx(struct v4l2_fh *fh)
> > --
> > 2.25.1
> > 
_______________________________________________
linux-arm-kernel mailing list
linux-arm-kernel@lists.infradead.org
http://lists.infradead.org/mailman/listinfo/linux-arm-kernel

  reply	other threads:[~2021-12-13  8:40 UTC|newest]

Thread overview: 116+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2021-12-02  3:45 [PATCH v12, 00/19] Support multi hardware decode using of_platform_populate Yunfei Dong
2021-12-02  3:45 ` Yunfei Dong
2021-12-02  3:45 ` Yunfei Dong
2021-12-02  3:45 ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 01/19] media: mtk-vcodec: Get numbers of register bases from DT Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 02/19] media: mtk-vcodec: Align vcodec wake up interrupt interface Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 03/19] media: mtk-vcodec: Refactor vcodec pm interface Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 04/19] media: mtk-vcodec: export decoder pm functions Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 05/19] media: mtk-vcodec: Support MT8192 Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 06/19] media: mtk-vcodec: Add to support multi hardware decode Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-09 23:29   ` Steve Cho
2021-12-09 23:29     ` Steve Cho
2021-12-09 23:29     ` Steve Cho
2021-12-09 23:29     ` Steve Cho
2021-12-13  8:40     ` yunfei.dong [this message]
2021-12-13  8:40       ` yunfei.dong
2021-12-13  8:40       ` yunfei.dong
2021-12-13  8:40       ` yunfei.dong
2021-12-13 23:55       ` Steve Cho
2021-12-13 23:55         ` Steve Cho
2021-12-13 23:55         ` Steve Cho
2021-12-13 23:55         ` Steve Cho
2021-12-02  3:45 ` [PATCH v12, 07/19] dt-bindings: media: mtk-vcodec: Separate video encoder and decoder dt-bindings Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 08/19] media: mtk-vcodec: Use pure single core for MT8183 Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 09/19] media: mtk-vcodec: Add irq interface for multi hardware Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 10/19] media: mtk-vcodec: Add msg queue feature for lat and core architecture Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 11/19] media: mtk-vcodec: Generalize power and clock on/off interfaces Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 12/19] media: mtk-vcodec: Add new interface to lock different hardware Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 13/19] media: mtk-vcodec: Add work queue for core hardware decode Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-09 23:44   ` Steve Cho
2021-12-09 23:44     ` Steve Cho
2021-12-09 23:44     ` Steve Cho
2021-12-09 23:44     ` Steve Cho
2021-12-13  8:52     ` yunfei.dong
2021-12-13  8:52       ` yunfei.dong
2021-12-13  8:52       ` yunfei.dong
2021-12-13  8:52       ` yunfei.dong
2021-12-13 23:49       ` Steve Cho
2021-12-13 23:49         ` Steve Cho
2021-12-13 23:49         ` Steve Cho
2021-12-13 23:49         ` Steve Cho
2021-12-02  3:45 ` [PATCH v12,14/19] media: mtk-vcodec: Support 34bits dma address for vdec Yunfei Dong
2021-12-02  3:45   ` [PATCH v12, 14/19] " Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 15/19] dt-bindings: media: mtk-vcodec: Adds decoder dt-bindings for mt8192 Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-09 23:57   ` Steve Cho
2021-12-09 23:57     ` Steve Cho
2021-12-09 23:57     ` Steve Cho
2021-12-09 23:57     ` Steve Cho
2021-12-10 16:49   ` Rob Herring
2021-12-10 16:49     ` Rob Herring
2021-12-10 16:49     ` Rob Herring
2021-12-10 16:49     ` Rob Herring
2021-12-13  8:32     ` yunfei.dong
2021-12-13  8:32       ` yunfei.dong
2021-12-13  8:32       ` yunfei.dong
2021-12-13  8:32       ` yunfei.dong
2021-12-02  3:45 ` [PATCH v12, 16/19] media: mtk-vcodec: Add core dec and dec end ipi msg Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 17/19] media: mtk-vcodec: Use codec type to separate different hardware Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 18/19] media: mtk-vcodec: Remove mtk_vcodec_release_dec_pm Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45 ` [PATCH v12, 19/19] media: mtk-vcodec: Remove mtk_vcodec_release_enc_pm Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong
2021-12-02  3:45   ` Yunfei Dong

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=87c70b080162b63454609d647ff1e2b39d06d406.camel@mediatek.com \
    --to=yunfei.dong@mediatek.com \
    --cc=Project_Global_Chrome_Upstream_Group@mediatek.com \
    --cc=acourbot@chromium.org \
    --cc=andrew-ct.chen@mediatek.com \
    --cc=angelogioacchino.delregno@collabora.com \
    --cc=benjamin.gaignard@collabora.com \
    --cc=dafna.hirschfeld@collabora.com \
    --cc=daniel@ffwll.ch \
    --cc=devicetree@vger.kernel.org \
    --cc=dri-devel@lists.freedesktop.org \
    --cc=frkoenig@chromium.org \
    --cc=hsinyi@chromium.org \
    --cc=hverkuil-cisco@xs4all.nl \
    --cc=irui.wang@mediatek.com \
    --cc=linux-arm-kernel@lists.infradead.org \
    --cc=linux-kernel@vger.kernel.org \
    --cc=linux-media@vger.kernel.org \
    --cc=linux-mediatek@lists.infradead.org \
    --cc=matthias.bgg@gmail.com \
    --cc=mchehab@kernel.org \
    --cc=robh+dt@kernel.org \
    --cc=srv_heupstream@mediatek.com \
    --cc=stevecho@chromium.org \
    --cc=tfiga@google.com \
    --cc=tiffany.lin@mediatek.com \
    --cc=tzungbi@chromium.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.