mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 057e212eae
			
		
	
	
		057e212eae
		
	
	
	
	
		
			
			When calling debugfs functions, there is no need to ever check the return value. The function can work or not, but the code logic should never do something different based on this. Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org> Signed-off-by: Hans Verkuil <hverkuil-cisco@xs4all.nl> Signed-off-by: Mauro Carvalho Chehab <mchehab+huawei@kernel.org>
		
			
				
	
	
		
			106 lines
		
	
	
		
			2.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			106 lines
		
	
	
		
			2.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| /*
 | |
|  *      uvc_debugfs.c --  USB Video Class driver - Debugging support
 | |
|  *
 | |
|  *      Copyright (C) 2011
 | |
|  *          Laurent Pinchart (laurent.pinchart@ideasonboard.com)
 | |
|  */
 | |
| 
 | |
| #include <linux/module.h>
 | |
| #include <linux/debugfs.h>
 | |
| #include <linux/slab.h>
 | |
| #include <linux/usb.h>
 | |
| 
 | |
| #include "uvcvideo.h"
 | |
| 
 | |
| /* -----------------------------------------------------------------------------
 | |
|  * Statistics
 | |
|  */
 | |
| 
 | |
| #define UVC_DEBUGFS_BUF_SIZE	1024
 | |
| 
 | |
| struct uvc_debugfs_buffer {
 | |
| 	size_t count;
 | |
| 	char data[UVC_DEBUGFS_BUF_SIZE];
 | |
| };
 | |
| 
 | |
| static int uvc_debugfs_stats_open(struct inode *inode, struct file *file)
 | |
| {
 | |
| 	struct uvc_streaming *stream = inode->i_private;
 | |
| 	struct uvc_debugfs_buffer *buf;
 | |
| 
 | |
| 	buf = kmalloc(sizeof(*buf), GFP_KERNEL);
 | |
| 	if (buf == NULL)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	buf->count = uvc_video_stats_dump(stream, buf->data, sizeof(buf->data));
 | |
| 
 | |
| 	file->private_data = buf;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static ssize_t uvc_debugfs_stats_read(struct file *file, char __user *user_buf,
 | |
| 				      size_t nbytes, loff_t *ppos)
 | |
| {
 | |
| 	struct uvc_debugfs_buffer *buf = file->private_data;
 | |
| 
 | |
| 	return simple_read_from_buffer(user_buf, nbytes, ppos, buf->data,
 | |
| 				       buf->count);
 | |
| }
 | |
| 
 | |
| static int uvc_debugfs_stats_release(struct inode *inode, struct file *file)
 | |
| {
 | |
| 	kfree(file->private_data);
 | |
| 	file->private_data = NULL;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static const struct file_operations uvc_debugfs_stats_fops = {
 | |
| 	.owner = THIS_MODULE,
 | |
| 	.open = uvc_debugfs_stats_open,
 | |
| 	.llseek = no_llseek,
 | |
| 	.read = uvc_debugfs_stats_read,
 | |
| 	.release = uvc_debugfs_stats_release,
 | |
| };
 | |
| 
 | |
| /* -----------------------------------------------------------------------------
 | |
|  * Global and stream initialization/cleanup
 | |
|  */
 | |
| 
 | |
| static struct dentry *uvc_debugfs_root_dir;
 | |
| 
 | |
| void uvc_debugfs_init_stream(struct uvc_streaming *stream)
 | |
| {
 | |
| 	struct usb_device *udev = stream->dev->udev;
 | |
| 	char dir_name[33];
 | |
| 
 | |
| 	if (uvc_debugfs_root_dir == NULL)
 | |
| 		return;
 | |
| 
 | |
| 	snprintf(dir_name, sizeof(dir_name), "%u-%u-%u", udev->bus->busnum,
 | |
| 		 udev->devnum, stream->intfnum);
 | |
| 
 | |
| 	stream->debugfs_dir = debugfs_create_dir(dir_name,
 | |
| 						 uvc_debugfs_root_dir);
 | |
| 
 | |
| 	debugfs_create_file("stats", 0444, stream->debugfs_dir, stream,
 | |
| 			    &uvc_debugfs_stats_fops);
 | |
| }
 | |
| 
 | |
| void uvc_debugfs_cleanup_stream(struct uvc_streaming *stream)
 | |
| {
 | |
| 	debugfs_remove_recursive(stream->debugfs_dir);
 | |
| 	stream->debugfs_dir = NULL;
 | |
| }
 | |
| 
 | |
| void uvc_debugfs_init(void)
 | |
| {
 | |
| 	uvc_debugfs_root_dir = debugfs_create_dir("uvcvideo", usb_debug_root);
 | |
| }
 | |
| 
 | |
| void uvc_debugfs_cleanup(void)
 | |
| {
 | |
| 	debugfs_remove_recursive(uvc_debugfs_root_dir);
 | |
| }
 |