652 lines
18 KiB
C
652 lines
18 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright (c) 2016 MediaTek Inc.
|
|
* Author: PC Chen <pc.chen@mediatek.com>
|
|
* Tiffany Lin <tiffany.lin@mediatek.com>
|
|
*/
|
|
|
|
#include <linux/slab.h>
|
|
#include <linux/module.h>
|
|
#include <linux/of_device.h>
|
|
#include <linux/of.h>
|
|
#include <media/v4l2-event.h>
|
|
#include <media/v4l2-mem2mem.h>
|
|
#include <media/videobuf2-dma-contig.h>
|
|
#include <linux/pm_runtime.h>
|
|
#include <linux/pm_wakeup.h>
|
|
#include <linux/iommu.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/suspend.h>
|
|
#include <linux/semaphore.h>
|
|
|
|
#include "mtk_vcodec_drv.h"
|
|
#include "mtk_vcodec_enc.h"
|
|
#include "mtk_vcodec_enc_pm.h"
|
|
#include "mtk_vcodec_enc_pm_plat.h"
|
|
#include "mtk_vcodec_intr.h"
|
|
#include "mtk_vcodec_util.h"
|
|
#include "mtk_vcu.h"
|
|
|
|
module_param(mtk_v4l2_dbg_level, int, 0644); //S_IRUGO | S_IWUSR
|
|
module_param(mtk_vcodec_dbg, bool, 0644); //S_IRUGO | S_IWUSR
|
|
module_param(mtk_vcodec_vcp, int, 0644); //S_IRUGO | S_IWUSR
|
|
char mtk_venc_property_prev[1024];
|
|
char mtk_venc_vcp_log_prev[1024];
|
|
|
|
static struct mtk_vcodec_dev *dev_ptr;
|
|
static int mtk_vcodec_vcp_log_write(const char *val, const struct kernel_param *kp)
|
|
{
|
|
if (!(val == NULL || strlen(val) == 0)) {
|
|
mtk_v4l2_debug(0, "val: %s, len: %zu", val, strlen(val));
|
|
mtk_vcodec_set_log(NULL, dev_ptr, val, MTK_VCODEC_LOG_INDEX_LOG, NULL);
|
|
}
|
|
return 0;
|
|
}
|
|
static struct kernel_param_ops vcodec_vcp_log_param_ops = {
|
|
.set = mtk_vcodec_vcp_log_write,
|
|
};
|
|
module_param_cb(mtk_venc_vcp_log, &vcodec_vcp_log_param_ops, &mtk_venc_vcp_log, 0644);
|
|
|
|
static int mtk_vcodec_vcp_property_write(const char *val, const struct kernel_param *kp)
|
|
{
|
|
if (!(val == NULL || strlen(val) == 0)) {
|
|
mtk_v4l2_debug(0, "val: %s, len: %zu", val, strlen(val));
|
|
mtk_vcodec_set_log(NULL, dev_ptr, val, MTK_VCODEC_LOG_INDEX_PROP, NULL);
|
|
}
|
|
return 0;
|
|
}
|
|
static struct kernel_param_ops vcodec_vcp_prop_param_ops = {
|
|
.set = mtk_vcodec_vcp_property_write,
|
|
};
|
|
module_param_cb(mtk_venc_property, &vcodec_vcp_prop_param_ops, &mtk_venc_property, 0644);
|
|
|
|
|
|
static int fops_vcodec_open(struct file *file)
|
|
{
|
|
struct mtk_vcodec_dev *dev = video_drvdata(file);
|
|
struct mtk_vcodec_ctx *ctx = NULL;
|
|
struct mtk_video_enc_buf *mtk_buf = NULL;
|
|
struct vb2_queue *src_vq;
|
|
int ret = 0;
|
|
|
|
ctx = kzalloc(sizeof(*ctx), GFP_KERNEL);
|
|
if (!ctx)
|
|
return -ENOMEM;
|
|
mtk_buf = kzalloc(sizeof(*mtk_buf), GFP_KERNEL);
|
|
if (!mtk_buf) {
|
|
kfree(ctx);
|
|
return -ENOMEM;
|
|
}
|
|
#if IS_ENABLED(CONFIG_MTK_TINYSYS_VCP_SUPPORT)
|
|
if (mtk_vcodec_is_vcp(MTK_INST_ENCODER)) {
|
|
ret = vcp_register_feature(VENC_FEATURE_ID);
|
|
if (ret) {
|
|
mtk_v4l2_err("Failed to vcp_register_feature");
|
|
kfree(ctx);
|
|
kfree(mtk_buf);
|
|
ctx = NULL;
|
|
mtk_buf = NULL;
|
|
return -EPERM;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
mutex_lock(&dev->dev_mutex);
|
|
/*
|
|
* Use simple counter to uniquely identify this context. Only
|
|
* used for logging.
|
|
*/
|
|
ctx->enc_flush_buf = mtk_buf;
|
|
dev->id_counter++;
|
|
if (dev->id_counter == 0)
|
|
dev->id_counter++;
|
|
ctx->id = dev->id_counter;
|
|
v4l2_fh_init(&ctx->fh, video_devdata(file));
|
|
file->private_data = &ctx->fh;
|
|
v4l2_fh_add(&ctx->fh);
|
|
INIT_LIST_HEAD(&ctx->list);
|
|
ctx->dev = dev;
|
|
init_waitqueue_head(&ctx->queue[0]);
|
|
init_waitqueue_head(&ctx->queue[1]);
|
|
mutex_init(&ctx->buf_lock);
|
|
mutex_init(&ctx->worker_lock);
|
|
mutex_init(&ctx->hw_status);
|
|
mutex_init(&ctx->q_mutex);
|
|
|
|
ctx->type = MTK_INST_ENCODER;
|
|
ret = mtk_vcodec_enc_ctrls_setup(ctx);
|
|
if (ret) {
|
|
mtk_v4l2_err("Failed to setup controls() (%d)",
|
|
ret);
|
|
goto err_ctrls_setup;
|
|
}
|
|
ctx->m2m_ctx = v4l2_m2m_ctx_init(dev->m2m_dev_enc, ctx,
|
|
&mtk_vcodec_enc_queue_init);
|
|
if (IS_ERR((__force void *)ctx->m2m_ctx)) {
|
|
ret = PTR_ERR((__force void *)ctx->m2m_ctx);
|
|
mtk_v4l2_err("Failed to v4l2_m2m_ctx_init() (%d)",
|
|
ret);
|
|
goto err_m2m_ctx_init;
|
|
}
|
|
src_vq = v4l2_m2m_get_vq(ctx->m2m_ctx,
|
|
V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE);
|
|
|
|
ctx->enc_flush_buf->vb.vb2_buf.vb2_queue = src_vq;
|
|
ctx->enc_flush_buf->lastframe = NON_EOS;
|
|
ctx->enc_flush_buf->vb.vb2_buf.planes[0].bytesused = 1;
|
|
mtk_vcodec_enc_set_default_params(ctx);
|
|
|
|
#if IS_ENABLED(CONFIG_VIDEO_MEDIATEK_VCU)
|
|
if (v4l2_fh_is_singular(&ctx->fh) && VCU_FPTR(vcu_load_firmware)) {
|
|
/*
|
|
* vcu_load_firmware checks if it was loaded already and
|
|
* does nothing in that case
|
|
*/
|
|
ret = VCU_FPTR(vcu_load_firmware)(dev->vcu_plat_dev);
|
|
if (ret < 0) {
|
|
/*
|
|
* Return 0 if downloading firmware successfully,
|
|
* otherwise it is failed
|
|
*/
|
|
mtk_v4l2_err("vcu_load_firmware failed!");
|
|
goto err_load_fw;
|
|
}
|
|
}
|
|
#endif
|
|
mtk_v4l2_debug(2, "Create instance [%d]@%p m2m_ctx=%p ",
|
|
ctx->id, ctx, ctx->m2m_ctx);
|
|
|
|
dev->enc_cnt++;
|
|
|
|
mutex_unlock(&dev->dev_mutex);
|
|
mtk_v4l2_debug(0, "%s encoder [%d][%d]", dev_name(&dev->plat_dev->dev),
|
|
ctx->id, dev->enc_cnt);
|
|
return ret;
|
|
|
|
/* Deinit when failure occurred */
|
|
#if IS_ENABLED(CONFIG_VIDEO_MEDIATEK_VCU)
|
|
err_load_fw:
|
|
v4l2_m2m_ctx_release(ctx->m2m_ctx);
|
|
mtk_vcodec_del_ctx_list(ctx);
|
|
#endif
|
|
err_m2m_ctx_init:
|
|
v4l2_ctrl_handler_free(&ctx->ctrl_hdl);
|
|
err_ctrls_setup:
|
|
v4l2_fh_del(&ctx->fh);
|
|
v4l2_fh_exit(&ctx->fh);
|
|
kfree(ctx->enc_flush_buf);
|
|
kfree(ctx);
|
|
mutex_unlock(&dev->dev_mutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int fops_vcodec_release(struct file *file)
|
|
{
|
|
struct mtk_vcodec_dev *dev = video_drvdata(file);
|
|
struct mtk_vcodec_ctx *ctx = fh_to_ctx(file->private_data);
|
|
#if IS_ENABLED(CONFIG_MTK_TINYSYS_VCP_SUPPORT)
|
|
int ret = 0;
|
|
#endif
|
|
|
|
mtk_v4l2_debug(0, "[%d][%d] encoder", ctx->id, dev->enc_cnt);
|
|
mutex_lock(&dev->dev_mutex);
|
|
|
|
/*
|
|
* Check no more ipi in progress, to avoid inst abort since vcp
|
|
* wdt (maybe cause by vdec) but still has ipi waiting timeout
|
|
*/
|
|
mutex_lock(&dev->ipi_mutex);
|
|
mutex_unlock(&dev->ipi_mutex);
|
|
|
|
mtk_vcodec_enc_empty_queues(file, ctx);
|
|
mutex_lock(&ctx->worker_lock);
|
|
v4l2_m2m_ctx_release(ctx->m2m_ctx);
|
|
mutex_unlock(&ctx->worker_lock);
|
|
mtk_vcodec_enc_release(ctx);
|
|
v4l2_fh_del(&ctx->fh);
|
|
v4l2_fh_exit(&ctx->fh);
|
|
v4l2_ctrl_handler_free(&ctx->ctrl_hdl);
|
|
|
|
kfree(ctx->enc_flush_buf);
|
|
kfree(ctx);
|
|
if (dev->enc_cnt > 0)
|
|
dev->enc_cnt--;
|
|
mutex_unlock(&dev->dev_mutex);
|
|
#if IS_ENABLED(CONFIG_MTK_TINYSYS_VCP_SUPPORT)
|
|
if (mtk_vcodec_is_vcp(MTK_INST_ENCODER)) {
|
|
ret = vcp_deregister_feature(VENC_FEATURE_ID);
|
|
if (ret) {
|
|
mtk_v4l2_err("Failed to vcp_deregister_feature");
|
|
return -EPERM;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct v4l2_file_operations mtk_vcodec_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = fops_vcodec_open,
|
|
.release = fops_vcodec_release,
|
|
.poll = v4l2_m2m_fop_poll,
|
|
.unlocked_ioctl = video_ioctl2,
|
|
.mmap = v4l2_m2m_fop_mmap,
|
|
};
|
|
|
|
/**
|
|
* Suspend callbacks after user space processes are frozen
|
|
* Since user space processes are frozen, there is no need and cannot hold same
|
|
* mutex that protects lock owner while checking status.
|
|
* If video codec hardware is still active now, must not to enter suspend.
|
|
**/
|
|
static int mtk_vcodec_enc_suspend(struct device *pDev)
|
|
{
|
|
int val, i;
|
|
struct mtk_vcodec_dev *dev = dev_get_drvdata(pDev);
|
|
|
|
for (i = 0; i < MTK_VENC_HW_NUM; i++) {
|
|
val = down_trylock(&dev->enc_sem[i]);
|
|
if (val == 1) {
|
|
mtk_v4l2_debug(0, "fail due to videocodec activity");
|
|
return -EBUSY;
|
|
}
|
|
up(&dev->enc_sem[i]);
|
|
}
|
|
|
|
mtk_v4l2_debug(1, "done");
|
|
return 0;
|
|
}
|
|
|
|
static int mtk_vcodec_enc_resume(struct device *pDev)
|
|
{
|
|
mtk_v4l2_debug(1, "done");
|
|
return 0;
|
|
}
|
|
|
|
static int mtk_vcodec_enc_suspend_notifier(struct notifier_block *nb,
|
|
unsigned long action, void *data)
|
|
{
|
|
int wait_cnt = 0;
|
|
int val = 0;
|
|
int i;
|
|
struct mtk_vcodec_dev *dev =
|
|
container_of(nb, struct mtk_vcodec_dev, pm_notifier);
|
|
|
|
mtk_v4l2_debug(1, "action = %ld", action);
|
|
switch (action) {
|
|
case PM_SUSPEND_PREPARE:
|
|
dev->is_codec_suspending = 1;
|
|
for (i = 0; i < MTK_VENC_HW_NUM; i++) {
|
|
val = down_trylock(&dev->enc_sem[i]);
|
|
while (val == 1) {
|
|
usleep_range(10000, 20000);
|
|
wait_cnt++;
|
|
/* Current task is still not finished, don't
|
|
* care, will check again in real suspend
|
|
*/
|
|
if (wait_cnt > 5) {
|
|
mtk_v4l2_err("waiting fail");
|
|
return NOTIFY_DONE;
|
|
}
|
|
val = down_trylock(&dev->enc_sem[i]);
|
|
}
|
|
up(&dev->enc_sem[i]);
|
|
}
|
|
return NOTIFY_OK;
|
|
case PM_POST_SUSPEND:
|
|
dev->is_codec_suspending = 0;
|
|
return NOTIFY_OK;
|
|
default:
|
|
return NOTIFY_DONE;
|
|
}
|
|
return NOTIFY_DONE;
|
|
}
|
|
|
|
#if IS_ENABLED(CONFIG_MTK_TINYSYS_VCP_SUPPORT)
|
|
extern void venc_vcp_probe(struct mtk_vcodec_dev *dev);
|
|
extern void venc_vcp_remove(struct mtk_vcodec_dev *dev);
|
|
#endif
|
|
|
|
static int mtk_vcodec_enc_probe(struct platform_device *pdev)
|
|
{
|
|
struct mtk_vcodec_dev *dev;
|
|
struct video_device *vfd_enc;
|
|
struct resource *res;
|
|
int i = 0, reg_index = 0, ret;
|
|
int port_num[MTK_VENC_HW_NUM] = {0};
|
|
const char *name = NULL;
|
|
int port_args_num = 0, port_data_len = 0, total_port_num = 0;
|
|
unsigned int offset = 0;
|
|
unsigned int core_id = 0, ram_type = 0, port_id = 0;
|
|
|
|
dev = devm_kzalloc(&pdev->dev, sizeof(*dev), GFP_KERNEL);
|
|
if (!dev)
|
|
return -ENOMEM;
|
|
|
|
INIT_LIST_HEAD(&dev->ctx_list);
|
|
dev->plat_dev = pdev;
|
|
#if IS_ENABLED(CONFIG_VIDEO_MEDIATEK_VCU)
|
|
if (VCU_FPTR(vcu_get_plat_device)) {
|
|
dev->vcu_plat_dev = VCU_FPTR(vcu_get_plat_device)(dev->plat_dev);
|
|
if (dev->vcu_plat_dev == NULL) {
|
|
mtk_v4l2_err("[VCU] vcu device in not ready");
|
|
return -EPROBE_DEFER;
|
|
}
|
|
}
|
|
#endif
|
|
ret = of_property_read_string(pdev->dev.of_node, "mediatek,platform", &dev->platform);
|
|
if (ret != 0) {
|
|
mtk_v4l2_err("failed to find mediatek,platform\n");
|
|
return ret;
|
|
}
|
|
mtk_v4l2_debug(0, "%s", dev->platform);
|
|
|
|
ret = of_property_read_u32(pdev->dev.of_node, "mediatek,ipm", &dev->venc_hw_ipm);
|
|
if (ret != 0 || dev->venc_hw_ipm > VCODEC_IPM_MAX) {
|
|
mtk_v4l2_debug(0, "default use ipm v1");
|
|
dev->venc_hw_ipm = VCODEC_IPM_V1;
|
|
}
|
|
mtk_v4l2_debug(0, "hw ipm: %d", dev->venc_hw_ipm);
|
|
|
|
ret = mtk_vcodec_init_enc_pm(dev);
|
|
if (ret < 0) {
|
|
dev_info(&pdev->dev, "Failed to get mt vcodec clock source!");
|
|
return ret;
|
|
}
|
|
|
|
|
|
while (!of_property_read_string_index(pdev->dev.of_node, "reg-names", i, &name)) {
|
|
if (!strcmp(MTK_VDEC_REG_NAME_VENC_SYS, name)) {
|
|
reg_index = VENC_SYS;
|
|
} else if (!strcmp(MTK_VDEC_REG_NAME_VENC_C1_SYS, name)) {
|
|
reg_index = VENC_C1_SYS;
|
|
} else if (!strcmp(MTK_VDEC_REG_NAME_VENC_C2_SYS, name)) {
|
|
reg_index = VENC_C2_SYS;
|
|
} else if (!strcmp(MTK_VDEC_REG_NAME_VENC_GCON, name)) {
|
|
reg_index = VENC_GCON;
|
|
} else {
|
|
dev_info(&pdev->dev, "invalid reg name: %s, index: %d", name, i);
|
|
return -EINVAL;
|
|
}
|
|
|
|
res = platform_get_resource(pdev, IORESOURCE_MEM, i);
|
|
if (i == VENC_SYS && res == NULL) {
|
|
dev_info(&pdev->dev,
|
|
"get memory resource failed. idx:%d", i);
|
|
ret = -ENXIO;
|
|
goto err_res;
|
|
} else if (res == NULL) {
|
|
mtk_v4l2_debug(0, "try next resource. idx:%d", i);
|
|
continue;
|
|
}
|
|
|
|
dev->enc_reg_base[reg_index] =
|
|
devm_ioremap_resource(&pdev->dev, res);
|
|
if (IS_ERR((__force void *)dev->enc_reg_base[reg_index])) {
|
|
ret = PTR_ERR(
|
|
(__force void *)dev->enc_reg_base[reg_index]);
|
|
goto err_res;
|
|
}
|
|
mtk_v4l2_debug(2, "reg[%d] base=0x%x",
|
|
reg_index, dev->enc_reg_base[reg_index]);
|
|
|
|
i++;
|
|
}
|
|
|
|
ret = of_property_read_u32(pdev->dev.of_node, "support-wfd-region", &support_wfd_region);
|
|
if (ret) {
|
|
mtk_v4l2_debug(0, "[VENC] Cannot get support-wfd-region, skip");
|
|
support_wfd_region = 0;
|
|
}
|
|
|
|
res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
|
|
if (res == NULL) {
|
|
dev_info(&pdev->dev, "failed to get irq resource");
|
|
ret = -ENOENT;
|
|
goto err_res;
|
|
}
|
|
|
|
ret = mtk_vcodec_enc_irq_setup(pdev, dev);
|
|
if (ret)
|
|
goto err_res;
|
|
|
|
ret = of_property_read_u32(pdev->dev.of_node, "port-arg-num", &port_args_num);
|
|
if (ret != 0)
|
|
dev_info(&pdev->dev, "Failed to get port_arg_num!");
|
|
|
|
pr_info("after get port_arg_num %d\n", port_args_num);
|
|
if (!of_get_property(pdev->dev.of_node, "port-def", &port_data_len))
|
|
dev_info(&pdev->dev, "Failed to get port-def!");
|
|
|
|
pr_info("after get port-def port_data_len %d\n", port_data_len);
|
|
if (port_args_num)
|
|
total_port_num = port_data_len / (sizeof(u32) * port_args_num);
|
|
|
|
for (i = 0; i < total_port_num; i++) {
|
|
offset = i * port_args_num;
|
|
if (of_property_read_u32_index(pdev->dev.of_node, "port-def",
|
|
offset, &core_id)) {
|
|
dev_info(&pdev->dev, "fail core id offset %d i %d!", offset, i);
|
|
goto err_res;
|
|
}
|
|
if (of_property_read_u32_index(pdev->dev.of_node, "port-def",
|
|
offset + 1, &port_id)) {
|
|
dev_info(&pdev->dev, "fail port id offset %d i %d!", offset, i);
|
|
goto err_res;
|
|
}
|
|
if (of_property_read_u32_index(pdev->dev.of_node, "port-def",
|
|
offset + 2, &ram_type)) {
|
|
dev_info(&pdev->dev, "fail ram type offset %d i %d!", offset, i);
|
|
goto err_res;
|
|
}
|
|
|
|
if (core_id < MTK_VENC_HW_NUM) {
|
|
dev->venc_ports[core_id].port_id[port_num[core_id]] = port_id;
|
|
dev->venc_ports[core_id].ram_type[port_num[core_id]] = ram_type;
|
|
port_num[core_id]++;
|
|
}
|
|
|
|
}
|
|
for (i = 0; i < MTK_VENC_HW_NUM; i++) {
|
|
dev->venc_ports[i].total_port_num = port_num[i];
|
|
pr_info("after get port-def port num [%d] %d\n", i, port_num[i]);
|
|
}
|
|
|
|
for (i = 0; i < MTK_VENC_HW_NUM; i++) {
|
|
sema_init(&dev->enc_sem[i], 1);
|
|
spin_lock_init(&dev->enc_power_lock[i]);
|
|
dev->enc_is_power_on[i] = false;
|
|
}
|
|
|
|
mutex_init(&dev->ctx_mutex);
|
|
mutex_init(&dev->dev_mutex);
|
|
mutex_init(&dev->ipi_mutex);
|
|
mutex_init(&dev->enc_dvfs_mutex);
|
|
spin_lock_init(&dev->irqlock);
|
|
|
|
snprintf(dev->v4l2_dev.name, sizeof(dev->v4l2_dev.name), "%s",
|
|
"[MTK_V4L2_VENC]");
|
|
|
|
ret = v4l2_device_register(&pdev->dev, &dev->v4l2_dev);
|
|
if (ret) {
|
|
mtk_v4l2_err("v4l2_device_register err=%d", ret);
|
|
goto err_res;
|
|
}
|
|
|
|
/* allocate video device for encoder and register it */
|
|
vfd_enc = video_device_alloc();
|
|
if (!vfd_enc) {
|
|
mtk_v4l2_err("Failed to allocate video device");
|
|
ret = -ENOMEM;
|
|
goto err_enc_alloc;
|
|
}
|
|
vfd_enc->fops = &mtk_vcodec_fops;
|
|
vfd_enc->ioctl_ops = &mtk_venc_ioctl_ops;
|
|
vfd_enc->release = video_device_release;
|
|
vfd_enc->lock = &dev->dev_mutex;
|
|
vfd_enc->v4l2_dev = &dev->v4l2_dev;
|
|
vfd_enc->vfl_dir = VFL_DIR_M2M;
|
|
vfd_enc->device_caps = V4L2_CAP_VIDEO_M2M_MPLANE |
|
|
V4L2_CAP_STREAMING;
|
|
|
|
snprintf(vfd_enc->name, sizeof(vfd_enc->name), "%s",
|
|
MTK_VCODEC_ENC_NAME);
|
|
video_set_drvdata(vfd_enc, dev);
|
|
dev->vfd_enc = vfd_enc;
|
|
platform_set_drvdata(pdev, dev);
|
|
|
|
dev->m2m_dev_enc = v4l2_m2m_init(&mtk_venc_m2m_ops);
|
|
if (IS_ERR((__force void *)dev->m2m_dev_enc)) {
|
|
mtk_v4l2_err("Failed to init mem2mem enc device");
|
|
ret = PTR_ERR((__force void *)dev->m2m_dev_enc);
|
|
goto err_enc_mem_init;
|
|
}
|
|
|
|
dev->encode_workqueue =
|
|
alloc_ordered_workqueue(MTK_VCODEC_ENC_NAME,
|
|
WQ_MEM_RECLAIM |
|
|
WQ_FREEZABLE);
|
|
if (!dev->encode_workqueue) {
|
|
mtk_v4l2_err("Failed to create encode workqueue");
|
|
ret = -EINVAL;
|
|
goto err_event_workq;
|
|
}
|
|
|
|
ret = video_register_device(vfd_enc, VFL_TYPE_VIDEO, -1);
|
|
if (ret) {
|
|
mtk_v4l2_err("Failed to register video device");
|
|
goto err_enc_reg;
|
|
}
|
|
|
|
#if IS_ENABLED(CONFIG_MTK_IOMMU)
|
|
dev->io_domain = iommu_get_domain_for_dev(&pdev->dev);
|
|
if (dev->io_domain == NULL) {
|
|
mtk_v4l2_err("Failed to get io_domain\n");
|
|
return -EPROBE_DEFER;
|
|
}
|
|
|
|
ret = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(34));
|
|
if (ret) {
|
|
ret = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(34));
|
|
if (ret) {
|
|
dev_info(&pdev->dev, "64-bit DMA enable failed\n");
|
|
return ret;
|
|
}
|
|
}
|
|
if (!pdev->dev.dma_parms) {
|
|
pdev->dev.dma_parms =
|
|
devm_kzalloc(&pdev->dev, sizeof(*pdev->dev.dma_parms), GFP_KERNEL);
|
|
}
|
|
if (pdev->dev.dma_parms) {
|
|
ret = dma_set_max_seg_size(&pdev->dev, (unsigned int)DMA_BIT_MASK(34));
|
|
if (ret)
|
|
dev_info(&pdev->dev, "Failed to set DMA segment size\n");
|
|
}
|
|
#endif
|
|
mtk_v4l2_debug(0, "encoder registered as /dev/video%d",
|
|
vfd_enc->num);
|
|
|
|
#if IS_ENABLED(CONFIG_MTK_IOMMU)
|
|
mtk_venc_translation_fault_callback_setting(dev);
|
|
#endif
|
|
|
|
mtk_prepare_venc_dvfs(dev);
|
|
mtk_prepare_venc_emi_bw(dev);
|
|
dev->pm_notifier.notifier_call = mtk_vcodec_enc_suspend_notifier;
|
|
register_pm_notifier(&dev->pm_notifier);
|
|
dev->is_codec_suspending = 0;
|
|
dev->enc_cnt = 0;
|
|
|
|
#if IS_ENABLED(CONFIG_MTK_TINYSYS_VCP_SUPPORT)
|
|
venc_vcp_probe(dev);
|
|
#endif
|
|
|
|
INIT_LIST_HEAD(&dev->log_param_list);
|
|
mutex_init(&dev->log_param_mutex);
|
|
INIT_LIST_HEAD(&dev->prop_param_list);
|
|
mutex_init(&dev->prop_param_mutex);
|
|
dev_ptr = dev;
|
|
|
|
return 0;
|
|
|
|
err_enc_reg:
|
|
destroy_workqueue(dev->encode_workqueue);
|
|
err_event_workq:
|
|
v4l2_m2m_release(dev->m2m_dev_enc);
|
|
err_enc_mem_init:
|
|
video_unregister_device(vfd_enc);
|
|
err_enc_alloc:
|
|
v4l2_device_unregister(&dev->v4l2_dev);
|
|
err_res:
|
|
mtk_vcodec_release_enc_pm(dev);
|
|
return ret;
|
|
}
|
|
|
|
static const struct of_device_id mtk_vcodec_enc_match[] = {
|
|
{.compatible = "mediatek,mt8173-vcodec-enc",},
|
|
{.compatible = "mediatek,mt2712-vcodec-enc",},
|
|
{.compatible = "mediatek,mt8167-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6771-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6885-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6873-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6853-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6983-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6879-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6895-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6855-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6985-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6886-vcodec-enc",},
|
|
{.compatible = "mediatek,mt8195-vcodec-enc",},
|
|
{.compatible = "mediatek,mt6835-vcodec-enc",},
|
|
{.compatible = "mediatek,venc_gcon",},
|
|
{},
|
|
};
|
|
MODULE_DEVICE_TABLE(of, mtk_vcodec_enc_match);
|
|
|
|
static int mtk_vcodec_enc_remove(struct platform_device *pdev)
|
|
{
|
|
struct mtk_vcodec_dev *dev = platform_get_drvdata(pdev);
|
|
|
|
mtk_unprepare_venc_emi_bw(dev);
|
|
mtk_unprepare_venc_dvfs(dev);
|
|
|
|
mtk_v4l2_debug_enter();
|
|
flush_workqueue(dev->encode_workqueue);
|
|
destroy_workqueue(dev->encode_workqueue);
|
|
if (dev->m2m_dev_enc)
|
|
v4l2_m2m_release(dev->m2m_dev_enc);
|
|
|
|
if (dev->vfd_enc)
|
|
video_unregister_device(dev->vfd_enc);
|
|
|
|
v4l2_device_unregister(&dev->v4l2_dev);
|
|
mtk_vcodec_release_enc_pm(dev);
|
|
|
|
#if IS_ENABLED(CONFIG_MTK_TINYSYS_VCP_SUPPORT)
|
|
venc_vcp_remove(dev);
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
static const struct dev_pm_ops mtk_vcodec_enc_pm_ops = {
|
|
.suspend = mtk_vcodec_enc_suspend,
|
|
.resume = mtk_vcodec_enc_resume,
|
|
};
|
|
|
|
static struct platform_driver mtk_vcodec_enc_driver = {
|
|
.probe = mtk_vcodec_enc_probe,
|
|
.remove = mtk_vcodec_enc_remove,
|
|
.driver = {
|
|
.name = MTK_VCODEC_ENC_NAME,
|
|
.pm = &mtk_vcodec_enc_pm_ops,
|
|
.of_match_table = mtk_vcodec_enc_match,
|
|
},
|
|
};
|
|
|
|
module_platform_driver(mtk_vcodec_enc_driver);
|
|
|
|
|
|
MODULE_LICENSE("GPL v2");
|
|
MODULE_DESCRIPTION("Mediatek video codec V4L2 encoder driver");
|