Merge git://git.kernel.org/pub/scm/linux/kernel/git/mchehab/v4l-dvb
* git://git.kernel.org/pub/scm/linux/kernel/git/mchehab/v4l-dvb: V4L/DVB (8178): uvc: Fix compilation breakage for the other drivers, if uvc is selected V4L/DVB (8145a): USB Video Class driver
This commit is contained in:
commit
3a57a78875
|
@ -4314,6 +4314,14 @@ L: netdev@vger.kernel.org
|
|||
W: http://www.linux-usb.org/usbnet
|
||||
S: Maintained
|
||||
|
||||
USB VIDEO CLASS
|
||||
P: Laurent Pinchart
|
||||
M: laurent.pinchart@skynet.be
|
||||
L: linx-uvc-devel@berlios.de
|
||||
L: video4linux-list@redhat.com
|
||||
W: http://linux-uvc.berlios.de
|
||||
S: Maintained
|
||||
|
||||
USB W996[87]CF DRIVER
|
||||
P: Luca Risolia
|
||||
M: luca.risolia@studio.unibo.it
|
||||
|
|
|
@ -793,6 +793,14 @@ menuconfig V4L_USB_DRIVERS
|
|||
|
||||
if V4L_USB_DRIVERS && USB
|
||||
|
||||
config USB_VIDEO_CLASS
|
||||
tristate "USB Video Class (UVC)"
|
||||
---help---
|
||||
Support for the USB Video Class (UVC). Currently only video
|
||||
input devices, such as webcams, are supported.
|
||||
|
||||
For more information see: <http://linux-uvc.berlios.de/>
|
||||
|
||||
source "drivers/media/video/pvrusb2/Kconfig"
|
||||
|
||||
source "drivers/media/video/em28xx/Kconfig"
|
||||
|
|
|
@ -136,6 +136,8 @@ obj-$(CONFIG_SOC_CAMERA_MT9V022) += mt9v022.o
|
|||
|
||||
obj-$(CONFIG_VIDEO_AU0828) += au0828/
|
||||
|
||||
obj-$(CONFIG_USB_VIDEO_CLASS) += uvc/
|
||||
|
||||
EXTRA_CFLAGS += -Idrivers/media/dvb/dvb-core
|
||||
EXTRA_CFLAGS += -Idrivers/media/dvb/frontends
|
||||
EXTRA_CFLAGS += -Idrivers/media/common/tuners
|
||||
|
|
|
@ -0,0 +1,3 @@
|
|||
uvcvideo-objs := uvc_driver.o uvc_queue.o uvc_v4l2.o uvc_video.o uvc_ctrl.o \
|
||||
uvc_status.o uvc_isight.o
|
||||
obj-$(CONFIG_USB_VIDEO_CLASS) += uvcvideo.o
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,134 @@
|
|||
/*
|
||||
* uvc_isight.c -- USB Video Class driver - iSight support
|
||||
*
|
||||
* Copyright (C) 2006-2007
|
||||
* Ivan N. Zlatev <contact@i-nz.net>
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/usb.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/mm.h>
|
||||
|
||||
#include "uvcvideo.h"
|
||||
|
||||
/* Built-in iSight webcams implements most of UVC 1.0 except a
|
||||
* different packet format. Instead of sending a header at the
|
||||
* beginning of each isochronous transfer payload, the webcam sends a
|
||||
* single header per image (on its own in a packet), followed by
|
||||
* packets containing data only.
|
||||
*
|
||||
* Offset Size (bytes) Description
|
||||
* ------------------------------------------------------------------
|
||||
* 0x00 1 Header length
|
||||
* 0x01 1 Flags (UVC-compliant)
|
||||
* 0x02 4 Always equal to '11223344'
|
||||
* 0x06 8 Always equal to 'deadbeefdeadface'
|
||||
* 0x0e 16 Unknown
|
||||
*
|
||||
* The header can be prefixed by an optional, unknown-purpose byte.
|
||||
*/
|
||||
|
||||
static int isight_decode(struct uvc_video_queue *queue, struct uvc_buffer *buf,
|
||||
const __u8 *data, unsigned int len)
|
||||
{
|
||||
static const __u8 hdr[] = {
|
||||
0x11, 0x22, 0x33, 0x44,
|
||||
0xde, 0xad, 0xbe, 0xef,
|
||||
0xde, 0xad, 0xfa, 0xce
|
||||
};
|
||||
|
||||
unsigned int maxlen, nbytes;
|
||||
__u8 *mem;
|
||||
int is_header = 0;
|
||||
|
||||
if (buf == NULL)
|
||||
return 0;
|
||||
|
||||
if ((len >= 14 && memcmp(&data[2], hdr, 12) == 0) ||
|
||||
(len >= 15 && memcmp(&data[3], hdr, 12) == 0)) {
|
||||
uvc_trace(UVC_TRACE_FRAME, "iSight header found\n");
|
||||
is_header = 1;
|
||||
}
|
||||
|
||||
/* Synchronize to the input stream by waiting for a header packet. */
|
||||
if (buf->state != UVC_BUF_STATE_ACTIVE) {
|
||||
if (!is_header) {
|
||||
uvc_trace(UVC_TRACE_FRAME, "Dropping packet (out of "
|
||||
"sync).\n");
|
||||
return 0;
|
||||
}
|
||||
|
||||
buf->state = UVC_BUF_STATE_ACTIVE;
|
||||
}
|
||||
|
||||
/* Mark the buffer as done if we're at the beginning of a new frame.
|
||||
*
|
||||
* Empty buffers (bytesused == 0) don't trigger end of frame detection
|
||||
* as it doesn't make sense to return an empty buffer.
|
||||
*/
|
||||
if (is_header && buf->buf.bytesused != 0) {
|
||||
buf->state = UVC_BUF_STATE_DONE;
|
||||
return -EAGAIN;
|
||||
}
|
||||
|
||||
/* Copy the video data to the buffer. Skip header packets, as they
|
||||
* contain no data.
|
||||
*/
|
||||
if (!is_header) {
|
||||
maxlen = buf->buf.length - buf->buf.bytesused;
|
||||
mem = queue->mem + buf->buf.m.offset + buf->buf.bytesused;
|
||||
nbytes = min(len, maxlen);
|
||||
memcpy(mem, data, nbytes);
|
||||
buf->buf.bytesused += nbytes;
|
||||
|
||||
if (len > maxlen || buf->buf.bytesused == buf->buf.length) {
|
||||
uvc_trace(UVC_TRACE_FRAME, "Frame complete "
|
||||
"(overflow).\n");
|
||||
buf->state = UVC_BUF_STATE_DONE;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void uvc_video_decode_isight(struct urb *urb, struct uvc_video_device *video,
|
||||
struct uvc_buffer *buf)
|
||||
{
|
||||
int ret, i;
|
||||
|
||||
for (i = 0; i < urb->number_of_packets; ++i) {
|
||||
if (urb->iso_frame_desc[i].status < 0) {
|
||||
uvc_trace(UVC_TRACE_FRAME, "USB isochronous frame "
|
||||
"lost (%d).\n",
|
||||
urb->iso_frame_desc[i].status);
|
||||
}
|
||||
|
||||
/* Decode the payload packet.
|
||||
* uvc_video_decode is entered twice when a frame transition
|
||||
* has been detected because the end of frame can only be
|
||||
* reliably detected when the first packet of the new frame
|
||||
* is processed. The first pass detects the transition and
|
||||
* closes the previous frame's buffer, the second pass
|
||||
* processes the data of the first payload of the new frame.
|
||||
*/
|
||||
do {
|
||||
ret = isight_decode(&video->queue, buf,
|
||||
urb->transfer_buffer +
|
||||
urb->iso_frame_desc[i].offset,
|
||||
urb->iso_frame_desc[i].actual_length);
|
||||
|
||||
if (buf == NULL)
|
||||
break;
|
||||
|
||||
if (buf->state == UVC_BUF_STATE_DONE ||
|
||||
buf->state == UVC_BUF_STATE_ERROR)
|
||||
buf = uvc_queue_next_buffer(&video->queue, buf);
|
||||
} while (ret == -EAGAIN);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,477 @@
|
|||
/*
|
||||
* uvc_queue.c -- USB Video Class driver - Buffers management
|
||||
*
|
||||
* Copyright (C) 2005-2008
|
||||
* Laurent Pinchart (laurent.pinchart@skynet.be)
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/version.h>
|
||||
#include <linux/list.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/usb.h>
|
||||
#include <linux/videodev2.h>
|
||||
#include <linux/vmalloc.h>
|
||||
#include <linux/wait.h>
|
||||
#include <asm/atomic.h>
|
||||
|
||||
#include "uvcvideo.h"
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* Video buffers queue management.
|
||||
*
|
||||
* Video queues is initialized by uvc_queue_init(). The function performs
|
||||
* basic initialization of the uvc_video_queue struct and never fails.
|
||||
*
|
||||
* Video buffer allocation and freeing are performed by uvc_alloc_buffers and
|
||||
* uvc_free_buffers respectively. The former acquires the video queue lock,
|
||||
* while the later must be called with the lock held (so that allocation can
|
||||
* free previously allocated buffers). Trying to free buffers that are mapped
|
||||
* to user space will return -EBUSY.
|
||||
*
|
||||
* Video buffers are managed using two queues. However, unlike most USB video
|
||||
* drivers which use an in queue and an out queue, we use a main queue which
|
||||
* holds all queued buffers (both 'empty' and 'done' buffers), and an irq
|
||||
* queue which holds empty buffers. This design (copied from video-buf)
|
||||
* minimizes locking in interrupt, as only one queue is shared between
|
||||
* interrupt and user contexts.
|
||||
*
|
||||
* Use cases
|
||||
* ---------
|
||||
*
|
||||
* Unless stated otherwise, all operations which modify the irq buffers queue
|
||||
* are protected by the irq spinlock.
|
||||
*
|
||||
* 1. The user queues the buffers, starts streaming and dequeues a buffer.
|
||||
*
|
||||
* The buffers are added to the main and irq queues. Both operations are
|
||||
* protected by the queue lock, and the latert is protected by the irq
|
||||
* spinlock as well.
|
||||
*
|
||||
* The completion handler fetches a buffer from the irq queue and fills it
|
||||
* with video data. If no buffer is available (irq queue empty), the handler
|
||||
* returns immediately.
|
||||
*
|
||||
* When the buffer is full, the completion handler removes it from the irq
|
||||
* queue, marks it as ready (UVC_BUF_STATE_DONE) and wake its wait queue.
|
||||
* At that point, any process waiting on the buffer will be woken up. If a
|
||||
* process tries to dequeue a buffer after it has been marked ready, the
|
||||
* dequeing will succeed immediately.
|
||||
*
|
||||
* 2. Buffers are queued, user is waiting on a buffer and the device gets
|
||||
* disconnected.
|
||||
*
|
||||
* When the device is disconnected, the kernel calls the completion handler
|
||||
* with an appropriate status code. The handler marks all buffers in the
|
||||
* irq queue as being erroneous (UVC_BUF_STATE_ERROR) and wakes them up so
|
||||
* that any process waiting on a buffer gets woken up.
|
||||
*
|
||||
* Waking up up the first buffer on the irq list is not enough, as the
|
||||
* process waiting on the buffer might restart the dequeue operation
|
||||
* immediately.
|
||||
*
|
||||
*/
|
||||
|
||||
void uvc_queue_init(struct uvc_video_queue *queue)
|
||||
{
|
||||
mutex_init(&queue->mutex);
|
||||
spin_lock_init(&queue->irqlock);
|
||||
INIT_LIST_HEAD(&queue->mainqueue);
|
||||
INIT_LIST_HEAD(&queue->irqqueue);
|
||||
}
|
||||
|
||||
/*
|
||||
* Allocate the video buffers.
|
||||
*
|
||||
* Pages are reserved to make sure they will not be swaped, as they will be
|
||||
* filled in URB completion handler.
|
||||
*
|
||||
* Buffers will be individually mapped, so they must all be page aligned.
|
||||
*/
|
||||
int uvc_alloc_buffers(struct uvc_video_queue *queue, unsigned int nbuffers,
|
||||
unsigned int buflength)
|
||||
{
|
||||
unsigned int bufsize = PAGE_ALIGN(buflength);
|
||||
unsigned int i;
|
||||
void *mem = NULL;
|
||||
int ret;
|
||||
|
||||
if (nbuffers > UVC_MAX_VIDEO_BUFFERS)
|
||||
nbuffers = UVC_MAX_VIDEO_BUFFERS;
|
||||
|
||||
mutex_lock(&queue->mutex);
|
||||
|
||||
if ((ret = uvc_free_buffers(queue)) < 0)
|
||||
goto done;
|
||||
|
||||
/* Bail out if no buffers should be allocated. */
|
||||
if (nbuffers == 0)
|
||||
goto done;
|
||||
|
||||
/* Decrement the number of buffers until allocation succeeds. */
|
||||
for (; nbuffers > 0; --nbuffers) {
|
||||
mem = vmalloc_32(nbuffers * bufsize);
|
||||
if (mem != NULL)
|
||||
break;
|
||||
}
|
||||
|
||||
if (mem == NULL) {
|
||||
ret = -ENOMEM;
|
||||
goto done;
|
||||
}
|
||||
|
||||
for (i = 0; i < nbuffers; ++i) {
|
||||
memset(&queue->buffer[i], 0, sizeof queue->buffer[i]);
|
||||
queue->buffer[i].buf.index = i;
|
||||
queue->buffer[i].buf.m.offset = i * bufsize;
|
||||
queue->buffer[i].buf.length = buflength;
|
||||
queue->buffer[i].buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
||||
queue->buffer[i].buf.sequence = 0;
|
||||
queue->buffer[i].buf.field = V4L2_FIELD_NONE;
|
||||
queue->buffer[i].buf.memory = V4L2_MEMORY_MMAP;
|
||||
queue->buffer[i].buf.flags = 0;
|
||||
init_waitqueue_head(&queue->buffer[i].wait);
|
||||
}
|
||||
|
||||
queue->mem = mem;
|
||||
queue->count = nbuffers;
|
||||
queue->buf_size = bufsize;
|
||||
ret = nbuffers;
|
||||
|
||||
done:
|
||||
mutex_unlock(&queue->mutex);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* Free the video buffers.
|
||||
*
|
||||
* This function must be called with the queue lock held.
|
||||
*/
|
||||
int uvc_free_buffers(struct uvc_video_queue *queue)
|
||||
{
|
||||
unsigned int i;
|
||||
|
||||
for (i = 0; i < queue->count; ++i) {
|
||||
if (queue->buffer[i].vma_use_count != 0)
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
if (queue->count) {
|
||||
vfree(queue->mem);
|
||||
queue->count = 0;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void __uvc_query_buffer(struct uvc_buffer *buf,
|
||||
struct v4l2_buffer *v4l2_buf)
|
||||
{
|
||||
memcpy(v4l2_buf, &buf->buf, sizeof *v4l2_buf);
|
||||
|
||||
if (buf->vma_use_count)
|
||||
v4l2_buf->flags |= V4L2_BUF_FLAG_MAPPED;
|
||||
|
||||
switch (buf->state) {
|
||||
case UVC_BUF_STATE_ERROR:
|
||||
case UVC_BUF_STATE_DONE:
|
||||
v4l2_buf->flags |= V4L2_BUF_FLAG_DONE;
|
||||
break;
|
||||
case UVC_BUF_STATE_QUEUED:
|
||||
case UVC_BUF_STATE_ACTIVE:
|
||||
v4l2_buf->flags |= V4L2_BUF_FLAG_QUEUED;
|
||||
break;
|
||||
case UVC_BUF_STATE_IDLE:
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
int uvc_query_buffer(struct uvc_video_queue *queue,
|
||||
struct v4l2_buffer *v4l2_buf)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
mutex_lock(&queue->mutex);
|
||||
if (v4l2_buf->index >= queue->count) {
|
||||
ret = -EINVAL;
|
||||
goto done;
|
||||
}
|
||||
|
||||
__uvc_query_buffer(&queue->buffer[v4l2_buf->index], v4l2_buf);
|
||||
|
||||
done:
|
||||
mutex_unlock(&queue->mutex);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* Queue a video buffer. Attempting to queue a buffer that has already been
|
||||
* queued will return -EINVAL.
|
||||
*/
|
||||
int uvc_queue_buffer(struct uvc_video_queue *queue,
|
||||
struct v4l2_buffer *v4l2_buf)
|
||||
{
|
||||
struct uvc_buffer *buf;
|
||||
unsigned long flags;
|
||||
int ret = 0;
|
||||
|
||||
uvc_trace(UVC_TRACE_CAPTURE, "Queuing buffer %u.\n", v4l2_buf->index);
|
||||
|
||||
if (v4l2_buf->type != V4L2_BUF_TYPE_VIDEO_CAPTURE ||
|
||||
v4l2_buf->memory != V4L2_MEMORY_MMAP) {
|
||||
uvc_trace(UVC_TRACE_CAPTURE, "[E] Invalid buffer type (%u) "
|
||||
"and/or memory (%u).\n", v4l2_buf->type,
|
||||
v4l2_buf->memory);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
mutex_lock(&queue->mutex);
|
||||
if (v4l2_buf->index >= queue->count) {
|
||||
uvc_trace(UVC_TRACE_CAPTURE, "[E] Out of range index.\n");
|
||||
ret = -EINVAL;
|
||||
goto done;
|
||||
}
|
||||
|
||||
buf = &queue->buffer[v4l2_buf->index];
|
||||
if (buf->state != UVC_BUF_STATE_IDLE) {
|
||||
uvc_trace(UVC_TRACE_CAPTURE, "[E] Invalid buffer state "
|
||||
"(%u).\n", buf->state);
|
||||
ret = -EINVAL;
|
||||
goto done;
|
||||
}
|
||||
|
||||
spin_lock_irqsave(&queue->irqlock, flags);
|
||||
if (queue->flags & UVC_QUEUE_DISCONNECTED) {
|
||||
spin_unlock_irqrestore(&queue->irqlock, flags);
|
||||
ret = -ENODEV;
|
||||
goto done;
|
||||
}
|
||||
buf->state = UVC_BUF_STATE_QUEUED;
|
||||
buf->buf.bytesused = 0;
|
||||
list_add_tail(&buf->stream, &queue->mainqueue);
|
||||
list_add_tail(&buf->queue, &queue->irqqueue);
|
||||
spin_unlock_irqrestore(&queue->irqlock, flags);
|
||||
|
||||
done:
|
||||
mutex_unlock(&queue->mutex);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int uvc_queue_waiton(struct uvc_buffer *buf, int nonblocking)
|
||||
{
|
||||
if (nonblocking) {
|
||||
return (buf->state != UVC_BUF_STATE_QUEUED &&
|
||||
buf->state != UVC_BUF_STATE_ACTIVE)
|
||||
? 0 : -EAGAIN;
|
||||
}
|
||||
|
||||
return wait_event_interruptible(buf->wait,
|
||||
buf->state != UVC_BUF_STATE_QUEUED &&
|
||||
buf->state != UVC_BUF_STATE_ACTIVE);
|
||||
}
|
||||
|
||||
/*
|
||||
* Dequeue a video buffer. If nonblocking is false, block until a buffer is
|
||||
* available.
|
||||
*/
|
||||
int uvc_dequeue_buffer(struct uvc_video_queue *queue,
|
||||
struct v4l2_buffer *v4l2_buf, int nonblocking)
|
||||
{
|
||||
struct uvc_buffer *buf;
|
||||
int ret = 0;
|
||||
|
||||
if (v4l2_buf->type != V4L2_BUF_TYPE_VIDEO_CAPTURE ||
|
||||
v4l2_buf->memory != V4L2_MEMORY_MMAP) {
|
||||
uvc_trace(UVC_TRACE_CAPTURE, "[E] Invalid buffer type (%u) "
|
||||
"and/or memory (%u).\n", v4l2_buf->type,
|
||||
v4l2_buf->memory);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
mutex_lock(&queue->mutex);
|
||||
if (list_empty(&queue->mainqueue)) {
|
||||
uvc_trace(UVC_TRACE_CAPTURE, "[E] Empty buffer queue.\n");
|
||||
ret = -EINVAL;
|
||||
goto done;
|
||||
}
|
||||
|
||||
buf = list_first_entry(&queue->mainqueue, struct uvc_buffer, stream);
|
||||
if ((ret = uvc_queue_waiton(buf, nonblocking)) < 0)
|
||||
goto done;
|
||||
|
||||
uvc_trace(UVC_TRACE_CAPTURE, "Dequeuing buffer %u (%u, %u bytes).\n",
|
||||
buf->buf.index, buf->state, buf->buf.bytesused);
|
||||
|
||||
switch (buf->state) {
|
||||
case UVC_BUF_STATE_ERROR:
|
||||
uvc_trace(UVC_TRACE_CAPTURE, "[W] Corrupted data "
|
||||
"(transmission error).\n");
|
||||
ret = -EIO;
|
||||
case UVC_BUF_STATE_DONE:
|
||||
buf->state = UVC_BUF_STATE_IDLE;
|
||||
break;
|
||||
|
||||
case UVC_BUF_STATE_IDLE:
|
||||
case UVC_BUF_STATE_QUEUED:
|
||||
case UVC_BUF_STATE_ACTIVE:
|
||||
default:
|
||||
uvc_trace(UVC_TRACE_CAPTURE, "[E] Invalid buffer state %u "
|
||||
"(driver bug?).\n", buf->state);
|
||||
ret = -EINVAL;
|
||||
goto done;
|
||||
}
|
||||
|
||||
list_del(&buf->stream);
|
||||
__uvc_query_buffer(buf, v4l2_buf);
|
||||
|
||||
done:
|
||||
mutex_unlock(&queue->mutex);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* Poll the video queue.
|
||||
*
|
||||
* This function implements video queue polling and is intended to be used by
|
||||
* the device poll handler.
|
||||
*/
|
||||
unsigned int uvc_queue_poll(struct uvc_video_queue *queue, struct file *file,
|
||||
poll_table *wait)
|
||||
{
|
||||
struct uvc_buffer *buf;
|
||||
unsigned int mask = 0;
|
||||
|
||||
mutex_lock(&queue->mutex);
|
||||
if (list_empty(&queue->mainqueue)) {
|
||||
mask |= POLLERR;
|
||||
goto done;
|
||||
}
|
||||
buf = list_first_entry(&queue->mainqueue, struct uvc_buffer, stream);
|
||||
|
||||
poll_wait(file, &buf->wait, wait);
|
||||
if (buf->state == UVC_BUF_STATE_DONE ||
|
||||
buf->state == UVC_BUF_STATE_ERROR)
|
||||
mask |= POLLIN | POLLRDNORM;
|
||||
|
||||
done:
|
||||
mutex_unlock(&queue->mutex);
|
||||
return mask;
|
||||
}
|
||||
|
||||
/*
|
||||
* Enable or disable the video buffers queue.
|
||||
*
|
||||
* The queue must be enabled before starting video acquisition and must be
|
||||
* disabled after stopping it. This ensures that the video buffers queue
|
||||
* state can be properly initialized before buffers are accessed from the
|
||||
* interrupt handler.
|
||||
*
|
||||
* Enabling the video queue initializes parameters (such as sequence number,
|
||||
* sync pattern, ...). If the queue is already enabled, return -EBUSY.
|
||||
*
|
||||
* Disabling the video queue cancels the queue and removes all buffers from
|
||||
* the main queue.
|
||||
*
|
||||
* This function can't be called from interrupt context. Use
|
||||
* uvc_queue_cancel() instead.
|
||||
*/
|
||||
int uvc_queue_enable(struct uvc_video_queue *queue, int enable)
|
||||
{
|
||||
unsigned int i;
|
||||
int ret = 0;
|
||||
|
||||
mutex_lock(&queue->mutex);
|
||||
if (enable) {
|
||||
if (uvc_queue_streaming(queue)) {
|
||||
ret = -EBUSY;
|
||||
goto done;
|
||||
}
|
||||
queue->sequence = 0;
|
||||
queue->flags |= UVC_QUEUE_STREAMING;
|
||||
} else {
|
||||
uvc_queue_cancel(queue, 0);
|
||||
INIT_LIST_HEAD(&queue->mainqueue);
|
||||
|
||||
for (i = 0; i < queue->count; ++i)
|
||||
queue->buffer[i].state = UVC_BUF_STATE_IDLE;
|
||||
|
||||
queue->flags &= ~UVC_QUEUE_STREAMING;
|
||||
}
|
||||
|
||||
done:
|
||||
mutex_unlock(&queue->mutex);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* Cancel the video buffers queue.
|
||||
*
|
||||
* Cancelling the queue marks all buffers on the irq queue as erroneous,
|
||||
* wakes them up and remove them from the queue.
|
||||
*
|
||||
* If the disconnect parameter is set, further calls to uvc_queue_buffer will
|
||||
* fail with -ENODEV.
|
||||
*
|
||||
* This function acquires the irq spinlock and can be called from interrupt
|
||||
* context.
|
||||
*/
|
||||
void uvc_queue_cancel(struct uvc_video_queue *queue, int disconnect)
|
||||
{
|
||||
struct uvc_buffer *buf;
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&queue->irqlock, flags);
|
||||
while (!list_empty(&queue->irqqueue)) {
|
||||
buf = list_first_entry(&queue->irqqueue, struct uvc_buffer,
|
||||
queue);
|
||||
list_del(&buf->queue);
|
||||
buf->state = UVC_BUF_STATE_ERROR;
|
||||
wake_up(&buf->wait);
|
||||
}
|
||||
/* This must be protected by the irqlock spinlock to avoid race
|
||||
* conditions between uvc_queue_buffer and the disconnection event that
|
||||
* could result in an interruptible wait in uvc_dequeue_buffer. Do not
|
||||
* blindly replace this logic by checking for the UVC_DEV_DISCONNECTED
|
||||
* state outside the queue code.
|
||||
*/
|
||||
if (disconnect)
|
||||
queue->flags |= UVC_QUEUE_DISCONNECTED;
|
||||
spin_unlock_irqrestore(&queue->irqlock, flags);
|
||||
}
|
||||
|
||||
struct uvc_buffer *uvc_queue_next_buffer(struct uvc_video_queue *queue,
|
||||
struct uvc_buffer *buf)
|
||||
{
|
||||
struct uvc_buffer *nextbuf;
|
||||
unsigned long flags;
|
||||
|
||||
if ((queue->flags & UVC_QUEUE_DROP_INCOMPLETE) &&
|
||||
buf->buf.length != buf->buf.bytesused) {
|
||||
buf->state = UVC_BUF_STATE_QUEUED;
|
||||
buf->buf.bytesused = 0;
|
||||
return buf;
|
||||
}
|
||||
|
||||
spin_lock_irqsave(&queue->irqlock, flags);
|
||||
list_del(&buf->queue);
|
||||
if (!list_empty(&queue->irqqueue))
|
||||
nextbuf = list_first_entry(&queue->irqqueue, struct uvc_buffer,
|
||||
queue);
|
||||
else
|
||||
nextbuf = NULL;
|
||||
spin_unlock_irqrestore(&queue->irqlock, flags);
|
||||
|
||||
buf->buf.sequence = queue->sequence++;
|
||||
do_gettimeofday(&buf->buf.timestamp);
|
||||
|
||||
wake_up(&buf->wait);
|
||||
return nextbuf;
|
||||
}
|
|
@ -0,0 +1,207 @@
|
|||
/*
|
||||
* uvc_status.c -- USB Video Class driver - Status endpoint
|
||||
*
|
||||
* Copyright (C) 2007-2008
|
||||
* Laurent Pinchart (laurent.pinchart@skynet.be)
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/version.h>
|
||||
#include <linux/input.h>
|
||||
#include <linux/usb.h>
|
||||
#include <linux/usb/input.h>
|
||||
|
||||
#include "uvcvideo.h"
|
||||
|
||||
/* --------------------------------------------------------------------------
|
||||
* Input device
|
||||
*/
|
||||
static int uvc_input_init(struct uvc_device *dev)
|
||||
{
|
||||
struct usb_device *udev = dev->udev;
|
||||
struct input_dev *input;
|
||||
char *phys = NULL;
|
||||
int ret;
|
||||
|
||||
input = input_allocate_device();
|
||||
if (input == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
phys = kmalloc(6 + strlen(udev->bus->bus_name) + strlen(udev->devpath),
|
||||
GFP_KERNEL);
|
||||
if (phys == NULL) {
|
||||
ret = -ENOMEM;
|
||||
goto error;
|
||||
}
|
||||
sprintf(phys, "usb-%s-%s", udev->bus->bus_name, udev->devpath);
|
||||
|
||||
input->name = dev->name;
|
||||
input->phys = phys;
|
||||
usb_to_input_id(udev, &input->id);
|
||||
input->dev.parent = &dev->intf->dev;
|
||||
|
||||
set_bit(EV_KEY, input->evbit);
|
||||
set_bit(BTN_0, input->keybit);
|
||||
|
||||
if ((ret = input_register_device(input)) < 0)
|
||||
goto error;
|
||||
|
||||
dev->input = input;
|
||||
return 0;
|
||||
|
||||
error:
|
||||
input_free_device(input);
|
||||
kfree(phys);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void uvc_input_cleanup(struct uvc_device *dev)
|
||||
{
|
||||
if (dev->input)
|
||||
input_unregister_device(dev->input);
|
||||
}
|
||||
|
||||
/* --------------------------------------------------------------------------
|
||||
* Status interrupt endpoint
|
||||
*/
|
||||
static void uvc_event_streaming(struct uvc_device *dev, __u8 *data, int len)
|
||||
{
|
||||
if (len < 3) {
|
||||
uvc_trace(UVC_TRACE_STATUS, "Invalid streaming status event "
|
||||
"received.\n");
|
||||
return;
|
||||
}
|
||||
|
||||
if (data[2] == 0) {
|
||||
if (len < 4)
|
||||
return;
|
||||
uvc_trace(UVC_TRACE_STATUS, "Button (intf %u) %s len %d\n",
|
||||
data[1], data[3] ? "pressed" : "released", len);
|
||||
if (dev->input)
|
||||
input_report_key(dev->input, BTN_0, data[3]);
|
||||
} else {
|
||||
uvc_trace(UVC_TRACE_STATUS, "Stream %u error event %02x %02x "
|
||||
"len %d.\n", data[1], data[2], data[3], len);
|
||||
}
|
||||
}
|
||||
|
||||
static void uvc_event_control(struct uvc_device *dev, __u8 *data, int len)
|
||||
{
|
||||
char *attrs[3] = { "value", "info", "failure" };
|
||||
|
||||
if (len < 6 || data[2] != 0 || data[4] > 2) {
|
||||
uvc_trace(UVC_TRACE_STATUS, "Invalid control status event "
|
||||
"received.\n");
|
||||
return;
|
||||
}
|
||||
|
||||
uvc_trace(UVC_TRACE_STATUS, "Control %u/%u %s change len %d.\n",
|
||||
data[1], data[3], attrs[data[4]], len);
|
||||
}
|
||||
|
||||
static void uvc_status_complete(struct urb *urb)
|
||||
{
|
||||
struct uvc_device *dev = urb->context;
|
||||
int len, ret;
|
||||
|
||||
switch (urb->status) {
|
||||
case 0:
|
||||
break;
|
||||
|
||||
case -ENOENT: /* usb_kill_urb() called. */
|
||||
case -ECONNRESET: /* usb_unlink_urb() called. */
|
||||
case -ESHUTDOWN: /* The endpoint is being disabled. */
|
||||
case -EPROTO: /* Device is disconnected (reported by some
|
||||
* host controller). */
|
||||
return;
|
||||
|
||||
default:
|
||||
uvc_printk(KERN_WARNING, "Non-zero status (%d) in status "
|
||||
"completion handler.\n", urb->status);
|
||||
return;
|
||||
}
|
||||
|
||||
len = urb->actual_length;
|
||||
if (len > 0) {
|
||||
switch (dev->status[0] & 0x0f) {
|
||||
case UVC_STATUS_TYPE_CONTROL:
|
||||
uvc_event_control(dev, dev->status, len);
|
||||
break;
|
||||
|
||||
case UVC_STATUS_TYPE_STREAMING:
|
||||
uvc_event_streaming(dev, dev->status, len);
|
||||
break;
|
||||
|
||||
default:
|
||||
uvc_printk(KERN_INFO, "unknown event type %u.\n",
|
||||
dev->status[0]);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
/* Resubmit the URB. */
|
||||
urb->interval = dev->int_ep->desc.bInterval;
|
||||
if ((ret = usb_submit_urb(urb, GFP_ATOMIC)) < 0) {
|
||||
uvc_printk(KERN_ERR, "Failed to resubmit status URB (%d).\n",
|
||||
ret);
|
||||
}
|
||||
}
|
||||
|
||||
int uvc_status_init(struct uvc_device *dev)
|
||||
{
|
||||
struct usb_host_endpoint *ep = dev->int_ep;
|
||||
unsigned int pipe;
|
||||
int interval;
|
||||
|
||||
if (ep == NULL)
|
||||
return 0;
|
||||
|
||||
uvc_input_init(dev);
|
||||
|
||||
dev->int_urb = usb_alloc_urb(0, GFP_KERNEL);
|
||||
if (dev->int_urb == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
pipe = usb_rcvintpipe(dev->udev, ep->desc.bEndpointAddress);
|
||||
|
||||
/* For high-speed interrupt endpoints, the bInterval value is used as
|
||||
* an exponent of two. Some developers forgot about it.
|
||||
*/
|
||||
interval = ep->desc.bInterval;
|
||||
if (interval > 16 && dev->udev->speed == USB_SPEED_HIGH &&
|
||||
(dev->quirks & UVC_QUIRK_STATUS_INTERVAL))
|
||||
interval = fls(interval) - 1;
|
||||
|
||||
usb_fill_int_urb(dev->int_urb, dev->udev, pipe,
|
||||
dev->status, sizeof dev->status, uvc_status_complete,
|
||||
dev, interval);
|
||||
|
||||
return usb_submit_urb(dev->int_urb, GFP_KERNEL);
|
||||
}
|
||||
|
||||
void uvc_status_cleanup(struct uvc_device *dev)
|
||||
{
|
||||
usb_kill_urb(dev->int_urb);
|
||||
usb_free_urb(dev->int_urb);
|
||||
uvc_input_cleanup(dev);
|
||||
}
|
||||
|
||||
int uvc_status_suspend(struct uvc_device *dev)
|
||||
{
|
||||
usb_kill_urb(dev->int_urb);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int uvc_status_resume(struct uvc_device *dev)
|
||||
{
|
||||
if (dev->int_urb == NULL)
|
||||
return 0;
|
||||
|
||||
return usb_submit_urb(dev->int_urb, GFP_KERNEL);
|
||||
}
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,934 @@
|
|||
/*
|
||||
* uvc_video.c -- USB Video Class driver - Video handling
|
||||
*
|
||||
* Copyright (C) 2005-2008
|
||||
* Laurent Pinchart (laurent.pinchart@skynet.be)
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/version.h>
|
||||
#include <linux/list.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/usb.h>
|
||||
#include <linux/videodev2.h>
|
||||
#include <linux/vmalloc.h>
|
||||
#include <linux/wait.h>
|
||||
#include <asm/atomic.h>
|
||||
#include <asm/unaligned.h>
|
||||
|
||||
#include <media/v4l2-common.h>
|
||||
|
||||
#include "uvcvideo.h"
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* UVC Controls
|
||||
*/
|
||||
|
||||
static int __uvc_query_ctrl(struct uvc_device *dev, __u8 query, __u8 unit,
|
||||
__u8 intfnum, __u8 cs, void *data, __u16 size,
|
||||
int timeout)
|
||||
{
|
||||
__u8 type = USB_TYPE_CLASS | USB_RECIP_INTERFACE;
|
||||
unsigned int pipe;
|
||||
int ret;
|
||||
|
||||
pipe = (query & 0x80) ? usb_rcvctrlpipe(dev->udev, 0)
|
||||
: usb_sndctrlpipe(dev->udev, 0);
|
||||
type |= (query & 0x80) ? USB_DIR_IN : USB_DIR_OUT;
|
||||
|
||||
ret = usb_control_msg(dev->udev, pipe, query, type, cs << 8,
|
||||
unit << 8 | intfnum, data, size, timeout);
|
||||
|
||||
if (ret != size) {
|
||||
uvc_printk(KERN_ERR, "Failed to query (%u) UVC control %u "
|
||||
"(unit %u) : %d (exp. %u).\n", query, cs, unit, ret,
|
||||
size);
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int uvc_query_ctrl(struct uvc_device *dev, __u8 query, __u8 unit,
|
||||
__u8 intfnum, __u8 cs, void *data, __u16 size)
|
||||
{
|
||||
return __uvc_query_ctrl(dev, query, unit, intfnum, cs, data, size,
|
||||
UVC_CTRL_CONTROL_TIMEOUT);
|
||||
}
|
||||
|
||||
static void uvc_fixup_buffer_size(struct uvc_video_device *video,
|
||||
struct uvc_streaming_control *ctrl)
|
||||
{
|
||||
struct uvc_format *format;
|
||||
struct uvc_frame *frame;
|
||||
|
||||
if (ctrl->bFormatIndex <= 0 ||
|
||||
ctrl->bFormatIndex > video->streaming->nformats)
|
||||
return;
|
||||
|
||||
format = &video->streaming->format[ctrl->bFormatIndex - 1];
|
||||
|
||||
if (ctrl->bFrameIndex <= 0 ||
|
||||
ctrl->bFrameIndex > format->nframes)
|
||||
return;
|
||||
|
||||
frame = &format->frame[ctrl->bFrameIndex - 1];
|
||||
|
||||
if (!(format->flags & UVC_FMT_FLAG_COMPRESSED) ||
|
||||
(ctrl->dwMaxVideoFrameSize == 0 &&
|
||||
video->dev->uvc_version < 0x0110))
|
||||
ctrl->dwMaxVideoFrameSize =
|
||||
frame->dwMaxVideoFrameBufferSize;
|
||||
}
|
||||
|
||||
static int uvc_get_video_ctrl(struct uvc_video_device *video,
|
||||
struct uvc_streaming_control *ctrl, int probe, __u8 query)
|
||||
{
|
||||
__u8 data[34];
|
||||
__u8 size;
|
||||
int ret;
|
||||
|
||||
size = video->dev->uvc_version >= 0x0110 ? 34 : 26;
|
||||
ret = __uvc_query_ctrl(video->dev, query, 0, video->streaming->intfnum,
|
||||
probe ? VS_PROBE_CONTROL : VS_COMMIT_CONTROL, &data, size,
|
||||
UVC_CTRL_STREAMING_TIMEOUT);
|
||||
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ctrl->bmHint = le16_to_cpup((__le16 *)&data[0]);
|
||||
ctrl->bFormatIndex = data[2];
|
||||
ctrl->bFrameIndex = data[3];
|
||||
ctrl->dwFrameInterval = le32_to_cpup((__le32 *)&data[4]);
|
||||
ctrl->wKeyFrameRate = le16_to_cpup((__le16 *)&data[8]);
|
||||
ctrl->wPFrameRate = le16_to_cpup((__le16 *)&data[10]);
|
||||
ctrl->wCompQuality = le16_to_cpup((__le16 *)&data[12]);
|
||||
ctrl->wCompWindowSize = le16_to_cpup((__le16 *)&data[14]);
|
||||
ctrl->wDelay = le16_to_cpup((__le16 *)&data[16]);
|
||||
ctrl->dwMaxVideoFrameSize =
|
||||
le32_to_cpu(get_unaligned((__le32 *)&data[18]));
|
||||
ctrl->dwMaxPayloadTransferSize =
|
||||
le32_to_cpu(get_unaligned((__le32 *)&data[22]));
|
||||
|
||||
if (size == 34) {
|
||||
ctrl->dwClockFrequency =
|
||||
le32_to_cpu(get_unaligned((__le32 *)&data[26]));
|
||||
ctrl->bmFramingInfo = data[30];
|
||||
ctrl->bPreferedVersion = data[31];
|
||||
ctrl->bMinVersion = data[32];
|
||||
ctrl->bMaxVersion = data[33];
|
||||
} else {
|
||||
ctrl->dwClockFrequency = video->dev->clock_frequency;
|
||||
ctrl->bmFramingInfo = 0;
|
||||
ctrl->bPreferedVersion = 0;
|
||||
ctrl->bMinVersion = 0;
|
||||
ctrl->bMaxVersion = 0;
|
||||
}
|
||||
|
||||
/* Some broken devices return a null or wrong dwMaxVideoFrameSize.
|
||||
* Try to get the value from the format and frame descriptor.
|
||||
*/
|
||||
uvc_fixup_buffer_size(video, ctrl);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int uvc_set_video_ctrl(struct uvc_video_device *video,
|
||||
struct uvc_streaming_control *ctrl, int probe)
|
||||
{
|
||||
__u8 data[34];
|
||||
__u8 size;
|
||||
|
||||
size = video->dev->uvc_version >= 0x0110 ? 34 : 26;
|
||||
memset(data, 0, sizeof data);
|
||||
|
||||
*(__le16 *)&data[0] = cpu_to_le16(ctrl->bmHint);
|
||||
data[2] = ctrl->bFormatIndex;
|
||||
data[3] = ctrl->bFrameIndex;
|
||||
*(__le32 *)&data[4] = cpu_to_le32(ctrl->dwFrameInterval);
|
||||
*(__le16 *)&data[8] = cpu_to_le16(ctrl->wKeyFrameRate);
|
||||
*(__le16 *)&data[10] = cpu_to_le16(ctrl->wPFrameRate);
|
||||
*(__le16 *)&data[12] = cpu_to_le16(ctrl->wCompQuality);
|
||||
*(__le16 *)&data[14] = cpu_to_le16(ctrl->wCompWindowSize);
|
||||
*(__le16 *)&data[16] = cpu_to_le16(ctrl->wDelay);
|
||||
/* Note: Some of the fields below are not required for IN devices (see
|
||||
* UVC spec, 4.3.1.1), but we still copy them in case support for OUT
|
||||
* devices is added in the future. */
|
||||
put_unaligned(cpu_to_le32(ctrl->dwMaxVideoFrameSize),
|
||||
(__le32 *)&data[18]);
|
||||
put_unaligned(cpu_to_le32(ctrl->dwMaxPayloadTransferSize),
|
||||
(__le32 *)&data[22]);
|
||||
|
||||
if (size == 34) {
|
||||
put_unaligned(cpu_to_le32(ctrl->dwClockFrequency),
|
||||
(__le32 *)&data[26]);
|
||||
data[30] = ctrl->bmFramingInfo;
|
||||
data[31] = ctrl->bPreferedVersion;
|
||||
data[32] = ctrl->bMinVersion;
|
||||
data[33] = ctrl->bMaxVersion;
|
||||
}
|
||||
|
||||
return __uvc_query_ctrl(video->dev, SET_CUR, 0,
|
||||
video->streaming->intfnum,
|
||||
probe ? VS_PROBE_CONTROL : VS_COMMIT_CONTROL, &data, size,
|
||||
UVC_CTRL_STREAMING_TIMEOUT);
|
||||
}
|
||||
|
||||
int uvc_probe_video(struct uvc_video_device *video,
|
||||
struct uvc_streaming_control *probe)
|
||||
{
|
||||
struct uvc_streaming_control probe_min, probe_max;
|
||||
__u16 bandwidth;
|
||||
unsigned int i;
|
||||
int ret;
|
||||
|
||||
mutex_lock(&video->streaming->mutex);
|
||||
|
||||
/* Perform probing. The device should adjust the requested values
|
||||
* according to its capabilities. However, some devices, namely the
|
||||
* first generation UVC Logitech webcams, don't implement the Video
|
||||
* Probe control properly, and just return the needed bandwidth. For
|
||||
* that reason, if the needed bandwidth exceeds the maximum available
|
||||
* bandwidth, try to lower the quality.
|
||||
*/
|
||||
if ((ret = uvc_set_video_ctrl(video, probe, 1)) < 0)
|
||||
goto done;
|
||||
|
||||
/* Get the minimum and maximum values for compression settings. */
|
||||
if (!(video->dev->quirks & UVC_QUIRK_PROBE_MINMAX)) {
|
||||
ret = uvc_get_video_ctrl(video, &probe_min, 1, GET_MIN);
|
||||
if (ret < 0)
|
||||
goto done;
|
||||
ret = uvc_get_video_ctrl(video, &probe_max, 1, GET_MAX);
|
||||
if (ret < 0)
|
||||
goto done;
|
||||
|
||||
probe->wCompQuality = probe_max.wCompQuality;
|
||||
}
|
||||
|
||||
for (i = 0; i < 2; ++i) {
|
||||
if ((ret = uvc_set_video_ctrl(video, probe, 1)) < 0 ||
|
||||
(ret = uvc_get_video_ctrl(video, probe, 1, GET_CUR)) < 0)
|
||||
goto done;
|
||||
|
||||
if (video->streaming->intf->num_altsetting == 1)
|
||||
break;
|
||||
|
||||
bandwidth = probe->dwMaxPayloadTransferSize;
|
||||
if (bandwidth <= video->streaming->maxpsize)
|
||||
break;
|
||||
|
||||
if (video->dev->quirks & UVC_QUIRK_PROBE_MINMAX) {
|
||||
ret = -ENOSPC;
|
||||
goto done;
|
||||
}
|
||||
|
||||
/* TODO: negotiate compression parameters */
|
||||
probe->wKeyFrameRate = probe_min.wKeyFrameRate;
|
||||
probe->wPFrameRate = probe_min.wPFrameRate;
|
||||
probe->wCompQuality = probe_max.wCompQuality;
|
||||
probe->wCompWindowSize = probe_min.wCompWindowSize;
|
||||
}
|
||||
|
||||
done:
|
||||
mutex_unlock(&video->streaming->mutex);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* Video codecs
|
||||
*/
|
||||
|
||||
/* Values for bmHeaderInfo (Video and Still Image Payload Headers, 2.4.3.3) */
|
||||
#define UVC_STREAM_EOH (1 << 7)
|
||||
#define UVC_STREAM_ERR (1 << 6)
|
||||
#define UVC_STREAM_STI (1 << 5)
|
||||
#define UVC_STREAM_RES (1 << 4)
|
||||
#define UVC_STREAM_SCR (1 << 3)
|
||||
#define UVC_STREAM_PTS (1 << 2)
|
||||
#define UVC_STREAM_EOF (1 << 1)
|
||||
#define UVC_STREAM_FID (1 << 0)
|
||||
|
||||
/* Video payload decoding is handled by uvc_video_decode_start(),
|
||||
* uvc_video_decode_data() and uvc_video_decode_end().
|
||||
*
|
||||
* uvc_video_decode_start is called with URB data at the start of a bulk or
|
||||
* isochronous payload. It processes header data and returns the header size
|
||||
* in bytes if successful. If an error occurs, it returns a negative error
|
||||
* code. The following error codes have special meanings.
|
||||
*
|
||||
* - EAGAIN informs the caller that the current video buffer should be marked
|
||||
* as done, and that the function should be called again with the same data
|
||||
* and a new video buffer. This is used when end of frame conditions can be
|
||||
* reliably detected at the beginning of the next frame only.
|
||||
*
|
||||
* If an error other than -EAGAIN is returned, the caller will drop the current
|
||||
* payload. No call to uvc_video_decode_data and uvc_video_decode_end will be
|
||||
* made until the next payload. -ENODATA can be used to drop the current
|
||||
* payload if no other error code is appropriate.
|
||||
*
|
||||
* uvc_video_decode_data is called for every URB with URB data. It copies the
|
||||
* data to the video buffer.
|
||||
*
|
||||
* uvc_video_decode_end is called with header data at the end of a bulk or
|
||||
* isochronous payload. It performs any additional header data processing and
|
||||
* returns 0 or a negative error code if an error occured. As header data have
|
||||
* already been processed by uvc_video_decode_start, this functions isn't
|
||||
* required to perform sanity checks a second time.
|
||||
*
|
||||
* For isochronous transfers where a payload is always transfered in a single
|
||||
* URB, the three functions will be called in a row.
|
||||
*
|
||||
* To let the decoder process header data and update its internal state even
|
||||
* when no video buffer is available, uvc_video_decode_start must be prepared
|
||||
* to be called with a NULL buf parameter. uvc_video_decode_data and
|
||||
* uvc_video_decode_end will never be called with a NULL buffer.
|
||||
*/
|
||||
static int uvc_video_decode_start(struct uvc_video_device *video,
|
||||
struct uvc_buffer *buf, const __u8 *data, int len)
|
||||
{
|
||||
__u8 fid;
|
||||
|
||||
/* Sanity checks:
|
||||
* - packet must be at least 2 bytes long
|
||||
* - bHeaderLength value must be at least 2 bytes (see above)
|
||||
* - bHeaderLength value can't be larger than the packet size.
|
||||
*/
|
||||
if (len < 2 || data[0] < 2 || data[0] > len)
|
||||
return -EINVAL;
|
||||
|
||||
/* Skip payloads marked with the error bit ("error frames"). */
|
||||
if (data[1] & UVC_STREAM_ERR) {
|
||||
uvc_trace(UVC_TRACE_FRAME, "Dropping payload (error bit "
|
||||
"set).\n");
|
||||
return -ENODATA;
|
||||
}
|
||||
|
||||
fid = data[1] & UVC_STREAM_FID;
|
||||
|
||||
/* Store the payload FID bit and return immediately when the buffer is
|
||||
* NULL.
|
||||
*/
|
||||
if (buf == NULL) {
|
||||
video->last_fid = fid;
|
||||
return -ENODATA;
|
||||
}
|
||||
|
||||
/* Synchronize to the input stream by waiting for the FID bit to be
|
||||
* toggled when the the buffer state is not UVC_BUF_STATE_ACTIVE.
|
||||
* queue->last_fid is initialized to -1, so the first isochronous
|
||||
* frame will always be in sync.
|
||||
*
|
||||
* If the device doesn't toggle the FID bit, invert video->last_fid
|
||||
* when the EOF bit is set to force synchronisation on the next packet.
|
||||
*/
|
||||
if (buf->state != UVC_BUF_STATE_ACTIVE) {
|
||||
if (fid == video->last_fid) {
|
||||
uvc_trace(UVC_TRACE_FRAME, "Dropping payload (out of "
|
||||
"sync).\n");
|
||||
if ((video->dev->quirks & UVC_QUIRK_STREAM_NO_FID) &&
|
||||
(data[1] & UVC_STREAM_EOF))
|
||||
video->last_fid ^= UVC_STREAM_FID;
|
||||
return -ENODATA;
|
||||
}
|
||||
|
||||
/* TODO: Handle PTS and SCR. */
|
||||
buf->state = UVC_BUF_STATE_ACTIVE;
|
||||
}
|
||||
|
||||
/* Mark the buffer as done if we're at the beginning of a new frame.
|
||||
* End of frame detection is better implemented by checking the EOF
|
||||
* bit (FID bit toggling is delayed by one frame compared to the EOF
|
||||
* bit), but some devices don't set the bit at end of frame (and the
|
||||
* last payload can be lost anyway). We thus must check if the FID has
|
||||
* been toggled.
|
||||
*
|
||||
* queue->last_fid is initialized to -1, so the first isochronous
|
||||
* frame will never trigger an end of frame detection.
|
||||
*
|
||||
* Empty buffers (bytesused == 0) don't trigger end of frame detection
|
||||
* as it doesn't make sense to return an empty buffer. This also
|
||||
* avoids detecting and of frame conditions at FID toggling if the
|
||||
* previous payload had the EOF bit set.
|
||||
*/
|
||||
if (fid != video->last_fid && buf->buf.bytesused != 0) {
|
||||
uvc_trace(UVC_TRACE_FRAME, "Frame complete (FID bit "
|
||||
"toggled).\n");
|
||||
buf->state = UVC_BUF_STATE_DONE;
|
||||
return -EAGAIN;
|
||||
}
|
||||
|
||||
video->last_fid = fid;
|
||||
|
||||
return data[0];
|
||||
}
|
||||
|
||||
static void uvc_video_decode_data(struct uvc_video_device *video,
|
||||
struct uvc_buffer *buf, const __u8 *data, int len)
|
||||
{
|
||||
struct uvc_video_queue *queue = &video->queue;
|
||||
unsigned int maxlen, nbytes;
|
||||
void *mem;
|
||||
|
||||
if (len <= 0)
|
||||
return;
|
||||
|
||||
/* Copy the video data to the buffer. */
|
||||
maxlen = buf->buf.length - buf->buf.bytesused;
|
||||
mem = queue->mem + buf->buf.m.offset + buf->buf.bytesused;
|
||||
nbytes = min((unsigned int)len, maxlen);
|
||||
memcpy(mem, data, nbytes);
|
||||
buf->buf.bytesused += nbytes;
|
||||
|
||||
/* Complete the current frame if the buffer size was exceeded. */
|
||||
if (len > maxlen) {
|
||||
uvc_trace(UVC_TRACE_FRAME, "Frame complete (overflow).\n");
|
||||
buf->state = UVC_BUF_STATE_DONE;
|
||||
}
|
||||
}
|
||||
|
||||
static void uvc_video_decode_end(struct uvc_video_device *video,
|
||||
struct uvc_buffer *buf, const __u8 *data, int len)
|
||||
{
|
||||
/* Mark the buffer as done if the EOF marker is set. */
|
||||
if (data[1] & UVC_STREAM_EOF && buf->buf.bytesused != 0) {
|
||||
uvc_trace(UVC_TRACE_FRAME, "Frame complete (EOF found).\n");
|
||||
if (data[0] == len)
|
||||
uvc_trace(UVC_TRACE_FRAME, "EOF in empty payload.\n");
|
||||
buf->state = UVC_BUF_STATE_DONE;
|
||||
if (video->dev->quirks & UVC_QUIRK_STREAM_NO_FID)
|
||||
video->last_fid ^= UVC_STREAM_FID;
|
||||
}
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* URB handling
|
||||
*/
|
||||
|
||||
/*
|
||||
* Completion handler for video URBs.
|
||||
*/
|
||||
static void uvc_video_decode_isoc(struct urb *urb,
|
||||
struct uvc_video_device *video, struct uvc_buffer *buf)
|
||||
{
|
||||
u8 *mem;
|
||||
int ret, i;
|
||||
|
||||
for (i = 0; i < urb->number_of_packets; ++i) {
|
||||
if (urb->iso_frame_desc[i].status < 0) {
|
||||
uvc_trace(UVC_TRACE_FRAME, "USB isochronous frame "
|
||||
"lost (%d).\n", urb->iso_frame_desc[i].status);
|
||||
continue;
|
||||
}
|
||||
|
||||
/* Decode the payload header. */
|
||||
mem = urb->transfer_buffer + urb->iso_frame_desc[i].offset;
|
||||
do {
|
||||
ret = uvc_video_decode_start(video, buf, mem,
|
||||
urb->iso_frame_desc[i].actual_length);
|
||||
if (ret == -EAGAIN)
|
||||
buf = uvc_queue_next_buffer(&video->queue, buf);
|
||||
} while (ret == -EAGAIN);
|
||||
|
||||
if (ret < 0)
|
||||
continue;
|
||||
|
||||
/* Decode the payload data. */
|
||||
uvc_video_decode_data(video, buf, mem + ret,
|
||||
urb->iso_frame_desc[i].actual_length - ret);
|
||||
|
||||
/* Process the header again. */
|
||||
uvc_video_decode_end(video, buf, mem, ret);
|
||||
|
||||
if (buf->state == UVC_BUF_STATE_DONE ||
|
||||
buf->state == UVC_BUF_STATE_ERROR)
|
||||
buf = uvc_queue_next_buffer(&video->queue, buf);
|
||||
}
|
||||
}
|
||||
|
||||
static void uvc_video_decode_bulk(struct urb *urb,
|
||||
struct uvc_video_device *video, struct uvc_buffer *buf)
|
||||
{
|
||||
u8 *mem;
|
||||
int len, ret;
|
||||
|
||||
mem = urb->transfer_buffer;
|
||||
len = urb->actual_length;
|
||||
video->bulk.payload_size += len;
|
||||
|
||||
/* If the URB is the first of its payload, decode and save the
|
||||
* header.
|
||||
*/
|
||||
if (video->bulk.header_size == 0) {
|
||||
do {
|
||||
ret = uvc_video_decode_start(video, buf, mem, len);
|
||||
if (ret == -EAGAIN)
|
||||
buf = uvc_queue_next_buffer(&video->queue, buf);
|
||||
} while (ret == -EAGAIN);
|
||||
|
||||
/* If an error occured skip the rest of the payload. */
|
||||
if (ret < 0 || buf == NULL) {
|
||||
video->bulk.skip_payload = 1;
|
||||
return;
|
||||
}
|
||||
|
||||
video->bulk.header_size = ret;
|
||||
memcpy(video->bulk.header, mem, video->bulk.header_size);
|
||||
|
||||
mem += ret;
|
||||
len -= ret;
|
||||
}
|
||||
|
||||
/* The buffer queue might have been cancelled while a bulk transfer
|
||||
* was in progress, so we can reach here with buf equal to NULL. Make
|
||||
* sure buf is never dereferenced if NULL.
|
||||
*/
|
||||
|
||||
/* Process video data. */
|
||||
if (!video->bulk.skip_payload && buf != NULL)
|
||||
uvc_video_decode_data(video, buf, mem, len);
|
||||
|
||||
/* Detect the payload end by a URB smaller than the maximum size (or
|
||||
* a payload size equal to the maximum) and process the header again.
|
||||
*/
|
||||
if (urb->actual_length < urb->transfer_buffer_length ||
|
||||
video->bulk.payload_size >= video->bulk.max_payload_size) {
|
||||
if (!video->bulk.skip_payload && buf != NULL) {
|
||||
uvc_video_decode_end(video, buf, video->bulk.header,
|
||||
video->bulk.header_size);
|
||||
if (buf->state == UVC_BUF_STATE_DONE ||
|
||||
buf->state == UVC_BUF_STATE_ERROR)
|
||||
buf = uvc_queue_next_buffer(&video->queue, buf);
|
||||
}
|
||||
|
||||
video->bulk.header_size = 0;
|
||||
video->bulk.skip_payload = 0;
|
||||
video->bulk.payload_size = 0;
|
||||
}
|
||||
}
|
||||
|
||||
static void uvc_video_complete(struct urb *urb)
|
||||
{
|
||||
struct uvc_video_device *video = urb->context;
|
||||
struct uvc_video_queue *queue = &video->queue;
|
||||
struct uvc_buffer *buf = NULL;
|
||||
unsigned long flags;
|
||||
int ret;
|
||||
|
||||
switch (urb->status) {
|
||||
case 0:
|
||||
break;
|
||||
|
||||
default:
|
||||
uvc_printk(KERN_WARNING, "Non-zero status (%d) in video "
|
||||
"completion handler.\n", urb->status);
|
||||
|
||||
case -ENOENT: /* usb_kill_urb() called. */
|
||||
if (video->frozen)
|
||||
return;
|
||||
|
||||
case -ECONNRESET: /* usb_unlink_urb() called. */
|
||||
case -ESHUTDOWN: /* The endpoint is being disabled. */
|
||||
uvc_queue_cancel(queue, urb->status == -ESHUTDOWN);
|
||||
return;
|
||||
}
|
||||
|
||||
spin_lock_irqsave(&queue->irqlock, flags);
|
||||
if (!list_empty(&queue->irqqueue))
|
||||
buf = list_first_entry(&queue->irqqueue, struct uvc_buffer,
|
||||
queue);
|
||||
spin_unlock_irqrestore(&queue->irqlock, flags);
|
||||
|
||||
video->decode(urb, video, buf);
|
||||
|
||||
if ((ret = usb_submit_urb(urb, GFP_ATOMIC)) < 0) {
|
||||
uvc_printk(KERN_ERR, "Failed to resubmit video URB (%d).\n",
|
||||
ret);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Uninitialize isochronous/bulk URBs and free transfer buffers.
|
||||
*/
|
||||
static void uvc_uninit_video(struct uvc_video_device *video)
|
||||
{
|
||||
struct urb *urb;
|
||||
unsigned int i;
|
||||
|
||||
for (i = 0; i < UVC_URBS; ++i) {
|
||||
if ((urb = video->urb[i]) == NULL)
|
||||
continue;
|
||||
|
||||
usb_kill_urb(urb);
|
||||
/* urb->transfer_buffer_length is not touched by USB core, so
|
||||
* we can use it here as the buffer length.
|
||||
*/
|
||||
if (video->urb_buffer[i]) {
|
||||
usb_buffer_free(video->dev->udev,
|
||||
urb->transfer_buffer_length,
|
||||
video->urb_buffer[i], urb->transfer_dma);
|
||||
video->urb_buffer[i] = NULL;
|
||||
}
|
||||
|
||||
usb_free_urb(urb);
|
||||
video->urb[i] = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Initialize isochronous URBs and allocate transfer buffers. The packet size
|
||||
* is given by the endpoint.
|
||||
*/
|
||||
static int uvc_init_video_isoc(struct uvc_video_device *video,
|
||||
struct usb_host_endpoint *ep)
|
||||
{
|
||||
struct urb *urb;
|
||||
unsigned int npackets, i, j;
|
||||
__u16 psize;
|
||||
__u32 size;
|
||||
|
||||
/* Compute the number of isochronous packets to allocate by dividing
|
||||
* the maximum video frame size by the packet size. Limit the result
|
||||
* to UVC_MAX_ISO_PACKETS.
|
||||
*/
|
||||
psize = le16_to_cpu(ep->desc.wMaxPacketSize);
|
||||
psize = (psize & 0x07ff) * (1 + ((psize >> 11) & 3));
|
||||
|
||||
size = video->streaming->ctrl.dwMaxVideoFrameSize;
|
||||
if (size > UVC_MAX_FRAME_SIZE)
|
||||
return -EINVAL;
|
||||
|
||||
npackets = (size + psize - 1) / psize;
|
||||
if (npackets > UVC_MAX_ISO_PACKETS)
|
||||
npackets = UVC_MAX_ISO_PACKETS;
|
||||
|
||||
size = npackets * psize;
|
||||
|
||||
for (i = 0; i < UVC_URBS; ++i) {
|
||||
urb = usb_alloc_urb(npackets, GFP_KERNEL);
|
||||
if (urb == NULL) {
|
||||
uvc_uninit_video(video);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
video->urb_buffer[i] = usb_buffer_alloc(video->dev->udev,
|
||||
size, GFP_KERNEL, &urb->transfer_dma);
|
||||
if (video->urb_buffer[i] == NULL) {
|
||||
usb_free_urb(urb);
|
||||
uvc_uninit_video(video);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
urb->dev = video->dev->udev;
|
||||
urb->context = video;
|
||||
urb->pipe = usb_rcvisocpipe(video->dev->udev,
|
||||
ep->desc.bEndpointAddress);
|
||||
urb->transfer_flags = URB_ISO_ASAP | URB_NO_TRANSFER_DMA_MAP;
|
||||
urb->interval = ep->desc.bInterval;
|
||||
urb->transfer_buffer = video->urb_buffer[i];
|
||||
urb->complete = uvc_video_complete;
|
||||
urb->number_of_packets = npackets;
|
||||
urb->transfer_buffer_length = size;
|
||||
|
||||
for (j = 0; j < npackets; ++j) {
|
||||
urb->iso_frame_desc[j].offset = j * psize;
|
||||
urb->iso_frame_desc[j].length = psize;
|
||||
}
|
||||
|
||||
video->urb[i] = urb;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Initialize bulk URBs and allocate transfer buffers. The packet size is
|
||||
* given by the endpoint.
|
||||
*/
|
||||
static int uvc_init_video_bulk(struct uvc_video_device *video,
|
||||
struct usb_host_endpoint *ep)
|
||||
{
|
||||
struct urb *urb;
|
||||
unsigned int pipe, i;
|
||||
__u16 psize;
|
||||
__u32 size;
|
||||
|
||||
/* Compute the bulk URB size. Some devices set the maximum payload
|
||||
* size to a value too high for memory-constrained devices. We must
|
||||
* then transfer the payload accross multiple URBs. To be consistant
|
||||
* with isochronous mode, allocate maximum UVC_MAX_ISO_PACKETS per bulk
|
||||
* URB.
|
||||
*/
|
||||
psize = le16_to_cpu(ep->desc.wMaxPacketSize) & 0x07ff;
|
||||
size = video->streaming->ctrl.dwMaxPayloadTransferSize;
|
||||
video->bulk.max_payload_size = size;
|
||||
if (size > psize * UVC_MAX_ISO_PACKETS)
|
||||
size = psize * UVC_MAX_ISO_PACKETS;
|
||||
|
||||
pipe = usb_rcvbulkpipe(video->dev->udev, ep->desc.bEndpointAddress);
|
||||
|
||||
for (i = 0; i < UVC_URBS; ++i) {
|
||||
urb = usb_alloc_urb(0, GFP_KERNEL);
|
||||
if (urb == NULL) {
|
||||
uvc_uninit_video(video);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
video->urb_buffer[i] = usb_buffer_alloc(video->dev->udev,
|
||||
size, GFP_KERNEL, &urb->transfer_dma);
|
||||
if (video->urb_buffer[i] == NULL) {
|
||||
usb_free_urb(urb);
|
||||
uvc_uninit_video(video);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
usb_fill_bulk_urb(urb, video->dev->udev, pipe,
|
||||
video->urb_buffer[i], size, uvc_video_complete,
|
||||
video);
|
||||
urb->transfer_flags = URB_NO_TRANSFER_DMA_MAP;
|
||||
|
||||
video->urb[i] = urb;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Initialize isochronous/bulk URBs and allocate transfer buffers.
|
||||
*/
|
||||
static int uvc_init_video(struct uvc_video_device *video)
|
||||
{
|
||||
struct usb_interface *intf = video->streaming->intf;
|
||||
struct usb_host_interface *alts;
|
||||
struct usb_host_endpoint *ep = NULL;
|
||||
int intfnum = video->streaming->intfnum;
|
||||
unsigned int bandwidth, psize, i;
|
||||
int ret;
|
||||
|
||||
video->last_fid = -1;
|
||||
video->bulk.header_size = 0;
|
||||
video->bulk.skip_payload = 0;
|
||||
video->bulk.payload_size = 0;
|
||||
|
||||
if (intf->num_altsetting > 1) {
|
||||
/* Isochronous endpoint, select the alternate setting. */
|
||||
bandwidth = video->streaming->ctrl.dwMaxPayloadTransferSize;
|
||||
|
||||
if (bandwidth == 0) {
|
||||
uvc_printk(KERN_WARNING, "device %s requested null "
|
||||
"bandwidth, defaulting to lowest.\n",
|
||||
video->vdev->name);
|
||||
bandwidth = 1;
|
||||
}
|
||||
|
||||
for (i = 0; i < intf->num_altsetting; ++i) {
|
||||
alts = &intf->altsetting[i];
|
||||
ep = uvc_find_endpoint(alts,
|
||||
video->streaming->header.bEndpointAddress);
|
||||
if (ep == NULL)
|
||||
continue;
|
||||
|
||||
/* Check if the bandwidth is high enough. */
|
||||
psize = le16_to_cpu(ep->desc.wMaxPacketSize);
|
||||
psize = (psize & 0x07ff) * (1 + ((psize >> 11) & 3));
|
||||
if (psize >= bandwidth)
|
||||
break;
|
||||
}
|
||||
|
||||
if (i >= intf->num_altsetting)
|
||||
return -EIO;
|
||||
|
||||
if ((ret = usb_set_interface(video->dev->udev, intfnum, i)) < 0)
|
||||
return ret;
|
||||
|
||||
ret = uvc_init_video_isoc(video, ep);
|
||||
} else {
|
||||
/* Bulk endpoint, proceed to URB initialization. */
|
||||
ep = uvc_find_endpoint(&intf->altsetting[0],
|
||||
video->streaming->header.bEndpointAddress);
|
||||
if (ep == NULL)
|
||||
return -EIO;
|
||||
|
||||
ret = uvc_init_video_bulk(video, ep);
|
||||
}
|
||||
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
/* Submit the URBs. */
|
||||
for (i = 0; i < UVC_URBS; ++i) {
|
||||
if ((ret = usb_submit_urb(video->urb[i], GFP_KERNEL)) < 0) {
|
||||
uvc_printk(KERN_ERR, "Failed to submit URB %u "
|
||||
"(%d).\n", i, ret);
|
||||
uvc_uninit_video(video);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* --------------------------------------------------------------------------
|
||||
* Suspend/resume
|
||||
*/
|
||||
|
||||
/*
|
||||
* Stop streaming without disabling the video queue.
|
||||
*
|
||||
* To let userspace applications resume without trouble, we must not touch the
|
||||
* video buffers in any way. We mark the device as frozen to make sure the URB
|
||||
* completion handler won't try to cancel the queue when we kill the URBs.
|
||||
*/
|
||||
int uvc_video_suspend(struct uvc_video_device *video)
|
||||
{
|
||||
if (!uvc_queue_streaming(&video->queue))
|
||||
return 0;
|
||||
|
||||
video->frozen = 1;
|
||||
uvc_uninit_video(video);
|
||||
usb_set_interface(video->dev->udev, video->streaming->intfnum, 0);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Reconfigure the video interface and restart streaming if it was enable
|
||||
* before suspend.
|
||||
*
|
||||
* If an error occurs, disable the video queue. This will wake all pending
|
||||
* buffers, making sure userspace applications are notified of the problem
|
||||
* instead of waiting forever.
|
||||
*/
|
||||
int uvc_video_resume(struct uvc_video_device *video)
|
||||
{
|
||||
int ret;
|
||||
|
||||
video->frozen = 0;
|
||||
|
||||
if ((ret = uvc_set_video_ctrl(video, &video->streaming->ctrl, 0)) < 0) {
|
||||
uvc_queue_enable(&video->queue, 0);
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (!uvc_queue_streaming(&video->queue))
|
||||
return 0;
|
||||
|
||||
if ((ret = uvc_init_video(video)) < 0)
|
||||
uvc_queue_enable(&video->queue, 0);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* Video device
|
||||
*/
|
||||
|
||||
/*
|
||||
* Initialize the UVC video device by retrieving the default format and
|
||||
* committing it.
|
||||
*
|
||||
* Some cameras (namely the Fuji Finepix) set the format and frame
|
||||
* indexes to zero. The UVC standard doesn't clearly make this a spec
|
||||
* violation, so try to silently fix the values if possible.
|
||||
*
|
||||
* This function is called before registering the device with V4L.
|
||||
*/
|
||||
int uvc_video_init(struct uvc_video_device *video)
|
||||
{
|
||||
struct uvc_streaming_control *probe = &video->streaming->ctrl;
|
||||
struct uvc_format *format = NULL;
|
||||
struct uvc_frame *frame = NULL;
|
||||
unsigned int i;
|
||||
int ret;
|
||||
|
||||
if (video->streaming->nformats == 0) {
|
||||
uvc_printk(KERN_INFO, "No supported video formats found.\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* Alternate setting 0 should be the default, yet the XBox Live Vision
|
||||
* Cam (and possibly other devices) crash or otherwise misbehave if
|
||||
* they don't receive a SET_INTERFACE request before any other video
|
||||
* control request.
|
||||
*/
|
||||
usb_set_interface(video->dev->udev, video->streaming->intfnum, 0);
|
||||
|
||||
/* Some webcams don't suport GET_DEF request on the probe control. We
|
||||
* fall back to GET_CUR if GET_DEF fails.
|
||||
*/
|
||||
if ((ret = uvc_get_video_ctrl(video, probe, 1, GET_DEF)) < 0 &&
|
||||
(ret = uvc_get_video_ctrl(video, probe, 1, GET_CUR)) < 0)
|
||||
return ret;
|
||||
|
||||
/* Check if the default format descriptor exists. Use the first
|
||||
* available format otherwise.
|
||||
*/
|
||||
for (i = video->streaming->nformats; i > 0; --i) {
|
||||
format = &video->streaming->format[i-1];
|
||||
if (format->index == probe->bFormatIndex)
|
||||
break;
|
||||
}
|
||||
|
||||
if (format->nframes == 0) {
|
||||
uvc_printk(KERN_INFO, "No frame descriptor found for the "
|
||||
"default format.\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* Zero bFrameIndex might be correct. Stream-based formats (including
|
||||
* MPEG-2 TS and DV) do not support frames but have a dummy frame
|
||||
* descriptor with bFrameIndex set to zero. If the default frame
|
||||
* descriptor is not found, use the first avalable frame.
|
||||
*/
|
||||
for (i = format->nframes; i > 0; --i) {
|
||||
frame = &format->frame[i-1];
|
||||
if (frame->bFrameIndex == probe->bFrameIndex)
|
||||
break;
|
||||
}
|
||||
|
||||
/* Commit the default settings. */
|
||||
probe->bFormatIndex = format->index;
|
||||
probe->bFrameIndex = frame->bFrameIndex;
|
||||
if ((ret = uvc_set_video_ctrl(video, probe, 0)) < 0)
|
||||
return ret;
|
||||
|
||||
video->streaming->cur_format = format;
|
||||
video->streaming->cur_frame = frame;
|
||||
atomic_set(&video->active, 0);
|
||||
|
||||
/* Select the video decoding function */
|
||||
if (video->dev->quirks & UVC_QUIRK_BUILTIN_ISIGHT)
|
||||
video->decode = uvc_video_decode_isight;
|
||||
else if (video->streaming->intf->num_altsetting > 1)
|
||||
video->decode = uvc_video_decode_isoc;
|
||||
else
|
||||
video->decode = uvc_video_decode_bulk;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Enable or disable the video stream.
|
||||
*/
|
||||
int uvc_video_enable(struct uvc_video_device *video, int enable)
|
||||
{
|
||||
int ret;
|
||||
|
||||
if (!enable) {
|
||||
uvc_uninit_video(video);
|
||||
usb_set_interface(video->dev->udev,
|
||||
video->streaming->intfnum, 0);
|
||||
uvc_queue_enable(&video->queue, 0);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if ((ret = uvc_queue_enable(&video->queue, 1)) < 0)
|
||||
return ret;
|
||||
|
||||
return uvc_init_video(video);
|
||||
}
|
|
@ -0,0 +1,796 @@
|
|||
#ifndef _USB_VIDEO_H_
|
||||
#define _USB_VIDEO_H_
|
||||
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/videodev2.h>
|
||||
|
||||
|
||||
/*
|
||||
* Dynamic controls
|
||||
*/
|
||||
|
||||
/* Data types for UVC control data */
|
||||
#define UVC_CTRL_DATA_TYPE_RAW 0
|
||||
#define UVC_CTRL_DATA_TYPE_SIGNED 1
|
||||
#define UVC_CTRL_DATA_TYPE_UNSIGNED 2
|
||||
#define UVC_CTRL_DATA_TYPE_BOOLEAN 3
|
||||
#define UVC_CTRL_DATA_TYPE_ENUM 4
|
||||
#define UVC_CTRL_DATA_TYPE_BITMASK 5
|
||||
|
||||
/* Control flags */
|
||||
#define UVC_CONTROL_SET_CUR (1 << 0)
|
||||
#define UVC_CONTROL_GET_CUR (1 << 1)
|
||||
#define UVC_CONTROL_GET_MIN (1 << 2)
|
||||
#define UVC_CONTROL_GET_MAX (1 << 3)
|
||||
#define UVC_CONTROL_GET_RES (1 << 4)
|
||||
#define UVC_CONTROL_GET_DEF (1 << 5)
|
||||
/* Control should be saved at suspend and restored at resume. */
|
||||
#define UVC_CONTROL_RESTORE (1 << 6)
|
||||
/* Control can be updated by the camera. */
|
||||
#define UVC_CONTROL_AUTO_UPDATE (1 << 7)
|
||||
|
||||
#define UVC_CONTROL_GET_RANGE (UVC_CONTROL_GET_CUR | UVC_CONTROL_GET_MIN | \
|
||||
UVC_CONTROL_GET_MAX | UVC_CONTROL_GET_RES | \
|
||||
UVC_CONTROL_GET_DEF)
|
||||
|
||||
struct uvc_xu_control_info {
|
||||
__u8 entity[16];
|
||||
__u8 index;
|
||||
__u8 selector;
|
||||
__u16 size;
|
||||
__u32 flags;
|
||||
};
|
||||
|
||||
struct uvc_xu_control_mapping {
|
||||
__u32 id;
|
||||
__u8 name[32];
|
||||
__u8 entity[16];
|
||||
__u8 selector;
|
||||
|
||||
__u8 size;
|
||||
__u8 offset;
|
||||
enum v4l2_ctrl_type v4l2_type;
|
||||
__u32 data_type;
|
||||
};
|
||||
|
||||
struct uvc_xu_control {
|
||||
__u8 unit;
|
||||
__u8 selector;
|
||||
__u16 size;
|
||||
__u8 __user *data;
|
||||
};
|
||||
|
||||
#define UVCIOC_CTRL_ADD _IOW('U', 1, struct uvc_xu_control_info)
|
||||
#define UVCIOC_CTRL_MAP _IOWR('U', 2, struct uvc_xu_control_mapping)
|
||||
#define UVCIOC_CTRL_GET _IOWR('U', 3, struct uvc_xu_control)
|
||||
#define UVCIOC_CTRL_SET _IOW('U', 4, struct uvc_xu_control)
|
||||
|
||||
#ifdef __KERNEL__
|
||||
|
||||
#include <linux/poll.h>
|
||||
|
||||
/* --------------------------------------------------------------------------
|
||||
* UVC constants
|
||||
*/
|
||||
|
||||
#define SC_UNDEFINED 0x00
|
||||
#define SC_VIDEOCONTROL 0x01
|
||||
#define SC_VIDEOSTREAMING 0x02
|
||||
#define SC_VIDEO_INTERFACE_COLLECTION 0x03
|
||||
|
||||
#define PC_PROTOCOL_UNDEFINED 0x00
|
||||
|
||||
#define CS_UNDEFINED 0x20
|
||||
#define CS_DEVICE 0x21
|
||||
#define CS_CONFIGURATION 0x22
|
||||
#define CS_STRING 0x23
|
||||
#define CS_INTERFACE 0x24
|
||||
#define CS_ENDPOINT 0x25
|
||||
|
||||
/* VideoControl class specific interface descriptor */
|
||||
#define VC_DESCRIPTOR_UNDEFINED 0x00
|
||||
#define VC_HEADER 0x01
|
||||
#define VC_INPUT_TERMINAL 0x02
|
||||
#define VC_OUTPUT_TERMINAL 0x03
|
||||
#define VC_SELECTOR_UNIT 0x04
|
||||
#define VC_PROCESSING_UNIT 0x05
|
||||
#define VC_EXTENSION_UNIT 0x06
|
||||
|
||||
/* VideoStreaming class specific interface descriptor */
|
||||
#define VS_UNDEFINED 0x00
|
||||
#define VS_INPUT_HEADER 0x01
|
||||
#define VS_OUTPUT_HEADER 0x02
|
||||
#define VS_STILL_IMAGE_FRAME 0x03
|
||||
#define VS_FORMAT_UNCOMPRESSED 0x04
|
||||
#define VS_FRAME_UNCOMPRESSED 0x05
|
||||
#define VS_FORMAT_MJPEG 0x06
|
||||
#define VS_FRAME_MJPEG 0x07
|
||||
#define VS_FORMAT_MPEG2TS 0x0a
|
||||
#define VS_FORMAT_DV 0x0c
|
||||
#define VS_COLORFORMAT 0x0d
|
||||
#define VS_FORMAT_FRAME_BASED 0x10
|
||||
#define VS_FRAME_FRAME_BASED 0x11
|
||||
#define VS_FORMAT_STREAM_BASED 0x12
|
||||
|
||||
/* Endpoint type */
|
||||
#define EP_UNDEFINED 0x00
|
||||
#define EP_GENERAL 0x01
|
||||
#define EP_ENDPOINT 0x02
|
||||
#define EP_INTERRUPT 0x03
|
||||
|
||||
/* Request codes */
|
||||
#define RC_UNDEFINED 0x00
|
||||
#define SET_CUR 0x01
|
||||
#define GET_CUR 0x81
|
||||
#define GET_MIN 0x82
|
||||
#define GET_MAX 0x83
|
||||
#define GET_RES 0x84
|
||||
#define GET_LEN 0x85
|
||||
#define GET_INFO 0x86
|
||||
#define GET_DEF 0x87
|
||||
|
||||
/* VideoControl interface controls */
|
||||
#define VC_CONTROL_UNDEFINED 0x00
|
||||
#define VC_VIDEO_POWER_MODE_CONTROL 0x01
|
||||
#define VC_REQUEST_ERROR_CODE_CONTROL 0x02
|
||||
|
||||
/* Terminal controls */
|
||||
#define TE_CONTROL_UNDEFINED 0x00
|
||||
|
||||
/* Selector Unit controls */
|
||||
#define SU_CONTROL_UNDEFINED 0x00
|
||||
#define SU_INPUT_SELECT_CONTROL 0x01
|
||||
|
||||
/* Camera Terminal controls */
|
||||
#define CT_CONTROL_UNDEFINED 0x00
|
||||
#define CT_SCANNING_MODE_CONTROL 0x01
|
||||
#define CT_AE_MODE_CONTROL 0x02
|
||||
#define CT_AE_PRIORITY_CONTROL 0x03
|
||||
#define CT_EXPOSURE_TIME_ABSOLUTE_CONTROL 0x04
|
||||
#define CT_EXPOSURE_TIME_RELATIVE_CONTROL 0x05
|
||||
#define CT_FOCUS_ABSOLUTE_CONTROL 0x06
|
||||
#define CT_FOCUS_RELATIVE_CONTROL 0x07
|
||||
#define CT_FOCUS_AUTO_CONTROL 0x08
|
||||
#define CT_IRIS_ABSOLUTE_CONTROL 0x09
|
||||
#define CT_IRIS_RELATIVE_CONTROL 0x0a
|
||||
#define CT_ZOOM_ABSOLUTE_CONTROL 0x0b
|
||||
#define CT_ZOOM_RELATIVE_CONTROL 0x0c
|
||||
#define CT_PANTILT_ABSOLUTE_CONTROL 0x0d
|
||||
#define CT_PANTILT_RELATIVE_CONTROL 0x0e
|
||||
#define CT_ROLL_ABSOLUTE_CONTROL 0x0f
|
||||
#define CT_ROLL_RELATIVE_CONTROL 0x10
|
||||
#define CT_PRIVACY_CONTROL 0x11
|
||||
|
||||
/* Processing Unit controls */
|
||||
#define PU_CONTROL_UNDEFINED 0x00
|
||||
#define PU_BACKLIGHT_COMPENSATION_CONTROL 0x01
|
||||
#define PU_BRIGHTNESS_CONTROL 0x02
|
||||
#define PU_CONTRAST_CONTROL 0x03
|
||||
#define PU_GAIN_CONTROL 0x04
|
||||
#define PU_POWER_LINE_FREQUENCY_CONTROL 0x05
|
||||
#define PU_HUE_CONTROL 0x06
|
||||
#define PU_SATURATION_CONTROL 0x07
|
||||
#define PU_SHARPNESS_CONTROL 0x08
|
||||
#define PU_GAMMA_CONTROL 0x09
|
||||
#define PU_WHITE_BALANCE_TEMPERATURE_CONTROL 0x0a
|
||||
#define PU_WHITE_BALANCE_TEMPERATURE_AUTO_CONTROL 0x0b
|
||||
#define PU_WHITE_BALANCE_COMPONENT_CONTROL 0x0c
|
||||
#define PU_WHITE_BALANCE_COMPONENT_AUTO_CONTROL 0x0d
|
||||
#define PU_DIGITAL_MULTIPLIER_CONTROL 0x0e
|
||||
#define PU_DIGITAL_MULTIPLIER_LIMIT_CONTROL 0x0f
|
||||
#define PU_HUE_AUTO_CONTROL 0x10
|
||||
#define PU_ANALOG_VIDEO_STANDARD_CONTROL 0x11
|
||||
#define PU_ANALOG_LOCK_STATUS_CONTROL 0x12
|
||||
|
||||
#define LXU_MOTOR_PANTILT_RELATIVE_CONTROL 0x01
|
||||
#define LXU_MOTOR_PANTILT_RESET_CONTROL 0x02
|
||||
#define LXU_MOTOR_FOCUS_MOTOR_CONTROL 0x03
|
||||
|
||||
/* VideoStreaming interface controls */
|
||||
#define VS_CONTROL_UNDEFINED 0x00
|
||||
#define VS_PROBE_CONTROL 0x01
|
||||
#define VS_COMMIT_CONTROL 0x02
|
||||
#define VS_STILL_PROBE_CONTROL 0x03
|
||||
#define VS_STILL_COMMIT_CONTROL 0x04
|
||||
#define VS_STILL_IMAGE_TRIGGER_CONTROL 0x05
|
||||
#define VS_STREAM_ERROR_CODE_CONTROL 0x06
|
||||
#define VS_GENERATE_KEY_FRAME_CONTROL 0x07
|
||||
#define VS_UPDATE_FRAME_SEGMENT_CONTROL 0x08
|
||||
#define VS_SYNC_DELAY_CONTROL 0x09
|
||||
|
||||
#define TT_VENDOR_SPECIFIC 0x0100
|
||||
#define TT_STREAMING 0x0101
|
||||
|
||||
/* Input Terminal types */
|
||||
#define ITT_VENDOR_SPECIFIC 0x0200
|
||||
#define ITT_CAMERA 0x0201
|
||||
#define ITT_MEDIA_TRANSPORT_INPUT 0x0202
|
||||
|
||||
/* Output Terminal types */
|
||||
#define OTT_VENDOR_SPECIFIC 0x0300
|
||||
#define OTT_DISPLAY 0x0301
|
||||
#define OTT_MEDIA_TRANSPORT_OUTPUT 0x0302
|
||||
|
||||
/* External Terminal types */
|
||||
#define EXTERNAL_VENDOR_SPECIFIC 0x0400
|
||||
#define COMPOSITE_CONNECTOR 0x0401
|
||||
#define SVIDEO_CONNECTOR 0x0402
|
||||
#define COMPONENT_CONNECTOR 0x0403
|
||||
|
||||
#define UVC_TERM_INPUT 0x0000
|
||||
#define UVC_TERM_OUTPUT 0x8000
|
||||
|
||||
#define UVC_ENTITY_TYPE(entity) ((entity)->type & 0x7fff)
|
||||
#define UVC_ENTITY_IS_UNIT(entity) (((entity)->type & 0xff00) == 0)
|
||||
#define UVC_ENTITY_IS_TERM(entity) (((entity)->type & 0xff00) != 0)
|
||||
#define UVC_ENTITY_IS_ITERM(entity) \
|
||||
(((entity)->type & 0x8000) == UVC_TERM_INPUT)
|
||||
#define UVC_ENTITY_IS_OTERM(entity) \
|
||||
(((entity)->type & 0x8000) == UVC_TERM_OUTPUT)
|
||||
|
||||
#define UVC_STATUS_TYPE_CONTROL 1
|
||||
#define UVC_STATUS_TYPE_STREAMING 2
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* GUIDs
|
||||
*/
|
||||
#define UVC_GUID_UVC_CAMERA \
|
||||
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, \
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01}
|
||||
#define UVC_GUID_UVC_OUTPUT \
|
||||
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, \
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02}
|
||||
#define UVC_GUID_UVC_MEDIA_TRANSPORT_INPUT \
|
||||
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, \
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03}
|
||||
#define UVC_GUID_UVC_PROCESSING \
|
||||
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, \
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01}
|
||||
#define UVC_GUID_UVC_SELECTOR \
|
||||
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, \
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x02}
|
||||
|
||||
#define UVC_GUID_LOGITECH_DEV_INFO \
|
||||
{0x82, 0x06, 0x61, 0x63, 0x70, 0x50, 0xab, 0x49, \
|
||||
0xb8, 0xcc, 0xb3, 0x85, 0x5e, 0x8d, 0x22, 0x1e}
|
||||
#define UVC_GUID_LOGITECH_USER_HW \
|
||||
{0x82, 0x06, 0x61, 0x63, 0x70, 0x50, 0xab, 0x49, \
|
||||
0xb8, 0xcc, 0xb3, 0x85, 0x5e, 0x8d, 0x22, 0x1f}
|
||||
#define UVC_GUID_LOGITECH_VIDEO \
|
||||
{0x82, 0x06, 0x61, 0x63, 0x70, 0x50, 0xab, 0x49, \
|
||||
0xb8, 0xcc, 0xb3, 0x85, 0x5e, 0x8d, 0x22, 0x50}
|
||||
#define UVC_GUID_LOGITECH_MOTOR \
|
||||
{0x82, 0x06, 0x61, 0x63, 0x70, 0x50, 0xab, 0x49, \
|
||||
0xb8, 0xcc, 0xb3, 0x85, 0x5e, 0x8d, 0x22, 0x56}
|
||||
|
||||
#define UVC_GUID_FORMAT_MJPEG \
|
||||
{ 'M', 'J', 'P', 'G', 0x00, 0x00, 0x10, 0x00, \
|
||||
0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}
|
||||
#define UVC_GUID_FORMAT_YUY2 \
|
||||
{ 'Y', 'U', 'Y', '2', 0x00, 0x00, 0x10, 0x00, \
|
||||
0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}
|
||||
#define UVC_GUID_FORMAT_NV12 \
|
||||
{ 'N', 'V', '1', '2', 0x00, 0x00, 0x10, 0x00, \
|
||||
0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}
|
||||
#define UVC_GUID_FORMAT_YV12 \
|
||||
{ 'Y', 'V', '1', '2', 0x00, 0x00, 0x10, 0x00, \
|
||||
0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}
|
||||
#define UVC_GUID_FORMAT_I420 \
|
||||
{ 'I', '4', '2', '0', 0x00, 0x00, 0x10, 0x00, \
|
||||
0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}
|
||||
#define UVC_GUID_FORMAT_UYVY \
|
||||
{ 'U', 'Y', 'V', 'Y', 0x00, 0x00, 0x10, 0x00, \
|
||||
0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}
|
||||
#define UVC_GUID_FORMAT_Y800 \
|
||||
{ 'Y', '8', '0', '0', 0x00, 0x00, 0x10, 0x00, \
|
||||
0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}
|
||||
#define UVC_GUID_FORMAT_BY8 \
|
||||
{ 'B', 'Y', '8', ' ', 0x00, 0x00, 0x10, 0x00, \
|
||||
0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}
|
||||
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* Driver specific constants.
|
||||
*/
|
||||
|
||||
#define DRIVER_VERSION_NUMBER KERNEL_VERSION(0, 1, 0)
|
||||
|
||||
/* Number of isochronous URBs. */
|
||||
#define UVC_URBS 5
|
||||
/* Maximum number of packets per isochronous URB. */
|
||||
#define UVC_MAX_ISO_PACKETS 40
|
||||
/* Maximum frame size in bytes, for sanity checking. */
|
||||
#define UVC_MAX_FRAME_SIZE (16*1024*1024)
|
||||
/* Maximum number of video buffers. */
|
||||
#define UVC_MAX_VIDEO_BUFFERS 32
|
||||
|
||||
#define UVC_CTRL_CONTROL_TIMEOUT 300
|
||||
#define UVC_CTRL_STREAMING_TIMEOUT 1000
|
||||
|
||||
/* Devices quirks */
|
||||
#define UVC_QUIRK_STATUS_INTERVAL 0x00000001
|
||||
#define UVC_QUIRK_PROBE_MINMAX 0x00000002
|
||||
#define UVC_QUIRK_PROBE_EXTRAFIELDS 0x00000004
|
||||
#define UVC_QUIRK_BUILTIN_ISIGHT 0x00000008
|
||||
#define UVC_QUIRK_STREAM_NO_FID 0x00000010
|
||||
#define UVC_QUIRK_IGNORE_SELECTOR_UNIT 0x00000020
|
||||
|
||||
/* Format flags */
|
||||
#define UVC_FMT_FLAG_COMPRESSED 0x00000001
|
||||
#define UVC_FMT_FLAG_STREAM 0x00000002
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* Structures.
|
||||
*/
|
||||
|
||||
struct uvc_device;
|
||||
|
||||
/* TODO: Put the most frequently accessed fields at the beginning of
|
||||
* structures to maximize cache efficiency.
|
||||
*/
|
||||
struct uvc_streaming_control {
|
||||
__u16 bmHint;
|
||||
__u8 bFormatIndex;
|
||||
__u8 bFrameIndex;
|
||||
__u32 dwFrameInterval;
|
||||
__u16 wKeyFrameRate;
|
||||
__u16 wPFrameRate;
|
||||
__u16 wCompQuality;
|
||||
__u16 wCompWindowSize;
|
||||
__u16 wDelay;
|
||||
__u32 dwMaxVideoFrameSize;
|
||||
__u32 dwMaxPayloadTransferSize;
|
||||
__u32 dwClockFrequency;
|
||||
__u8 bmFramingInfo;
|
||||
__u8 bPreferedVersion;
|
||||
__u8 bMinVersion;
|
||||
__u8 bMaxVersion;
|
||||
};
|
||||
|
||||
struct uvc_menu_info {
|
||||
__u32 value;
|
||||
__u8 name[32];
|
||||
};
|
||||
|
||||
struct uvc_control_info {
|
||||
struct list_head list;
|
||||
struct list_head mappings;
|
||||
|
||||
__u8 entity[16];
|
||||
__u8 index;
|
||||
__u8 selector;
|
||||
|
||||
__u16 size;
|
||||
__u32 flags;
|
||||
};
|
||||
|
||||
struct uvc_control_mapping {
|
||||
struct list_head list;
|
||||
|
||||
struct uvc_control_info *ctrl;
|
||||
|
||||
__u32 id;
|
||||
__u8 name[32];
|
||||
__u8 entity[16];
|
||||
__u8 selector;
|
||||
|
||||
__u8 size;
|
||||
__u8 offset;
|
||||
enum v4l2_ctrl_type v4l2_type;
|
||||
__u32 data_type;
|
||||
|
||||
struct uvc_menu_info *menu_info;
|
||||
__u32 menu_count;
|
||||
};
|
||||
|
||||
struct uvc_control {
|
||||
struct uvc_entity *entity;
|
||||
struct uvc_control_info *info;
|
||||
|
||||
__u8 index; /* Used to match the uvc_control entry with a
|
||||
uvc_control_info. */
|
||||
__u8 dirty : 1,
|
||||
loaded : 1,
|
||||
modified : 1;
|
||||
|
||||
__u8 *data;
|
||||
};
|
||||
|
||||
struct uvc_format_desc {
|
||||
char *name;
|
||||
__u8 guid[16];
|
||||
__u32 fcc;
|
||||
};
|
||||
|
||||
/* The term 'entity' refers to both UVC units and UVC terminals.
|
||||
*
|
||||
* The type field is either the terminal type (wTerminalType in the terminal
|
||||
* descriptor), or the unit type (bDescriptorSubtype in the unit descriptor).
|
||||
* As the bDescriptorSubtype field is one byte long, the type value will
|
||||
* always have a null MSB for units. All terminal types defined by the UVC
|
||||
* specification have a non-null MSB, so it is safe to use the MSB to
|
||||
* differentiate between units and terminals as long as the descriptor parsing
|
||||
* code makes sure terminal types have a non-null MSB.
|
||||
*
|
||||
* For terminals, the type's most significant bit stores the terminal
|
||||
* direction (either UVC_TERM_INPUT or UVC_TERM_OUTPUT). The type field should
|
||||
* always be accessed with the UVC_ENTITY_* macros and never directly.
|
||||
*/
|
||||
|
||||
struct uvc_entity {
|
||||
struct list_head list; /* Entity as part of a UVC device. */
|
||||
struct list_head chain; /* Entity as part of a video device
|
||||
* chain. */
|
||||
__u8 id;
|
||||
__u16 type;
|
||||
char name[64];
|
||||
|
||||
union {
|
||||
struct {
|
||||
__u16 wObjectiveFocalLengthMin;
|
||||
__u16 wObjectiveFocalLengthMax;
|
||||
__u16 wOcularFocalLength;
|
||||
__u8 bControlSize;
|
||||
__u8 *bmControls;
|
||||
} camera;
|
||||
|
||||
struct {
|
||||
__u8 bControlSize;
|
||||
__u8 *bmControls;
|
||||
__u8 bTransportModeSize;
|
||||
__u8 *bmTransportModes;
|
||||
} media;
|
||||
|
||||
struct {
|
||||
__u8 bSourceID;
|
||||
} output;
|
||||
|
||||
struct {
|
||||
__u8 bSourceID;
|
||||
__u16 wMaxMultiplier;
|
||||
__u8 bControlSize;
|
||||
__u8 *bmControls;
|
||||
__u8 bmVideoStandards;
|
||||
} processing;
|
||||
|
||||
struct {
|
||||
__u8 bNrInPins;
|
||||
__u8 *baSourceID;
|
||||
} selector;
|
||||
|
||||
struct {
|
||||
__u8 guidExtensionCode[16];
|
||||
__u8 bNumControls;
|
||||
__u8 bNrInPins;
|
||||
__u8 *baSourceID;
|
||||
__u8 bControlSize;
|
||||
__u8 *bmControls;
|
||||
__u8 *bmControlsType;
|
||||
} extension;
|
||||
};
|
||||
|
||||
unsigned int ncontrols;
|
||||
struct uvc_control *controls;
|
||||
};
|
||||
|
||||
struct uvc_frame {
|
||||
__u8 bFrameIndex;
|
||||
__u8 bmCapabilities;
|
||||
__u16 wWidth;
|
||||
__u16 wHeight;
|
||||
__u32 dwMinBitRate;
|
||||
__u32 dwMaxBitRate;
|
||||
__u32 dwMaxVideoFrameBufferSize;
|
||||
__u8 bFrameIntervalType;
|
||||
__u32 dwDefaultFrameInterval;
|
||||
__u32 *dwFrameInterval;
|
||||
};
|
||||
|
||||
struct uvc_format {
|
||||
__u8 type;
|
||||
__u8 index;
|
||||
__u8 bpp;
|
||||
__u8 colorspace;
|
||||
__u32 fcc;
|
||||
__u32 flags;
|
||||
|
||||
char name[32];
|
||||
|
||||
unsigned int nframes;
|
||||
struct uvc_frame *frame;
|
||||
};
|
||||
|
||||
struct uvc_streaming_header {
|
||||
__u8 bNumFormats;
|
||||
__u8 bEndpointAddress;
|
||||
__u8 bTerminalLink;
|
||||
__u8 bControlSize;
|
||||
__u8 *bmaControls;
|
||||
/* The following fields are used by input headers only. */
|
||||
__u8 bmInfo;
|
||||
__u8 bStillCaptureMethod;
|
||||
__u8 bTriggerSupport;
|
||||
__u8 bTriggerUsage;
|
||||
};
|
||||
|
||||
struct uvc_streaming {
|
||||
struct list_head list;
|
||||
|
||||
struct usb_interface *intf;
|
||||
int intfnum;
|
||||
__u16 maxpsize;
|
||||
|
||||
struct uvc_streaming_header header;
|
||||
|
||||
unsigned int nformats;
|
||||
struct uvc_format *format;
|
||||
|
||||
struct uvc_streaming_control ctrl;
|
||||
struct uvc_format *cur_format;
|
||||
struct uvc_frame *cur_frame;
|
||||
|
||||
struct mutex mutex;
|
||||
};
|
||||
|
||||
enum uvc_buffer_state {
|
||||
UVC_BUF_STATE_IDLE = 0,
|
||||
UVC_BUF_STATE_QUEUED = 1,
|
||||
UVC_BUF_STATE_ACTIVE = 2,
|
||||
UVC_BUF_STATE_DONE = 3,
|
||||
UVC_BUF_STATE_ERROR = 4,
|
||||
};
|
||||
|
||||
struct uvc_buffer {
|
||||
unsigned long vma_use_count;
|
||||
struct list_head stream;
|
||||
|
||||
/* Touched by interrupt handler. */
|
||||
struct v4l2_buffer buf;
|
||||
struct list_head queue;
|
||||
wait_queue_head_t wait;
|
||||
enum uvc_buffer_state state;
|
||||
};
|
||||
|
||||
#define UVC_QUEUE_STREAMING (1 << 0)
|
||||
#define UVC_QUEUE_DISCONNECTED (1 << 1)
|
||||
#define UVC_QUEUE_DROP_INCOMPLETE (1 << 2)
|
||||
|
||||
struct uvc_video_queue {
|
||||
void *mem;
|
||||
unsigned int flags;
|
||||
__u32 sequence;
|
||||
|
||||
unsigned int count;
|
||||
unsigned int buf_size;
|
||||
struct uvc_buffer buffer[UVC_MAX_VIDEO_BUFFERS];
|
||||
struct mutex mutex; /* protects buffers and mainqueue */
|
||||
spinlock_t irqlock; /* protects irqqueue */
|
||||
|
||||
struct list_head mainqueue;
|
||||
struct list_head irqqueue;
|
||||
};
|
||||
|
||||
struct uvc_video_device {
|
||||
struct uvc_device *dev;
|
||||
struct video_device *vdev;
|
||||
atomic_t active;
|
||||
unsigned int frozen : 1;
|
||||
|
||||
struct list_head iterms;
|
||||
struct uvc_entity *oterm;
|
||||
struct uvc_entity *processing;
|
||||
struct uvc_entity *selector;
|
||||
struct list_head extensions;
|
||||
struct mutex ctrl_mutex;
|
||||
|
||||
struct uvc_video_queue queue;
|
||||
|
||||
/* Video streaming object, must always be non-NULL. */
|
||||
struct uvc_streaming *streaming;
|
||||
|
||||
void (*decode) (struct urb *urb, struct uvc_video_device *video,
|
||||
struct uvc_buffer *buf);
|
||||
|
||||
/* Context data used by the bulk completion handler. */
|
||||
struct {
|
||||
__u8 header[256];
|
||||
unsigned int header_size;
|
||||
int skip_payload;
|
||||
__u32 payload_size;
|
||||
__u32 max_payload_size;
|
||||
} bulk;
|
||||
|
||||
struct urb *urb[UVC_URBS];
|
||||
char *urb_buffer[UVC_URBS];
|
||||
|
||||
__u8 last_fid;
|
||||
};
|
||||
|
||||
enum uvc_device_state {
|
||||
UVC_DEV_DISCONNECTED = 1,
|
||||
};
|
||||
|
||||
struct uvc_device {
|
||||
struct usb_device *udev;
|
||||
struct usb_interface *intf;
|
||||
__u32 quirks;
|
||||
int intfnum;
|
||||
char name[32];
|
||||
|
||||
enum uvc_device_state state;
|
||||
struct kref kref;
|
||||
struct list_head list;
|
||||
|
||||
/* Video control interface */
|
||||
__u16 uvc_version;
|
||||
__u32 clock_frequency;
|
||||
|
||||
struct list_head entities;
|
||||
|
||||
struct uvc_video_device video;
|
||||
|
||||
/* Status Interrupt Endpoint */
|
||||
struct usb_host_endpoint *int_ep;
|
||||
struct urb *int_urb;
|
||||
__u8 status[16];
|
||||
struct input_dev *input;
|
||||
|
||||
/* Video Streaming interfaces */
|
||||
struct list_head streaming;
|
||||
};
|
||||
|
||||
enum uvc_handle_state {
|
||||
UVC_HANDLE_PASSIVE = 0,
|
||||
UVC_HANDLE_ACTIVE = 1,
|
||||
};
|
||||
|
||||
struct uvc_fh {
|
||||
struct uvc_video_device *device;
|
||||
enum uvc_handle_state state;
|
||||
};
|
||||
|
||||
struct uvc_driver {
|
||||
struct usb_driver driver;
|
||||
|
||||
struct mutex open_mutex; /* protects from open/disconnect race */
|
||||
|
||||
struct list_head devices; /* struct uvc_device list */
|
||||
struct list_head controls; /* struct uvc_control_info list */
|
||||
struct mutex ctrl_mutex; /* protects controls and devices
|
||||
lists */
|
||||
};
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* Debugging, printing and logging
|
||||
*/
|
||||
|
||||
#define UVC_TRACE_PROBE (1 << 0)
|
||||
#define UVC_TRACE_DESCR (1 << 1)
|
||||
#define UVC_TRACE_CONTROL (1 << 2)
|
||||
#define UVC_TRACE_FORMAT (1 << 3)
|
||||
#define UVC_TRACE_CAPTURE (1 << 4)
|
||||
#define UVC_TRACE_CALLS (1 << 5)
|
||||
#define UVC_TRACE_IOCTL (1 << 6)
|
||||
#define UVC_TRACE_FRAME (1 << 7)
|
||||
#define UVC_TRACE_SUSPEND (1 << 8)
|
||||
#define UVC_TRACE_STATUS (1 << 9)
|
||||
|
||||
extern unsigned int uvc_trace_param;
|
||||
|
||||
#define uvc_trace(flag, msg...) \
|
||||
do { \
|
||||
if (uvc_trace_param & flag) \
|
||||
printk(KERN_DEBUG "uvcvideo: " msg); \
|
||||
} while (0)
|
||||
|
||||
#define uvc_printk(level, msg...) \
|
||||
printk(level "uvcvideo: " msg)
|
||||
|
||||
#define UVC_GUID_FORMAT "%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-" \
|
||||
"%02x%02x%02x%02x%02x%02x"
|
||||
#define UVC_GUID_ARGS(guid) \
|
||||
(guid)[3], (guid)[2], (guid)[1], (guid)[0], \
|
||||
(guid)[5], (guid)[4], \
|
||||
(guid)[7], (guid)[6], \
|
||||
(guid)[8], (guid)[9], \
|
||||
(guid)[10], (guid)[11], (guid)[12], \
|
||||
(guid)[13], (guid)[14], (guid)[15]
|
||||
|
||||
/* --------------------------------------------------------------------------
|
||||
* Internal functions.
|
||||
*/
|
||||
|
||||
/* Core driver */
|
||||
extern struct uvc_driver uvc_driver;
|
||||
extern void uvc_delete(struct kref *kref);
|
||||
|
||||
/* Video buffers queue management. */
|
||||
extern void uvc_queue_init(struct uvc_video_queue *queue);
|
||||
extern int uvc_alloc_buffers(struct uvc_video_queue *queue,
|
||||
unsigned int nbuffers, unsigned int buflength);
|
||||
extern int uvc_free_buffers(struct uvc_video_queue *queue);
|
||||
extern int uvc_query_buffer(struct uvc_video_queue *queue,
|
||||
struct v4l2_buffer *v4l2_buf);
|
||||
extern int uvc_queue_buffer(struct uvc_video_queue *queue,
|
||||
struct v4l2_buffer *v4l2_buf);
|
||||
extern int uvc_dequeue_buffer(struct uvc_video_queue *queue,
|
||||
struct v4l2_buffer *v4l2_buf, int nonblocking);
|
||||
extern int uvc_queue_enable(struct uvc_video_queue *queue, int enable);
|
||||
extern void uvc_queue_cancel(struct uvc_video_queue *queue, int disconnect);
|
||||
extern struct uvc_buffer *uvc_queue_next_buffer(struct uvc_video_queue *queue,
|
||||
struct uvc_buffer *buf);
|
||||
extern unsigned int uvc_queue_poll(struct uvc_video_queue *queue,
|
||||
struct file *file, poll_table *wait);
|
||||
static inline int uvc_queue_streaming(struct uvc_video_queue *queue)
|
||||
{
|
||||
return queue->flags & UVC_QUEUE_STREAMING;
|
||||
}
|
||||
|
||||
/* V4L2 interface */
|
||||
extern struct file_operations uvc_fops;
|
||||
|
||||
/* Video */
|
||||
extern int uvc_video_init(struct uvc_video_device *video);
|
||||
extern int uvc_video_suspend(struct uvc_video_device *video);
|
||||
extern int uvc_video_resume(struct uvc_video_device *video);
|
||||
extern int uvc_video_enable(struct uvc_video_device *video, int enable);
|
||||
extern int uvc_probe_video(struct uvc_video_device *video,
|
||||
struct uvc_streaming_control *probe);
|
||||
extern int uvc_query_ctrl(struct uvc_device *dev, __u8 query, __u8 unit,
|
||||
__u8 intfnum, __u8 cs, void *data, __u16 size);
|
||||
extern int uvc_set_video_ctrl(struct uvc_video_device *video,
|
||||
struct uvc_streaming_control *ctrl, int probe);
|
||||
|
||||
/* Status */
|
||||
extern int uvc_status_init(struct uvc_device *dev);
|
||||
extern void uvc_status_cleanup(struct uvc_device *dev);
|
||||
extern int uvc_status_suspend(struct uvc_device *dev);
|
||||
extern int uvc_status_resume(struct uvc_device *dev);
|
||||
|
||||
/* Controls */
|
||||
extern struct uvc_control *uvc_find_control(struct uvc_video_device *video,
|
||||
__u32 v4l2_id, struct uvc_control_mapping **mapping);
|
||||
extern int uvc_query_v4l2_ctrl(struct uvc_video_device *video,
|
||||
struct v4l2_queryctrl *v4l2_ctrl);
|
||||
|
||||
extern int uvc_ctrl_add_info(struct uvc_control_info *info);
|
||||
extern int uvc_ctrl_add_mapping(struct uvc_control_mapping *mapping);
|
||||
extern int uvc_ctrl_init_device(struct uvc_device *dev);
|
||||
extern void uvc_ctrl_cleanup_device(struct uvc_device *dev);
|
||||
extern int uvc_ctrl_resume_device(struct uvc_device *dev);
|
||||
extern void uvc_ctrl_init(void);
|
||||
|
||||
extern int uvc_ctrl_begin(struct uvc_video_device *video);
|
||||
extern int __uvc_ctrl_commit(struct uvc_video_device *video, int rollback);
|
||||
static inline int uvc_ctrl_commit(struct uvc_video_device *video)
|
||||
{
|
||||
return __uvc_ctrl_commit(video, 0);
|
||||
}
|
||||
static inline int uvc_ctrl_rollback(struct uvc_video_device *video)
|
||||
{
|
||||
return __uvc_ctrl_commit(video, 1);
|
||||
}
|
||||
|
||||
extern int uvc_ctrl_get(struct uvc_video_device *video,
|
||||
struct v4l2_ext_control *xctrl);
|
||||
extern int uvc_ctrl_set(struct uvc_video_device *video,
|
||||
struct v4l2_ext_control *xctrl);
|
||||
|
||||
extern int uvc_xu_ctrl_query(struct uvc_video_device *video,
|
||||
struct uvc_xu_control *ctrl, int set);
|
||||
|
||||
/* Utility functions */
|
||||
extern void uvc_simplify_fraction(uint32_t *numerator, uint32_t *denominator,
|
||||
unsigned int n_terms, unsigned int threshold);
|
||||
extern uint32_t uvc_fraction_to_interval(uint32_t numerator,
|
||||
uint32_t denominator);
|
||||
extern struct usb_host_endpoint *uvc_find_endpoint(
|
||||
struct usb_host_interface *alts, __u8 epaddr);
|
||||
|
||||
/* Quirks support */
|
||||
void uvc_video_decode_isight(struct urb *urb, struct uvc_video_device *video,
|
||||
struct uvc_buffer *buf);
|
||||
|
||||
#endif /* __KERNEL__ */
|
||||
|
||||
#endif
|
Loading…
Reference in New Issue