18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci *      uvc_metadata.c  --  USB Video Class driver - Metadata handling
48c2ecf20Sopenharmony_ci *
58c2ecf20Sopenharmony_ci *      Copyright (C) 2016
68c2ecf20Sopenharmony_ci *          Guennadi Liakhovetski (guennadi.liakhovetski@intel.com)
78c2ecf20Sopenharmony_ci */
88c2ecf20Sopenharmony_ci
98c2ecf20Sopenharmony_ci#include <linux/kernel.h>
108c2ecf20Sopenharmony_ci#include <linux/list.h>
118c2ecf20Sopenharmony_ci#include <linux/module.h>
128c2ecf20Sopenharmony_ci#include <linux/usb.h>
138c2ecf20Sopenharmony_ci#include <linux/videodev2.h>
148c2ecf20Sopenharmony_ci
158c2ecf20Sopenharmony_ci#include <media/v4l2-ioctl.h>
168c2ecf20Sopenharmony_ci#include <media/videobuf2-v4l2.h>
178c2ecf20Sopenharmony_ci#include <media/videobuf2-vmalloc.h>
188c2ecf20Sopenharmony_ci
198c2ecf20Sopenharmony_ci#include "uvcvideo.h"
208c2ecf20Sopenharmony_ci
218c2ecf20Sopenharmony_ci/* -----------------------------------------------------------------------------
228c2ecf20Sopenharmony_ci * V4L2 ioctls
238c2ecf20Sopenharmony_ci */
248c2ecf20Sopenharmony_ci
258c2ecf20Sopenharmony_cistatic int uvc_meta_v4l2_querycap(struct file *file, void *fh,
268c2ecf20Sopenharmony_ci				  struct v4l2_capability *cap)
278c2ecf20Sopenharmony_ci{
288c2ecf20Sopenharmony_ci	struct v4l2_fh *vfh = file->private_data;
298c2ecf20Sopenharmony_ci	struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
308c2ecf20Sopenharmony_ci	struct uvc_video_chain *chain = stream->chain;
318c2ecf20Sopenharmony_ci
328c2ecf20Sopenharmony_ci	strscpy(cap->driver, "uvcvideo", sizeof(cap->driver));
338c2ecf20Sopenharmony_ci	strscpy(cap->card, vfh->vdev->name, sizeof(cap->card));
348c2ecf20Sopenharmony_ci	usb_make_path(stream->dev->udev, cap->bus_info, sizeof(cap->bus_info));
358c2ecf20Sopenharmony_ci	cap->capabilities = V4L2_CAP_DEVICE_CAPS | V4L2_CAP_STREAMING
368c2ecf20Sopenharmony_ci			  | chain->caps;
378c2ecf20Sopenharmony_ci
388c2ecf20Sopenharmony_ci	return 0;
398c2ecf20Sopenharmony_ci}
408c2ecf20Sopenharmony_ci
418c2ecf20Sopenharmony_cistatic int uvc_meta_v4l2_get_format(struct file *file, void *fh,
428c2ecf20Sopenharmony_ci				    struct v4l2_format *format)
438c2ecf20Sopenharmony_ci{
448c2ecf20Sopenharmony_ci	struct v4l2_fh *vfh = file->private_data;
458c2ecf20Sopenharmony_ci	struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
468c2ecf20Sopenharmony_ci	struct v4l2_meta_format *fmt = &format->fmt.meta;
478c2ecf20Sopenharmony_ci
488c2ecf20Sopenharmony_ci	if (format->type != vfh->vdev->queue->type)
498c2ecf20Sopenharmony_ci		return -EINVAL;
508c2ecf20Sopenharmony_ci
518c2ecf20Sopenharmony_ci	memset(fmt, 0, sizeof(*fmt));
528c2ecf20Sopenharmony_ci
538c2ecf20Sopenharmony_ci	fmt->dataformat = stream->meta.format;
548c2ecf20Sopenharmony_ci	fmt->buffersize = UVC_METADATA_BUF_SIZE;
558c2ecf20Sopenharmony_ci
568c2ecf20Sopenharmony_ci	return 0;
578c2ecf20Sopenharmony_ci}
588c2ecf20Sopenharmony_ci
598c2ecf20Sopenharmony_cistatic int uvc_meta_v4l2_try_format(struct file *file, void *fh,
608c2ecf20Sopenharmony_ci				    struct v4l2_format *format)
618c2ecf20Sopenharmony_ci{
628c2ecf20Sopenharmony_ci	struct v4l2_fh *vfh = file->private_data;
638c2ecf20Sopenharmony_ci	struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
648c2ecf20Sopenharmony_ci	struct uvc_device *dev = stream->dev;
658c2ecf20Sopenharmony_ci	struct v4l2_meta_format *fmt = &format->fmt.meta;
668c2ecf20Sopenharmony_ci	u32 fmeta = fmt->dataformat;
678c2ecf20Sopenharmony_ci
688c2ecf20Sopenharmony_ci	if (format->type != vfh->vdev->queue->type)
698c2ecf20Sopenharmony_ci		return -EINVAL;
708c2ecf20Sopenharmony_ci
718c2ecf20Sopenharmony_ci	memset(fmt, 0, sizeof(*fmt));
728c2ecf20Sopenharmony_ci
738c2ecf20Sopenharmony_ci	fmt->dataformat = fmeta == dev->info->meta_format
748c2ecf20Sopenharmony_ci			? fmeta : V4L2_META_FMT_UVC;
758c2ecf20Sopenharmony_ci	fmt->buffersize = UVC_METADATA_BUF_SIZE;
768c2ecf20Sopenharmony_ci
778c2ecf20Sopenharmony_ci	return 0;
788c2ecf20Sopenharmony_ci}
798c2ecf20Sopenharmony_ci
808c2ecf20Sopenharmony_cistatic int uvc_meta_v4l2_set_format(struct file *file, void *fh,
818c2ecf20Sopenharmony_ci				    struct v4l2_format *format)
828c2ecf20Sopenharmony_ci{
838c2ecf20Sopenharmony_ci	struct v4l2_fh *vfh = file->private_data;
848c2ecf20Sopenharmony_ci	struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
858c2ecf20Sopenharmony_ci	struct v4l2_meta_format *fmt = &format->fmt.meta;
868c2ecf20Sopenharmony_ci	int ret;
878c2ecf20Sopenharmony_ci
888c2ecf20Sopenharmony_ci	ret = uvc_meta_v4l2_try_format(file, fh, format);
898c2ecf20Sopenharmony_ci	if (ret < 0)
908c2ecf20Sopenharmony_ci		return ret;
918c2ecf20Sopenharmony_ci
928c2ecf20Sopenharmony_ci	/*
938c2ecf20Sopenharmony_ci	 * We could in principle switch at any time, also during streaming.
948c2ecf20Sopenharmony_ci	 * Metadata buffers would still be perfectly parseable, but it's more
958c2ecf20Sopenharmony_ci	 * consistent and cleaner to disallow that.
968c2ecf20Sopenharmony_ci	 */
978c2ecf20Sopenharmony_ci	mutex_lock(&stream->mutex);
988c2ecf20Sopenharmony_ci
998c2ecf20Sopenharmony_ci	if (uvc_queue_allocated(&stream->queue))
1008c2ecf20Sopenharmony_ci		ret = -EBUSY;
1018c2ecf20Sopenharmony_ci	else
1028c2ecf20Sopenharmony_ci		stream->meta.format = fmt->dataformat;
1038c2ecf20Sopenharmony_ci
1048c2ecf20Sopenharmony_ci	mutex_unlock(&stream->mutex);
1058c2ecf20Sopenharmony_ci
1068c2ecf20Sopenharmony_ci	return ret;
1078c2ecf20Sopenharmony_ci}
1088c2ecf20Sopenharmony_ci
1098c2ecf20Sopenharmony_cistatic int uvc_meta_v4l2_enum_formats(struct file *file, void *fh,
1108c2ecf20Sopenharmony_ci				      struct v4l2_fmtdesc *fdesc)
1118c2ecf20Sopenharmony_ci{
1128c2ecf20Sopenharmony_ci	struct v4l2_fh *vfh = file->private_data;
1138c2ecf20Sopenharmony_ci	struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
1148c2ecf20Sopenharmony_ci	struct uvc_device *dev = stream->dev;
1158c2ecf20Sopenharmony_ci	u32 index = fdesc->index;
1168c2ecf20Sopenharmony_ci
1178c2ecf20Sopenharmony_ci	if (fdesc->type != vfh->vdev->queue->type ||
1188c2ecf20Sopenharmony_ci	    index > 1U || (index && !dev->info->meta_format))
1198c2ecf20Sopenharmony_ci		return -EINVAL;
1208c2ecf20Sopenharmony_ci
1218c2ecf20Sopenharmony_ci	memset(fdesc, 0, sizeof(*fdesc));
1228c2ecf20Sopenharmony_ci
1238c2ecf20Sopenharmony_ci	fdesc->type = vfh->vdev->queue->type;
1248c2ecf20Sopenharmony_ci	fdesc->index = index;
1258c2ecf20Sopenharmony_ci	fdesc->pixelformat = index ? dev->info->meta_format : V4L2_META_FMT_UVC;
1268c2ecf20Sopenharmony_ci
1278c2ecf20Sopenharmony_ci	return 0;
1288c2ecf20Sopenharmony_ci}
1298c2ecf20Sopenharmony_ci
1308c2ecf20Sopenharmony_cistatic const struct v4l2_ioctl_ops uvc_meta_ioctl_ops = {
1318c2ecf20Sopenharmony_ci	.vidioc_querycap		= uvc_meta_v4l2_querycap,
1328c2ecf20Sopenharmony_ci	.vidioc_g_fmt_meta_cap		= uvc_meta_v4l2_get_format,
1338c2ecf20Sopenharmony_ci	.vidioc_s_fmt_meta_cap		= uvc_meta_v4l2_set_format,
1348c2ecf20Sopenharmony_ci	.vidioc_try_fmt_meta_cap	= uvc_meta_v4l2_try_format,
1358c2ecf20Sopenharmony_ci	.vidioc_enum_fmt_meta_cap	= uvc_meta_v4l2_enum_formats,
1368c2ecf20Sopenharmony_ci	.vidioc_reqbufs			= vb2_ioctl_reqbufs,
1378c2ecf20Sopenharmony_ci	.vidioc_querybuf		= vb2_ioctl_querybuf,
1388c2ecf20Sopenharmony_ci	.vidioc_qbuf			= vb2_ioctl_qbuf,
1398c2ecf20Sopenharmony_ci	.vidioc_dqbuf			= vb2_ioctl_dqbuf,
1408c2ecf20Sopenharmony_ci	.vidioc_create_bufs		= vb2_ioctl_create_bufs,
1418c2ecf20Sopenharmony_ci	.vidioc_prepare_buf		= vb2_ioctl_prepare_buf,
1428c2ecf20Sopenharmony_ci	.vidioc_streamon		= vb2_ioctl_streamon,
1438c2ecf20Sopenharmony_ci	.vidioc_streamoff		= vb2_ioctl_streamoff,
1448c2ecf20Sopenharmony_ci};
1458c2ecf20Sopenharmony_ci
1468c2ecf20Sopenharmony_ci/* -----------------------------------------------------------------------------
1478c2ecf20Sopenharmony_ci * V4L2 File Operations
1488c2ecf20Sopenharmony_ci */
1498c2ecf20Sopenharmony_ci
1508c2ecf20Sopenharmony_cistatic const struct v4l2_file_operations uvc_meta_fops = {
1518c2ecf20Sopenharmony_ci	.owner = THIS_MODULE,
1528c2ecf20Sopenharmony_ci	.unlocked_ioctl = video_ioctl2,
1538c2ecf20Sopenharmony_ci	.open = v4l2_fh_open,
1548c2ecf20Sopenharmony_ci	.release = vb2_fop_release,
1558c2ecf20Sopenharmony_ci	.poll = vb2_fop_poll,
1568c2ecf20Sopenharmony_ci	.mmap = vb2_fop_mmap,
1578c2ecf20Sopenharmony_ci};
1588c2ecf20Sopenharmony_ci
1598c2ecf20Sopenharmony_ciint uvc_meta_register(struct uvc_streaming *stream)
1608c2ecf20Sopenharmony_ci{
1618c2ecf20Sopenharmony_ci	struct uvc_device *dev = stream->dev;
1628c2ecf20Sopenharmony_ci	struct video_device *vdev = &stream->meta.vdev;
1638c2ecf20Sopenharmony_ci	struct uvc_video_queue *queue = &stream->meta.queue;
1648c2ecf20Sopenharmony_ci
1658c2ecf20Sopenharmony_ci	stream->meta.format = V4L2_META_FMT_UVC;
1668c2ecf20Sopenharmony_ci
1678c2ecf20Sopenharmony_ci	/*
1688c2ecf20Sopenharmony_ci	 * The video interface queue uses manual locking and thus does not set
1698c2ecf20Sopenharmony_ci	 * the queue pointer. Set it manually here.
1708c2ecf20Sopenharmony_ci	 */
1718c2ecf20Sopenharmony_ci	vdev->queue = &queue->queue;
1728c2ecf20Sopenharmony_ci
1738c2ecf20Sopenharmony_ci	return uvc_register_video_device(dev, stream, vdev, queue,
1748c2ecf20Sopenharmony_ci					 V4L2_BUF_TYPE_META_CAPTURE,
1758c2ecf20Sopenharmony_ci					 &uvc_meta_fops, &uvc_meta_ioctl_ops);
1768c2ecf20Sopenharmony_ci}
177