mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 650bf52208
			
		
	
	
		650bf52208
		
	
	
	
	
		
			
			If the string is invalid, this should return -EINVAL instead of 0.
Fixes: 73517cf49b ("usb: gadget: add RNDIS configfs options for class/subclass/protocol")
Cc: stable <stable@vger.kernel.org>
Acked-by: Lorenzo Colitti <lorenzo@google.com>
Signed-off-by: Dan Carpenter <dan.carpenter@oracle.com>
Link: https://lore.kernel.org/r/YCqZ3P53yyIg5cn7@mwanda
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
		
	
			
		
			
				
	
	
		
			201 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			201 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* SPDX-License-Identifier: GPL-2.0 */
 | |
| /*
 | |
|  * u_ether_configfs.h
 | |
|  *
 | |
|  * Utility definitions for configfs support in USB Ethernet functions
 | |
|  *
 | |
|  * Copyright (c) 2013 Samsung Electronics Co., Ltd.
 | |
|  *		http://www.samsung.com
 | |
|  *
 | |
|  * Author: Andrzej Pietrasiewicz <andrzejtp2010@gmail.com>
 | |
|  */
 | |
| 
 | |
| #ifndef __U_ETHER_CONFIGFS_H
 | |
| #define __U_ETHER_CONFIGFS_H
 | |
| 
 | |
| #define USB_ETHERNET_CONFIGFS_ITEM(_f_)					\
 | |
| 	static void _f_##_attr_release(struct config_item *item)	\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 									\
 | |
| 		usb_put_function_instance(&opts->func_inst);		\
 | |
| 	}								\
 | |
| 									\
 | |
| 	static struct configfs_item_operations _f_##_item_ops = {	\
 | |
| 		.release	= _f_##_attr_release,			\
 | |
| 	}
 | |
| 
 | |
| #define USB_ETHERNET_CONFIGFS_ITEM_ATTR_DEV_ADDR(_f_)			\
 | |
| 	static ssize_t _f_##_opts_dev_addr_show(struct config_item *item, \
 | |
| 						char *page)		\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 		int result;						\
 | |
| 									\
 | |
| 		mutex_lock(&opts->lock);				\
 | |
| 		result = gether_get_dev_addr(opts->net, page, PAGE_SIZE); \
 | |
| 		mutex_unlock(&opts->lock);				\
 | |
| 									\
 | |
| 		return result;						\
 | |
| 	}								\
 | |
| 									\
 | |
| 	static ssize_t _f_##_opts_dev_addr_store(struct config_item *item, \
 | |
| 						 const char *page, size_t len)\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 		int ret;						\
 | |
| 									\
 | |
| 		mutex_lock(&opts->lock);				\
 | |
| 		if (opts->refcnt) {					\
 | |
| 			mutex_unlock(&opts->lock);			\
 | |
| 			return -EBUSY;					\
 | |
| 		}							\
 | |
| 									\
 | |
| 		ret = gether_set_dev_addr(opts->net, page);		\
 | |
| 		mutex_unlock(&opts->lock);				\
 | |
| 		if (!ret)						\
 | |
| 			ret = len;					\
 | |
| 		return ret;						\
 | |
| 	}								\
 | |
| 									\
 | |
| 	CONFIGFS_ATTR(_f_##_opts_, dev_addr)
 | |
| 
 | |
| #define USB_ETHERNET_CONFIGFS_ITEM_ATTR_HOST_ADDR(_f_)			\
 | |
| 	static ssize_t _f_##_opts_host_addr_show(struct config_item *item, \
 | |
| 						 char *page)		\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 		int result;						\
 | |
| 									\
 | |
| 		mutex_lock(&opts->lock);				\
 | |
| 		result = gether_get_host_addr(opts->net, page, PAGE_SIZE); \
 | |
| 		mutex_unlock(&opts->lock);				\
 | |
| 									\
 | |
| 		return result;						\
 | |
| 	}								\
 | |
| 									\
 | |
| 	static ssize_t _f_##_opts_host_addr_store(struct config_item *item, \
 | |
| 						  const char *page, size_t len)\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 		int ret;						\
 | |
| 									\
 | |
| 		mutex_lock(&opts->lock);				\
 | |
| 		if (opts->refcnt) {					\
 | |
| 			mutex_unlock(&opts->lock);			\
 | |
| 			return -EBUSY;					\
 | |
| 		}							\
 | |
| 									\
 | |
| 		ret = gether_set_host_addr(opts->net, page);		\
 | |
| 		mutex_unlock(&opts->lock);				\
 | |
| 		if (!ret)						\
 | |
| 			ret = len;					\
 | |
| 		return ret;						\
 | |
| 	}								\
 | |
