mirror of
				https://github.com/torvalds/linux.git
				synced 2025-10-31 16:48:26 +02:00 
			
		
		
		
	 36ee98b555
			
		
	
	
		36ee98b555
		
	
	
	
	
		
			
			Use proper kernel-doc notation to prevent build warnings: lib/argv_split.c:36: warning: Function parameter or member 'argv' not described in 'argv_free' lib/argv_split.c:61: warning: No description found for return value of 'argv_split' Link: https://lkml.kernel.org/r/20230912060838.3794-1-rdunlap@infradead.org Signed-off-by: Randy Dunlap <rdunlap@infradead.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
		
			
				
	
	
		
			95 lines
		
	
	
	
		
			2.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			95 lines
		
	
	
	
		
			2.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Helper function for splitting a string into an argv-like array.
 | |
|  */
 | |
| 
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/ctype.h>
 | |
| #include <linux/string.h>
 | |
| #include <linux/slab.h>
 | |
| #include <linux/export.h>
 | |
| 
 | |
| static int count_argc(const char *str)
 | |
| {
 | |
| 	int count = 0;
 | |
| 	bool was_space;
 | |
| 
 | |
| 	for (was_space = true; *str; str++) {
 | |
| 		if (isspace(*str)) {
 | |
| 			was_space = true;
 | |
| 		} else if (was_space) {
 | |
| 			was_space = false;
 | |
| 			count++;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return count;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * argv_free - free an argv
 | |
|  * @argv: the argument vector to be freed
 | |
|  *
 | |
|  * Frees an argv and the strings it points to.
 | |
|  */
 | |
| void argv_free(char **argv)
 | |
| {
 | |
| 	argv--;
 | |
| 	kfree(argv[0]);
 | |
| 	kfree(argv);
 | |
| }
 | |
| EXPORT_SYMBOL(argv_free);
 | |
| 
 | |
| /**
 | |
|  * argv_split - split a string at whitespace, returning an argv
 | |
|  * @gfp: the GFP mask used to allocate memory
 | |
|  * @str: the string to be split
 | |
|  * @argcp: returned argument count
 | |
|  *
 | |
|  * Returns: an array of pointers to strings which are split out from
 | |
|  * @str.  This is performed by strictly splitting on white-space; no
 | |
|  * quote processing is performed.  Multiple whitespace characters are
 | |
|  * considered to be a single argument separator.  The returned array
 | |
|  * is always NULL-terminated.  Returns NULL on memory allocation
 | |
|  * failure.
 | |
|  *
 | |
|  * The source string at `str' may be undergoing concurrent alteration via
 | |
|  * userspace sysctl activity (at least).  The argv_split() implementation
 | |
|  * attempts to handle this gracefully by taking a local copy to work on.
 | |
|  */
 | |
| char **argv_split(gfp_t gfp, const char *str, int *argcp)
 | |
| {
 | |
| 	char *argv_str;
 | |
| 	bool was_space;
 | |
| 	char **argv, **argv_ret;
 | |
| 	int argc;
 | |
| 
 | |
| 	argv_str = kstrndup(str, KMALLOC_MAX_SIZE - 1, gfp);
 | |
| 	if (!argv_str)
 | |
| 		return NULL;
 | |
| 
 | |
| 	argc = count_argc(argv_str);
 | |
| 	argv = kmalloc_array(argc + 2, sizeof(*argv), gfp);
 | |
| 	if (!argv) {
 | |
| 		kfree(argv_str);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	*argv = argv_str;
 | |
| 	argv_ret = ++argv;
 | |
| 	for (was_space = true; *argv_str; argv_str++) {
 | |
| 		if (isspace(*argv_str)) {
 | |
| 			was_space = true;
 | |
| 			*argv_str = 0;
 | |
| 		} else if (was_space) {
 | |
| 			was_space = false;
 | |
| 			*argv++ = argv_str;
 | |
| 		}
 | |
| 	}
 | |
| 	*argv = NULL;
 | |
| 
 | |
| 	if (argcp)
 | |
| 		*argcp = argc;
 | |
| 	return argv_ret;
 | |
| }
 | |
| EXPORT_SYMBOL(argv_split);
 |