aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/kmod.c
diff options
context:
space:
mode:
authorOleg Nesterov <oleg@redhat.com>2014-12-10 18:54:39 -0500
committerLinus Torvalds <torvalds@linux-foundation.org>2014-12-10 20:41:16 -0500
commit7117bc8888aff73fb081956afa501edcc85a1552 (patch)
tree3d0af7c27c1b62e576aa3d6a59706f224920089d /kernel/kmod.c
parentddbc22e27e672b6b180757ea1d7f8481dbb88128 (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.c12
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)
223static int ____call_usermodehelper(void *data) 223static 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)
267out: 266out:
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 }