提交 0a8f36eb 编写于 作者: E Eric W. Biederman

exec: Factor out alloc_bprm

Currently it is necessary for the usermode helper code and the code
that launches init to use set_fs so that pages coming from the kernel
look like they are coming from userspace.

To allow that usage of set_fs to be removed cleanly the argument
copying from userspace needs to happen earlier.  Move the allocation
of the bprm into it's own function (alloc_bprm) and move the call of
alloc_bprm before unshare_files so that bprm can ultimately be
allocated, the arguments can be placed on the new stack, and then the
bprm can be passed into the core of exec.

Neither the allocation of struct binprm nor the unsharing depend upon each
other so swapping the order in which they are called is trivially safe.

To keep things consistent the order of cleanup at the end of
do_execve_common swapped to match the order of initialization.
Reviewed-by: NKees Cook <keescook@chromium.org>
Link: https://lkml.kernel.org/r/87pn8y6x9a.fsf@x220.int.ebiederm.orgSigned-off-by: N"Eric W. Biederman" <ebiederm@xmission.com>
上级 9746c9be
...@@ -1560,6 +1560,14 @@ static void free_bprm(struct linux_binprm *bprm) ...@@ -1560,6 +1560,14 @@ static void free_bprm(struct linux_binprm *bprm)
kfree(bprm); kfree(bprm);
} }
static struct linux_binprm *alloc_bprm(void)
{
struct linux_binprm *bprm = kzalloc(sizeof(*bprm), GFP_KERNEL);
if (!bprm)
return ERR_PTR(-ENOMEM);
return bprm;
}
int bprm_change_interp(const char *interp, struct linux_binprm *bprm) int bprm_change_interp(const char *interp, struct linux_binprm *bprm)
{ {
/* If a binfmt changed the interp, free it first. */ /* If a binfmt changed the interp, free it first. */
...@@ -1848,18 +1856,19 @@ static int do_execveat_common(int fd, struct filename *filename, ...@@ -1848,18 +1856,19 @@ static int do_execveat_common(int fd, struct filename *filename,
* further execve() calls fail. */ * further execve() calls fail. */
current->flags &= ~PF_NPROC_EXCEEDED; current->flags &= ~PF_NPROC_EXCEEDED;
retval = unshare_files(&displaced); bprm = alloc_bprm();
if (retval) if (IS_ERR(bprm)) {
retval = PTR_ERR(bprm);
goto out_ret; goto out_ret;
}
retval = -ENOMEM; retval = unshare_files(&displaced);
bprm = kzalloc(sizeof(*bprm), GFP_KERNEL); if (retval)
if (!bprm) goto out_free;
goto out_files;
retval = prepare_bprm_creds(bprm); retval = prepare_bprm_creds(bprm);
if (retval) if (retval)
goto out_free; goto out_files;
check_unsafe_exec(bprm); check_unsafe_exec(bprm);
current->in_execve = 1; current->in_execve = 1;
...@@ -1956,13 +1965,13 @@ static int do_execveat_common(int fd, struct filename *filename, ...@@ -1956,13 +1965,13 @@ static int do_execveat_common(int fd, struct filename *filename,
current->fs->in_exec = 0; current->fs->in_exec = 0;
current->in_execve = 0; current->in_execve = 0;
out_files:
if (displaced)
reset_files_struct(displaced);
out_free: out_free:
free_bprm(bprm); free_bprm(bprm);
kfree(pathbuf); kfree(pathbuf);
out_files:
if (displaced)
reset_files_struct(displaced);
out_ret: out_ret:
putname(filename); putname(filename);
return retval; return retval;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册