Commit 095d141b authored by Oleg Nesterov's avatar Oleg Nesterov Committed by Linus Torvalds

argv_split(): teach it to handle mutable strings

argv_split() allocates argv[count_argc(str)] array and assumes that it
will find the same number of arguments later.  This is obviously wrong if
this string can be changed, say, by sysctl.

With this patch argv_split() kstrndup's the whole string and does not
split it, we simply replace the spaces with zeroes and keep the allocated
memory in argv[-1] for argv_free(arg).

We do not use argv[0] because:

	- str can be all-spaces or empty. In fact this case is fine,
	  we could kfree() it before return, but:

	- str can have a space at the start, and we can not rely on
	  kstrndup(skip_spaces(str)) because it can equally race if
	  this string is mutable.

Also, simplify count_argc() and kill the no longer used skip_arg().
Signed-off-by: default avatarOleg Nesterov <oleg@redhat.com>
Cc: Andi Kleen <andi@firstfloor.org>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 30493cc9
...@@ -8,23 +8,17 @@ ...@@ -8,23 +8,17 @@
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/export.h> #include <linux/export.h>
static const char *skip_arg(const char *cp)
{
while (*cp && !isspace(*cp))
cp++;
return cp;
}
static int count_argc(const char *str) static int count_argc(const char *str)
{ {
int count = 0; int count = 0;
bool was_space;
while (*str) { for (was_space = true; *str; str++) {
str = skip_spaces(str); if (isspace(*str)) {
if (*str) { was_space = true;
} else if (was_space) {
was_space = false;
count++; count++;
str = skip_arg(str);
} }
} }
...@@ -39,10 +33,8 @@ static int count_argc(const char *str) ...@@ -39,10 +33,8 @@ static int count_argc(const char *str)
*/ */
void argv_free(char **argv) void argv_free(char **argv)
{ {
char **p; argv--;
for (p = argv; *p; p++) kfree(argv[0]);
kfree(*p);
kfree(argv); kfree(argv);
} }
EXPORT_SYMBOL(argv_free); EXPORT_SYMBOL(argv_free);
...@@ -59,43 +51,44 @@ EXPORT_SYMBOL(argv_free); ...@@ -59,43 +51,44 @@ EXPORT_SYMBOL(argv_free);
* considered to be a single argument separator. The returned array * considered to be a single argument separator. The returned array
* is always NULL-terminated. Returns NULL on memory allocation * is always NULL-terminated. Returns NULL on memory allocation
* failure. * 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_split(gfp_t gfp, const char *str, int *argcp)
{ {
int argc = count_argc(str); char *argv_str;
char **argv = kzalloc(sizeof(*argv) * (argc+1), gfp); bool was_space;
char **argvp; char **argv, **argv_ret;
int argc;
if (argv == NULL)
goto out;
if (argcp)
*argcp = argc;
argvp = argv; argv_str = kstrndup(str, KMALLOC_MAX_SIZE - 1, gfp);
if (!argv_str)
while (*str) { return NULL;
str = skip_spaces(str);
if (*str) {
const char *p = str;
char *t;
str = skip_arg(str); argc = count_argc(argv_str);
argv = kmalloc(sizeof(*argv) * (argc + 2), gfp);
if (!argv) {
kfree(argv_str);
return NULL;
}
t = kstrndup(p, str-p, gfp); *argv = argv_str;
if (t == NULL) argv_ret = ++argv;
goto fail; for (was_space = true; *argv_str; argv_str++) {
*argvp++ = t; if (isspace(*argv_str)) {
was_space = true;
*argv_str = 0;
} else if (was_space) {
was_space = false;
*argv++ = argv_str;
} }
} }
*argvp = NULL; *argv = NULL;
out: if (argcp)
return argv; *argcp = argc;
return argv_ret;
fail:
argv_free(argv);
return NULL;
} }
EXPORT_SYMBOL(argv_split); EXPORT_SYMBOL(argv_split);
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment