| 
									
										
										
										
											2007-07-17 18:37:02 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Helper function for splitting a string into an argv-like array. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/kernel.h>
 | 
					
						
							|  |  |  | #include <linux/ctype.h>
 | 
					
						
							| 
									
										
										
										
											2009-12-14 18:01:06 -08:00
										 |  |  | #include <linux/string.h>
 | 
					
						
							| 
									
										
										
										
											2007-10-20 00:25:12 +02:00
										 |  |  | #include <linux/slab.h>
 | 
					
						
							| 
									
										
										
										
											2011-11-16 21:29:17 -05:00
										 |  |  | #include <linux/export.h>
 | 
					
						
							| 
									
										
										
										
											2007-07-17 18:37:02 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int count_argc(const char *str) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int count = 0; | 
					
						
							| 
									
										
										
										
											2013-04-29 16:18:10 -07:00
										 |  |  | 	bool was_space; | 
					
						
							| 
									
										
										
										
											2007-07-17 18:37:02 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-29 16:18:10 -07:00
										 |  |  | 	for (was_space = true; *str; str++) { | 
					
						
							|  |  |  | 		if (isspace(*str)) { | 
					
						
							|  |  |  | 			was_space = true; | 
					
						
							|  |  |  | 		} else if (was_space) { | 
					
						
							|  |  |  | 			was_space = false; | 
					
						
							| 
									
										
										
										
											2007-07-17 18:37:02 -07:00
										 |  |  | 			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) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-29 16:18:10 -07:00
										 |  |  | 	argv--; | 
					
						
							|  |  |  | 	kfree(argv[0]); | 
					
						
							| 
									
										
										
										
											2007-07-17 18:37:02 -07:00
										 |  |  | 	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. | 
					
						
							| 
									
										
										
										
											2013-04-29 16:18:10 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							| 
									
										
										
										
											2007-07-17 18:37:02 -07:00
										 |  |  |  */ | 
					
						
							|  |  |  | char **argv_split(gfp_t gfp, const char *str, int *argcp) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-29 16:18:10 -07:00
										 |  |  | 	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(sizeof(*argv) * (argc + 2), gfp); | 
					
						
							|  |  |  | 	if (!argv) { | 
					
						
							|  |  |  | 		kfree(argv_str); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2007-07-17 18:37:02 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-29 16:18:10 -07:00
										 |  |  | 	*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; | 
					
						
							| 
									
										
										
										
											2007-07-17 18:37:02 -07:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-04-29 16:18:10 -07:00
										 |  |  | 	*argv = NULL; | 
					
						
							| 
									
										
										
										
											2007-07-17 18:37:02 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-29 16:18:10 -07:00
										 |  |  | 	if (argcp) | 
					
						
							|  |  |  | 		*argcp = argc; | 
					
						
							|  |  |  | 	return argv_ret; | 
					
						
							| 
									
										
										
										
											2007-07-17 18:37:02 -07:00
										 |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL(argv_split); |