| 									\
 | |
| 	CONFIGFS_ATTR(_f_##_opts_, host_addr)
 | |
| 
 | |
| #define USB_ETHERNET_CONFIGFS_ITEM_ATTR_QMULT(_f_)			\
 | |
| 	static ssize_t _f_##_opts_qmult_show(struct config_item *item,	\
 | |
| 					     char *page)		\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 		unsigned qmult;						\
 | |
| 									\
 | |
| 		mutex_lock(&opts->lock);				\
 | |
| 		qmult = gether_get_qmult(opts->net);			\
 | |
| 		mutex_unlock(&opts->lock);				\
 | |
| 		return sprintf(page, "%d\n", qmult);			\
 | |
| 	}								\
 | |
| 									\
 | |
| 	static ssize_t _f_##_opts_qmult_store(struct config_item *item, \
 | |
| 					      const char *page, size_t len)\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 		u8 val;							\
 | |
| 		int ret;						\
 | |
| 									\
 | |
| 		mutex_lock(&opts->lock);				\
 | |
| 		if (opts->refcnt) {					\
 | |
| 			ret = -EBUSY;					\
 | |
| 			goto out;					\
 | |
| 		}							\
 | |
| 									\
 | |
| 		ret = kstrtou8(page, 0, &val);				\
 | |
| 		if (ret)						\
 | |
| 			goto out;					\
 | |
| 									\
 | |
| 		gether_set_qmult(opts->net, val);			\
 | |
| 		ret = len;						\
 | |
| out:									\
 | |
| 		mutex_unlock(&opts->lock);				\
 | |
| 		return ret;						\
 | |
| 	}								\
 | |
| 									\
 | |
| 	CONFIGFS_ATTR(_f_##_opts_, qmult)
 | |
| 
 | |
| #define USB_ETHERNET_CONFIGFS_ITEM_ATTR_IFNAME(_f_)			\
 | |
| 	static ssize_t _f_##_opts_ifname_show(struct config_item *item, \
 | |
| 					      char *page)		\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 		int ret;						\
 | |
| 									\
 | |
| 		mutex_lock(&opts->lock);				\
 | |
| 		ret = gether_get_ifname(opts->net, page, PAGE_SIZE);	\
 | |
| 		mutex_unlock(&opts->lock);				\
 | |
| 									\
 | |
| 		return ret;						\
 | |
| 	}								\
 | |
| 									\
 | |
| 	static ssize_t _f_##_opts_ifname_store(struct config_item *item, \
 | |
| 					       const char *page, size_t len)\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 		int ret = -EBUSY;					\
 | |
| 									\
 | |
| 		mutex_lock(&opts->lock);				\
 | |
| 		if (!opts->refcnt)					\
 | |
| 			ret = gether_set_ifname(opts->net, page, len);	\
 | |
| 		mutex_unlock(&opts->lock);				\
 | |
| 		return ret ?: len;					\
 | |
| 	}								\
 | |
| 									\
 | |
| 	CONFIGFS_ATTR(_f_##_opts_, ifname)
 | |
| 
 | |
| #define USB_ETHER_CONFIGFS_ITEM_ATTR_U8_RW(_f_, _n_)			\
 | |
| 	static ssize_t _f_##_opts_##_n_##_show(struct config_item *item,\
 | |
| 					       char *page)		\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 		int ret;						\
 | |
| 									\
 | |
| 		mutex_lock(&opts->lock);				\
 | |
| 		ret = sprintf(page, "%02x\n", opts->_n_);		\
 | |
| 		mutex_unlock(&opts->lock);				\
 | |
| 									\
 | |
| 		return ret;						\
 | |
| 	}								\
 | |
| 									\
 | |
| 	static ssize_t _f_##_opts_##_n_##_store(struct config_item *item,\
 | |
| 						const char *page,	\
 | |
| 						size_t len)		\
 | |
| 	{								\
 | |
| 		struct f_##_f_##_opts *opts = to_f_##_f_##_opts(item);	\
 | |
| 		int ret = -EINVAL;					\
 | |
| 		u8 val;							\
 | |
| 									\
 | |
| 		mutex_lock(&opts->lock);				\
 | |
| 		if (sscanf(page, "%02hhx", &val) > 0) {			\
 | |
| 			opts->_n_ = val;				\
 | |
| 			ret = len;					\
 | |
| 		}							\
 | |
| 		mutex_unlock(&opts->lock);				\
 | |
| 									\
 | |
| 		return ret;						\
 | |
| 	}								\
 | |
| 									\
 | |
| 	CONFIGFS_ATTR(_f_##_opts_, _n_)
 | |
| 
 | |
| #endif /* __U_ETHER_CONFIGFS_H */
 |