diff options
Diffstat (limited to 'drivers/media/video/v4l2-dev.c')
-rw-r--r-- | drivers/media/video/v4l2-dev.c | 49 |
1 files changed, 45 insertions, 4 deletions
diff --git a/drivers/media/video/v4l2-dev.c b/drivers/media/video/v4l2-dev.c index abe04ef38066..e405b8094b94 100644 --- a/drivers/media/video/v4l2-dev.c +++ b/drivers/media/video/v4l2-dev.c | |||
@@ -303,6 +303,9 @@ static int v4l2_mmap(struct file *filp, struct vm_area_struct *vm) | |||
303 | static int v4l2_open(struct inode *inode, struct file *filp) | 303 | static int v4l2_open(struct inode *inode, struct file *filp) |
304 | { | 304 | { |
305 | struct video_device *vdev; | 305 | struct video_device *vdev; |
306 | #if defined(CONFIG_MEDIA_CONTROLLER) | ||
307 | struct media_entity *entity = NULL; | ||
308 | #endif | ||
306 | int ret = 0; | 309 | int ret = 0; |
307 | 310 | ||
308 | /* Check if the video device is available */ | 311 | /* Check if the video device is available */ |
@@ -316,6 +319,16 @@ static int v4l2_open(struct inode *inode, struct file *filp) | |||
316 | /* and increase the device refcount */ | 319 | /* and increase the device refcount */ |
317 | video_get(vdev); | 320 | video_get(vdev); |
318 | mutex_unlock(&videodev_lock); | 321 | mutex_unlock(&videodev_lock); |
322 | #if defined(CONFIG_MEDIA_CONTROLLER) | ||
323 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev) { | ||
324 | entity = media_entity_get(&vdev->entity); | ||
325 | if (!entity) { | ||
326 | ret = -EBUSY; | ||
327 | video_put(vdev); | ||
328 | return ret; | ||
329 | } | ||
330 | } | ||
331 | #endif | ||
319 | if (vdev->fops->open) { | 332 | if (vdev->fops->open) { |
320 | if (vdev->lock && mutex_lock_interruptible(vdev->lock)) { | 333 | if (vdev->lock && mutex_lock_interruptible(vdev->lock)) { |
321 | ret = -ERESTARTSYS; | 334 | ret = -ERESTARTSYS; |
@@ -331,8 +344,13 @@ static int v4l2_open(struct inode *inode, struct file *filp) | |||
331 | 344 | ||
332 | err: | 345 | err: |
333 | /* decrease the refcount in case of an error */ | 346 | /* decrease the refcount in case of an error */ |
334 | if (ret) | 347 | if (ret) { |
348 | #if defined(CONFIG_MEDIA_CONTROLLER) | ||
349 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev) | ||
350 | media_entity_put(entity); | ||
351 | #endif | ||
335 | video_put(vdev); | 352 | video_put(vdev); |
353 | } | ||
336 | return ret; | 354 | return ret; |
337 | } | 355 | } |
338 | 356 | ||
@@ -349,7 +367,10 @@ static int v4l2_release(struct inode *inode, struct file *filp) | |||
349 | if (vdev->lock) | 367 | if (vdev->lock) |
350 | mutex_unlock(vdev->lock); | 368 | mutex_unlock(vdev->lock); |
351 | } | 369 | } |
352 | 370 | #if defined(CONFIG_MEDIA_CONTROLLER) | |
371 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev) | ||
372 | media_entity_put(&vdev->entity); | ||
373 | #endif | ||
353 | /* decrease the refcount unconditionally since the release() | 374 | /* decrease the refcount unconditionally since the release() |
354 | return value is ignored. */ | 375 | return value is ignored. */ |
355 | video_put(vdev); | 376 | video_put(vdev); |
@@ -585,12 +606,27 @@ int __video_register_device(struct video_device *vdev, int type, int nr, | |||
585 | if (nr != -1 && nr != vdev->num && warn_if_nr_in_use) | 606 | if (nr != -1 && nr != vdev->num && warn_if_nr_in_use) |
586 | printk(KERN_WARNING "%s: requested %s%d, got %s\n", __func__, | 607 | printk(KERN_WARNING "%s: requested %s%d, got %s\n", __func__, |
587 | name_base, nr, video_device_node_name(vdev)); | 608 | name_base, nr, video_device_node_name(vdev)); |
588 | 609 | #if defined(CONFIG_MEDIA_CONTROLLER) | |
589 | /* Part 5: Activate this minor. The char device can now be used. */ | 610 | /* Part 5: Register the entity. */ |
611 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev) { | ||
612 | vdev->entity.type = MEDIA_ENT_T_DEVNODE_V4L; | ||
613 | vdev->entity.name = vdev->name; | ||
614 | vdev->entity.v4l.major = VIDEO_MAJOR; | ||
615 | vdev->entity.v4l.minor = vdev->minor; | ||
616 | ret = media_device_register_entity(vdev->v4l2_dev->mdev, | ||
617 | &vdev->entity); | ||
618 | if (ret < 0) | ||
619 | printk(KERN_WARNING | ||
620 | "%s: media_device_register_entity failed\n", | ||
621 | __func__); | ||
622 | } | ||
623 | #endif | ||
624 | /* Part 6: Activate this minor. The char device can now be used. */ | ||
590 | set_bit(V4L2_FL_REGISTERED, &vdev->flags); | 625 | set_bit(V4L2_FL_REGISTERED, &vdev->flags); |
591 | mutex_lock(&videodev_lock); | 626 | mutex_lock(&videodev_lock); |
592 | video_device[vdev->minor] = vdev; | 627 | video_device[vdev->minor] = vdev; |
593 | mutex_unlock(&videodev_lock); | 628 | mutex_unlock(&videodev_lock); |
629 | |||
594 | return 0; | 630 | return 0; |
595 | 631 | ||
596 | cleanup: | 632 | cleanup: |
@@ -618,6 +654,11 @@ void video_unregister_device(struct video_device *vdev) | |||
618 | if (!vdev || !video_is_registered(vdev)) | 654 | if (!vdev || !video_is_registered(vdev)) |
619 | return; | 655 | return; |
620 | 656 | ||
657 | #if defined(CONFIG_MEDIA_CONTROLLER) | ||
658 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev) | ||
659 | media_device_unregister_entity(&vdev->entity); | ||
660 | #endif | ||
661 | |||
621 | mutex_lock(&videodev_lock); | 662 | mutex_lock(&videodev_lock); |
622 | /* This must be in a critical section to prevent a race with v4l2_open. | 663 | /* This must be in a critical section to prevent a race with v4l2_open. |
623 | * Once this bit has been cleared video_get may never be called again. | 664 | * Once this bit has been cleared video_get may never be called again. |