2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* Endpoints (formerly known as AOX) se401 USB Camera Driver
|
|
|
|
*
|
|
|
|
* Copyright (c) 2000 Jeroen B. Vreeken (pe1rxq@amsat.org)
|
|
|
|
*
|
|
|
|
* Still somewhat based on the Linux ov511 driver.
|
2006-03-25 20:19:53 +08:00
|
|
|
*
|
2005-04-17 06:20:36 +08:00
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
|
|
|
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
* for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* Thanks to Endpoints Inc. (www.endpoints.com) for making documentation on
|
|
|
|
* their chipset available and supporting me while writing this driver.
|
|
|
|
* - Jeroen Vreeken
|
|
|
|
*/
|
|
|
|
|
|
|
|
static const char version[] = "0.24";
|
|
|
|
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/vmalloc.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/pagemap.h>
|
|
|
|
#include <linux/usb.h>
|
|
|
|
#include "se401.h"
|
|
|
|
|
2008-04-23 01:41:48 +08:00
|
|
|
static int flickerless;
|
2005-04-17 06:20:36 +08:00
|
|
|
static int video_nr = -1;
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
static struct usb_device_id device_table[] = {
|
2005-04-17 06:20:36 +08:00
|
|
|
{ USB_DEVICE(0x03e8, 0x0004) },/* Endpoints/Aox SE401 */
|
|
|
|
{ USB_DEVICE(0x0471, 0x030b) },/* Philips PCVC665K */
|
|
|
|
{ USB_DEVICE(0x047d, 0x5001) },/* Kensington 67014 */
|
|
|
|
{ USB_DEVICE(0x047d, 0x5002) },/* Kensington 6701(5/7) */
|
|
|
|
{ USB_DEVICE(0x047d, 0x5003) },/* Kensington 67016 */
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
|
|
|
MODULE_DEVICE_TABLE(usb, device_table);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Jeroen Vreeken <pe1rxq@amsat.org>");
|
|
|
|
MODULE_DESCRIPTION("SE401 USB Camera Driver");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
module_param(flickerless, int, 0);
|
2009-06-09 21:02:11 +08:00
|
|
|
MODULE_PARM_DESC(flickerless,
|
|
|
|
"Net frequency to adjust exposure time to (0/50/60)");
|
2005-04-17 06:20:36 +08:00
|
|
|
module_param(video_nr, int, 0);
|
|
|
|
|
|
|
|
static struct usb_driver se401_driver;
|
|
|
|
|
|
|
|
|
|
|
|
/**********************************************************************
|
|
|
|
*
|
|
|
|
* Memory management
|
|
|
|
*
|
|
|
|
**********************************************************************/
|
|
|
|
static void *rvmalloc(unsigned long size)
|
|
|
|
{
|
|
|
|
void *mem;
|
|
|
|
unsigned long adr;
|
|
|
|
|
|
|
|
size = PAGE_ALIGN(size);
|
|
|
|
mem = vmalloc_32(size);
|
|
|
|
if (!mem)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
memset(mem, 0, size); /* Clear the ram out, no junk to the user */
|
|
|
|
adr = (unsigned long) mem;
|
|
|
|
while (size > 0) {
|
|
|
|
SetPageReserved(vmalloc_to_page((void *)adr));
|
2009-06-09 21:02:11 +08:00
|
|
|
adr += PAGE_SIZE;
|
|
|
|
size -= PAGE_SIZE;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return mem;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void rvfree(void *mem, unsigned long size)
|
|
|
|
{
|
|
|
|
unsigned long adr;
|
|
|
|
|
|
|
|
if (!mem)
|
|
|
|
return;
|
|
|
|
|
|
|
|
adr = (unsigned long) mem;
|
|
|
|
while ((long) size > 0) {
|
|
|
|
ClearPageReserved(vmalloc_to_page((void *)adr));
|
2009-06-09 21:02:11 +08:00
|
|
|
adr += PAGE_SIZE;
|
|
|
|
size -= PAGE_SIZE;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
vfree(mem);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
*
|
|
|
|
* se401 register read/write functions
|
|
|
|
*
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
static int se401_sndctrl(int set, struct usb_se401 *se401, unsigned short req,
|
|
|
|
unsigned short value, unsigned char *cp, int size)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
return usb_control_msg(
|
2006-03-25 20:19:53 +08:00
|
|
|
se401->dev,
|
|
|
|
set ? usb_sndctrlpipe(se401->dev, 0) : usb_rcvctrlpipe(se401->dev, 0),
|
|
|
|
req,
|
|
|
|
(set ? USB_DIR_OUT : USB_DIR_IN) | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
|
|
|
|
value,
|
|
|
|
0,
|
|
|
|
cp,
|
|
|
|
size,
|
|
|
|
1000
|
|
|
|
);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int se401_set_feature(struct usb_se401 *se401, unsigned short selector,
|
|
|
|
unsigned short param)
|
|
|
|
{
|
|
|
|
/* specs say that the selector (address) should go in the value field
|
|
|
|
and the param in index, but in the logs of the windows driver they do
|
|
|
|
this the other way around...
|
|
|
|
*/
|
2009-06-09 21:02:11 +08:00
|
|
|
return usb_control_msg(
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->dev,
|
|
|
|
usb_sndctrlpipe(se401->dev, 0),
|
|
|
|
SE401_REQ_SET_EXT_FEATURE,
|
|
|
|
USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
|
|
|
|
param,
|
|
|
|
selector,
|
2006-03-25 20:19:53 +08:00
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
1000
|
|
|
|
);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
static unsigned short se401_get_feature(struct usb_se401 *se401,
|
|
|
|
unsigned short selector)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
/* For 'set' the selecetor should be in index, not sure if the spec is
|
|
|
|
wrong here to....
|
|
|
|
*/
|
|
|
|
unsigned char cp[2];
|
2009-06-09 21:02:11 +08:00
|
|
|
usb_control_msg(
|
2006-03-25 20:19:53 +08:00
|
|
|
se401->dev,
|
|
|
|
usb_rcvctrlpipe(se401->dev, 0),
|
|
|
|
SE401_REQ_GET_EXT_FEATURE,
|
|
|
|
USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
|
|
|
|
0,
|
|
|
|
selector,
|
|
|
|
cp,
|
|
|
|
2,
|
|
|
|
1000
|
|
|
|
);
|
2005-04-17 06:20:36 +08:00
|
|
|
return cp[0]+cp[1]*256;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
*
|
|
|
|
* Camera control
|
|
|
|
*
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
static int se401_send_pict(struct usb_se401 *se401)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
/* integration time low */
|
|
|
|
se401_set_feature(se401, HV7131_REG_TITL, se401->expose_l);
|
|
|
|
/* integration time mid */
|
|
|
|
se401_set_feature(se401, HV7131_REG_TITM, se401->expose_m);
|
|
|
|
/* integration time mid */
|
|
|
|
se401_set_feature(se401, HV7131_REG_TITU, se401->expose_h);
|
|
|
|
/* reset level value */
|
|
|
|
se401_set_feature(se401, HV7131_REG_ARLV, se401->resetlevel);
|
|
|
|
/* red color gain */
|
|
|
|
se401_set_feature(se401, HV7131_REG_ARCG, se401->rgain);
|
|
|
|
/* green color gain */
|
|
|
|
se401_set_feature(se401, HV7131_REG_AGCG, se401->ggain);
|
|
|
|
/* blue color gain */
|
|
|
|
se401_set_feature(se401, HV7131_REG_ABCG, se401->bgain);
|
2006-03-25 20:19:53 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void se401_set_exposure(struct usb_se401 *se401, int brightness)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
int integration = brightness << 5;
|
|
|
|
|
|
|
|
if (flickerless == 50)
|
|
|
|
integration = integration-integration % 106667;
|
|
|
|
if (flickerless == 60)
|
|
|
|
integration = integration-integration % 88889;
|
|
|
|
se401->brightness = integration >> 5;
|
|
|
|
se401->expose_h = (integration >> 16) & 0xff;
|
|
|
|
se401->expose_m = (integration >> 8) & 0xff;
|
|
|
|
se401->expose_l = integration & 0xff;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int se401_get_pict(struct usb_se401 *se401, struct video_picture *p)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
p->brightness = se401->brightness;
|
|
|
|
if (se401->enhance)
|
|
|
|
p->whiteness = 32768;
|
|
|
|
else
|
|
|
|
p->whiteness = 0;
|
|
|
|
|
|
|
|
p->colour = 65535;
|
|
|
|
p->contrast = 65535;
|
|
|
|
p->hue = se401->rgain << 10;
|
|
|
|
p->palette = se401->palette;
|
|
|
|
p->depth = 3; /* rgb24 */
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int se401_set_pict(struct usb_se401 *se401, struct video_picture *p)
|
|
|
|
{
|
|
|
|
if (p->palette != VIDEO_PALETTE_RGB24)
|
|
|
|
return 1;
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->palette = p->palette;
|
|
|
|
if (p->hue != se401->hue) {
|
|
|
|
se401->rgain = p->hue >> 10;
|
|
|
|
se401->bgain = 0x40-(p->hue >> 10);
|
|
|
|
se401->hue = p->hue;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
if (p->brightness != se401->brightness)
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_set_exposure(se401, p->brightness);
|
2009-06-09 21:02:11 +08:00
|
|
|
|
|
|
|
if (p->whiteness >= 32768)
|
|
|
|
se401->enhance = 1;
|
|
|
|
else
|
|
|
|
se401->enhance = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_send_pict(se401);
|
|
|
|
se401_send_pict(se401);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Hyundai have some really nice docs about this and other sensor related
|
|
|
|
stuff on their homepage: www.hei.co.kr
|
|
|
|
*/
|
|
|
|
static void se401_auto_resetlevel(struct usb_se401 *se401)
|
|
|
|
{
|
|
|
|
unsigned int ahrc, alrc;
|
2009-06-09 21:02:11 +08:00
|
|
|
int oldreset = se401->resetlevel;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* For some reason this normally read-only register doesn't get reset
|
|
|
|
to zero after reading them just once...
|
|
|
|
*/
|
2006-03-25 20:19:53 +08:00
|
|
|
se401_get_feature(se401, HV7131_REG_HIREFNOH);
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_get_feature(se401, HV7131_REG_HIREFNOL);
|
|
|
|
se401_get_feature(se401, HV7131_REG_LOREFNOH);
|
|
|
|
se401_get_feature(se401, HV7131_REG_LOREFNOL);
|
2009-06-09 21:02:11 +08:00
|
|
|
ahrc = 256*se401_get_feature(se401, HV7131_REG_HIREFNOH) +
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_get_feature(se401, HV7131_REG_HIREFNOL);
|
2009-06-09 21:02:11 +08:00
|
|
|
alrc = 256*se401_get_feature(se401, HV7131_REG_LOREFNOH) +
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_get_feature(se401, HV7131_REG_LOREFNOL);
|
|
|
|
|
|
|
|
/* Not an exact science, but it seems to work pretty well... */
|
|
|
|
if (alrc > 10) {
|
2009-06-09 21:02:11 +08:00
|
|
|
while (alrc >= 10 && se401->resetlevel < 63) {
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->resetlevel++;
|
2009-06-09 21:02:11 +08:00
|
|
|
alrc /= 2;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
} else if (ahrc > 20) {
|
2009-06-09 21:02:11 +08:00
|
|
|
while (ahrc >= 20 && se401->resetlevel > 0) {
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->resetlevel--;
|
2009-06-09 21:02:11 +08:00
|
|
|
ahrc /= 2;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->resetlevel != oldreset)
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_set_feature(se401, HV7131_REG_ARLV, se401->resetlevel);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* irq handler for snapshot button */
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 21:55:46 +08:00
|
|
|
static void se401_button_irq(struct urb *urb)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct usb_se401 *se401 = urb->context;
|
|
|
|
int status;
|
2006-03-25 20:19:53 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!se401->dev) {
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&urb->dev->dev, "device vapourished\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
return;
|
|
|
|
}
|
2006-03-25 20:19:53 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
switch (urb->status) {
|
|
|
|
case 0:
|
|
|
|
/* success */
|
|
|
|
break;
|
|
|
|
case -ECONNRESET:
|
|
|
|
case -ENOENT:
|
|
|
|
case -ESHUTDOWN:
|
|
|
|
/* this urb is terminated, clean up */
|
2009-06-09 21:02:11 +08:00
|
|
|
dbg("%s - urb shutting down with status: %d",
|
|
|
|
__func__, urb->status);
|
2005-04-17 06:20:36 +08:00
|
|
|
return;
|
|
|
|
default:
|
2009-06-09 21:02:11 +08:00
|
|
|
dbg("%s - nonzero urb status received: %d",
|
|
|
|
__func__, urb->status);
|
2005-04-17 06:20:36 +08:00
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
if (urb->actual_length >= 2)
|
2005-04-17 06:20:36 +08:00
|
|
|
if (se401->button)
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->buttonpressed = 1;
|
2005-04-17 06:20:36 +08:00
|
|
|
exit:
|
2009-06-09 21:02:11 +08:00
|
|
|
status = usb_submit_urb(urb, GFP_ATOMIC);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (status)
|
2009-06-09 21:02:11 +08:00
|
|
|
err("%s - usb_submit_urb failed with result %d",
|
2008-04-09 10:20:00 +08:00
|
|
|
__func__, status);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 21:55:46 +08:00
|
|
|
static void se401_video_irq(struct urb *urb)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct usb_se401 *se401 = urb->context;
|
|
|
|
int length = urb->actual_length;
|
|
|
|
|
|
|
|
/* ohoh... */
|
|
|
|
if (!se401->streaming)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!se401->dev) {
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&urb->dev->dev, "device vapourished\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 0 sized packets happen if we are to fast, but sometimes the camera
|
|
|
|
keeps sending them forever...
|
|
|
|
*/
|
|
|
|
if (length && !urb->status) {
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->nullpackets = 0;
|
|
|
|
switch (se401->scratch[se401->scratch_next].state) {
|
|
|
|
case BUFFER_READY:
|
|
|
|
case BUFFER_BUSY:
|
|
|
|
se401->dropped++;
|
|
|
|
break;
|
|
|
|
case BUFFER_UNUSED:
|
|
|
|
memcpy(se401->scratch[se401->scratch_next].data,
|
|
|
|
(unsigned char *)urb->transfer_buffer, length);
|
|
|
|
se401->scratch[se401->scratch_next].state
|
|
|
|
= BUFFER_READY;
|
|
|
|
se401->scratch[se401->scratch_next].offset
|
|
|
|
= se401->bayeroffset;
|
|
|
|
se401->scratch[se401->scratch_next].length = length;
|
|
|
|
if (waitqueue_active(&se401->wq))
|
|
|
|
wake_up_interruptible(&se401->wq);
|
|
|
|
se401->scratch_overflow = 0;
|
|
|
|
se401->scratch_next++;
|
|
|
|
if (se401->scratch_next >= SE401_NUMSCRATCH)
|
|
|
|
se401->scratch_next = 0;
|
|
|
|
break;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->bayeroffset += length;
|
|
|
|
if (se401->bayeroffset >= se401->cheight * se401->cwidth)
|
|
|
|
se401->bayeroffset = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
} else {
|
|
|
|
se401->nullpackets++;
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->nullpackets > SE401_MAX_NULLPACKETS)
|
|
|
|
if (waitqueue_active(&se401->wq))
|
2005-04-17 06:20:36 +08:00
|
|
|
wake_up_interruptible(&se401->wq);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Resubmit urb for new data */
|
2009-06-09 21:02:11 +08:00
|
|
|
urb->status = 0;
|
|
|
|
urb->dev = se401->dev;
|
|
|
|
if (usb_submit_urb(urb, GFP_KERNEL))
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&urb->dev->dev, "urb burned down\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void se401_send_size(struct usb_se401 *se401, int width, int height)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
int i = 0;
|
|
|
|
int mode = 0x03; /* No compression */
|
|
|
|
int sendheight = height;
|
|
|
|
int sendwidth = width;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* JangGu compression can only be used with the camera supported sizes,
|
|
|
|
but bayer seems to work with any size that fits on the sensor.
|
|
|
|
We check if we can use compression with the current size with either
|
|
|
|
4 or 16 times subcapturing, if not we use uncompressed bayer data
|
|
|
|
but this will result in cutouts of the maximum size....
|
|
|
|
*/
|
2009-06-09 21:02:11 +08:00
|
|
|
while (i < se401->sizes && !(se401->width[i] == width &&
|
|
|
|
se401->height[i] == height))
|
2005-04-17 06:20:36 +08:00
|
|
|
i++;
|
2009-06-09 21:02:11 +08:00
|
|
|
while (i < se401->sizes) {
|
|
|
|
if (se401->width[i] == width * 2 &&
|
|
|
|
se401->height[i] == height * 2) {
|
|
|
|
sendheight = se401->height[i];
|
|
|
|
sendwidth = se401->width[i];
|
|
|
|
mode = 0x40;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->width[i] == width * 4 &&
|
|
|
|
se401->height[i] == height * 4) {
|
|
|
|
sendheight = se401->height[i];
|
|
|
|
sendwidth = se401->width[i];
|
|
|
|
mode = 0x42;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
se401_sndctrl(1, se401, SE401_REQ_SET_WIDTH, sendwidth, NULL, 0);
|
|
|
|
se401_sndctrl(1, se401, SE401_REQ_SET_HEIGHT, sendheight, NULL, 0);
|
|
|
|
se401_set_feature(se401, SE401_OPERATINGMODE, mode);
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
if (mode == 0x03)
|
|
|
|
se401->format = FMT_BAYER;
|
|
|
|
else
|
|
|
|
se401->format = FMT_JANGGU;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
In this function se401_send_pict is called several times,
|
|
|
|
for some reason (depending on the state of the sensor and the phase of
|
|
|
|
the moon :) doing this only in either place doesn't always work...
|
|
|
|
*/
|
|
|
|
static int se401_start_stream(struct usb_se401 *se401)
|
|
|
|
{
|
|
|
|
struct urb *urb;
|
2009-06-09 21:02:11 +08:00
|
|
|
int err = 0, i;
|
|
|
|
se401->streaming = 1;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
se401_sndctrl(1, se401, SE401_REQ_CAMERA_POWER, 1, NULL, 0);
|
|
|
|
se401_sndctrl(1, se401, SE401_REQ_LED_CONTROL, 1, NULL, 0);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Set picture settings */
|
2009-06-09 21:02:11 +08:00
|
|
|
/* windowed + pix intg */
|
|
|
|
se401_set_feature(se401, HV7131_REG_MODE_B, 0x05);
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_send_pict(se401);
|
|
|
|
|
|
|
|
se401_send_size(se401, se401->cwidth, se401->cheight);
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
se401_sndctrl(1, se401, SE401_REQ_START_CONTINUOUS_CAPTURE,
|
|
|
|
0, NULL, 0);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Do some memory allocation */
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < SE401_NUMFRAMES; i++) {
|
|
|
|
se401->frame[i].data = se401->fbuf + i * se401->maxframesize;
|
|
|
|
se401->frame[i].curpix = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < SE401_NUMSBUF; i++) {
|
|
|
|
se401->sbuf[i].data = kmalloc(SE401_PACKETSIZE, GFP_KERNEL);
|
2007-03-31 04:48:59 +08:00
|
|
|
if (!se401->sbuf[i].data) {
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = i - 1; i >= 0; i--) {
|
2007-03-31 04:48:59 +08:00
|
|
|
kfree(se401->sbuf[i].data);
|
|
|
|
se401->sbuf[i].data = NULL;
|
|
|
|
}
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->bayeroffset = 0;
|
|
|
|
se401->scratch_next = 0;
|
|
|
|
se401->scratch_use = 0;
|
|
|
|
se401->scratch_overflow = 0;
|
|
|
|
for (i = 0; i < SE401_NUMSCRATCH; i++) {
|
|
|
|
se401->scratch[i].data = kmalloc(SE401_PACKETSIZE, GFP_KERNEL);
|
2007-03-31 04:48:59 +08:00
|
|
|
if (!se401->scratch[i].data) {
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = i - 1; i >= 0; i--) {
|
2007-03-31 04:48:59 +08:00
|
|
|
kfree(se401->scratch[i].data);
|
|
|
|
se401->scratch[i].data = NULL;
|
|
|
|
}
|
|
|
|
goto nomem_sbuf;
|
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->scratch[i].state = BUFFER_UNUSED;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < SE401_NUMSBUF; i++) {
|
|
|
|
urb = usb_alloc_urb(0, GFP_KERNEL);
|
|
|
|
if (!urb) {
|
|
|
|
for (i = i - 1; i >= 0; i--) {
|
2007-03-31 04:48:59 +08:00
|
|
|
usb_kill_urb(se401->urb[i]);
|
|
|
|
usb_free_urb(se401->urb[i]);
|
|
|
|
se401->urb[i] = NULL;
|
|
|
|
}
|
|
|
|
goto nomem_scratch;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
usb_fill_bulk_urb(urb, se401->dev,
|
|
|
|
usb_rcvbulkpipe(se401->dev, SE401_VIDEO_ENDPOINT),
|
|
|
|
se401->sbuf[i].data, SE401_PACKETSIZE,
|
|
|
|
se401_video_irq,
|
|
|
|
se401);
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->urb[i] = urb;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
err = usb_submit_urb(se401->urb[i], GFP_KERNEL);
|
|
|
|
if (err)
|
2005-04-17 06:20:36 +08:00
|
|
|
err("urb burned down");
|
|
|
|
}
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->framecount = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
return 0;
|
2007-03-31 04:48:59 +08:00
|
|
|
|
|
|
|
nomem_scratch:
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < SE401_NUMSCRATCH; i++) {
|
2007-03-31 04:48:59 +08:00
|
|
|
kfree(se401->scratch[i].data);
|
|
|
|
se401->scratch[i].data = NULL;
|
|
|
|
}
|
|
|
|
nomem_sbuf:
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < SE401_NUMSBUF; i++) {
|
2007-03-31 04:48:59 +08:00
|
|
|
kfree(se401->sbuf[i].data);
|
|
|
|
se401->sbuf[i].data = NULL;
|
|
|
|
}
|
|
|
|
return -ENOMEM;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int se401_stop_stream(struct usb_se401 *se401)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!se401->streaming || !se401->dev)
|
|
|
|
return 1;
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->streaming = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
se401_sndctrl(1, se401, SE401_REQ_STOP_CONTINUOUS_CAPTURE, 0, NULL, 0);
|
|
|
|
|
|
|
|
se401_sndctrl(1, se401, SE401_REQ_LED_CONTROL, 0, NULL, 0);
|
|
|
|
se401_sndctrl(1, se401, SE401_REQ_CAMERA_POWER, 0, NULL, 0);
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < SE401_NUMSBUF; i++)
|
|
|
|
if (se401->urb[i]) {
|
|
|
|
usb_kill_urb(se401->urb[i]);
|
|
|
|
usb_free_urb(se401->urb[i]);
|
|
|
|
se401->urb[i] = NULL;
|
|
|
|
kfree(se401->sbuf[i].data);
|
|
|
|
}
|
|
|
|
for (i = 0; i < SE401_NUMSCRATCH; i++) {
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree(se401->scratch[i].data);
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->scratch[i].data = NULL;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int se401_set_size(struct usb_se401 *se401, int width, int height)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
int wasstreaming = se401->streaming;
|
2005-04-17 06:20:36 +08:00
|
|
|
/* Check to see if we need to change */
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->cwidth == width && se401->cheight == height)
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* Check for a valid mode */
|
|
|
|
if (!width || !height)
|
|
|
|
return 1;
|
|
|
|
if ((width & 1) || (height & 1))
|
|
|
|
return 1;
|
2009-06-09 21:02:11 +08:00
|
|
|
if (width > se401->width[se401->sizes-1])
|
2005-04-17 06:20:36 +08:00
|
|
|
return 1;
|
2009-06-09 21:02:11 +08:00
|
|
|
if (height > se401->height[se401->sizes-1])
|
2005-04-17 06:20:36 +08:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* Stop a current stream and start it again at the new size */
|
|
|
|
if (wasstreaming)
|
|
|
|
se401_stop_stream(se401);
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->cwidth = width;
|
|
|
|
se401->cheight = height;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (wasstreaming)
|
|
|
|
se401_start_stream(se401);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
*
|
|
|
|
* Video Decoding
|
|
|
|
*
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
This shouldn't really be done in a v4l driver....
|
|
|
|
But it does make the image look a lot more usable.
|
|
|
|
Basically it lifts the dark pixels more than the light pixels.
|
|
|
|
*/
|
|
|
|
static inline void enhance_picture(unsigned char *frame, int len)
|
|
|
|
{
|
|
|
|
while (len--) {
|
2009-06-09 21:02:11 +08:00
|
|
|
*frame = (((*frame^255)*(*frame^255))/255)^255;
|
2005-04-17 06:20:36 +08:00
|
|
|
frame++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void decode_JangGu_integrate(struct usb_se401 *se401, int data)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
struct se401_frame *frame = &se401->frame[se401->curframe];
|
|
|
|
int linelength = se401->cwidth * 3;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
if (frame->curlinepix >= linelength) {
|
2009-06-09 21:02:11 +08:00
|
|
|
frame->curlinepix = 0;
|
|
|
|
frame->curline += linelength;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* First three are absolute, all others relative.
|
2006-03-25 20:19:53 +08:00
|
|
|
* Format is rgb from right to left (mirrorred image),
|
2005-04-17 06:20:36 +08:00
|
|
|
* we flip it to get bgr from left to right. */
|
2009-06-09 21:02:11 +08:00
|
|
|
if (frame->curlinepix < 3)
|
|
|
|
*(frame->curline-frame->curlinepix) = 1 + data * 4;
|
|
|
|
else
|
|
|
|
*(frame->curline-frame->curlinepix) =
|
|
|
|
*(frame->curline-frame->curlinepix + 3) + data * 4;
|
2005-04-17 06:20:36 +08:00
|
|
|
frame->curlinepix++;
|
|
|
|
}
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
static inline void decode_JangGu_vlc(struct usb_se401 *se401,
|
|
|
|
unsigned char *data, int bit_exp, int packetlength)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
int pos = 0;
|
|
|
|
int vlc_cod = 0;
|
|
|
|
int vlc_size = 0;
|
|
|
|
int vlc_data = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
int bit_cur;
|
|
|
|
int bit;
|
2009-06-09 21:02:11 +08:00
|
|
|
data += 4;
|
2005-04-17 06:20:36 +08:00
|
|
|
while (pos < packetlength) {
|
2009-06-09 21:02:11 +08:00
|
|
|
bit_cur = 8;
|
2005-04-17 06:20:36 +08:00
|
|
|
while (bit_cur && bit_exp) {
|
2009-06-09 21:02:11 +08:00
|
|
|
bit = ((*data) >> (bit_cur-1))&1;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!vlc_cod) {
|
|
|
|
if (bit) {
|
|
|
|
vlc_size++;
|
|
|
|
} else {
|
2009-06-09 21:02:11 +08:00
|
|
|
if (!vlc_size)
|
2005-04-17 06:20:36 +08:00
|
|
|
decode_JangGu_integrate(se401, 0);
|
2009-06-09 21:02:11 +08:00
|
|
|
else {
|
|
|
|
vlc_cod = 2;
|
|
|
|
vlc_data = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2009-06-09 21:02:11 +08:00
|
|
|
if (vlc_cod == 2) {
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!bit)
|
2009-06-09 21:02:11 +08:00
|
|
|
vlc_data = -(1 << vlc_size) + 1;
|
2005-04-17 06:20:36 +08:00
|
|
|
vlc_cod--;
|
|
|
|
}
|
|
|
|
vlc_size--;
|
2009-06-09 21:02:11 +08:00
|
|
|
vlc_data += bit << vlc_size;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!vlc_size) {
|
|
|
|
decode_JangGu_integrate(se401, vlc_data);
|
2009-06-09 21:02:11 +08:00
|
|
|
vlc_cod = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
bit_cur--;
|
|
|
|
bit_exp--;
|
|
|
|
}
|
|
|
|
pos++;
|
|
|
|
data++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
static inline void decode_JangGu(struct usb_se401 *se401,
|
|
|
|
struct se401_scratch *buffer)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
unsigned char *data = buffer->data;
|
|
|
|
int len = buffer->length;
|
|
|
|
int bit_exp = 0, pix_exp = 0, frameinfo = 0, packetlength = 0, size;
|
|
|
|
int datapos = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* New image? */
|
|
|
|
if (!se401->frame[se401->curframe].curpix) {
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->frame[se401->curframe].curlinepix = 0;
|
|
|
|
se401->frame[se401->curframe].curline =
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->frame[se401->curframe].data+
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->cwidth * 3 - 1;
|
|
|
|
if (se401->frame[se401->curframe].grabstate == FRAME_READY)
|
|
|
|
se401->frame[se401->curframe].grabstate = FRAME_GRABBING;
|
|
|
|
se401->vlcdatapos = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
while (datapos < len) {
|
2009-06-09 21:02:11 +08:00
|
|
|
size = 1024 - se401->vlcdatapos;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (size+datapos > len)
|
2009-06-09 21:02:11 +08:00
|
|
|
size = len-datapos;
|
2005-04-17 06:20:36 +08:00
|
|
|
memcpy(se401->vlcdata+se401->vlcdatapos, data+datapos, size);
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->vlcdatapos += size;
|
|
|
|
packetlength = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (se401->vlcdatapos >= 4) {
|
2009-06-09 21:02:11 +08:00
|
|
|
bit_exp = se401->vlcdata[3] + (se401->vlcdata[2] << 8);
|
|
|
|
pix_exp = se401->vlcdata[1] +
|
|
|
|
((se401->vlcdata[0] & 0x3f) << 8);
|
|
|
|
frameinfo = se401->vlcdata[0] & 0xc0;
|
|
|
|
packetlength = ((bit_exp + 47) >> 4) << 1;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (packetlength > 1024) {
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->vlcdatapos = 0;
|
|
|
|
datapos = len;
|
|
|
|
packetlength = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->error++;
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->frame[se401->curframe].curpix = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (packetlength && se401->vlcdatapos >= packetlength) {
|
2009-06-09 21:02:11 +08:00
|
|
|
decode_JangGu_vlc(se401, se401->vlcdata, bit_exp,
|
|
|
|
packetlength);
|
|
|
|
se401->frame[se401->curframe].curpix += pix_exp * 3;
|
|
|
|
datapos += size-(se401->vlcdatapos-packetlength);
|
|
|
|
se401->vlcdatapos = 0;
|
|
|
|
if (se401->frame[se401->curframe].curpix >= se401->cwidth * se401->cheight * 3) {
|
|
|
|
if (se401->frame[se401->curframe].curpix == se401->cwidth * se401->cheight * 3) {
|
|
|
|
if (se401->frame[se401->curframe].grabstate == FRAME_GRABBING) {
|
|
|
|
se401->frame[se401->curframe].grabstate = FRAME_DONE;
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->framecount++;
|
|
|
|
se401->readcount++;
|
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->frame[(se401->curframe + 1) & (SE401_NUMFRAMES - 1)].grabstate == FRAME_READY)
|
|
|
|
se401->curframe = (se401->curframe + 1) & (SE401_NUMFRAMES - 1);
|
|
|
|
} else
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->error++;
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->frame[se401->curframe].curpix = 0;
|
|
|
|
datapos = len;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
} else
|
|
|
|
datapos += size;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
static inline void decode_bayer(struct usb_se401 *se401,
|
|
|
|
struct se401_scratch *buffer)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
unsigned char *data = buffer->data;
|
|
|
|
int len = buffer->length;
|
|
|
|
int offset = buffer->offset;
|
|
|
|
int datasize = se401->cwidth * se401->cheight;
|
|
|
|
struct se401_frame *frame = &se401->frame[se401->curframe];
|
|
|
|
unsigned char *framedata = frame->data, *curline, *nextline;
|
|
|
|
int width = se401->cwidth;
|
|
|
|
int blineoffset = 0, bline;
|
|
|
|
int linelength = width * 3, i;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
if (frame->curpix == 0) {
|
|
|
|
if (frame->grabstate == FRAME_READY)
|
|
|
|
frame->grabstate = FRAME_GRABBING;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
frame->curline = framedata + linelength;
|
|
|
|
frame->curlinepix = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
if (offset != frame->curpix) {
|
2005-04-17 06:20:36 +08:00
|
|
|
/* Regard frame as lost :( */
|
2009-06-09 21:02:11 +08:00
|
|
|
frame->curpix = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->error++;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if we have to much data */
|
2009-06-09 21:02:11 +08:00
|
|
|
if (frame->curpix + len > datasize)
|
|
|
|
len = datasize-frame->curpix;
|
|
|
|
|
|
|
|
if (se401->cheight % 4)
|
|
|
|
blineoffset = 1;
|
|
|
|
bline = frame->curpix / se401->cwidth+blineoffset;
|
|
|
|
|
|
|
|
curline = frame->curline;
|
|
|
|
nextline = curline + linelength;
|
|
|
|
if (nextline >= framedata+datasize * 3)
|
|
|
|
nextline = curline;
|
2005-04-17 06:20:36 +08:00
|
|
|
while (len) {
|
2009-06-09 21:02:11 +08:00
|
|
|
if (frame->curlinepix >= width) {
|
|
|
|
frame->curlinepix -= width;
|
|
|
|
bline = frame->curpix / width + blineoffset;
|
|
|
|
curline += linelength*2;
|
|
|
|
nextline += linelength*2;
|
|
|
|
if (curline >= framedata+datasize * 3) {
|
2005-04-17 06:20:36 +08:00
|
|
|
frame->curlinepix++;
|
2009-06-09 21:02:11 +08:00
|
|
|
curline -= 3;
|
|
|
|
nextline -= 3;
|
2005-04-17 06:20:36 +08:00
|
|
|
len--;
|
|
|
|
data++;
|
|
|
|
frame->curpix++;
|
|
|
|
}
|
|
|
|
if (nextline >= framedata+datasize*3)
|
2009-06-09 21:02:11 +08:00
|
|
|
nextline = curline;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
if (bline & 1) {
|
|
|
|
if (frame->curlinepix & 1) {
|
|
|
|
*(curline + 2) = *data;
|
|
|
|
*(curline - 1) = *data;
|
|
|
|
*(nextline + 2) = *data;
|
|
|
|
*(nextline - 1) = *data;
|
2005-04-17 06:20:36 +08:00
|
|
|
} else {
|
2009-06-09 21:02:11 +08:00
|
|
|
*(curline + 1) =
|
|
|
|
(*(curline + 1) + *data) / 2;
|
|
|
|
*(curline-2) =
|
|
|
|
(*(curline - 2) + *data) / 2;
|
|
|
|
*(nextline + 1) = *data;
|
|
|
|
*(nextline - 2) = *data;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
} else {
|
2009-06-09 21:02:11 +08:00
|
|
|
if (frame->curlinepix & 1) {
|
|
|
|
*(curline + 1) =
|
|
|
|
(*(curline + 1) + *data) / 2;
|
|
|
|
*(curline - 2) =
|
|
|
|
(*(curline - 2) + *data) / 2;
|
|
|
|
*(nextline + 1) = *data;
|
|
|
|
*(nextline - 2) = *data;
|
2005-04-17 06:20:36 +08:00
|
|
|
} else {
|
2009-06-09 21:02:11 +08:00
|
|
|
*curline = *data;
|
|
|
|
*(curline - 3) = *data;
|
|
|
|
*nextline = *data;
|
|
|
|
*(nextline - 3) = *data;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
frame->curlinepix++;
|
2009-06-09 21:02:11 +08:00
|
|
|
curline -= 3;
|
|
|
|
nextline -= 3;
|
2005-04-17 06:20:36 +08:00
|
|
|
len--;
|
|
|
|
data++;
|
|
|
|
frame->curpix++;
|
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
frame->curline = curline;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
if (frame->curpix >= datasize) {
|
2005-04-17 06:20:36 +08:00
|
|
|
/* Fix the top line */
|
2009-06-09 21:02:11 +08:00
|
|
|
framedata += linelength;
|
|
|
|
for (i = 0; i < linelength; i++) {
|
2005-04-17 06:20:36 +08:00
|
|
|
framedata--;
|
2009-06-09 21:02:11 +08:00
|
|
|
*framedata = *(framedata + linelength);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
/* Fix the left side (green is already present) */
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < se401->cheight; i++) {
|
|
|
|
*framedata = *(framedata + 3);
|
|
|
|
*(framedata + 1) = *(framedata + 4);
|
|
|
|
*(framedata + 2) = *(framedata + 5);
|
|
|
|
framedata += linelength;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
frame->curpix = 0;
|
|
|
|
frame->grabstate = FRAME_DONE;
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->framecount++;
|
|
|
|
se401->readcount++;
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->frame[(se401->curframe + 1) &
|
|
|
|
(SE401_NUMFRAMES - 1)].grabstate == FRAME_READY) {
|
|
|
|
se401->curframe = (se401->curframe+1) &
|
|
|
|
(SE401_NUMFRAMES-1);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int se401_newframe(struct usb_se401 *se401, int framenr)
|
|
|
|
{
|
|
|
|
DECLARE_WAITQUEUE(wait, current);
|
2009-06-09 21:02:11 +08:00
|
|
|
int errors = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
while (se401->streaming &&
|
2009-06-09 21:02:11 +08:00
|
|
|
(se401->frame[framenr].grabstate == FRAME_READY ||
|
|
|
|
se401->frame[framenr].grabstate == FRAME_GRABBING)) {
|
|
|
|
if (!se401->frame[framenr].curpix)
|
2005-04-17 06:20:36 +08:00
|
|
|
errors++;
|
2009-06-09 21:02:11 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
wait_interruptible(
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->scratch[se401->scratch_use].state != BUFFER_READY,
|
|
|
|
&se401->wq, &wait);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (se401->nullpackets > SE401_MAX_NULLPACKETS) {
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->nullpackets = 0;
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&se401->dev->dev,
|
2009-06-09 21:02:11 +08:00
|
|
|
"too many null length packets, restarting capture\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_stop_stream(se401);
|
2006-03-25 20:19:53 +08:00
|
|
|
se401_start_stream(se401);
|
2005-04-17 06:20:36 +08:00
|
|
|
} else {
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->scratch[se401->scratch_use].state !=
|
|
|
|
BUFFER_READY) {
|
|
|
|
se401->frame[framenr].grabstate = FRAME_ERROR;
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EIO;
|
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->scratch[se401->scratch_use].state = BUFFER_BUSY;
|
|
|
|
if (se401->format == FMT_JANGGU)
|
|
|
|
decode_JangGu(se401,
|
|
|
|
&se401->scratch[se401->scratch_use]);
|
|
|
|
else
|
|
|
|
decode_bayer(se401,
|
|
|
|
&se401->scratch[se401->scratch_use]);
|
|
|
|
|
|
|
|
se401->scratch[se401->scratch_use].state =
|
|
|
|
BUFFER_UNUSED;
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->scratch_use++;
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->scratch_use >= SE401_NUMSCRATCH)
|
|
|
|
se401->scratch_use = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (errors > SE401_MAX_ERRORS) {
|
2009-06-09 21:02:11 +08:00
|
|
|
errors = 0;
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&se401->dev->dev,
|
2009-06-09 21:02:11 +08:00
|
|
|
"too many errors, restarting capture\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_stop_stream(se401);
|
|
|
|
se401_start_stream(se401);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->frame[framenr].grabstate == FRAME_DONE)
|
2005-04-17 06:20:36 +08:00
|
|
|
if (se401->enhance)
|
2009-06-09 21:02:11 +08:00
|
|
|
enhance_picture(se401->frame[framenr].data,
|
|
|
|
se401->cheight * se401->cwidth * 3);
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
static void usb_se401_remove_disconnected(struct usb_se401 *se401)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
se401->dev = NULL;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < SE401_NUMSBUF; i++)
|
2005-04-19 08:39:34 +08:00
|
|
|
if (se401->urb[i]) {
|
|
|
|
usb_kill_urb(se401->urb[i]);
|
|
|
|
usb_free_urb(se401->urb[i]);
|
|
|
|
se401->urb[i] = NULL;
|
|
|
|
kfree(se401->sbuf[i].data);
|
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
|
|
|
|
for (i = 0; i < SE401_NUMSCRATCH; i++)
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree(se401->scratch[i].data);
|
2009-06-09 21:02:11 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
if (se401->inturb) {
|
|
|
|
usb_kill_urb(se401->inturb);
|
|
|
|
usb_free_urb(se401->inturb);
|
|
|
|
}
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&se401->dev->dev, "%s disconnected", se401->camera_name);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
/* Free the memory */
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree(se401->width);
|
|
|
|
kfree(se401->height);
|
|
|
|
kfree(se401);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
*
|
|
|
|
* Video4Linux
|
|
|
|
*
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
2008-12-30 17:58:20 +08:00
|
|
|
static int se401_open(struct file *file)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct video_device *dev = video_devdata(file);
|
|
|
|
struct usb_se401 *se401 = (struct usb_se401 *)dev;
|
|
|
|
int err = 0;
|
|
|
|
|
2010-10-27 20:30:32 +08:00
|
|
|
mutex_lock(&se401->lock);
|
2008-07-30 19:43:36 +08:00
|
|
|
if (se401->user) {
|
2010-10-27 20:30:32 +08:00
|
|
|
mutex_unlock(&se401->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EBUSY;
|
2008-07-30 19:43:36 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->fbuf = rvmalloc(se401->maxframesize * SE401_NUMFRAMES);
|
|
|
|
if (se401->fbuf)
|
|
|
|
file->private_data = dev;
|
2006-03-25 20:19:53 +08:00
|
|
|
else
|
2005-04-17 06:20:36 +08:00
|
|
|
err = -ENOMEM;
|
|
|
|
se401->user = !err;
|
2010-10-27 20:30:32 +08:00
|
|
|
mutex_unlock(&se401->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2008-12-30 17:58:20 +08:00
|
|
|
static int se401_close(struct file *file)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct video_device *dev = file->private_data;
|
2006-03-25 20:19:53 +08:00
|
|
|
struct usb_se401 *se401 = (struct usb_se401 *)dev;
|
2005-04-17 06:20:36 +08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
rvfree(se401->fbuf, se401->maxframesize * SE401_NUMFRAMES);
|
2006-03-25 20:19:53 +08:00
|
|
|
if (se401->removed) {
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&se401->dev->dev, "device unregistered\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
usb_se401_remove_disconnected(se401);
|
|
|
|
} else {
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < SE401_NUMFRAMES; i++)
|
|
|
|
se401->frame[i].grabstate = FRAME_UNUSED;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (se401->streaming)
|
|
|
|
se401_stop_stream(se401);
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->user = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
file->private_data = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-12-30 18:04:34 +08:00
|
|
|
static long se401_do_ioctl(struct file *file, unsigned int cmd, void *arg)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct video_device *vdev = file->private_data;
|
2006-03-25 20:19:53 +08:00
|
|
|
struct usb_se401 *se401 = (struct usb_se401 *)vdev;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
if (!se401->dev)
|
|
|
|
return -EIO;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
switch (cmd) {
|
2005-04-17 06:20:36 +08:00
|
|
|
case VIDIOCGCAP:
|
|
|
|
{
|
|
|
|
struct video_capability *b = arg;
|
|
|
|
strcpy(b->name, se401->camera_name);
|
|
|
|
b->type = VID_TYPE_CAPTURE;
|
|
|
|
b->channels = 1;
|
|
|
|
b->audios = 0;
|
|
|
|
b->maxwidth = se401->width[se401->sizes-1];
|
|
|
|
b->maxheight = se401->height[se401->sizes-1];
|
|
|
|
b->minwidth = se401->width[0];
|
|
|
|
b->minheight = se401->height[0];
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
case VIDIOCGCHAN:
|
|
|
|
{
|
|
|
|
struct video_channel *v = arg;
|
|
|
|
|
|
|
|
if (v->channel != 0)
|
|
|
|
return -EINVAL;
|
|
|
|
v->flags = 0;
|
|
|
|
v->tuners = 0;
|
|
|
|
v->type = VIDEO_TYPE_CAMERA;
|
|
|
|
strcpy(v->name, "Camera");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
case VIDIOCSCHAN:
|
|
|
|
{
|
|
|
|
struct video_channel *v = arg;
|
|
|
|
|
|
|
|
if (v->channel != 0)
|
|
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
|
|
}
|
2006-03-25 20:19:53 +08:00
|
|
|
case VIDIOCGPICT:
|
|
|
|
{
|
2005-04-17 06:20:36 +08:00
|
|
|
struct video_picture *p = arg;
|
|
|
|
|
|
|
|
se401_get_pict(se401, p);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
case VIDIOCSPICT:
|
|
|
|
{
|
|
|
|
struct video_picture *p = arg;
|
|
|
|
|
|
|
|
if (se401_set_pict(se401, p))
|
|
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
case VIDIOCSWIN:
|
|
|
|
{
|
|
|
|
struct video_window *vw = arg;
|
|
|
|
|
|
|
|
if (vw->flags)
|
|
|
|
return -EINVAL;
|
|
|
|
if (vw->clipcount)
|
|
|
|
return -EINVAL;
|
|
|
|
if (se401_set_size(se401, vw->width, vw->height))
|
|
|
|
return -EINVAL;
|
|
|
|
return 0;
|
2006-03-25 20:19:53 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
case VIDIOCGWIN:
|
|
|
|
{
|
|
|
|
struct video_window *vw = arg;
|
|
|
|
|
|
|
|
vw->x = 0; /* FIXME */
|
|
|
|
vw->y = 0;
|
|
|
|
vw->chromakey = 0;
|
|
|
|
vw->flags = 0;
|
|
|
|
vw->clipcount = 0;
|
|
|
|
vw->width = se401->cwidth;
|
|
|
|
vw->height = se401->cheight;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
case VIDIOCGMBUF:
|
|
|
|
{
|
|
|
|
struct video_mbuf *vm = arg;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
memset(vm, 0, sizeof(*vm));
|
|
|
|
vm->size = SE401_NUMFRAMES * se401->maxframesize;
|
|
|
|
vm->frames = SE401_NUMFRAMES;
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < SE401_NUMFRAMES; i++)
|
2005-04-17 06:20:36 +08:00
|
|
|
vm->offsets[i] = se401->maxframesize * i;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
case VIDIOCMCAPTURE:
|
|
|
|
{
|
|
|
|
struct video_mmap *vm = arg;
|
|
|
|
|
|
|
|
if (vm->format != VIDEO_PALETTE_RGB24)
|
|
|
|
return -EINVAL;
|
|
|
|
if (vm->frame >= SE401_NUMFRAMES)
|
|
|
|
return -EINVAL;
|
|
|
|
if (se401->frame[vm->frame].grabstate != FRAME_UNUSED)
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
/* Is this according to the v4l spec??? */
|
|
|
|
if (se401_set_size(se401, vm->width, vm->height))
|
|
|
|
return -EINVAL;
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->frame[vm->frame].grabstate = FRAME_READY;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
if (!se401->streaming)
|
|
|
|
se401_start_stream(se401);
|
|
|
|
|
|
|
|
/* Set the picture properties */
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->framecount == 0)
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_send_pict(se401);
|
|
|
|
/* Calibrate the reset level after a few frames. */
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->framecount % 20 == 1)
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_auto_resetlevel(se401);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
case VIDIOCSYNC:
|
|
|
|
{
|
|
|
|
int *frame = arg;
|
2009-06-09 21:02:11 +08:00
|
|
|
int ret = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
if (*frame < 0 || *frame >= SE401_NUMFRAMES)
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EINVAL;
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
ret = se401_newframe(se401, *frame);
|
|
|
|
se401->frame[*frame].grabstate = FRAME_UNUSED;
|
2005-04-17 06:20:36 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
case VIDIOCGFBUF:
|
|
|
|
{
|
|
|
|
struct video_buffer *vb = arg;
|
|
|
|
|
|
|
|
memset(vb, 0, sizeof(*vb));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
case VIDIOCKEY:
|
|
|
|
return 0;
|
|
|
|
case VIDIOCCAPTURE:
|
|
|
|
return -EINVAL;
|
|
|
|
case VIDIOCSFBUF:
|
|
|
|
return -EINVAL;
|
|
|
|
case VIDIOCGTUNER:
|
|
|
|
case VIDIOCSTUNER:
|
|
|
|
return -EINVAL;
|
|
|
|
case VIDIOCGFREQ:
|
|
|
|
case VIDIOCSFREQ:
|
|
|
|
return -EINVAL;
|
|
|
|
case VIDIOCGAUDIO:
|
|
|
|
case VIDIOCSAUDIO:
|
|
|
|
return -EINVAL;
|
2006-03-25 20:19:53 +08:00
|
|
|
default:
|
|
|
|
return -ENOIOCTLCMD;
|
|
|
|
} /* end switch */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
return 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2008-12-30 18:04:34 +08:00
|
|
|
static long se401_ioctl(struct file *file,
|
2005-04-17 06:20:36 +08:00
|
|
|
unsigned int cmd, unsigned long arg)
|
|
|
|
{
|
2008-11-01 19:25:11 +08:00
|
|
|
return video_usercopy(file, cmd, arg, se401_do_ioctl);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t se401_read(struct file *file, char __user *buf,
|
|
|
|
size_t count, loff_t *ppos)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
int realcount = count, ret = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
struct video_device *dev = file->private_data;
|
|
|
|
struct usb_se401 *se401 = (struct usb_se401 *)dev;
|
|
|
|
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->dev == NULL)
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EIO;
|
|
|
|
if (realcount > se401->cwidth*se401->cheight*3)
|
2009-06-09 21:02:11 +08:00
|
|
|
realcount = se401->cwidth*se401->cheight*3;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Shouldn't happen: */
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->frame[0].grabstate == FRAME_GRABBING)
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EBUSY;
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->frame[0].grabstate = FRAME_READY;
|
|
|
|
se401->frame[1].grabstate = FRAME_UNUSED;
|
|
|
|
se401->curframe = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
if (!se401->streaming)
|
|
|
|
se401_start_stream(se401);
|
|
|
|
|
|
|
|
/* Set the picture properties */
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->framecount == 0)
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_send_pict(se401);
|
|
|
|
/* Calibrate the reset level after a few frames. */
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->framecount%20 == 1)
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_auto_resetlevel(se401);
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
ret = se401_newframe(se401, 0);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->frame[0].grabstate = FRAME_UNUSED;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (ret)
|
2006-03-25 20:19:53 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (copy_to_user(buf, se401->frame[0].data, realcount))
|
|
|
|
return -EFAULT;
|
|
|
|
|
|
|
|
return realcount;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int se401_mmap(struct file *file, struct vm_area_struct *vma)
|
|
|
|
{
|
|
|
|
struct video_device *dev = file->private_data;
|
|
|
|
struct usb_se401 *se401 = (struct usb_se401 *)dev;
|
|
|
|
unsigned long start = vma->vm_start;
|
|
|
|
unsigned long size = vma->vm_end-vma->vm_start;
|
|
|
|
unsigned long page, pos;
|
|
|
|
|
2006-01-11 22:55:29 +08:00
|
|
|
mutex_lock(&se401->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
if (se401->dev == NULL) {
|
2006-01-11 22:55:29 +08:00
|
|
|
mutex_unlock(&se401->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EIO;
|
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
if (size > (((SE401_NUMFRAMES * se401->maxframesize) + PAGE_SIZE - 1)
|
|
|
|
& ~(PAGE_SIZE - 1))) {
|
2006-01-11 22:55:29 +08:00
|
|
|
mutex_unlock(&se401->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
pos = (unsigned long)se401->fbuf;
|
|
|
|
while (size > 0) {
|
|
|
|
page = vmalloc_to_pfn((void *)pos);
|
|
|
|
if (remap_pfn_range(vma, start, page, PAGE_SIZE, PAGE_SHARED)) {
|
2006-01-11 22:55:29 +08:00
|
|
|
mutex_unlock(&se401->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EAGAIN;
|
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
start += PAGE_SIZE;
|
|
|
|
pos += PAGE_SIZE;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (size > PAGE_SIZE)
|
2009-06-09 21:02:11 +08:00
|
|
|
size -= PAGE_SIZE;
|
2005-04-17 06:20:36 +08:00
|
|
|
else
|
|
|
|
size = 0;
|
|
|
|
}
|
2006-01-11 22:55:29 +08:00
|
|
|
mutex_unlock(&se401->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
return 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2008-12-30 17:58:20 +08:00
|
|
|
static const struct v4l2_file_operations se401_fops = {
|
2009-06-09 21:02:11 +08:00
|
|
|
.owner = THIS_MODULE,
|
2006-03-25 20:19:53 +08:00
|
|
|
.open = se401_open,
|
|
|
|
.release = se401_close,
|
|
|
|
.read = se401_read,
|
|
|
|
.mmap = se401_mmap,
|
2005-04-17 06:20:36 +08:00
|
|
|
.ioctl = se401_ioctl,
|
|
|
|
};
|
|
|
|
static struct video_device se401_template = {
|
2006-03-25 20:19:53 +08:00
|
|
|
.name = "se401 USB camera",
|
2005-04-17 06:20:36 +08:00
|
|
|
.fops = &se401_fops,
|
2008-08-23 17:23:55 +08:00
|
|
|
.release = video_device_release_empty,
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***************************/
|
|
|
|
static int se401_init(struct usb_se401 *se401, int button)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
int i = 0, rc;
|
2006-03-25 20:19:53 +08:00
|
|
|
unsigned char cp[0x40];
|
2005-04-17 06:20:36 +08:00
|
|
|
char temp[200];
|
2009-06-11 22:04:11 +08:00
|
|
|
int slen;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* led on */
|
2006-03-25 20:19:53 +08:00
|
|
|
se401_sndctrl(1, se401, SE401_REQ_LED_CONTROL, 1, NULL, 0);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* get camera descriptor */
|
2009-06-09 21:02:11 +08:00
|
|
|
rc = se401_sndctrl(0, se401, SE401_REQ_GET_CAMERA_DESCRIPTOR, 0,
|
|
|
|
cp, sizeof(cp));
|
2009-06-11 22:04:11 +08:00
|
|
|
if (cp[1] != 0x41) {
|
2005-04-17 06:20:36 +08:00
|
|
|
err("Wrong descriptor type");
|
|
|
|
return 1;
|
|
|
|
}
|
2009-06-11 22:04:11 +08:00
|
|
|
slen = snprintf(temp, 200, "ExtraFeatures: %d", cp[3]);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->sizes = cp[4] + cp[5] * 256;
|
|
|
|
se401->width = kmalloc(se401->sizes*sizeof(int), GFP_KERNEL);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!se401->width)
|
|
|
|
return 1;
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->height = kmalloc(se401->sizes*sizeof(int), GFP_KERNEL);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!se401->height) {
|
|
|
|
kfree(se401->width);
|
|
|
|
return 1;
|
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
for (i = 0; i < se401->sizes; i++) {
|
|
|
|
se401->width[i] = cp[6 + i * 4 + 0] + cp[6 + i*4 + 1] * 256;
|
|
|
|
se401->height[i] = cp[6 + i * 4 + 2] + cp[6 + i * 4 + 3] * 256;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2009-06-09 21:02:11 +08:00
|
|
|
slen += snprintf(temp + slen, 200 - slen, " Sizes:");
|
|
|
|
for (i = 0; i < se401->sizes; i++) {
|
|
|
|
slen += snprintf(temp + slen, 200 - slen,
|
2009-06-11 22:04:11 +08:00
|
|
|
" %dx%d", se401->width[i], se401->height[i]);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&se401->dev->dev, "%s\n", temp);
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->maxframesize = se401->width[se401->sizes-1] *
|
|
|
|
se401->height[se401->sizes - 1] * 3;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
rc = se401_sndctrl(0, se401, SE401_REQ_GET_WIDTH, 0, cp, sizeof(cp));
|
|
|
|
se401->cwidth = cp[0]+cp[1]*256;
|
|
|
|
rc = se401_sndctrl(0, se401, SE401_REQ_GET_HEIGHT, 0, cp, sizeof(cp));
|
|
|
|
se401->cheight = cp[0]+cp[1]*256;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2008-04-23 01:46:02 +08:00
|
|
|
if (!(cp[2] & SE401_FORMAT_BAYER)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
err("Bayer format not supported!");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
/* set output mode (BAYER) */
|
2009-06-09 21:02:11 +08:00
|
|
|
se401_sndctrl(1, se401, SE401_REQ_SET_OUTPUT_MODE,
|
|
|
|
SE401_FORMAT_BAYER, NULL, 0);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
rc = se401_sndctrl(0, se401, SE401_REQ_GET_BRT, 0, cp, sizeof(cp));
|
|
|
|
se401->brightness = cp[0]+cp[1]*256;
|
2005-04-17 06:20:36 +08:00
|
|
|
/* some default values */
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->resetlevel = 0x2d;
|
|
|
|
se401->rgain = 0x20;
|
|
|
|
se401->ggain = 0x20;
|
|
|
|
se401->bgain = 0x20;
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_set_exposure(se401, 20000);
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->palette = VIDEO_PALETTE_RGB24;
|
|
|
|
se401->enhance = 1;
|
|
|
|
se401->dropped = 0;
|
|
|
|
se401->error = 0;
|
|
|
|
se401->framecount = 0;
|
|
|
|
se401->readcount = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Start interrupt transfers for snapshot button */
|
|
|
|
if (button) {
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->inturb = usb_alloc_urb(0, GFP_KERNEL);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!se401->inturb) {
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&se401->dev->dev,
|
|
|
|
"Allocation of inturb failed\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
usb_fill_int_urb(se401->inturb, se401->dev,
|
|
|
|
usb_rcvintpipe(se401->dev, SE401_BUTTON_ENDPOINT),
|
|
|
|
&se401->button, sizeof(se401->button),
|
|
|
|
se401_button_irq,
|
|
|
|
se401,
|
|
|
|
8
|
|
|
|
);
|
|
|
|
if (usb_submit_urb(se401->inturb, GFP_KERNEL)) {
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&se401->dev->dev, "int urb burned down\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
} else
|
2009-06-09 21:02:11 +08:00
|
|
|
se401->inturb = NULL;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
/* Flash the led */
|
|
|
|
se401_sndctrl(1, se401, SE401_REQ_CAMERA_POWER, 1, NULL, 0);
|
|
|
|
se401_sndctrl(1, se401, SE401_REQ_LED_CONTROL, 1, NULL, 0);
|
|
|
|
se401_sndctrl(1, se401, SE401_REQ_CAMERA_POWER, 0, NULL, 0);
|
2005-04-17 06:20:36 +08:00
|
|
|
se401_sndctrl(1, se401, SE401_REQ_LED_CONTROL, 0, NULL, 0);
|
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
return 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int se401_probe(struct usb_interface *intf,
|
|
|
|
const struct usb_device_id *id)
|
|
|
|
{
|
|
|
|
struct usb_device *dev = interface_to_usbdev(intf);
|
2006-03-25 20:19:53 +08:00
|
|
|
struct usb_interface_descriptor *interface;
|
|
|
|
struct usb_se401 *se401;
|
2009-06-09 21:02:11 +08:00
|
|
|
char *camera_name = NULL;
|
|
|
|
int button = 1;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
/* We don't handle multi-config cameras */
|
|
|
|
if (dev->descriptor.bNumConfigurations != 1)
|
|
|
|
return -ENODEV;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
interface = &intf->cur_altsetting->desc;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
/* Is it an se401? */
|
2009-06-09 21:02:11 +08:00
|
|
|
if (le16_to_cpu(dev->descriptor.idVendor) == 0x03e8 &&
|
|
|
|
le16_to_cpu(dev->descriptor.idProduct) == 0x0004) {
|
|
|
|
camera_name = "Endpoints/Aox SE401";
|
|
|
|
} else if (le16_to_cpu(dev->descriptor.idVendor) == 0x0471 &&
|
|
|
|
le16_to_cpu(dev->descriptor.idProduct) == 0x030b) {
|
|
|
|
camera_name = "Philips PCVC665K";
|
|
|
|
} else if (le16_to_cpu(dev->descriptor.idVendor) == 0x047d &&
|
|
|
|
le16_to_cpu(dev->descriptor.idProduct) == 0x5001) {
|
|
|
|
camera_name = "Kensington VideoCAM 67014";
|
|
|
|
} else if (le16_to_cpu(dev->descriptor.idVendor) == 0x047d &&
|
|
|
|
le16_to_cpu(dev->descriptor.idProduct) == 0x5002) {
|
|
|
|
camera_name = "Kensington VideoCAM 6701(5/7)";
|
|
|
|
} else if (le16_to_cpu(dev->descriptor.idVendor) == 0x047d &&
|
|
|
|
le16_to_cpu(dev->descriptor.idProduct) == 0x5003) {
|
|
|
|
camera_name = "Kensington VideoCAM 67016";
|
|
|
|
button = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
} else
|
|
|
|
return -ENODEV;
|
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
/* Checking vendor/product should be enough, but what the hell */
|
|
|
|
if (interface->bInterfaceClass != 0x00)
|
2005-04-17 06:20:36 +08:00
|
|
|
return -ENODEV;
|
2006-03-25 20:19:53 +08:00
|
|
|
if (interface->bInterfaceSubClass != 0x00)
|
2005-04-17 06:20:36 +08:00
|
|
|
return -ENODEV;
|
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
/* We found one */
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&intf->dev, "SE401 camera found: %s\n", camera_name);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
se401 = kzalloc(sizeof(*se401), GFP_KERNEL);
|
|
|
|
if (se401 == NULL) {
|
2006-03-25 20:19:53 +08:00
|
|
|
err("couldn't kmalloc se401 struct");
|
2005-04-17 06:20:36 +08:00
|
|
|
return -ENOMEM;
|
2006-03-25 20:19:53 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
se401->dev = dev;
|
|
|
|
se401->iface = interface->bInterfaceNumber;
|
|
|
|
se401->camera_name = camera_name;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2008-10-10 16:08:23 +08:00
|
|
|
dev_info(&intf->dev, "firmware version: %02x\n",
|
|
|
|
le16_to_cpu(dev->descriptor.bcdDevice) & 255);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-03-25 20:19:53 +08:00
|
|
|
if (se401_init(se401, button)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree(se401);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(&se401->vdev, &se401_template, sizeof(se401_template));
|
2009-06-09 21:02:11 +08:00
|
|
|
memcpy(se401->vdev.name, se401->camera_name,
|
|
|
|
strlen(se401->camera_name));
|
2005-04-17 06:20:36 +08:00
|
|
|
init_waitqueue_head(&se401->wq);
|
2006-01-11 22:55:29 +08:00
|
|
|
mutex_init(&se401->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
wmb();
|
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
if (video_register_device(&se401->vdev,
|
|
|
|
VFL_TYPE_GRABBER, video_nr) < 0) {
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree(se401);
|
|
|
|
err("video_register_device failed");
|
|
|
|
return -EIO;
|
|
|
|
}
|
2009-11-28 00:57:15 +08:00
|
|
|
dev_info(&intf->dev, "registered new video device: %s\n",
|
|
|
|
video_device_node_name(&se401->vdev));
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
usb_set_intfdata(intf, se401);
|
2006-03-25 20:19:53 +08:00
|
|
|
return 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void se401_disconnect(struct usb_interface *intf)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
struct usb_se401 *se401 = usb_get_intfdata(intf);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-09 21:02:11 +08:00
|
|
|
usb_set_intfdata(intf, NULL);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (se401) {
|
|
|
|
video_unregister_device(&se401->vdev);
|
2009-06-09 21:02:11 +08:00
|
|
|
if (!se401->user)
|
2005-04-17 06:20:36 +08:00
|
|
|
usb_se401_remove_disconnected(se401);
|
2009-06-09 21:02:11 +08:00
|
|
|
else {
|
2005-04-17 06:20:36 +08:00
|
|
|
se401->frame[0].grabstate = FRAME_ERROR;
|
|
|
|
se401->frame[0].grabstate = FRAME_ERROR;
|
|
|
|
|
|
|
|
se401->streaming = 0;
|
|
|
|
|
|
|
|
wake_up_interruptible(&se401->wq);
|
|
|
|
se401->removed = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct usb_driver se401_driver = {
|
2009-06-09 21:02:11 +08:00
|
|
|
.name = "se401",
|
|
|
|
.id_table = device_table,
|
|
|
|
.probe = se401_probe,
|
|
|
|
.disconnect = se401_disconnect,
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
*
|
|
|
|
* Module routines
|
|
|
|
*
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
static int __init usb_se401_init(void)
|
|
|
|
{
|
2009-06-09 21:02:11 +08:00
|
|
|
printk(KERN_INFO "SE401 usb camera driver version %s registering\n",
|
|
|
|
version);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (flickerless)
|
2009-06-09 21:02:11 +08:00
|
|
|
if (flickerless != 50 && flickerless != 60) {
|
2008-10-10 16:08:23 +08:00
|
|
|
printk(KERN_ERR "Invallid flickerless value, use 0, 50 or 60.\n");
|
2005-04-17 06:20:36 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return usb_register(&se401_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit usb_se401_exit(void)
|
|
|
|
{
|
|
|
|
usb_deregister(&se401_driver);
|
2008-10-10 16:08:23 +08:00
|
|
|
printk(KERN_INFO "SE401 driver deregistered\frame");
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
module_init(usb_se401_init);
|
|
|
|
module_exit(usb_se401_exit);
|