aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/signal.c
diff options
context:
space:
mode:
Diffstat (limited to 'kernel/signal.c')
-rw-r--r--kernel/signal.c31
1 files changed, 24 insertions, 7 deletions
diff --git a/kernel/signal.c b/kernel/signal.c
index 2f2c8f6ee01f..69d60540a680 100644
--- a/kernel/signal.c
+++ b/kernel/signal.c
@@ -1595,16 +1595,30 @@ int do_notify_parent(struct task_struct *tsk, int sig)
1595 return ret; 1595 return ret;
1596} 1596}
1597 1597
1598static void do_notify_parent_cldstop(struct task_struct *tsk, int why) 1598/**
1599 * do_notify_parent_cldstop - notify parent of stopped/continued state change
1600 * @tsk: task reporting the state change
1601 * @for_ptracer: the notification is for ptracer
1602 * @why: CLD_{CONTINUED|STOPPED|TRAPPED} to report
1603 *
1604 * Notify @tsk's parent that the stopped/continued state has changed. If
1605 * @for_ptracer is %false, @tsk's group leader notifies to its real parent.
1606 * If %true, @tsk reports to @tsk->parent which should be the ptracer.
1607 *
1608 * CONTEXT:
1609 * Must be called with tasklist_lock at least read locked.
1610 */
1611static void do_notify_parent_cldstop(struct task_struct *tsk,
1612 bool for_ptracer, int why)
1599{ 1613{
1600 struct siginfo info; 1614 struct siginfo info;
1601 unsigned long flags; 1615 unsigned long flags;
1602 struct task_struct *parent; 1616 struct task_struct *parent;
1603 struct sighand_struct *sighand; 1617 struct sighand_struct *sighand;
1604 1618
1605 if (task_ptrace(tsk)) 1619 if (for_ptracer) {
1606 parent = tsk->parent; 1620 parent = tsk->parent;
1607 else { 1621 } else {
1608 tsk = tsk->group_leader; 1622 tsk = tsk->group_leader;
1609 parent = tsk->real_parent; 1623 parent = tsk->real_parent;
1610 } 1624 }
@@ -1743,7 +1757,7 @@ static void ptrace_stop(int exit_code, int why, int clear_code, siginfo_t *info)
1743 spin_unlock_irq(&current->sighand->siglock); 1757 spin_unlock_irq(&current->sighand->siglock);
1744 read_lock(&tasklist_lock); 1758 read_lock(&tasklist_lock);
1745 if (may_ptrace_stop()) { 1759 if (may_ptrace_stop()) {
1746 do_notify_parent_cldstop(current, why); 1760 do_notify_parent_cldstop(current, task_ptrace(current), why);
1747 /* 1761 /*
1748 * Don't want to allow preemption here, because 1762 * Don't want to allow preemption here, because
1749 * sys_ptrace() needs this task to be inactive. 1763 * sys_ptrace() needs this task to be inactive.
@@ -1886,7 +1900,8 @@ retry:
1886 1900
1887 if (notify) { 1901 if (notify) {
1888 read_lock(&tasklist_lock); 1902 read_lock(&tasklist_lock);
1889 do_notify_parent_cldstop(current, notify); 1903 do_notify_parent_cldstop(current, task_ptrace(current),
1904 notify);
1890 read_unlock(&tasklist_lock); 1905 read_unlock(&tasklist_lock);
1891 } 1906 }
1892 1907
@@ -1982,6 +1997,7 @@ relock:
1982 * the CLD_ si_code into SIGNAL_CLD_MASK bits. 1997 * the CLD_ si_code into SIGNAL_CLD_MASK bits.
1983 */ 1998 */
1984 if (unlikely(signal->flags & SIGNAL_CLD_MASK)) { 1999 if (unlikely(signal->flags & SIGNAL_CLD_MASK)) {
2000 struct task_struct *leader;
1985 int why; 2001 int why;
1986 2002
1987 if (signal->flags & SIGNAL_CLD_CONTINUED) 2003 if (signal->flags & SIGNAL_CLD_CONTINUED)
@@ -1994,7 +2010,8 @@ relock:
1994 spin_unlock_irq(&sighand->siglock); 2010 spin_unlock_irq(&sighand->siglock);
1995 2011
1996 read_lock(&tasklist_lock); 2012 read_lock(&tasklist_lock);
1997 do_notify_parent_cldstop(current->group_leader, why); 2013 leader = current->group_leader;
2014 do_notify_parent_cldstop(leader, task_ptrace(leader), why);
1998 read_unlock(&tasklist_lock); 2015 read_unlock(&tasklist_lock);
1999 goto relock; 2016 goto relock;
2000 } 2017 }
@@ -2167,7 +2184,7 @@ out:
2167 2184
2168 if (unlikely(group_stop)) { 2185 if (unlikely(group_stop)) {
2169 read_lock(&tasklist_lock); 2186 read_lock(&tasklist_lock);
2170 do_notify_parent_cldstop(tsk, group_stop); 2187 do_notify_parent_cldstop(tsk, task_ptrace(tsk), group_stop);
2171 read_unlock(&tasklist_lock); 2188 read_unlock(&tasklist_lock);
2172 } 2189 }
2173} 2190}