xref: /openbmc/linux/drivers/media/test-drivers/visl/visl-core.c (revision 1ac731c529cd4d6adbce134754b51ff7d822b145)
10c078e31SDaniel Almeida // SPDX-License-Identifier: GPL-2.0
20c078e31SDaniel Almeida /*
30c078e31SDaniel Almeida  * A virtual stateless decoder device for stateless uAPI development purposes.
40c078e31SDaniel Almeida  *
50c078e31SDaniel Almeida  * This tool's objective is to help the development and testing of userspace
60c078e31SDaniel Almeida  * applications that use the V4L2 stateless API to decode media.
70c078e31SDaniel Almeida  *
80c078e31SDaniel Almeida  * A userspace implementation can use visl to run a decoding loop even when no
90c078e31SDaniel Almeida  * hardware is available or when the kernel uAPI for the codec has not been
100c078e31SDaniel Almeida  * upstreamed yet. This can reveal bugs at an early stage.
110c078e31SDaniel Almeida  *
120c078e31SDaniel Almeida  * This driver can also trace the contents of the V4L2 controls submitted to it.
130c078e31SDaniel Almeida  * It can also dump the contents of the vb2 buffers through a debugfs
140c078e31SDaniel Almeida  * interface. This is in many ways similar to the tracing infrastructure
150c078e31SDaniel Almeida  * available for other popular encode/decode APIs out there and can help develop
160c078e31SDaniel Almeida  * a userspace application by using another (working) one as a reference.
170c078e31SDaniel Almeida  *
180c078e31SDaniel Almeida  * Note that no actual decoding of video frames is performed by visl. The V4L2
190c078e31SDaniel Almeida  * test pattern generator is used to write various debug information to the
200c078e31SDaniel Almeida  * capture buffers instead.
210c078e31SDaniel Almeida  *
220c078e31SDaniel Almeida  * Copyright (C) 2022 Collabora, Ltd.
230c078e31SDaniel Almeida  *
240c078e31SDaniel Almeida  * Based on the vim2m driver, that is:
250c078e31SDaniel Almeida  *
260c078e31SDaniel Almeida  * Copyright (c) 2009-2010 Samsung Electronics Co., Ltd.
270c078e31SDaniel Almeida  * Pawel Osciak, <pawel@osciak.com>
280c078e31SDaniel Almeida  * Marek Szyprowski, <m.szyprowski@samsung.com>
290c078e31SDaniel Almeida  *
300c078e31SDaniel Almeida  * Based on the vicodec driver, that is:
310c078e31SDaniel Almeida  *
320c078e31SDaniel Almeida  * Copyright 2018 Cisco Systems, Inc. and/or its affiliates. All rights reserved.
330c078e31SDaniel Almeida  *
340c078e31SDaniel Almeida  * Based on the Cedrus VPU driver, that is:
350c078e31SDaniel Almeida  *
360c078e31SDaniel Almeida  * Copyright (C) 2016 Florent Revest <florent.revest@free-electrons.com>
370c078e31SDaniel Almeida  * Copyright (C) 2018 Paul Kocialkowski <paul.kocialkowski@bootlin.com>
380c078e31SDaniel Almeida  * Copyright (C) 2018 Bootlin
390c078e31SDaniel Almeida  */
400c078e31SDaniel Almeida 
410c078e31SDaniel Almeida #include <linux/debugfs.h>
420c078e31SDaniel Almeida #include <linux/module.h>
430c078e31SDaniel Almeida #include <linux/platform_device.h>
440c078e31SDaniel Almeida #include <media/v4l2-ctrls.h>
450c078e31SDaniel Almeida #include <media/v4l2-device.h>
460c078e31SDaniel Almeida #include <media/v4l2-ioctl.h>
470c078e31SDaniel Almeida #include <media/v4l2-mem2mem.h>
480c078e31SDaniel Almeida 
490c078e31SDaniel Almeida #include "visl.h"
500c078e31SDaniel Almeida #include "visl-dec.h"
510c078e31SDaniel Almeida #include "visl-debugfs.h"
520c078e31SDaniel Almeida #include "visl-video.h"
530c078e31SDaniel Almeida 
540c078e31SDaniel Almeida unsigned int visl_debug;
550c078e31SDaniel Almeida module_param(visl_debug, uint, 0644);
560c078e31SDaniel Almeida MODULE_PARM_DESC(visl_debug, " activates debug info");
570c078e31SDaniel Almeida 
580c078e31SDaniel Almeida unsigned int visl_transtime_ms;
590c078e31SDaniel Almeida module_param(visl_transtime_ms, uint, 0644);
600c078e31SDaniel Almeida MODULE_PARM_DESC(visl_transtime_ms, " simulated process time in milliseconds.");
610c078e31SDaniel Almeida 
620c078e31SDaniel Almeida /*
630c078e31SDaniel Almeida  * dprintk can be slow through serial. This lets one limit the tracing to a
640c078e31SDaniel Almeida  * particular number of frames
650c078e31SDaniel Almeida  */
660c078e31SDaniel Almeida int visl_dprintk_frame_start = -1;
670c078e31SDaniel Almeida module_param(visl_dprintk_frame_start, int, 0);
680c078e31SDaniel Almeida MODULE_PARM_DESC(visl_dprintk_frame_start,
690c078e31SDaniel Almeida 		 " a frame number to start tracing with dprintk");
700c078e31SDaniel Almeida 
710c078e31SDaniel Almeida unsigned int visl_dprintk_nframes;
720c078e31SDaniel Almeida module_param(visl_dprintk_nframes, uint, 0);
730c078e31SDaniel Almeida MODULE_PARM_DESC(visl_dprintk_nframes,
740c078e31SDaniel Almeida 		 " the number of frames to trace with dprintk");
750c078e31SDaniel Almeida 
760c078e31SDaniel Almeida bool keep_bitstream_buffers;
770c078e31SDaniel Almeida module_param(keep_bitstream_buffers, bool, false);
780c078e31SDaniel Almeida MODULE_PARM_DESC(keep_bitstream_buffers,
790c078e31SDaniel Almeida 		 " keep bitstream buffers in debugfs after streaming is stopped");
800c078e31SDaniel Almeida 
810c078e31SDaniel Almeida int bitstream_trace_frame_start = -1;
820c078e31SDaniel Almeida module_param(bitstream_trace_frame_start, int, 0);
830c078e31SDaniel Almeida MODULE_PARM_DESC(bitstream_trace_frame_start,
840c078e31SDaniel Almeida 		 " a frame number to start dumping the bitstream through debugfs");
850c078e31SDaniel Almeida 
860c078e31SDaniel Almeida unsigned int bitstream_trace_nframes;
870c078e31SDaniel Almeida module_param(bitstream_trace_nframes, uint, 0);
880c078e31SDaniel Almeida MODULE_PARM_DESC(bitstream_trace_nframes,
890c078e31SDaniel Almeida 		 " the number of frames to dump the bitstream through debugfs");
900c078e31SDaniel Almeida 
910c078e31SDaniel Almeida static const struct visl_ctrl_desc visl_fwht_ctrl_descs[] = {
920c078e31SDaniel Almeida 	{
930c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_FWHT_PARAMS,
940c078e31SDaniel Almeida 	},
950c078e31SDaniel Almeida };
960c078e31SDaniel Almeida 
970c078e31SDaniel Almeida const struct visl_ctrls visl_fwht_ctrls = {
980c078e31SDaniel Almeida 	.ctrls = visl_fwht_ctrl_descs,
990c078e31SDaniel Almeida 	.num_ctrls = ARRAY_SIZE(visl_fwht_ctrl_descs)
1000c078e31SDaniel Almeida };
1010c078e31SDaniel Almeida 
1020c078e31SDaniel Almeida static const struct visl_ctrl_desc visl_mpeg2_ctrl_descs[] = {
1030c078e31SDaniel Almeida 	{
1040c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_MPEG2_SEQUENCE,
1050c078e31SDaniel Almeida 	},
1060c078e31SDaniel Almeida 	{
1070c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_MPEG2_PICTURE,
1080c078e31SDaniel Almeida 	},
1090c078e31SDaniel Almeida 	{
1100c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_MPEG2_QUANTISATION,
1110c078e31SDaniel Almeida 	},
1120c078e31SDaniel Almeida };
1130c078e31SDaniel Almeida 
1140c078e31SDaniel Almeida const struct visl_ctrls visl_mpeg2_ctrls = {
1150c078e31SDaniel Almeida 	.ctrls = visl_mpeg2_ctrl_descs,
1160c078e31SDaniel Almeida 	.num_ctrls = ARRAY_SIZE(visl_mpeg2_ctrl_descs),
1170c078e31SDaniel Almeida };
1180c078e31SDaniel Almeida 
1190c078e31SDaniel Almeida static const struct visl_ctrl_desc visl_vp8_ctrl_descs[] = {
1200c078e31SDaniel Almeida 	{
1210c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_VP8_FRAME,
1220c078e31SDaniel Almeida 	},
1230c078e31SDaniel Almeida };
1240c078e31SDaniel Almeida 
1250c078e31SDaniel Almeida const struct visl_ctrls visl_vp8_ctrls = {
1260c078e31SDaniel Almeida 	.ctrls = visl_vp8_ctrl_descs,
1270c078e31SDaniel Almeida 	.num_ctrls = ARRAY_SIZE(visl_vp8_ctrl_descs),
1280c078e31SDaniel Almeida };
1290c078e31SDaniel Almeida 
1300c078e31SDaniel Almeida static const struct visl_ctrl_desc visl_vp9_ctrl_descs[] = {
1310c078e31SDaniel Almeida 	{
1320c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_VP9_FRAME,
1330c078e31SDaniel Almeida 	},
1340c078e31SDaniel Almeida 	{
1350c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_VP9_COMPRESSED_HDR,
1360c078e31SDaniel Almeida 	},
1370c078e31SDaniel Almeida };
1380c078e31SDaniel Almeida 
1390c078e31SDaniel Almeida const struct visl_ctrls visl_vp9_ctrls = {
1400c078e31SDaniel Almeida 	.ctrls = visl_vp9_ctrl_descs,
1410c078e31SDaniel Almeida 	.num_ctrls = ARRAY_SIZE(visl_vp9_ctrl_descs),
1420c078e31SDaniel Almeida };
1430c078e31SDaniel Almeida 
1440c078e31SDaniel Almeida static const struct visl_ctrl_desc visl_h264_ctrl_descs[] = {
1450c078e31SDaniel Almeida 	{
1460c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_H264_DECODE_PARAMS,
1470c078e31SDaniel Almeida 	},
1480c078e31SDaniel Almeida 	{
1490c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_H264_SPS,
1500c078e31SDaniel Almeida 	},
1510c078e31SDaniel Almeida 	{
1520c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_H264_PPS,
1530c078e31SDaniel Almeida 	},
1540c078e31SDaniel Almeida 	{
1550c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_H264_SCALING_MATRIX,
1560c078e31SDaniel Almeida 	},
1570c078e31SDaniel Almeida 	{
1580c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_H264_DECODE_MODE,
1590c078e31SDaniel Almeida 	},
1600c078e31SDaniel Almeida 	{
1610c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_H264_START_CODE,
1620c078e31SDaniel Almeida 	},
1630c078e31SDaniel Almeida 	{
1640c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_H264_SLICE_PARAMS,
1650c078e31SDaniel Almeida 	},
1660c078e31SDaniel Almeida 	{
1670c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_H264_PRED_WEIGHTS,
1680c078e31SDaniel Almeida 	},
1690c078e31SDaniel Almeida };
1700c078e31SDaniel Almeida 
1710c078e31SDaniel Almeida const struct visl_ctrls visl_h264_ctrls = {
1720c078e31SDaniel Almeida 	.ctrls = visl_h264_ctrl_descs,
1730c078e31SDaniel Almeida 	.num_ctrls = ARRAY_SIZE(visl_h264_ctrl_descs),
1740c078e31SDaniel Almeida };
1750c078e31SDaniel Almeida 
1760c078e31SDaniel Almeida static const struct visl_ctrl_desc visl_hevc_ctrl_descs[] = {
1770c078e31SDaniel Almeida 	{
1780c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_HEVC_SPS,
1790c078e31SDaniel Almeida 	},
1800c078e31SDaniel Almeida 	{
1810c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_HEVC_PPS,
1820c078e31SDaniel Almeida 	},
1830c078e31SDaniel Almeida 	{
1840c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_HEVC_SLICE_PARAMS,
1850c078e31SDaniel Almeida 		/* The absolute maximum for level > 6 */
1860c078e31SDaniel Almeida 		.cfg.dims = { 600 },
1870c078e31SDaniel Almeida 	},
1880c078e31SDaniel Almeida 	{
1890c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_HEVC_SCALING_MATRIX,
1900c078e31SDaniel Almeida 	},
1910c078e31SDaniel Almeida 	{
1920c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_HEVC_DECODE_PARAMS,
1930c078e31SDaniel Almeida 	},
1940c078e31SDaniel Almeida 	{
1950c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_HEVC_DECODE_MODE,
1960c078e31SDaniel Almeida 	},
1970c078e31SDaniel Almeida 	{
1980c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_HEVC_START_CODE,
1990c078e31SDaniel Almeida 	},
2000c078e31SDaniel Almeida 	{
2010c078e31SDaniel Almeida 		.cfg.id = V4L2_CID_STATELESS_HEVC_ENTRY_POINT_OFFSETS,
2020c078e31SDaniel Almeida 		.cfg.dims = { 256 },
2030c078e31SDaniel Almeida 		.cfg.max = 0xffffffff,
2040c078e31SDaniel Almeida 		.cfg.step = 1,
2050c078e31SDaniel Almeida 	},
2060c078e31SDaniel Almeida 
2070c078e31SDaniel Almeida };
2080c078e31SDaniel Almeida 
2090c078e31SDaniel Almeida const struct visl_ctrls visl_hevc_ctrls = {
2100c078e31SDaniel Almeida 	.ctrls = visl_hevc_ctrl_descs,
2110c078e31SDaniel Almeida 	.num_ctrls = ARRAY_SIZE(visl_hevc_ctrl_descs),
2120c078e31SDaniel Almeida };
2130c078e31SDaniel Almeida 
visl_find_control(struct visl_ctx * ctx,u32 id)2140c078e31SDaniel Almeida struct v4l2_ctrl *visl_find_control(struct visl_ctx *ctx, u32 id)
2150c078e31SDaniel Almeida {
2160c078e31SDaniel Almeida 	struct v4l2_ctrl_handler *hdl = &ctx->hdl;
2170c078e31SDaniel Almeida 
2180c078e31SDaniel Almeida 	return v4l2_ctrl_find(hdl, id);
2190c078e31SDaniel Almeida }
2200c078e31SDaniel Almeida 
visl_find_control_data(struct visl_ctx * ctx,u32 id)2210c078e31SDaniel Almeida void *visl_find_control_data(struct visl_ctx *ctx, u32 id)
2220c078e31SDaniel Almeida {
2230c078e31SDaniel Almeida 	struct v4l2_ctrl *ctrl;
2240c078e31SDaniel Almeida 
2250c078e31SDaniel Almeida 	ctrl = visl_find_control(ctx, id);
2260c078e31SDaniel Almeida 	if (ctrl)
2270c078e31SDaniel Almeida 		return ctrl->p_cur.p;
2280c078e31SDaniel Almeida 
2290c078e31SDaniel Almeida 	return NULL;
2300c078e31SDaniel Almeida }
2310c078e31SDaniel Almeida 
visl_control_num_elems(struct visl_ctx * ctx,u32 id)2320c078e31SDaniel Almeida u32 visl_control_num_elems(struct visl_ctx *ctx, u32 id)
2330c078e31SDaniel Almeida {
2340c078e31SDaniel Almeida 	struct v4l2_ctrl *ctrl;
2350c078e31SDaniel Almeida 
2360c078e31SDaniel Almeida 	ctrl = visl_find_control(ctx, id);
2370c078e31SDaniel Almeida 	if (ctrl)
2380c078e31SDaniel Almeida 		return ctrl->elems;
2390c078e31SDaniel Almeida 
2400c078e31SDaniel Almeida 	return 0;
2410c078e31SDaniel Almeida }
2420c078e31SDaniel Almeida 
visl_device_release(struct video_device * vdev)2430c078e31SDaniel Almeida static void visl_device_release(struct video_device *vdev)
2440c078e31SDaniel Almeida {
2450c078e31SDaniel Almeida 	struct visl_dev *dev = container_of(vdev, struct visl_dev, vfd);
2460c078e31SDaniel Almeida 
2470c078e31SDaniel Almeida 	v4l2_device_unregister(&dev->v4l2_dev);
2480c078e31SDaniel Almeida 	v4l2_m2m_release(dev->m2m_dev);
2490c078e31SDaniel Almeida 	media_device_cleanup(&dev->mdev);
2500c078e31SDaniel Almeida 	visl_debugfs_deinit(dev);
2510c078e31SDaniel Almeida 	kfree(dev);
2520c078e31SDaniel Almeida }
2530c078e31SDaniel Almeida 
2540c078e31SDaniel Almeida #define VISL_CONTROLS_COUNT	ARRAY_SIZE(visl_controls)
2550c078e31SDaniel Almeida 
visl_init_ctrls(struct visl_ctx * ctx)2560c078e31SDaniel Almeida static int visl_init_ctrls(struct visl_ctx *ctx)
2570c078e31SDaniel Almeida {
2580c078e31SDaniel Almeida 	struct visl_dev *dev = ctx->dev;
2590c078e31SDaniel Almeida 	struct v4l2_ctrl_handler *hdl = &ctx->hdl;
2600c078e31SDaniel Almeida 	unsigned int ctrl_cnt = 0;
2610c078e31SDaniel Almeida 	unsigned int i;
2620c078e31SDaniel Almeida 	unsigned int j;
2630c078e31SDaniel Almeida 	const struct visl_ctrls *ctrls;
2640c078e31SDaniel Almeida 
2650c078e31SDaniel Almeida 	for (i = 0; i < num_coded_fmts; i++)
2660c078e31SDaniel Almeida 		ctrl_cnt += visl_coded_fmts[i].ctrls->num_ctrls;
2670c078e31SDaniel Almeida 
2680c078e31SDaniel Almeida 	v4l2_ctrl_handler_init(hdl, ctrl_cnt);
2690c078e31SDaniel Almeida 
2700c078e31SDaniel Almeida 	for (i = 0; i < num_coded_fmts; i++) {
2710c078e31SDaniel Almeida 		ctrls = visl_coded_fmts[i].ctrls;
2720c078e31SDaniel Almeida 		for (j = 0; j < ctrls->num_ctrls; j++)
2730c078e31SDaniel Almeida 			v4l2_ctrl_new_custom(hdl, &ctrls->ctrls[j].cfg, NULL);
2740c078e31SDaniel Almeida 	}
2750c078e31SDaniel Almeida 
2760c078e31SDaniel Almeida 	if (hdl->error) {
2770c078e31SDaniel Almeida 		v4l2_err(&dev->v4l2_dev,
2780c078e31SDaniel Almeida 			 "Failed to initialize control handler\n");
2790c078e31SDaniel Almeida 		v4l2_ctrl_handler_free(hdl);
2800c078e31SDaniel Almeida 		return hdl->error;
2810c078e31SDaniel Almeida 	}
2820c078e31SDaniel Almeida 
2830c078e31SDaniel Almeida 	ctx->fh.ctrl_handler = hdl;
2840c078e31SDaniel Almeida 	v4l2_ctrl_handler_setup(hdl);
2850c078e31SDaniel Almeida 
2860c078e31SDaniel Almeida 	return 0;
2870c078e31SDaniel Almeida }
2880c078e31SDaniel Almeida 
visl_open(struct file * file)2890c078e31SDaniel Almeida static int visl_open(struct file *file)
2900c078e31SDaniel Almeida {
2910c078e31SDaniel Almeida 	struct visl_dev *dev = video_drvdata(file);
2920c078e31SDaniel Almeida 	struct visl_ctx *ctx = NULL;
2930c078e31SDaniel Almeida 	int rc = 0;
2940c078e31SDaniel Almeida 
2950c078e31SDaniel Almeida 	if (mutex_lock_interruptible(&dev->dev_mutex))
2960c078e31SDaniel Almeida 		return -ERESTARTSYS;
2970c078e31SDaniel Almeida 	ctx = kzalloc(sizeof(*ctx), GFP_KERNEL);
2980c078e31SDaniel Almeida 	if (!ctx) {
2990c078e31SDaniel Almeida 		rc = -ENOMEM;
3000c078e31SDaniel Almeida 		goto unlock;
3010c078e31SDaniel Almeida 	}
3020c078e31SDaniel Almeida 
3030c078e31SDaniel Almeida 	ctx->tpg_str_buf = kzalloc(TPG_STR_BUF_SZ, GFP_KERNEL);
3040c078e31SDaniel Almeida 
3050c078e31SDaniel Almeida 	v4l2_fh_init(&ctx->fh, video_devdata(file));
3060c078e31SDaniel Almeida 	file->private_data = &ctx->fh;
3070c078e31SDaniel Almeida 	ctx->dev = dev;
3080c078e31SDaniel Almeida 
3090c078e31SDaniel Almeida 	rc = visl_init_ctrls(ctx);
3100c078e31SDaniel Almeida 	if (rc)
3110c078e31SDaniel Almeida 		goto free_ctx;
3120c078e31SDaniel Almeida 
3130c078e31SDaniel Almeida 	ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(dev->m2m_dev, ctx, &visl_queue_init);
3140c078e31SDaniel Almeida 
3150c078e31SDaniel Almeida 	mutex_init(&ctx->vb_mutex);
3160c078e31SDaniel Almeida 
3170c078e31SDaniel Almeida 	if (IS_ERR(ctx->fh.m2m_ctx)) {
3180c078e31SDaniel Almeida 		rc = PTR_ERR(ctx->fh.m2m_ctx);
3190c078e31SDaniel Almeida 		goto free_hdl;
3200c078e31SDaniel Almeida 	}
3210c078e31SDaniel Almeida 
3220c078e31SDaniel Almeida 	rc = visl_set_default_format(ctx);
3230c078e31SDaniel Almeida 	if (rc)
3240c078e31SDaniel Almeida 		goto free_m2m_ctx;
3250c078e31SDaniel Almeida 
3260c078e31SDaniel Almeida 	v4l2_fh_add(&ctx->fh);
3270c078e31SDaniel Almeida 
3280c078e31SDaniel Almeida 	dprintk(dev, "Created instance: %p, m2m_ctx: %p\n",
3290c078e31SDaniel Almeida 		ctx, ctx->fh.m2m_ctx);
3300c078e31SDaniel Almeida 
3310c078e31SDaniel Almeida 	mutex_unlock(&dev->dev_mutex);
3320c078e31SDaniel Almeida 	return rc;
3330c078e31SDaniel Almeida 
3340c078e31SDaniel Almeida free_m2m_ctx:
3350c078e31SDaniel Almeida 	v4l2_m2m_ctx_release(ctx->fh.m2m_ctx);
3360c078e31SDaniel Almeida free_hdl:
3370c078e31SDaniel Almeida 	v4l2_ctrl_handler_free(&ctx->hdl);
3380c078e31SDaniel Almeida 	v4l2_fh_exit(&ctx->fh);
3390c078e31SDaniel Almeida free_ctx:
3400c078e31SDaniel Almeida 	kfree(ctx->tpg_str_buf);
3410c078e31SDaniel Almeida 	kfree(ctx);
3420c078e31SDaniel Almeida unlock:
3430c078e31SDaniel Almeida 	mutex_unlock(&dev->dev_mutex);
3440c078e31SDaniel Almeida 	return rc;
3450c078e31SDaniel Almeida }
3460c078e31SDaniel Almeida 
visl_release(struct file * file)3470c078e31SDaniel Almeida static int visl_release(struct file *file)
3480c078e31SDaniel Almeida {
3490c078e31SDaniel Almeida 	struct visl_dev *dev = video_drvdata(file);
3500c078e31SDaniel Almeida 	struct visl_ctx *ctx = visl_file_to_ctx(file);
3510c078e31SDaniel Almeida 
3520c078e31SDaniel Almeida 	dprintk(dev, "Releasing instance %p\n", ctx);
3530c078e31SDaniel Almeida 
3540c078e31SDaniel Almeida 	tpg_free(&ctx->tpg);
3550c078e31SDaniel Almeida 	v4l2_fh_del(&ctx->fh);
3560c078e31SDaniel Almeida 	v4l2_fh_exit(&ctx->fh);
3570c078e31SDaniel Almeida 	v4l2_ctrl_handler_free(&ctx->hdl);
3580c078e31SDaniel Almeida 	mutex_lock(&dev->dev_mutex);
3590c078e31SDaniel Almeida 	v4l2_m2m_ctx_release(ctx->fh.m2m_ctx);
3600c078e31SDaniel Almeida 	mutex_unlock(&dev->dev_mutex);
3610c078e31SDaniel Almeida 
3620c078e31SDaniel Almeida 	kfree(ctx->tpg_str_buf);
3630c078e31SDaniel Almeida 	kfree(ctx);
3640c078e31SDaniel Almeida 
3650c078e31SDaniel Almeida 	return 0;
3660c078e31SDaniel Almeida }
3670c078e31SDaniel Almeida 
3680c078e31SDaniel Almeida static const struct v4l2_file_operations visl_fops = {
3690c078e31SDaniel Almeida 	.owner		= THIS_MODULE,
3700c078e31SDaniel Almeida 	.open		= visl_open,
3710c078e31SDaniel Almeida 	.release	= visl_release,
3720c078e31SDaniel Almeida 	.poll		= v4l2_m2m_fop_poll,
3730c078e31SDaniel Almeida 	.unlocked_ioctl	= video_ioctl2,
3740c078e31SDaniel Almeida 	.mmap		= v4l2_m2m_fop_mmap,
3750c078e31SDaniel Almeida };
3760c078e31SDaniel Almeida 
3770c078e31SDaniel Almeida static const struct video_device visl_videodev = {
3780c078e31SDaniel Almeida 	.name		= VISL_NAME,
3790c078e31SDaniel Almeida 	.vfl_dir	= VFL_DIR_M2M,
3800c078e31SDaniel Almeida 	.fops		= &visl_fops,
3810c078e31SDaniel Almeida 	.ioctl_ops	= &visl_ioctl_ops,
3820c078e31SDaniel Almeida 	.minor		= -1,
3830c078e31SDaniel Almeida 	.release	= visl_device_release,
3840c078e31SDaniel Almeida 	.device_caps	= V4L2_CAP_VIDEO_M2M_MPLANE | V4L2_CAP_STREAMING,
3850c078e31SDaniel Almeida };
3860c078e31SDaniel Almeida 
3870c078e31SDaniel Almeida static const struct v4l2_m2m_ops visl_m2m_ops = {
3880c078e31SDaniel Almeida 	.device_run	= visl_device_run,
3890c078e31SDaniel Almeida };
3900c078e31SDaniel Almeida 
3910c078e31SDaniel Almeida static const struct media_device_ops visl_m2m_media_ops = {
3920c078e31SDaniel Almeida 	.req_validate	= visl_request_validate,
3930c078e31SDaniel Almeida 	.req_queue	= v4l2_m2m_request_queue,
3940c078e31SDaniel Almeida };
3950c078e31SDaniel Almeida 
visl_probe(struct platform_device * pdev)3960c078e31SDaniel Almeida static int visl_probe(struct platform_device *pdev)
3970c078e31SDaniel Almeida {
3980c078e31SDaniel Almeida 	struct visl_dev *dev;
3990c078e31SDaniel Almeida 	struct video_device *vfd;
4000c078e31SDaniel Almeida 	int ret;
4010c078e31SDaniel Almeida 	int rc;
4020c078e31SDaniel Almeida 
4030c078e31SDaniel Almeida 	dev = kzalloc(sizeof(*dev), GFP_KERNEL);
4040c078e31SDaniel Almeida 	if (!dev)
4050c078e31SDaniel Almeida 		return -ENOMEM;
4060c078e31SDaniel Almeida 
4070c078e31SDaniel Almeida 	ret = v4l2_device_register(&pdev->dev, &dev->v4l2_dev);
4080c078e31SDaniel Almeida 	if (ret)
4090c078e31SDaniel Almeida 		goto error_visl_dev;
4100c078e31SDaniel Almeida 
4110c078e31SDaniel Almeida 	mutex_init(&dev->dev_mutex);
4120c078e31SDaniel Almeida 
4130c078e31SDaniel Almeida 	dev->vfd = visl_videodev;
4140c078e31SDaniel Almeida 	vfd = &dev->vfd;
4150c078e31SDaniel Almeida 	vfd->lock = &dev->dev_mutex;
4160c078e31SDaniel Almeida 	vfd->v4l2_dev = &dev->v4l2_dev;
4170c078e31SDaniel Almeida 
4180c078e31SDaniel Almeida 	video_set_drvdata(vfd, dev);
4190c078e31SDaniel Almeida 
4200c078e31SDaniel Almeida 	platform_set_drvdata(pdev, dev);
4210c078e31SDaniel Almeida 
4220c078e31SDaniel Almeida 	dev->m2m_dev = v4l2_m2m_init(&visl_m2m_ops);
4230c078e31SDaniel Almeida 	if (IS_ERR(dev->m2m_dev)) {
4240c078e31SDaniel Almeida 		v4l2_err(&dev->v4l2_dev, "Failed to init mem2mem device\n");
4250c078e31SDaniel Almeida 		ret = PTR_ERR(dev->m2m_dev);
4260c078e31SDaniel Almeida 		dev->m2m_dev = NULL;
4270c078e31SDaniel Almeida 		goto error_dev;
4280c078e31SDaniel Almeida 	}
4290c078e31SDaniel Almeida 
4300c078e31SDaniel Almeida 	dev->mdev.dev = &pdev->dev;
4310c078e31SDaniel Almeida 	strscpy(dev->mdev.model, "visl", sizeof(dev->mdev.model));
4320c078e31SDaniel Almeida 	strscpy(dev->mdev.bus_info, "platform:visl",
4330c078e31SDaniel Almeida 		sizeof(dev->mdev.bus_info));
4340c078e31SDaniel Almeida 	media_device_init(&dev->mdev);
4350c078e31SDaniel Almeida 	dev->mdev.ops = &visl_m2m_media_ops;
4360c078e31SDaniel Almeida 	dev->v4l2_dev.mdev = &dev->mdev;
4370c078e31SDaniel Almeida 
4380c078e31SDaniel Almeida 	ret = video_register_device(vfd, VFL_TYPE_VIDEO, -1);
4390c078e31SDaniel Almeida 	if (ret) {
4400c078e31SDaniel Almeida 		v4l2_err(&dev->v4l2_dev, "Failed to register video device\n");
4410c078e31SDaniel Almeida 		goto error_m2m;
4420c078e31SDaniel Almeida 	}
4430c078e31SDaniel Almeida 
4440c078e31SDaniel Almeida 	v4l2_info(&dev->v4l2_dev,
4450c078e31SDaniel Almeida 		  "Device registered as /dev/video%d\n", vfd->num);
4460c078e31SDaniel Almeida 
4470c078e31SDaniel Almeida 	ret = v4l2_m2m_register_media_controller(dev->m2m_dev, vfd,
4480c078e31SDaniel Almeida 						 MEDIA_ENT_F_PROC_VIDEO_DECODER);
4490c078e31SDaniel Almeida 	if (ret) {
4500c078e31SDaniel Almeida 		v4l2_err(&dev->v4l2_dev, "Failed to init mem2mem media controller\n");
4510c078e31SDaniel Almeida 		goto error_v4l2;
4520c078e31SDaniel Almeida 	}
4530c078e31SDaniel Almeida 
4540c078e31SDaniel Almeida 	ret = media_device_register(&dev->mdev);
4550c078e31SDaniel Almeida 	if (ret) {
4560c078e31SDaniel Almeida 		v4l2_err(&dev->v4l2_dev, "Failed to register mem2mem media device\n");
4570c078e31SDaniel Almeida 		goto error_m2m_mc;
4580c078e31SDaniel Almeida 	}
4590c078e31SDaniel Almeida 
4600c078e31SDaniel Almeida 	rc = visl_debugfs_init(dev);
4610c078e31SDaniel Almeida 	if (rc)
4620c078e31SDaniel Almeida 		dprintk(dev, "visl_debugfs_init failed: %d\n"
4630c078e31SDaniel Almeida 			"Continuing without debugfs support\n", rc);
4640c078e31SDaniel Almeida 
4650c078e31SDaniel Almeida 	return 0;
4660c078e31SDaniel Almeida 
4670c078e31SDaniel Almeida error_m2m_mc:
4680c078e31SDaniel Almeida 	v4l2_m2m_unregister_media_controller(dev->m2m_dev);
4690c078e31SDaniel Almeida error_v4l2:
4700c078e31SDaniel Almeida 	video_unregister_device(&dev->vfd);
4710c078e31SDaniel Almeida 	/* visl_device_release called by video_unregister_device to release various objects */
4720c078e31SDaniel Almeida 	return ret;
4730c078e31SDaniel Almeida error_m2m:
4740c078e31SDaniel Almeida 	v4l2_m2m_release(dev->m2m_dev);
4750c078e31SDaniel Almeida error_dev:
4760c078e31SDaniel Almeida 	v4l2_device_unregister(&dev->v4l2_dev);
4770c078e31SDaniel Almeida error_visl_dev:
4780c078e31SDaniel Almeida 	kfree(dev);
4790c078e31SDaniel Almeida 
4800c078e31SDaniel Almeida 	return ret;
4810c078e31SDaniel Almeida }
4820c078e31SDaniel Almeida 
visl_remove(struct platform_device * pdev)483*1727b716SUwe Kleine-König static void visl_remove(struct platform_device *pdev)
4840c078e31SDaniel Almeida {
4850c078e31SDaniel Almeida 	struct visl_dev *dev = platform_get_drvdata(pdev);
4860c078e31SDaniel Almeida 
4870c078e31SDaniel Almeida 	v4l2_info(&dev->v4l2_dev, "Removing " VISL_NAME);
4880c078e31SDaniel Almeida 
4890c078e31SDaniel Almeida #ifdef CONFIG_MEDIA_CONTROLLER
4900c078e31SDaniel Almeida 	if (media_devnode_is_registered(dev->mdev.devnode)) {
4910c078e31SDaniel Almeida 		media_device_unregister(&dev->mdev);
4920c078e31SDaniel Almeida 		v4l2_m2m_unregister_media_controller(dev->m2m_dev);
4930c078e31SDaniel Almeida 	}
4940c078e31SDaniel Almeida #endif
4950c078e31SDaniel Almeida 	video_unregister_device(&dev->vfd);
4960c078e31SDaniel Almeida }
4970c078e31SDaniel Almeida 
4980c078e31SDaniel Almeida static struct platform_driver visl_pdrv = {
4990c078e31SDaniel Almeida 	.probe		= visl_probe,
500*1727b716SUwe Kleine-König 	.remove_new	= visl_remove,
5010c078e31SDaniel Almeida 	.driver		= {
5020c078e31SDaniel Almeida 		.name	= VISL_NAME,
5030c078e31SDaniel Almeida 	},
5040c078e31SDaniel Almeida };
5050c078e31SDaniel Almeida 
visl_dev_release(struct device * dev)5060c078e31SDaniel Almeida static void visl_dev_release(struct device *dev) {}
5070c078e31SDaniel Almeida 
5080c078e31SDaniel Almeida static struct platform_device visl_pdev = {
5090c078e31SDaniel Almeida 	.name		= VISL_NAME,
5100c078e31SDaniel Almeida 	.dev.release	= visl_dev_release,
5110c078e31SDaniel Almeida };
5120c078e31SDaniel Almeida 
visl_exit(void)5130c078e31SDaniel Almeida static void __exit visl_exit(void)
5140c078e31SDaniel Almeida {
5150c078e31SDaniel Almeida 	platform_driver_unregister(&visl_pdrv);
5160c078e31SDaniel Almeida 	platform_device_unregister(&visl_pdev);
5170c078e31SDaniel Almeida }
5180c078e31SDaniel Almeida 
visl_init(void)5190c078e31SDaniel Almeida static int __init visl_init(void)
5200c078e31SDaniel Almeida {
5210c078e31SDaniel Almeida 	int ret;
5220c078e31SDaniel Almeida 
5230c078e31SDaniel Almeida 	ret = platform_device_register(&visl_pdev);
5240c078e31SDaniel Almeida 	if (ret)
5250c078e31SDaniel Almeida 		return ret;
5260c078e31SDaniel Almeida 
5270c078e31SDaniel Almeida 	ret = platform_driver_register(&visl_pdrv);
5280c078e31SDaniel Almeida 	if (ret)
5290c078e31SDaniel Almeida 		platform_device_unregister(&visl_pdev);
5300c078e31SDaniel Almeida 
5310c078e31SDaniel Almeida 	return ret;
5320c078e31SDaniel Almeida }
5330c078e31SDaniel Almeida 
5340c078e31SDaniel Almeida MODULE_DESCRIPTION("Virtual stateless decoder device");
5350c078e31SDaniel Almeida MODULE_AUTHOR("Daniel Almeida <daniel.almeida@collabora.com>");
5360c078e31SDaniel Almeida MODULE_LICENSE("GPL");
5370c078e31SDaniel Almeida 
5380c078e31SDaniel Almeida module_init(visl_init);
5390c078e31SDaniel Almeida module_exit(visl_exit);
540