From: Tejun Heo <tj@kernel.org>
To: Oleg Nesterov <oleg@redhat.com>
Cc: roland@redhat.com, linux-kernel@vger.kernel.org,
torvalds@linux-foundation.org, akpm@linux-foundation.org,
rjw@sisk.pl, jan.kratochvil@redhat.com
Subject: Re: [PATCH 14/16] ptrace: make SIGCONT notification reliable against ptrace
Date: Wed, 22 Dec 2010 11:35:22 +0100 [thread overview]
Message-ID: <20101222103522.GC4684@htj.dyndns.org> (raw)
In-Reply-To: <20101221172516.GA16681@redhat.com>
Hello, Oleg.
On Tue, Dec 21, 2010 at 06:25:16PM +0100, Oleg Nesterov wrote:
> Two threads, T1 and T2. T1 is ptraced, T2 is not.
>
> SIGSTOP stops them both. T1 sleeps in TASK_TRACED, T2 in TASK_STOPPED.
>
> prepare_signal(SIGCONT) sets SIGNAL_NOTIFY_CONT + SIGNAL_CLD_CONTINUED,
> and wakes T2 up.
>
> T2 notifies its ->real_parent, clears SIGNAL_NOTIFY_CONT.
>
> Debugger does ptrace(PTRACE_DETACH, T1), sees SIGNAL_CLD_MASK, and
> restores SIGNAL_NOTIFY_CONT.
>
> T1 resends the (bogus) notification to its (and T2's) real_parent.
You're right. Any thread which notifies the real parent should clear
the pending status.
> Even if I missed something,
>
> > @@ -245,6 +273,14 @@ int ptrace_attach(struct task_struct *task)
> > signal_wake_up(task, 1);
> > }
> >
> > + /*
> > + * Clear SIGNAL_CLD_MASK if NOTIFY_CONT is not set. This is
> > + * used to preserve SIGCONT notification across ptrace
> > + * attach/detach. Read the comment in __ptrace_unlink().
> > + */
> > + if (!(task->signal->flags & SIGNAL_NOTIFY_CONT))
> > + task->signal->flags &= ~SIGNAL_CLD_MASK;
>
> What if there is another ptraced sub-thread in this group who "owes"
> the notification ?
Okay, hmmm... yeah, this is problematic. Man, I suck.
Combined with the previously pointed out issue with indeterministic
behavior regarding stop notification, maybe it's better to reimplement
something which blocks both stop and cont notifications while any
thread in the group is ptraced and reissues them when all detach? So
that we can get both transparent and more predictable behavior?
> > + * Force the tracee into signal delivery path so that
> > + * the notification is delievered ASAP. This wakeup
> > + * is unintrusive as SIGCONT delivery would have
> > + * caused the same effect.
> > + */
> > + if (!woken_up)
> > + signal_wake_up(child, 0);
>
> Well, signal_wake_up() can really force the tracee into signal delivery.
> It only sets TIF_SIGPENDING, but this can race with recalc_sigpending().
Indeed.
> Oh. This reminds me: http://marc.info/?t=123411921400004
>
> > @@ -1639,7 +1642,24 @@ static void do_notify_parent_cldstop(struct task_struct *tsk, int why)
> >
> > switch (why) {
> > case CLD_CONTINUED:
> > - notify = why;
> > + /*
> > + * Notify once if NOTIFY_CONT is set regardless of ptrace.
> > + * NOTIFY_CONT will be reinstated on detach if necessary.
> > + */
> > + if (!(sig->flags & SIGNAL_NOTIFY_CONT))
> > + break;
> > +
> > + /*
> > + * If ptraced, always report CLD_CONTINUED; otherwise,
> > + * prepare_signal(SIGCONT) encodes the CLD_ si_code into
> > + * SIGNAL_CLD_MASK bits.
> > + */
> > + if (task_ptrace(tsk) || (sig->flags & SIGNAL_CLD_CONTINUED))
> > + notify = CLD_CONTINUED;
>
> See the comment on 4/16
Will update.
> > @@ -2015,31 +2035,18 @@ relock:
> > */
> > try_to_freeze();
> >
> > - spin_lock_irq(&sighand->siglock);
> > /*
> > - * Every stopped thread goes here after wakeup. Check to see if
> > - * we should notify the parent, prepare_signal(SIGCONT) encodes
> > - * the CLD_ si_code into SIGNAL_CLD_MASK bits.
> > + * Every stopped thread should go through this function after
> > + * waking up. Check to see if we should notify the parent.
> > */
> > - if (unlikely(signal->flags & SIGNAL_CLD_MASK)) {
> > - int why;
> > -
> > - if (task_ptrace(current) ||
> > - (signal->flags & SIGNAL_CLD_CONTINUED))
> > - why = CLD_CONTINUED;
> > - else
> > - why = CLD_STOPPED;
> > -
> > - signal->flags &= ~SIGNAL_CLD_MASK;
> > -
> > - spin_unlock_irq(&sighand->siglock);
> > -
> > + if (unlikely(current->signal->flags & SIGNAL_NOTIFY_CONT)) {
>
> I am not sure it is OK to check SIGNAL_NOTIFY_CONT without ->siglock.
> If we return from do_signal_stop(), everything is fine.
>
> But if we got here because of __ptrace_unlink()->signal_wake_up(1),
> we can miss SIGNAL_NOTIFY_CONT.
This probably should be resolved together with the above
recalc_sigpending() issue. I'll think about it.
Thank you.
--
tejun
next prev parent reply other threads:[~2010-12-22 10:35 UTC|newest]
Thread overview: 62+ messages / expand[flat|nested] mbox.gz Atom feed top
2010-12-06 16:56 [PATCHSET] ptrace,signal: sane interaction between ptrace and job control signals, take#2 Tejun Heo
2010-12-06 16:56 ` [PATCH 01/16] signal: fix SIGCONT notification code Tejun Heo
2010-12-06 16:56 ` [PATCH 02/16] signal: fix CLD_CONTINUED notification target Tejun Heo
2010-12-20 14:58 ` Oleg Nesterov
2010-12-21 16:31 ` Tejun Heo
2010-12-06 16:56 ` [PATCH 03/16] signal: remove superflous try_to_freeze() loop in do_signal_stop() Tejun Heo
2010-12-20 14:59 ` Oleg Nesterov
2010-12-06 16:56 ` [PATCH 04/16] ptrace: kill tracehook_notify_jctl() Tejun Heo
2010-12-20 14:59 ` Oleg Nesterov
2010-12-21 17:00 ` Tejun Heo
2010-12-06 16:56 ` [PATCH 05/16] ptrace: add @why to ptrace_stop() Tejun Heo
2010-12-06 16:56 ` [PATCH 06/16] signal: fix premature completion of group stop when interfered by ptrace Tejun Heo
2010-12-20 15:00 ` Oleg Nesterov
2010-12-21 17:04 ` Tejun Heo
2010-12-06 16:56 ` [PATCH 07/16] signal: use GROUP_STOP_PENDING to stop once for a single group stop Tejun Heo
2010-12-06 16:56 ` [PATCH 08/16] ptrace: participate in group stop from ptrace_stop() iff the task is trapping for " Tejun Heo
2010-12-06 16:56 ` [PATCH 09/16] ptrace: make do_signal_stop() use ptrace_stop() if the task is being ptraced Tejun Heo
2010-12-23 12:26 ` Oleg Nesterov
2010-12-23 13:53 ` Tejun Heo
2010-12-23 16:06 ` Oleg Nesterov
2010-12-23 16:33 ` Tejun Heo
2011-01-17 22:09 ` Roland McGrath
2011-01-27 13:56 ` Tejun Heo
2011-01-28 20:30 ` Roland McGrath
2011-01-31 14:39 ` Tejun Heo
2010-12-06 16:56 ` [PATCH 10/16] ptrace: clean transitions between TASK_STOPPED and TRACED Tejun Heo
2010-12-20 15:00 ` Oleg Nesterov
2010-12-21 17:31 ` Tejun Heo
2010-12-21 17:32 ` Tejun Heo
2010-12-22 10:54 ` Tejun Heo
2010-12-22 11:39 ` Oleg Nesterov
2010-12-22 15:14 ` Tejun Heo
2010-12-22 16:00 ` Oleg Nesterov
2010-12-22 16:21 ` Tejun Heo
2010-12-06 16:56 ` [PATCH 11/16] signal: prepare for CLD_* notification changes Tejun Heo
2010-12-20 16:21 ` Oleg Nesterov
2010-12-20 16:23 ` Oleg Nesterov
2010-12-21 17:35 ` Tejun Heo
2010-12-06 16:57 ` [PATCH 12/16] ptrace: make group stop notification reliable against ptrace Tejun Heo
2010-12-20 17:34 ` Oleg Nesterov
2010-12-21 17:43 ` Tejun Heo
2010-12-22 11:54 ` Oleg Nesterov
2010-12-22 15:26 ` Tejun Heo
2010-12-22 16:02 ` Oleg Nesterov
2010-12-06 16:57 ` [PATCH 13/16] ptrace: reorganize __ptrace_unlink() and ptrace_untrace() Tejun Heo
2010-12-20 18:15 ` Oleg Nesterov
2010-12-21 17:54 ` Tejun Heo
2010-12-06 16:57 ` [PATCH 14/16] ptrace: make SIGCONT notification reliable against ptrace Tejun Heo
2010-12-20 19:43 ` Oleg Nesterov
2010-12-21 17:48 ` Tejun Heo
2010-12-22 12:16 ` Oleg Nesterov
2010-12-21 17:25 ` Oleg Nesterov
2010-12-22 10:35 ` Tejun Heo [this message]
2010-12-06 16:57 ` [PATCH 15/16] ptrace: make sure SIGNAL_NOTIFY_CONT is checked after ptrace_signal() Tejun Heo
2010-12-06 16:57 ` [PATCH 16/16] ptrace: remove the extra wake_up_process() from ptrace_detach() Tejun Heo
2010-12-07 0:10 ` Roland McGrath
2010-12-07 13:43 ` Tejun Heo
2010-12-21 17:54 ` Oleg Nesterov
2010-12-22 10:36 ` Tejun Heo
2010-12-14 17:36 ` [PATCHSET] ptrace,signal: sane interaction between ptrace and job control signals, take#2 Oleg Nesterov
2010-12-14 17:46 ` Tejun Heo
2010-12-22 15:20 ` Oleg Nesterov
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20101222103522.GC4684@htj.dyndns.org \
--to=tj@kernel.org \
--cc=akpm@linux-foundation.org \
--cc=jan.kratochvil@redhat.com \
--cc=linux-kernel@vger.kernel.org \
--cc=oleg@redhat.com \
--cc=rjw@sisk.pl \
--cc=roland@redhat.com \
--cc=torvalds@linux-foundation.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.