mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-08-05 16:54:27 +00:00

The code dealing with the 0 terminated meta_formats array is a bit klunky especially for the uvc_meta_v4l2_enum_formats() case. Instead of 0 terminating add an unsigned int nmeta_formats member to struct uvc_device and use that. This leads to slightly cleaner code. Signed-off-by: Hans de Goede <hansg@kernel.org> Reviewed-by: Ricardo Ribalda <ribalda@chrium.org> Tested-by: Ricardo Ribalda <ribalda@chromium.org> # Camera with MSXU_CONTROL_METADATA Link: https://lore.kernel.org/r/20250708104622.73237-2-hansg@kernel.org Signed-off-by: Hans Verkuil <hverkuil@xs4all.nl>
268 lines
7 KiB
C
268 lines
7 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* uvc_metadata.c -- USB Video Class driver - Metadata handling
|
|
*
|
|
* Copyright (C) 2016
|
|
* Guennadi Liakhovetski (guennadi.liakhovetski@intel.com)
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/list.h>
|
|
#include <linux/module.h>
|
|
#include <linux/usb.h>
|
|
#include <linux/usb/uvc.h>
|
|
#include <linux/videodev2.h>
|
|
|
|
#include <media/v4l2-ioctl.h>
|
|
#include <media/videobuf2-v4l2.h>
|
|
#include <media/videobuf2-vmalloc.h>
|
|
|
|
#include "uvcvideo.h"
|
|
|
|
/* -----------------------------------------------------------------------------
|
|
* V4L2 ioctls
|
|
*/
|
|
|
|
static int uvc_meta_v4l2_querycap(struct file *file, void *fh,
|
|
struct v4l2_capability *cap)
|
|
{
|
|
struct v4l2_fh *vfh = file->private_data;
|
|
struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
|
|
struct uvc_video_chain *chain = stream->chain;
|
|
|
|
strscpy(cap->driver, "uvcvideo", sizeof(cap->driver));
|
|
strscpy(cap->card, stream->dev->name, sizeof(cap->card));
|
|
usb_make_path(stream->dev->udev, cap->bus_info, sizeof(cap->bus_info));
|
|
cap->capabilities = V4L2_CAP_DEVICE_CAPS | V4L2_CAP_STREAMING
|
|
| chain->caps;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int uvc_meta_v4l2_get_format(struct file *file, void *fh,
|
|
struct v4l2_format *format)
|
|
{
|
|
struct v4l2_fh *vfh = file->private_data;
|
|
struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
|
|
struct v4l2_meta_format *fmt = &format->fmt.meta;
|
|
|
|
if (format->type != vfh->vdev->queue->type)
|
|
return -EINVAL;
|
|
|
|
memset(fmt, 0, sizeof(*fmt));
|
|
|
|
fmt->dataformat = stream->meta.format;
|
|
fmt->buffersize = UVC_METADATA_BUF_SIZE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int uvc_meta_v4l2_try_format(struct file *file, void *fh,
|
|
struct v4l2_format *format)
|
|
{
|
|
struct v4l2_fh *vfh = file->private_data;
|
|
struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
|
|
struct uvc_device *dev = stream->dev;
|
|
struct v4l2_meta_format *fmt = &format->fmt.meta;
|
|
u32 fmeta = V4L2_META_FMT_UVC;
|
|
|
|
if (format->type != vfh->vdev->queue->type)
|
|
return -EINVAL;
|
|
|
|
for (unsigned int i = 0; i < dev->nmeta_formats; i++)
|
|
if (dev->meta_formats[i] == fmt->dataformat) {
|
|
fmeta = fmt->dataformat;
|
|
break;
|
|
}
|
|
|
|
memset(fmt, 0, sizeof(*fmt));
|
|
|
|
fmt->dataformat = fmeta;
|
|
fmt->buffersize = UVC_METADATA_BUF_SIZE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int uvc_meta_v4l2_set_format(struct file *file, void *fh,
|
|
struct v4l2_format *format)
|
|
{
|
|
struct v4l2_fh *vfh = file->private_data;
|
|
struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
|
|
struct v4l2_meta_format *fmt = &format->fmt.meta;
|
|
int ret;
|
|
|
|
ret = uvc_meta_v4l2_try_format(file, fh, format);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
/*
|
|
* We could in principle switch at any time, also during streaming.
|
|
* Metadata buffers would still be perfectly parseable, but it's more
|
|
* consistent and cleaner to disallow that.
|
|
*/
|
|
mutex_lock(&stream->mutex);
|
|
|
|
if (vb2_is_busy(&stream->meta.queue.queue))
|
|
ret = -EBUSY;
|
|
else
|
|
stream->meta.format = fmt->dataformat;
|
|
|
|
mutex_unlock(&stream->mutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int uvc_meta_v4l2_enum_formats(struct file *file, void *fh,
|
|
struct v4l2_fmtdesc *fdesc)
|
|
{
|
|
struct v4l2_fh *vfh = file->private_data;
|
|
struct uvc_streaming *stream = video_get_drvdata(vfh->vdev);
|
|
struct uvc_device *dev = stream->dev;
|
|
u32 i = fdesc->index;
|
|
|
|
if (fdesc->type != vfh->vdev->queue->type)
|
|
return -EINVAL;
|
|
|
|
if (i >= dev->nmeta_formats)
|
|
return -EINVAL;
|
|
|
|
memset(fdesc, 0, sizeof(*fdesc));
|
|
|
|
fdesc->type = vfh->vdev->queue->type;
|
|
fdesc->index = i;
|
|
fdesc->pixelformat = dev->meta_formats[i];
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct v4l2_ioctl_ops uvc_meta_ioctl_ops = {
|
|
.vidioc_querycap = uvc_meta_v4l2_querycap,
|
|
.vidioc_g_fmt_meta_cap = uvc_meta_v4l2_get_format,
|
|
.vidioc_s_fmt_meta_cap = uvc_meta_v4l2_set_format,
|
|
.vidioc_try_fmt_meta_cap = uvc_meta_v4l2_try_format,
|
|
.vidioc_enum_fmt_meta_cap = uvc_meta_v4l2_enum_formats,
|
|
.vidioc_reqbufs = vb2_ioctl_reqbufs,
|
|
.vidioc_querybuf = vb2_ioctl_querybuf,
|
|
.vidioc_qbuf = vb2_ioctl_qbuf,
|
|
.vidioc_dqbuf = vb2_ioctl_dqbuf,
|
|
.vidioc_create_bufs = vb2_ioctl_create_bufs,
|
|
.vidioc_prepare_buf = vb2_ioctl_prepare_buf,
|
|
.vidioc_streamon = vb2_ioctl_streamon,
|
|
.vidioc_streamoff = vb2_ioctl_streamoff,
|
|
};
|
|
|
|
/* -----------------------------------------------------------------------------
|
|
* V4L2 File Operations
|
|
*/
|
|
|
|
static const struct v4l2_file_operations uvc_meta_fops = {
|
|
.owner = THIS_MODULE,
|
|
.unlocked_ioctl = video_ioctl2,
|
|
.open = v4l2_fh_open,
|
|
.release = vb2_fop_release,
|
|
.poll = vb2_fop_poll,
|
|
.mmap = vb2_fop_mmap,
|
|
};
|
|
|
|
static struct uvc_entity *uvc_meta_find_msxu(struct uvc_device *dev)
|
|
{
|
|
static const u8 uvc_msxu_guid[16] = UVC_GUID_MSXU_1_5;
|
|
struct uvc_entity *entity;
|
|
|
|
list_for_each_entry(entity, &dev->entities, list) {
|
|
if (!memcmp(entity->guid, uvc_msxu_guid, sizeof(entity->guid)))
|
|
return entity;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#define MSXU_CONTROL_METADATA 0x9
|
|
static int uvc_meta_detect_msxu(struct uvc_device *dev)
|
|
{
|
|
u32 *data __free(kfree) = NULL;
|
|
struct uvc_entity *entity;
|
|
int ret;
|
|
|
|
entity = uvc_meta_find_msxu(dev);
|
|
if (!entity)
|
|
return 0;
|
|
|
|
/*
|
|
* USB requires buffers aligned in a special way, simplest way is to
|
|
* make sure that query_ctrl will work is to kmalloc() them.
|
|
*/
|
|
data = kmalloc(sizeof(*data), GFP_KERNEL);
|
|
if (!data)
|
|
return -ENOMEM;
|
|
|
|
/* Check if the metadata is already enabled. */
|
|
ret = uvc_query_ctrl(dev, UVC_GET_CUR, entity->id, dev->intfnum,
|
|
MSXU_CONTROL_METADATA, data, sizeof(*data));
|
|
if (ret)
|
|
return 0;
|
|
|
|
if (*data) {
|
|
dev->quirks |= UVC_QUIRK_MSXU_META;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* We have seen devices that require 1 to enable the metadata, others
|
|
* requiring a value != 1 and others requiring a value >1. Luckily for
|
|
* us, the value from GET_MAX seems to work all the time.
|
|
*/
|
|
ret = uvc_query_ctrl(dev, UVC_GET_MAX, entity->id, dev->intfnum,
|
|
MSXU_CONTROL_METADATA, data, sizeof(*data));
|
|
if (ret || !*data)
|
|
return 0;
|
|
|
|
/*
|
|
* If we can set MSXU_CONTROL_METADATA, the device will report
|
|
* metadata.
|
|
*/
|
|
ret = uvc_query_ctrl(dev, UVC_SET_CUR, entity->id, dev->intfnum,
|
|
MSXU_CONTROL_METADATA, data, sizeof(*data));
|
|
if (!ret)
|
|
dev->quirks |= UVC_QUIRK_MSXU_META;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int uvc_meta_register(struct uvc_streaming *stream)
|
|
{
|
|
struct uvc_device *dev = stream->dev;
|
|
struct video_device *vdev = &stream->meta.vdev;
|
|
struct uvc_video_queue *queue = &stream->meta.queue;
|
|
|
|
stream->meta.format = V4L2_META_FMT_UVC;
|
|
|
|
return uvc_register_video_device(dev, stream, vdev, queue,
|
|
V4L2_BUF_TYPE_META_CAPTURE,
|
|
&uvc_meta_fops, &uvc_meta_ioctl_ops);
|
|
}
|
|
|
|
int uvc_meta_init(struct uvc_device *dev)
|
|
{
|
|
unsigned int i = 0;
|
|
int ret;
|
|
|
|
ret = uvc_meta_detect_msxu(dev);
|
|
if (ret)
|
|
return ret;
|
|
|
|
dev->meta_formats[i++] = V4L2_META_FMT_UVC;
|
|
|
|
if (dev->info->meta_format &&
|
|
!WARN_ON(dev->info->meta_format == V4L2_META_FMT_UVC))
|
|
dev->meta_formats[i++] = dev->info->meta_format;
|
|
|
|
if (dev->quirks & UVC_QUIRK_MSXU_META &&
|
|
!WARN_ON(dev->info->meta_format == V4L2_META_FMT_UVC_MSXU_1_5))
|
|
dev->meta_formats[i++] = V4L2_META_FMT_UVC_MSXU_1_5;
|
|
|
|
/* IMPORTANT: for new meta-formats update UVC_MAX_META_DATA_FORMATS. */
|
|
dev->nmeta_formats = i;
|
|
|
|
return 0;
|
|
}
|