| // 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); |