mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	media: uvcvideo: Convert from using an atomic variable to a reference count
When adding support for metadata nodes, we'll have to keep video devices registered until all metadata nodes are closed too. Since this has nothing to do with stream counting, replace the nstreams atomic variable with a reference counter. Signed-off-by: Guennadi Liakhovetski <guennadi.liakhovetski@intel.com> Reviewed-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com> Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com> Signed-off-by: Mauro Carvalho Chehab <mchehab@s-opensource.com>
This commit is contained in:
		
							parent
							
								
									dfa6f2400c
								
							
						
					
					
						commit
						9d15cd958c
					
				| @ -1802,8 +1802,9 @@ static int uvc_scan_device(struct uvc_device *dev) | |||||||
|  * already been canceled by the USB core. There is no need to kill the |  * already been canceled by the USB core. There is no need to kill the | ||||||
|  * interrupt URB manually. |  * interrupt URB manually. | ||||||
|  */ |  */ | ||||||
| static void uvc_delete(struct uvc_device *dev) | static void uvc_delete(struct kref *kref) | ||||||
| { | { | ||||||
|  | 	struct uvc_device *dev = container_of(kref, struct uvc_device, ref); | ||||||
| 	struct list_head *p, *n; | 	struct list_head *p, *n; | ||||||
| 
 | 
 | ||||||
| 	uvc_status_cleanup(dev); | 	uvc_status_cleanup(dev); | ||||||
| @ -1854,11 +1855,7 @@ static void uvc_release(struct video_device *vdev) | |||||||
| 	struct uvc_streaming *stream = video_get_drvdata(vdev); | 	struct uvc_streaming *stream = video_get_drvdata(vdev); | ||||||
| 	struct uvc_device *dev = stream->dev; | 	struct uvc_device *dev = stream->dev; | ||||||
| 
 | 
 | ||||||
| 	/* Decrement the registered streams count and delete the device when it
 | 	kref_put(&dev->ref, uvc_delete); | ||||||
| 	 * reaches zero. |  | ||||||
| 	 */ |  | ||||||
| 	if (atomic_dec_and_test(&dev->nstreams)) |  | ||||||
| 		uvc_delete(dev); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
| @ -1870,10 +1867,10 @@ static void uvc_unregister_video(struct uvc_device *dev) | |||||||
| 
 | 
 | ||||||
| 	/* Unregistering all video devices might result in uvc_delete() being
 | 	/* Unregistering all video devices might result in uvc_delete() being
 | ||||||
| 	 * called from inside the loop if there's no open file handle. To avoid | 	 * called from inside the loop if there's no open file handle. To avoid | ||||||
| 	 * that, increment the stream count before iterating over the streams | 	 * that, increment the refcount before iterating over the streams and | ||||||
| 	 * and decrement it when done. | 	 * decrement it when done. | ||||||
| 	 */ | 	 */ | ||||||
| 	atomic_inc(&dev->nstreams); | 	kref_get(&dev->ref); | ||||||
| 
 | 
 | ||||||
| 	list_for_each_entry(stream, &dev->streams, list) { | 	list_for_each_entry(stream, &dev->streams, list) { | ||||||
| 		if (!video_is_registered(&stream->vdev)) | 		if (!video_is_registered(&stream->vdev)) | ||||||
| @ -1884,11 +1881,7 @@ static void uvc_unregister_video(struct uvc_device *dev) | |||||||
| 		uvc_debugfs_cleanup_stream(stream); | 		uvc_debugfs_cleanup_stream(stream); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/* Decrement the stream count and call uvc_delete explicitly if there
 | 	kref_put(&dev->ref, uvc_delete); | ||||||
| 	 * are no stream left. |  | ||||||
| 	 */ |  | ||||||
| 	if (atomic_dec_and_test(&dev->nstreams)) |  | ||||||
| 		uvc_delete(dev); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int uvc_register_video(struct uvc_device *dev, | static int uvc_register_video(struct uvc_device *dev, | ||||||
| @ -1946,7 +1939,7 @@ static int uvc_register_video(struct uvc_device *dev, | |||||||
| 	else | 	else | ||||||
| 		stream->chain->caps |= V4L2_CAP_VIDEO_OUTPUT; | 		stream->chain->caps |= V4L2_CAP_VIDEO_OUTPUT; | ||||||
| 
 | 
 | ||||||
| 	atomic_inc(&dev->nstreams); | 	kref_get(&dev->ref); | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -2031,7 +2024,7 @@ static int uvc_probe(struct usb_interface *intf, | |||||||
| 	INIT_LIST_HEAD(&dev->entities); | 	INIT_LIST_HEAD(&dev->entities); | ||||||
| 	INIT_LIST_HEAD(&dev->chains); | 	INIT_LIST_HEAD(&dev->chains); | ||||||
| 	INIT_LIST_HEAD(&dev->streams); | 	INIT_LIST_HEAD(&dev->streams); | ||||||
| 	atomic_set(&dev->nstreams, 0); | 	kref_init(&dev->ref); | ||||||
| 	atomic_set(&dev->nmappings, 0); | 	atomic_set(&dev->nmappings, 0); | ||||||
| 	mutex_init(&dev->lock); | 	mutex_init(&dev->lock); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -575,7 +575,7 @@ struct uvc_device { | |||||||
| 
 | 
 | ||||||
| 	/* Video Streaming interfaces */ | 	/* Video Streaming interfaces */ | ||||||
| 	struct list_head streams; | 	struct list_head streams; | ||||||
| 	atomic_t nstreams; | 	struct kref ref; | ||||||
| 
 | 
 | ||||||
| 	/* Status Interrupt Endpoint */ | 	/* Status Interrupt Endpoint */ | ||||||
| 	struct usb_host_endpoint *int_ep; | 	struct usb_host_endpoint *int_ep; | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Guennadi Liakhovetski
						Guennadi Liakhovetski