Commit a906f62f authored by Hans Verkuil's avatar Hans Verkuil Committed by Mauro Carvalho Chehab

[media] zr364xx: introduce v4l2_device

Implement struct v4l2_device: use the core lock and use the v4l2_device
release() callback.
Signed-off-by: default avatarHans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@redhat.com>
parent 2b99251f
...@@ -37,6 +37,7 @@ ...@@ -37,6 +37,7 @@
#include <linux/highmem.h> #include <linux/highmem.h>
#include <media/v4l2-common.h> #include <media/v4l2-common.h>
#include <media/v4l2-ioctl.h> #include <media/v4l2-ioctl.h>
#include <media/v4l2-device.h>
#include <media/videobuf-vmalloc.h> #include <media/videobuf-vmalloc.h>
...@@ -173,6 +174,7 @@ static const struct zr364xx_fmt formats[] = { ...@@ -173,6 +174,7 @@ static const struct zr364xx_fmt formats[] = {
struct zr364xx_camera { struct zr364xx_camera {
struct usb_device *udev; /* save off the usb device pointer */ struct usb_device *udev; /* save off the usb device pointer */
struct usb_interface *interface;/* the interface for this device */ struct usb_interface *interface;/* the interface for this device */
struct v4l2_device v4l2_dev;
struct video_device vdev; /* v4l video device */ struct video_device vdev; /* v4l video device */
int nb; int nb;
struct zr364xx_bufferi buffer; struct zr364xx_bufferi buffer;
...@@ -181,7 +183,6 @@ struct zr364xx_camera { ...@@ -181,7 +183,6 @@ struct zr364xx_camera {
int height; int height;
int method; int method;
struct mutex lock; struct mutex lock;
struct mutex open_lock;
int users; int users;
spinlock_t slock; spinlock_t slock;
...@@ -230,11 +231,6 @@ static int send_control_msg(struct usb_device *udev, u8 request, u16 value, ...@@ -230,11 +231,6 @@ static int send_control_msg(struct usb_device *udev, u8 request, u16 value,
transfer_buffer, size, CTRL_TIMEOUT); transfer_buffer, size, CTRL_TIMEOUT);
kfree(transfer_buffer); kfree(transfer_buffer);
if (status < 0)
dev_err(&udev->dev,
"Failed sending control message, error %d.\n", status);
return status; return status;
} }
...@@ -468,6 +464,7 @@ static ssize_t zr364xx_read(struct file *file, char __user *buf, size_t count, ...@@ -468,6 +464,7 @@ static ssize_t zr364xx_read(struct file *file, char __user *buf, size_t count,
loff_t * ppos) loff_t * ppos)
{ {
struct zr364xx_camera *cam = video_drvdata(file); struct zr364xx_camera *cam = video_drvdata(file);
int err = 0;
_DBG("%s\n", __func__); _DBG("%s\n", __func__);
...@@ -477,17 +474,20 @@ static ssize_t zr364xx_read(struct file *file, char __user *buf, size_t count, ...@@ -477,17 +474,20 @@ static ssize_t zr364xx_read(struct file *file, char __user *buf, size_t count,
if (!count) if (!count)
return -EINVAL; return -EINVAL;
if (mutex_lock_interruptible(&cam->lock))
return -ERESTARTSYS;
if (cam->type == V4L2_BUF_TYPE_VIDEO_CAPTURE && if (cam->type == V4L2_BUF_TYPE_VIDEO_CAPTURE &&
zr364xx_vidioc_streamon(file, cam, cam->type) == 0) { zr364xx_vidioc_streamon(file, cam, cam->type) == 0) {
DBG("%s: reading %d bytes at pos %d.\n", __func__, (int) count, DBG("%s: reading %d bytes at pos %d.\n", __func__, (int) count,
(int) *ppos); (int) *ppos);
/* NoMan Sux ! */ /* NoMan Sux ! */
return videobuf_read_one(&cam->vb_vidq, buf, count, ppos, err = videobuf_read_one(&cam->vb_vidq, buf, count, ppos,
file->f_flags & O_NONBLOCK); file->f_flags & O_NONBLOCK);
} }
mutex_unlock(&cam->lock);
return 0; return err;
} }
/* video buffer vmalloc implementation based partly on VIVI driver which is /* video buffer vmalloc implementation based partly on VIVI driver which is
...@@ -705,16 +705,13 @@ static int zr364xx_read_video_callback(struct zr364xx_camera *cam, ...@@ -705,16 +705,13 @@ static int zr364xx_read_video_callback(struct zr364xx_camera *cam,
static int res_get(struct zr364xx_camera *cam) static int res_get(struct zr364xx_camera *cam)
{ {
/* is it free? */ /* is it free? */
mutex_lock(&cam->lock);
if (cam->resources) { if (cam->resources) {
/* no, someone else uses it */ /* no, someone else uses it */
mutex_unlock(&cam->lock);
return 0; return 0;
} }
/* it's free, grab it */ /* it's free, grab it */
cam->resources = 1; cam->resources = 1;
_DBG("res: get\n"); _DBG("res: get\n");
mutex_unlock(&cam->lock);
return 1; return 1;
} }
...@@ -725,9 +722,7 @@ static inline int res_check(struct zr364xx_camera *cam) ...@@ -725,9 +722,7 @@ static inline int res_check(struct zr364xx_camera *cam)
static void res_free(struct zr364xx_camera *cam) static void res_free(struct zr364xx_camera *cam)
{ {
mutex_lock(&cam->lock);
cam->resources = 0; cam->resources = 0;
mutex_unlock(&cam->lock);
_DBG("res: put\n"); _DBG("res: put\n");
} }
...@@ -811,11 +806,9 @@ static int zr364xx_vidioc_s_ctrl(struct file *file, void *priv, ...@@ -811,11 +806,9 @@ static int zr364xx_vidioc_s_ctrl(struct file *file, void *priv,
case V4L2_CID_BRIGHTNESS: case V4L2_CID_BRIGHTNESS:
cam->mode.brightness = c->value; cam->mode.brightness = c->value;
/* hardware brightness */ /* hardware brightness */
mutex_lock(&cam->lock);
send_control_msg(cam->udev, 1, 0x2001, 0, NULL, 0); send_control_msg(cam->udev, 1, 0x2001, 0, NULL, 0);
temp = (0x60 << 8) + 127 - cam->mode.brightness; temp = (0x60 << 8) + 127 - cam->mode.brightness;
send_control_msg(cam->udev, 1, temp, 0, NULL, 0); send_control_msg(cam->udev, 1, temp, 0, NULL, 0);
mutex_unlock(&cam->lock);
break; break;
default: default:
return -EINVAL; return -EINVAL;
...@@ -1270,7 +1263,8 @@ static int zr364xx_open(struct file *file) ...@@ -1270,7 +1263,8 @@ static int zr364xx_open(struct file *file)
DBG("%s\n", __func__); DBG("%s\n", __func__);
mutex_lock(&cam->open_lock); if (mutex_lock_interruptible(&cam->lock))
return -ERESTARTSYS;
if (cam->users) { if (cam->users) {
err = -EBUSY; err = -EBUSY;
...@@ -1299,7 +1293,7 @@ static int zr364xx_open(struct file *file) ...@@ -1299,7 +1293,7 @@ static int zr364xx_open(struct file *file)
NULL, &cam->slock, NULL, &cam->slock,
cam->type, cam->type,
V4L2_FIELD_NONE, V4L2_FIELD_NONE,
sizeof(struct zr364xx_buffer), cam, NULL); sizeof(struct zr364xx_buffer), cam, &cam->lock);
/* Added some delay here, since opening/closing the camera quickly, /* Added some delay here, since opening/closing the camera quickly,
* like Ekiga does during its startup, can crash the webcam * like Ekiga does during its startup, can crash the webcam
...@@ -1308,27 +1302,20 @@ static int zr364xx_open(struct file *file) ...@@ -1308,27 +1302,20 @@ static int zr364xx_open(struct file *file)
err = 0; err = 0;
out: out:
mutex_unlock(&cam->open_lock); mutex_unlock(&cam->lock);
DBG("%s: %d\n", __func__, err); DBG("%s: %d\n", __func__, err);
return err; return err;
} }
static void zr364xx_destroy(struct zr364xx_camera *cam) static void zr364xx_release(struct v4l2_device *v4l2_dev)
{ {
struct zr364xx_camera *cam =
container_of(v4l2_dev, struct zr364xx_camera, v4l2_dev);
unsigned long i; unsigned long i;
if (!cam) { v4l2_device_unregister(&cam->v4l2_dev);
printk(KERN_ERR KBUILD_MODNAME ", %s: no device\n", __func__);
return;
}
mutex_lock(&cam->open_lock);
video_unregister_device(&cam->vdev);
/* stops the read pipe if it is running */
if (cam->b_acquire)
zr364xx_stop_acquire(cam);
zr364xx_stop_readpipe(cam); videobuf_mmap_free(&cam->vb_vidq);
/* release sys buffers */ /* release sys buffers */
for (i = 0; i < FRAMES; i++) { for (i = 0; i < FRAMES; i++) {
...@@ -1341,25 +1328,20 @@ static void zr364xx_destroy(struct zr364xx_camera *cam) ...@@ -1341,25 +1328,20 @@ static void zr364xx_destroy(struct zr364xx_camera *cam)
/* release transfer buffer */ /* release transfer buffer */
kfree(cam->pipe->transfer_buffer); kfree(cam->pipe->transfer_buffer);
cam->pipe->transfer_buffer = NULL;
mutex_unlock(&cam->open_lock);
kfree(cam); kfree(cam);
} }
/* release the camera */ /* release the camera */
static int zr364xx_release(struct file *file) static int zr364xx_close(struct file *file)
{ {
struct zr364xx_camera *cam; struct zr364xx_camera *cam;
struct usb_device *udev; struct usb_device *udev;
int i, err; int i;
DBG("%s\n", __func__); DBG("%s\n", __func__);
cam = video_drvdata(file); cam = video_drvdata(file);
if (!cam) mutex_lock(&cam->lock);
return -ENODEV;
mutex_lock(&cam->open_lock);
udev = cam->udev; udev = cam->udev;
/* turn off stream */ /* turn off stream */
...@@ -1374,26 +1356,17 @@ static int zr364xx_release(struct file *file) ...@@ -1374,26 +1356,17 @@ static int zr364xx_release(struct file *file)
file->private_data = NULL; file->private_data = NULL;
for (i = 0; i < 2; i++) { for (i = 0; i < 2; i++) {
err =
send_control_msg(udev, 1, init[cam->method][i].value, send_control_msg(udev, 1, init[cam->method][i].value,
0, init[cam->method][i].bytes, 0, init[cam->method][i].bytes,
init[cam->method][i].size); init[cam->method][i].size);
if (err < 0) {
dev_err(&udev->dev, "error during release sequence\n");
goto out;
}
} }
/* Added some delay here, since opening/closing the camera quickly, /* Added some delay here, since opening/closing the camera quickly,
* like Ekiga does during its startup, can crash the webcam * like Ekiga does during its startup, can crash the webcam
*/ */
mdelay(100); mdelay(100);
err = 0; mutex_unlock(&cam->lock);
return 0;
out:
mutex_unlock(&cam->open_lock);
return err;
} }
...@@ -1432,10 +1405,10 @@ static unsigned int zr364xx_poll(struct file *file, ...@@ -1432,10 +1405,10 @@ static unsigned int zr364xx_poll(struct file *file,
static const struct v4l2_file_operations zr364xx_fops = { static const struct v4l2_file_operations zr364xx_fops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.open = zr364xx_open, .open = zr364xx_open,
.release = zr364xx_release, .release = zr364xx_close,
.read = zr364xx_read, .read = zr364xx_read,
.mmap = zr364xx_mmap, .mmap = zr364xx_mmap,
.ioctl = video_ioctl2, .unlocked_ioctl = video_ioctl2,
.poll = zr364xx_poll, .poll = zr364xx_poll,
}; };
...@@ -1552,10 +1525,20 @@ static int zr364xx_probe(struct usb_interface *intf, ...@@ -1552,10 +1525,20 @@ static int zr364xx_probe(struct usb_interface *intf,
dev_err(&udev->dev, "cam: out of memory !\n"); dev_err(&udev->dev, "cam: out of memory !\n");
return -ENOMEM; return -ENOMEM;
} }
cam->v4l2_dev.release = zr364xx_release;
err = v4l2_device_register(&intf->dev, &cam->v4l2_dev);
if (err < 0) {
dev_err(&udev->dev, "couldn't register v4l2_device\n");
kfree(cam);
return err;
}
/* save the init method used by this camera */ /* save the init method used by this camera */
cam->method = id->driver_info; cam->method = id->driver_info;
mutex_init(&cam->lock);
cam->vdev = zr364xx_template; cam->vdev = zr364xx_template;
cam->vdev.parent = &intf->dev; cam->vdev.lock = &cam->lock;
cam->vdev.v4l2_dev = &cam->v4l2_dev;
video_set_drvdata(&cam->vdev, cam); video_set_drvdata(&cam->vdev, cam);
if (debug) if (debug)
cam->vdev.debug = V4L2_DEBUG_IOCTL | V4L2_DEBUG_IOCTL_ARG; cam->vdev.debug = V4L2_DEBUG_IOCTL | V4L2_DEBUG_IOCTL_ARG;
...@@ -1607,8 +1590,6 @@ static int zr364xx_probe(struct usb_interface *intf, ...@@ -1607,8 +1590,6 @@ static int zr364xx_probe(struct usb_interface *intf,
cam->users = 0; cam->users = 0;
cam->nb = 0; cam->nb = 0;
cam->mode.brightness = 64; cam->mode.brightness = 64;
mutex_init(&cam->lock);
mutex_init(&cam->open_lock);
DBG("dev: %p, udev %p interface %p\n", cam, cam->udev, intf); DBG("dev: %p, udev %p interface %p\n", cam, cam->udev, intf);
...@@ -1625,6 +1606,7 @@ static int zr364xx_probe(struct usb_interface *intf, ...@@ -1625,6 +1606,7 @@ static int zr364xx_probe(struct usb_interface *intf,
if (!cam->read_endpoint) { if (!cam->read_endpoint) {
dev_err(&intf->dev, "Could not find bulk-in endpoint\n"); dev_err(&intf->dev, "Could not find bulk-in endpoint\n");
v4l2_device_unregister(&cam->v4l2_dev);
kfree(cam); kfree(cam);
return -ENOMEM; return -ENOMEM;
} }
...@@ -1647,6 +1629,7 @@ static int zr364xx_probe(struct usb_interface *intf, ...@@ -1647,6 +1629,7 @@ static int zr364xx_probe(struct usb_interface *intf,
err = video_register_device(&cam->vdev, VFL_TYPE_GRABBER, -1); err = video_register_device(&cam->vdev, VFL_TYPE_GRABBER, -1);
if (err) { if (err) {
dev_err(&udev->dev, "video_register_device failed\n"); dev_err(&udev->dev, "video_register_device failed\n");
v4l2_device_unregister(&cam->v4l2_dev);
kfree(cam); kfree(cam);
return err; return err;
} }
...@@ -1660,10 +1643,20 @@ static int zr364xx_probe(struct usb_interface *intf, ...@@ -1660,10 +1643,20 @@ static int zr364xx_probe(struct usb_interface *intf,
static void zr364xx_disconnect(struct usb_interface *intf) static void zr364xx_disconnect(struct usb_interface *intf)
{ {
struct zr364xx_camera *cam = usb_get_intfdata(intf); struct zr364xx_camera *cam = usb_get_intfdata(intf);
videobuf_mmap_free(&cam->vb_vidq);
mutex_lock(&cam->lock);
usb_set_intfdata(intf, NULL); usb_set_intfdata(intf, NULL);
dev_info(&intf->dev, DRIVER_DESC " webcam unplugged\n"); dev_info(&intf->dev, DRIVER_DESC " webcam unplugged\n");
zr364xx_destroy(cam); video_unregister_device(&cam->vdev);
v4l2_device_disconnect(&cam->v4l2_dev);
/* stops the read pipe if it is running */
if (cam->b_acquire)
zr364xx_stop_acquire(cam);
zr364xx_stop_readpipe(cam);
mutex_unlock(&cam->lock);
v4l2_device_put(&cam->v4l2_dev);
} }
......
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