diff options
author | Oleg Nesterov <oleg@redhat.com> | 2014-12-10 18:54:39 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2014-12-10 20:41:16 -0500 |
commit | 7117bc8888aff73fb081956afa501edcc85a1552 (patch) | |
tree | 3d0af7c27c1b62e576aa3d6a59706f224920089d /kernel/kmod.c | |
parent | ddbc22e27e672b6b180757ea1d7f8481dbb88128 (diff) |
usermodehelper: don't use CLONE_VFORK for ____call_usermodehelper()
After "kernel/kmod: fix use-after-free of the sub_infostructure"
CLONE_VFORK in __call_usermodehelper() buys nothing, we rely on on
umh_complete() in ____call_usermodehelper() anyway.
Remove it. This also eliminates the unnecessary sleep/wakeup in the
likely case, and this allows the next change.
While at it, kill the "int wait" locals in ____call_usermodehelper() and
__call_usermodehelper(), they can safely use sub_info->wait.
Signed-off-by: Oleg Nesterov <oleg@redhat.com>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'kernel/kmod.c')
-rw-r--r-- | kernel/kmod.c | 12 |
1 files changed, 3 insertions, 9 deletions
diff --git a/kernel/kmod.c b/kernel/kmod.c index 80f7a6d00519..4621771b43b9 100644 --- a/kernel/kmod.c +++ b/kernel/kmod.c | |||
@@ -223,7 +223,6 @@ static void umh_complete(struct subprocess_info *sub_info) | |||
223 | static int ____call_usermodehelper(void *data) | 223 | static int ____call_usermodehelper(void *data) |
224 | { | 224 | { |
225 | struct subprocess_info *sub_info = data; | 225 | struct subprocess_info *sub_info = data; |
226 | int wait = sub_info->wait & ~UMH_KILLABLE; | ||
227 | struct cred *new; | 226 | struct cred *new; |
228 | int retval; | 227 | int retval; |
229 | 228 | ||
@@ -267,7 +266,7 @@ static int ____call_usermodehelper(void *data) | |||
267 | out: | 266 | out: |
268 | sub_info->retval = retval; | 267 | sub_info->retval = retval; |
269 | /* wait_for_helper() will call umh_complete if UHM_WAIT_PROC. */ | 268 | /* wait_for_helper() will call umh_complete if UHM_WAIT_PROC. */ |
270 | if (wait != UMH_WAIT_PROC) | 269 | if (!(sub_info->wait & UMH_WAIT_PROC)) |
271 | umh_complete(sub_info); | 270 | umh_complete(sub_info); |
272 | if (!retval) | 271 | if (!retval) |
273 | return 0; | 272 | return 0; |
@@ -323,18 +322,13 @@ static void __call_usermodehelper(struct work_struct *work) | |||
323 | { | 322 | { |
324 | struct subprocess_info *sub_info = | 323 | struct subprocess_info *sub_info = |
325 | container_of(work, struct subprocess_info, work); | 324 | container_of(work, struct subprocess_info, work); |
326 | int wait = sub_info->wait & ~UMH_KILLABLE; | ||
327 | pid_t pid; | 325 | pid_t pid; |
328 | 326 | ||
329 | /* CLONE_VFORK: wait until the usermode helper has execve'd | 327 | if (sub_info->wait & UMH_WAIT_PROC) |
330 | * successfully We need the data structures to stay around | ||
331 | * until that is done. */ | ||
332 | if (wait == UMH_WAIT_PROC) | ||
333 | pid = kernel_thread(wait_for_helper, sub_info, | 328 | pid = kernel_thread(wait_for_helper, sub_info, |
334 | CLONE_FS | CLONE_FILES | SIGCHLD); | 329 | CLONE_FS | CLONE_FILES | SIGCHLD); |
335 | else { | 330 | else { |
336 | pid = kernel_thread(call_helper, sub_info, | 331 | pid = kernel_thread(call_helper, sub_info, SIGCHLD); |
337 | CLONE_VFORK | SIGCHLD); | ||
338 | /* Worker thread stopped blocking khelper thread. */ | 332 | /* Worker thread stopped blocking khelper thread. */ |
339 | kmod_thread_locker = NULL; | 333 | kmod_thread_locker = NULL; |
340 | } | 334 | } |