Commit 0bd5a41d authored by Gerd Knorr's avatar Gerd Knorr Committed by Linus Torvalds

[PATCH] v4l:

This patch adapts the se401 driver to the videodev changes and fixes the
unplug race.
parent e631dac6
...@@ -1000,29 +1000,30 @@ static int se401_newframe(struct usb_se401 *se401, int framenr) ...@@ -1000,29 +1000,30 @@ static int se401_newframe(struct usb_se401 *se401, int framenr)
***************************************************************************/ ***************************************************************************/
static int se401_open(struct video_device *dev, int flags) static int se401_open(struct inode *inode, struct file *file)
{ {
struct video_device *dev = video_devdata(file);
struct usb_se401 *se401 = (struct usb_se401 *)dev; struct usb_se401 *se401 = (struct usb_se401 *)dev;
int err = 0; int err = 0;
/* we are called with the BKL held */ if (se401->user)
MOD_INC_USE_COUNT; return -EBUSY;
se401->user=1; se401->user=1;
se401->fbuf=rvmalloc(se401->maxframesize * SE401_NUMFRAMES); se401->fbuf=rvmalloc(se401->maxframesize * SE401_NUMFRAMES);
if(!se401->fbuf) err=-ENOMEM; if(!se401->fbuf) err=-ENOMEM;
if (err) { if (0 != err) {
MOD_DEC_USE_COUNT;
se401->user = 0; se401->user = 0;
} else {
file->private_data = dev;
} }
return err; return err;
} }
static void se401_close(struct video_device *dev) static int se401_close(struct inode *inode, struct file *file)
{ {
/* called with BKL held */ struct video_device *dev = file->private_data;
struct usb_se401 *se401 = (struct usb_se401 *)dev; struct usb_se401 *se401 = (struct usb_se401 *)dev;
int i; int i;
...@@ -1035,34 +1036,20 @@ static void se401_close(struct video_device *dev) ...@@ -1035,34 +1036,20 @@ static void se401_close(struct video_device *dev)
se401->user=0; se401->user=0;
if (se401->removed) { if (se401->removed) {
video_unregister_device(&se401->vdev);
kfree(se401->width); kfree(se401->width);
kfree(se401->height); kfree(se401->height);
kfree(se401); kfree(se401);
se401 = NULL; se401 = NULL;
info("device unregistered"); info("device unregistered");
} }
file->private_data = NULL;
MOD_DEC_USE_COUNT; return 0;
}
static int se401_init_done(struct video_device *dev)
{
#if defined(CONFIG_PROC_FS) && defined(CONFIG_VIDEO_PROC_FS)
create_proc_se401_cam((struct usb_se401 *)dev);
#endif
return 0;
}
static long se401_write(struct video_device *dev, const char *buf, unsigned long
count, int noblock)
{
return -EINVAL;
} }
static int se401_ioctl(struct video_device *vdev, unsigned int cmd, void *arg) static int se401_ioctl(struct inode *inode, struct file *file,
unsigned int cmd, void *arg)
{ {
struct video_device *vdev = file->private_data;
struct usb_se401 *se401 = (struct usb_se401 *)vdev; struct usb_se401 *se401 = (struct usb_se401 *)vdev;
if (!se401->dev) if (!se401->dev)
...@@ -1071,138 +1058,104 @@ static int se401_ioctl(struct video_device *vdev, unsigned int cmd, void *arg) ...@@ -1071,138 +1058,104 @@ static int se401_ioctl(struct video_device *vdev, unsigned int cmd, void *arg)
switch (cmd) { switch (cmd) {
case VIDIOCGCAP: case VIDIOCGCAP:
{ {
struct video_capability b; struct video_capability *b = arg;
strcpy(b.name, se401->camera_name); strcpy(b->name, se401->camera_name);
b.type = VID_TYPE_CAPTURE; b->type = VID_TYPE_CAPTURE;
b.channels = 1; b->channels = 1;
b.audios = 0; b->audios = 0;
b.maxwidth = se401->width[se401->sizes-1]; b->maxwidth = se401->width[se401->sizes-1];
b.maxheight = se401->height[se401->sizes-1]; b->maxheight = se401->height[se401->sizes-1];
b.minwidth = se401->width[0]; b->minwidth = se401->width[0];
b.minheight = se401->height[0]; b->minheight = se401->height[0];
if (copy_to_user(arg, &b, sizeof(b)))
return -EFAULT;
return 0; return 0;
} }
case VIDIOCGCHAN: case VIDIOCGCHAN:
{ {
struct video_channel v; struct video_channel *v = arg;
if (copy_from_user(&v, arg, sizeof(v))) if (v->channel != 0)
return -EFAULT;
if (v.channel != 0)
return -EINVAL; return -EINVAL;
v->flags = 0;
v.flags = 0; v->tuners = 0;
v.tuners = 0; v->type = VIDEO_TYPE_CAMERA;
v.type = VIDEO_TYPE_CAMERA; strcpy(v->name, "Camera");
strcpy(v.name, "Camera");
if (copy_to_user(arg, &v, sizeof(v)))
return -EFAULT;
return 0; return 0;
} }
case VIDIOCSCHAN: case VIDIOCSCHAN:
{ {
int v; struct video_channel *v = arg;
if (copy_from_user(&v, arg, sizeof(v)))
return -EFAULT;
if (v != 0) if (v->channel != 0)
return -EINVAL; return -EINVAL;
return 0; return 0;
} }
case VIDIOCGPICT: case VIDIOCGPICT:
{ {
struct video_picture p; struct video_picture *p = arg;
se401_get_pict(se401, &p);
if (copy_to_user(arg, &p, sizeof(p))) se401_get_pict(se401, p);
return -EFAULT;
return 0; return 0;
} }
case VIDIOCSPICT: case VIDIOCSPICT:
{ {
struct video_picture p; struct video_picture *p = arg;
if (copy_from_user(&p, arg, sizeof(p))) if (se401_set_pict(se401, p))
return -EFAULT;
if (se401_set_pict(se401, &p))
return -EINVAL; return -EINVAL;
return 0; return 0;
} }
case VIDIOCSWIN: case VIDIOCSWIN:
{ {
struct video_window vw; struct video_window *vw = arg;
if (copy_from_user(&vw, arg, sizeof(vw))) if (vw->flags)
return -EFAULT;
if (vw.flags)
return -EINVAL; return -EINVAL;
if (vw.clipcount) if (vw->clipcount)
return -EINVAL; return -EINVAL;
if (se401_set_size(se401, vw.width, vw.height)) if (se401_set_size(se401, vw->width, vw->height))
return -EINVAL; return -EINVAL;
return 0; return 0;
} }
case VIDIOCGWIN: case VIDIOCGWIN:
{ {
struct video_window vw; struct video_window *vw = arg;
vw.x = 0; /* FIXME */ vw->x = 0; /* FIXME */
vw.y = 0; vw->y = 0;
vw.chromakey = 0; vw->chromakey = 0;
vw.flags = 0; vw->flags = 0;
vw.clipcount = 0; vw->clipcount = 0;
vw.width = se401->cwidth; vw->width = se401->cwidth;
vw.height = se401->cheight; vw->height = se401->cheight;
if (copy_to_user(arg, &vw, sizeof(vw)))
return -EFAULT;
return 0; return 0;
} }
case VIDIOCGMBUF: case VIDIOCGMBUF:
{ {
struct video_mbuf vm; struct video_mbuf *vm = arg;
int i; int i;
memset(&vm, 0, sizeof(vm)); memset(vm, 0, sizeof(*vm));
vm.size = SE401_NUMFRAMES * se401->maxframesize; vm->size = SE401_NUMFRAMES * se401->maxframesize;
vm.frames = SE401_NUMFRAMES; vm->frames = SE401_NUMFRAMES;
for (i=0; i<SE401_NUMFRAMES; i++) for (i=0; i<SE401_NUMFRAMES; i++)
vm.offsets[i] = se401->maxframesize * i; vm->offsets[i] = se401->maxframesize * i;
if (copy_to_user((void *)arg, (void *)&vm, sizeof(vm)))
return -EFAULT;
return 0; return 0;
} }
case VIDIOCMCAPTURE: case VIDIOCMCAPTURE:
{ {
struct video_mmap vm; struct video_mmap *vm = arg;
if (copy_from_user(&vm, arg, sizeof(vm))) if (vm->format != VIDEO_PALETTE_RGB24)
return -EFAULT;
if (vm.format != VIDEO_PALETTE_RGB24)
return -EINVAL; return -EINVAL;
if (vm.frame >= SE401_NUMFRAMES) if (vm->frame >= SE401_NUMFRAMES)
return -EINVAL; return -EINVAL;
if (se401->frame[vm.frame].grabstate != FRAME_UNUSED) if (se401->frame[vm->frame].grabstate != FRAME_UNUSED)
return -EBUSY; return -EBUSY;
/* Is this according to the v4l spec??? */ /* Is this according to the v4l spec??? */
if (se401_set_size(se401, vm.width, vm.height)) if (se401_set_size(se401, vm->width, vm->height))
return -EINVAL; return -EINVAL;
se401->frame[vm.frame].grabstate=FRAME_READY; se401->frame[vm->frame].grabstate=FRAME_READY;
if (!se401->streaming) if (!se401->streaming)
se401_start_stream(se401); se401_start_stream(se401);
...@@ -1218,28 +1171,21 @@ static int se401_ioctl(struct video_device *vdev, unsigned int cmd, void *arg) ...@@ -1218,28 +1171,21 @@ static int se401_ioctl(struct video_device *vdev, unsigned int cmd, void *arg)
} }
case VIDIOCSYNC: case VIDIOCSYNC:
{ {
int frame, ret=0; int *frame = arg;
int ret=0;
if (copy_from_user((void *)&frame, arg, sizeof(int))) if(*frame <0 || *frame >= SE401_NUMFRAMES)
return -EFAULT;
if(frame <0 || frame >= SE401_NUMFRAMES)
return -EINVAL; return -EINVAL;
ret=se401_newframe(se401, frame); ret=se401_newframe(se401, *frame);
se401->frame[frame].grabstate=FRAME_UNUSED; se401->frame[*frame].grabstate=FRAME_UNUSED;
return ret; return ret;
} }
case VIDIOCGFBUF: case VIDIOCGFBUF:
{ {
struct video_buffer vb; struct video_buffer *vb = arg;
memset(&vb, 0, sizeof(vb));
vb.base = NULL; /* frame buffer not supported, not used */
if (copy_to_user((void *)arg, (void *)&vb, sizeof(vb)))
return -EFAULT;
memset(vb, 0, sizeof(*vb));
return 0; return 0;
} }
case VIDIOCKEY: case VIDIOCKEY:
...@@ -1264,10 +1210,11 @@ static int se401_ioctl(struct video_device *vdev, unsigned int cmd, void *arg) ...@@ -1264,10 +1210,11 @@ static int se401_ioctl(struct video_device *vdev, unsigned int cmd, void *arg)
return 0; return 0;
} }
static long se401_read(struct video_device *dev, char *buf, unsigned long count, static int se401_read(struct file *file, char *buf,
int noblock) size_t count, loff_t *ppos)
{ {
int realcount=count, ret=0; int realcount=count, ret=0;
struct video_device *dev = file->private_data;
struct usb_se401 *se401 = (struct usb_se401 *)dev; struct usb_se401 *se401 = (struct usb_se401 *)dev;
...@@ -1304,11 +1251,12 @@ static long se401_read(struct video_device *dev, char *buf, unsigned long count, ...@@ -1304,11 +1251,12 @@ static long se401_read(struct video_device *dev, char *buf, unsigned long count,
return realcount; return realcount;
} }
static int se401_mmap(struct vm_area_struct *vma, struct video_device *dev, const char *adr, static int se401_mmap(struct file *file, struct vm_area_struct *vma)
unsigned long size)
{ {
struct video_device *dev = file->private_data;
struct usb_se401 *se401 = (struct usb_se401 *)dev; struct usb_se401 *se401 = (struct usb_se401 *)dev;
unsigned long start = (unsigned long)adr; unsigned long start = vma->vm_start;
unsigned long size = vma->vm_end-vma->vm_start;
unsigned long page, pos; unsigned long page, pos;
down(&se401->lock); down(&se401->lock);
...@@ -1340,17 +1288,22 @@ static int se401_mmap(struct vm_area_struct *vma, struct video_device *dev, cons ...@@ -1340,17 +1288,22 @@ static int se401_mmap(struct vm_area_struct *vma, struct video_device *dev, cons
return 0; return 0;
} }
static struct file_operations se401_fops = {
owner: THIS_MODULE,
open: se401_open,
release: se401_close,
read: se401_read,
mmap: se401_mmap,
ioctl: video_generic_ioctl,
llseek: no_llseek,
};
static struct video_device se401_template = { static struct video_device se401_template = {
owner: THIS_MODULE,
name: "se401 USB camera", name: "se401 USB camera",
type: VID_TYPE_CAPTURE, type: VID_TYPE_CAPTURE,
hardware: VID_HARDWARE_SE401, hardware: VID_HARDWARE_SE401,
open: se401_open, fops: &se401_fops,
close: se401_close, kernel_ioctl: se401_ioctl,
read: se401_read,
write: se401_write,
ioctl: se401_ioctl,
mmap: se401_mmap,
initialize: se401_init_done,
}; };
...@@ -1515,6 +1468,9 @@ static void* __devinit se401_probe(struct usb_device *dev, unsigned int ifnum, ...@@ -1515,6 +1468,9 @@ static void* __devinit se401_probe(struct usb_device *dev, unsigned int ifnum,
err("video_register_device failed"); err("video_register_device failed");
return NULL; return NULL;
} }
#if defined(CONFIG_PROC_FS) && defined(CONFIG_VIDEO_PROC_FS)
create_proc_se401_cam(se401);
#endif
info("registered new video device: video%d", se401->vdev.minor); info("registered new video device: video%d", se401->vdev.minor);
return se401; return se401;
...@@ -1524,15 +1480,12 @@ static void se401_disconnect(struct usb_device *dev, void *ptr) ...@@ -1524,15 +1480,12 @@ static void se401_disconnect(struct usb_device *dev, void *ptr)
{ {
struct usb_se401 *se401 = (struct usb_se401 *) ptr; struct usb_se401 *se401 = (struct usb_se401 *) ptr;
lock_kernel(); video_unregister_device(&se401->vdev);
/* We don't want people trying to open up the device */
if (!se401->user){ if (!se401->user){
video_unregister_device(&se401->vdev);
usb_se401_remove_disconnected(se401); usb_se401_remove_disconnected(se401);
} else { } else {
se401->removed = 1; se401->removed = 1;
} }
unlock_kernel();
} }
static inline void usb_se401_remove_disconnected (struct usb_se401 *se401) static inline void usb_se401_remove_disconnected (struct usb_se401 *se401)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment