]> git.ipfire.org Git - thirdparty/kernel/stable-queue.git/commitdiff
4.19-stable patches
authorGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 6 Nov 2020 11:23:11 +0000 (12:23 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 6 Nov 2020 11:23:11 +0000 (12:23 +0100)
added patches:
ptrace-fix-task_join_group_stop-for-the-case-when-current-is-traced.patch

queue-4.19/ptrace-fix-task_join_group_stop-for-the-case-when-current-is-traced.patch [new file with mode: 0644]
queue-4.19/series

diff --git a/queue-4.19/ptrace-fix-task_join_group_stop-for-the-case-when-current-is-traced.patch b/queue-4.19/ptrace-fix-task_join_group_stop-for-the-case-when-current-is-traced.patch
new file mode 100644 (file)
index 0000000..cb47e8b
--- /dev/null
@@ -0,0 +1,116 @@
+From 7b3c36fc4c231ca532120bbc0df67a12f09c1d96 Mon Sep 17 00:00:00 2001
+From: Oleg Nesterov <oleg@redhat.com>
+Date: Sun, 1 Nov 2020 17:07:44 -0800
+Subject: ptrace: fix task_join_group_stop() for the case when current is traced
+
+From: Oleg Nesterov <oleg@redhat.com>
+
+commit 7b3c36fc4c231ca532120bbc0df67a12f09c1d96 upstream.
+
+This testcase
+
+       #include <stdio.h>
+       #include <unistd.h>
+       #include <signal.h>
+       #include <sys/ptrace.h>
+       #include <sys/wait.h>
+       #include <pthread.h>
+       #include <assert.h>
+
+       void *tf(void *arg)
+       {
+               return NULL;
+       }
+
+       int main(void)
+       {
+               int pid = fork();
+               if (!pid) {
+                       kill(getpid(), SIGSTOP);
+
+                       pthread_t th;
+                       pthread_create(&th, NULL, tf, NULL);
+
+                       return 0;
+               }
+
+               waitpid(pid, NULL, WSTOPPED);
+
+               ptrace(PTRACE_SEIZE, pid, 0, PTRACE_O_TRACECLONE);
+               waitpid(pid, NULL, 0);
+
+               ptrace(PTRACE_CONT, pid, 0,0);
+               waitpid(pid, NULL, 0);
+
+               int status;
+               int thread = waitpid(-1, &status, 0);
+               assert(thread > 0 && thread != pid);
+               assert(status == 0x80137f);
+
+               return 0;
+       }
+
+fails and triggers WARN_ON_ONCE(!signr) in do_jobctl_trap().
+
+This is because task_join_group_stop() has 2 problems when current is traced:
+
+       1. We can't rely on the "JOBCTL_STOP_PENDING" check, a stopped tracee
+          can be woken up by debugger and it can clone another thread which
+          should join the group-stop.
+
+          We need to check group_stop_count || SIGNAL_STOP_STOPPED.
+
+       2. If SIGNAL_STOP_STOPPED is already set, we should not increment
+          sig->group_stop_count and add JOBCTL_STOP_CONSUME. The new thread
+          should stop without another do_notify_parent_cldstop() report.
+
+To clarify, the problem is very old and we should blame
+ptrace_init_task().  But now that we have task_join_group_stop() it makes
+more sense to fix this helper to avoid the code duplication.
+
+Reported-by: syzbot+3485e3773f7da290eecc@syzkaller.appspotmail.com
+Signed-off-by: Oleg Nesterov <oleg@redhat.com>
+Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
+Cc: Jens Axboe <axboe@kernel.dk>
+Cc: Christian Brauner <christian@brauner.io>
+Cc: "Eric W . Biederman" <ebiederm@xmission.com>
+Cc: Zhiqiang Liu <liuzhiqiang26@huawei.com>
+Cc: Tejun Heo <tj@kernel.org>
+Cc: <stable@vger.kernel.org>
+Link: https://lkml.kernel.org/r/20201019134237.GA18810@redhat.com
+Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
+Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
+
+---
+ kernel/signal.c |   19 ++++++++++---------
+ 1 file changed, 10 insertions(+), 9 deletions(-)
+
+--- a/kernel/signal.c
++++ b/kernel/signal.c
+@@ -385,16 +385,17 @@ static bool task_participate_group_stop(
+ void task_join_group_stop(struct task_struct *task)
+ {
++      unsigned long mask = current->jobctl & JOBCTL_STOP_SIGMASK;
++      struct signal_struct *sig = current->signal;
++
++      if (sig->group_stop_count) {
++              sig->group_stop_count++;
++              mask |= JOBCTL_STOP_CONSUME;
++      } else if (!(sig->flags & SIGNAL_STOP_STOPPED))
++              return;
++
+       /* Have the new thread join an on-going signal group stop */
+-      unsigned long jobctl = current->jobctl;
+-      if (jobctl & JOBCTL_STOP_PENDING) {
+-              struct signal_struct *sig = current->signal;
+-              unsigned long signr = jobctl & JOBCTL_STOP_SIGMASK;
+-              unsigned long gstop = JOBCTL_STOP_PENDING | JOBCTL_STOP_CONSUME;
+-              if (task_set_jobctl_pending(task, signr | gstop)) {
+-                      sig->group_stop_count++;
+-              }
+-      }
++      task_set_jobctl_pending(task, mask | JOBCTL_STOP_PENDING);
+ }
+ /*
index c03008e3df471a697bc5824d691777e3191a2b2f..c01cd731a15ebd8aee01923efc2d0c44c099a006 100644 (file)
@@ -1,2 +1,3 @@
 drm-i915-break-up-error-capture-compression-loops-with-cond_resched.patch
 tipc-fix-use-after-free-in-tipc_bcast_get_mode.patch
+ptrace-fix-task_join_group_stop-for-the-case-when-current-is-traced.patch