162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci *      uvc_metadata.c  --  USB Video Class driver - Metadata handling
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci *      Copyright (C) 2016
662306a36Sopenharmony_ci *          Guennadi Liakhovetski (guennadi.liakhovetski@intel.com)
762306a36Sopenharmony_ci */
862306a36Sopenharmony_ci
962306a36Sopenharmony_ci#include <linux/kernel.h>
1062306a36Sopenharmony_ci#include <linux/list.h>
1162306a36Sopenharmony_ci#include <linux/module.h>
1262306a36Sopenharmony_ci#include <linux/usb.h>
1362306a36Sopenharmony_ci#include <linux/videodev2.h>
1462306a36Sopenharmony_ci
1562306a36Sopenharmony_ci#include <media/v4l2-ioctl.h>
1662306a36Sopenharmony_ci#include <media/videobuf2-v4l2.h>
1762306a36Sopenharmony_ci#include <media/videobuf2-vmalloc.h>
1862306a36Sopenharmony_ci
1962306a36Sopenharmony_ci#include "uvcvideo.h"
2062306a36Sopenharmony_ci
2162306a36Sopenharmony_ci/* -----------------------------------------------------------------------------
2262306a36Sopenharmony_ci * V4L2 ioctls
2362306a36Sopenharmony_ci */
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_cistatic int uvc_meta_v4l2_querycap(struct file *file, void *fh,
2662306a36Sopenharmony_ci				  struct v4l2_capability *cap)
2762306a36Sopenharmony_ci{
2862306a36Sopenharmony_ci	struct v4l2_fh *vfh = file->private_data;
2962306a36Sopenharmony_ci	struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
3062306a36Sopenharmony_ci	struct uvc_video_chain *chain = stream->chain;
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_ci	strscpy(cap->driver, "uvcvideo", sizeof(cap->driver));
3362306a36Sopenharmony_ci	strscpy(cap->card, stream->dev->name, sizeof(cap->card));
3462306a36Sopenharmony_ci	usb_make_path(stream->dev->udev, cap->bus_info, sizeof(cap->bus_info));
3562306a36Sopenharmony_ci	cap->capabilities = V4L2_CAP_DEVICE_CAPS | V4L2_CAP_STREAMING
3662306a36Sopenharmony_ci			  | chain->caps;
3762306a36Sopenharmony_ci
3862306a36Sopenharmony_ci	return 0;
3962306a36Sopenharmony_ci}
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_cistatic int uvc_meta_v4l2_get_format(struct file *file, void *fh,
4262306a36Sopenharmony_ci				    struct v4l2_format *format)
4362306a36Sopenharmony_ci{
4462306a36Sopenharmony_ci	struct v4l2_fh *vfh = file->private_data;
4562306a36Sopenharmony_ci	struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
4662306a36Sopenharmony_ci	struct v4l2_meta_format *fmt = &format->fmt.meta;
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_ci	if (format->type != vfh->vdev->queue->type)
4962306a36Sopenharmony_ci		return -EINVAL;
5062306a36Sopenharmony_ci
5162306a36Sopenharmony_ci	memset(fmt, 0, sizeof(*fmt));
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_ci	fmt->dataformat = stream->meta.format;
5462306a36Sopenharmony_ci	fmt->buffersize = UVC_METADATA_BUF_SIZE;
5562306a36Sopenharmony_ci
5662306a36Sopenharmony_ci	return 0;
5762306a36Sopenharmony_ci}
5862306a36Sopenharmony_ci
5962306a36Sopenharmony_cistatic int uvc_meta_v4l2_try_format(struct file *file, void *fh,
6062306a36Sopenharmony_ci				    struct v4l2_format *format)
6162306a36Sopenharmony_ci{
6262306a36Sopenharmony_ci	struct v4l2_fh *vfh = file->private_data;
6362306a36Sopenharmony_ci	struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
6462306a36Sopenharmony_ci	struct uvc_device *dev = stream->dev;
6562306a36Sopenharmony_ci	struct v4l2_meta_format *fmt = &format->fmt.meta;
6662306a36Sopenharmony_ci	u32 fmeta = fmt->dataformat;
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_ci	if (format->type != vfh->vdev->queue->type)
6962306a36Sopenharmony_ci		return -EINVAL;
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci	memset(fmt, 0, sizeof(*fmt));
7262306a36Sopenharmony_ci
7362306a36Sopenharmony_ci	fmt->dataformat = fmeta == dev->info->meta_format
7462306a36Sopenharmony_ci			? fmeta : V4L2_META_FMT_UVC;
7562306a36Sopenharmony_ci	fmt->buffersize = UVC_METADATA_BUF_SIZE;
7662306a36Sopenharmony_ci
7762306a36Sopenharmony_ci	return 0;
7862306a36Sopenharmony_ci}
7962306a36Sopenharmony_ci
8062306a36Sopenharmony_cistatic int uvc_meta_v4l2_set_format(struct file *file, void *fh,
8162306a36Sopenharmony_ci				    struct v4l2_format *format)
8262306a36Sopenharmony_ci{
8362306a36Sopenharmony_ci	struct v4l2_fh *vfh = file->private_data;
8462306a36Sopenharmony_ci	struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
8562306a36Sopenharmony_ci	struct v4l2_meta_format *fmt = &format->fmt.meta;
8662306a36Sopenharmony_ci	int ret;
8762306a36Sopenharmony_ci
8862306a36Sopenharmony_ci	ret = uvc_meta_v4l2_try_format(file, fh, format);
8962306a36Sopenharmony_ci	if (ret < 0)
9062306a36Sopenharmony_ci		return ret;
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_ci	/*
9362306a36Sopenharmony_ci	 * We could in principle switch at any time, also during streaming.
9462306a36Sopenharmony_ci	 * Metadata buffers would still be perfectly parseable, but it's more
9562306a36Sopenharmony_ci	 * consistent and cleaner to disallow that.
9662306a36Sopenharmony_ci	 */
9762306a36Sopenharmony_ci	mutex_lock(&stream->mutex);
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ci	if (uvc_queue_allocated(&stream->queue))
10062306a36Sopenharmony_ci		ret = -EBUSY;
10162306a36Sopenharmony_ci	else
10262306a36Sopenharmony_ci		stream->meta.format = fmt->dataformat;
10362306a36Sopenharmony_ci
10462306a36Sopenharmony_ci	mutex_unlock(&stream->mutex);
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ci	return ret;
10762306a36Sopenharmony_ci}
10862306a36Sopenharmony_ci
10962306a36Sopenharmony_cistatic int uvc_meta_v4l2_enum_formats(struct file *file, void *fh,
11062306a36Sopenharmony_ci				      struct v4l2_fmtdesc *fdesc)
11162306a36Sopenharmony_ci{
11262306a36Sopenharmony_ci	struct v4l2_fh *vfh = file->private_data;
11362306a36Sopenharmony_ci	struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
11462306a36Sopenharmony_ci	struct uvc_device *dev = stream->dev;
11562306a36Sopenharmony_ci	u32 index = fdesc->index;
11662306a36Sopenharmony_ci
11762306a36Sopenharmony_ci	if (fdesc->type != vfh->vdev->queue->type ||
11862306a36Sopenharmony_ci	    index > 1U || (index && !dev->info->meta_format))
11962306a36Sopenharmony_ci		return -EINVAL;
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_ci	memset(fdesc, 0, sizeof(*fdesc));
12262306a36Sopenharmony_ci
12362306a36Sopenharmony_ci	fdesc->type = vfh->vdev->queue->type;
12462306a36Sopenharmony_ci	fdesc->index = index;
12562306a36Sopenharmony_ci	fdesc->pixelformat = index ? dev->info->meta_format : V4L2_META_FMT_UVC;
12662306a36Sopenharmony_ci
12762306a36Sopenharmony_ci	return 0;
12862306a36Sopenharmony_ci}
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_cistatic const struct v4l2_ioctl_ops uvc_meta_ioctl_ops = {
13162306a36Sopenharmony_ci	.vidioc_querycap		= uvc_meta_v4l2_querycap,
13262306a36Sopenharmony_ci	.vidioc_g_fmt_meta_cap		= uvc_meta_v4l2_get_format,
13362306a36Sopenharmony_ci	.vidioc_s_fmt_meta_cap		= uvc_meta_v4l2_set_format,
13462306a36Sopenharmony_ci	.vidioc_try_fmt_meta_cap	= uvc_meta_v4l2_try_format,
13562306a36Sopenharmony_ci	.vidioc_enum_fmt_meta_cap	= uvc_meta_v4l2_enum_formats,
13662306a36Sopenharmony_ci	.vidioc_reqbufs			= vb2_ioctl_reqbufs,
13762306a36Sopenharmony_ci	.vidioc_querybuf		= vb2_ioctl_querybuf,
13862306a36Sopenharmony_ci	.vidioc_qbuf			= vb2_ioctl_qbuf,
13962306a36Sopenharmony_ci	.vidioc_dqbuf			= vb2_ioctl_dqbuf,
14062306a36Sopenharmony_ci	.vidioc_create_bufs		= vb2_ioctl_create_bufs,
14162306a36Sopenharmony_ci	.vidioc_prepare_buf		= vb2_ioctl_prepare_buf,
14262306a36Sopenharmony_ci	.vidioc_streamon		= vb2_ioctl_streamon,
14362306a36Sopenharmony_ci	.vidioc_streamoff		= vb2_ioctl_streamoff,
14462306a36Sopenharmony_ci};
14562306a36Sopenharmony_ci
14662306a36Sopenharmony_ci/* -----------------------------------------------------------------------------
14762306a36Sopenharmony_ci * V4L2 File Operations
14862306a36Sopenharmony_ci */
14962306a36Sopenharmony_ci
15062306a36Sopenharmony_cistatic const struct v4l2_file_operations uvc_meta_fops = {
15162306a36Sopenharmony_ci	.owner = THIS_MODULE,
15262306a36Sopenharmony_ci	.unlocked_ioctl = video_ioctl2,
15362306a36Sopenharmony_ci	.open = v4l2_fh_open,
15462306a36Sopenharmony_ci	.release = vb2_fop_release,
15562306a36Sopenharmony_ci	.poll = vb2_fop_poll,
15662306a36Sopenharmony_ci	.mmap = vb2_fop_mmap,
15762306a36Sopenharmony_ci};
15862306a36Sopenharmony_ci
15962306a36Sopenharmony_ciint uvc_meta_register(struct uvc_streaming *stream)
16062306a36Sopenharmony_ci{
16162306a36Sopenharmony_ci	struct uvc_device *dev = stream->dev;
16262306a36Sopenharmony_ci	struct video_device *vdev = &stream->meta.vdev;
16362306a36Sopenharmony_ci	struct uvc_video_queue *queue = &stream->meta.queue;
16462306a36Sopenharmony_ci
16562306a36Sopenharmony_ci	stream->meta.format = V4L2_META_FMT_UVC;
16662306a36Sopenharmony_ci
16762306a36Sopenharmony_ci	/*
16862306a36Sopenharmony_ci	 * The video interface queue uses manual locking and thus does not set
16962306a36Sopenharmony_ci	 * the queue pointer. Set it manually here.
17062306a36Sopenharmony_ci	 */
17162306a36Sopenharmony_ci	vdev->queue = &queue->queue;
17262306a36Sopenharmony_ci
17362306a36Sopenharmony_ci	return uvc_register_video_device(dev, stream, vdev, queue,
17462306a36Sopenharmony_ci					 V4L2_BUF_TYPE_META_CAPTURE,
17562306a36Sopenharmony_ci					 &uvc_meta_fops, &uvc_meta_ioctl_ops);
17662306a36Sopenharmony_ci}
177