diff options
Diffstat (limited to 'kernel')
| -rw-r--r-- | kernel/exit.c | 2 | ||||
| -rw-r--r-- | kernel/signal.c | 27 |
2 files changed, 27 insertions, 2 deletions
diff --git a/kernel/exit.c b/kernel/exit.c index d7815f57088..8f3bf53a5b4 100644 --- a/kernel/exit.c +++ b/kernel/exit.c | |||
| @@ -947,7 +947,7 @@ fastcall NORET_TYPE void do_exit(long code) | |||
| 947 | schedule(); | 947 | schedule(); |
| 948 | } | 948 | } |
| 949 | 949 | ||
| 950 | tsk->flags |= PF_EXITING; | 950 | exit_signals(tsk); /* sets PF_EXITING */ |
| 951 | /* | 951 | /* |
| 952 | * tsk->flags are checked in the futex code to protect against | 952 | * tsk->flags are checked in the futex code to protect against |
| 953 | * an exiting task cleaning up the robust pi futexes. | 953 | * an exiting task cleaning up the robust pi futexes. |
diff --git a/kernel/signal.c b/kernel/signal.c index 3fca710a5cd..209eec11eef 100644 --- a/kernel/signal.c +++ b/kernel/signal.c | |||
| @@ -1739,7 +1739,7 @@ static int do_signal_stop(int signr) | |||
| 1739 | * stop is always done with the siglock held, | 1739 | * stop is always done with the siglock held, |
| 1740 | * so this check has no races. | 1740 | * so this check has no races. |
| 1741 | */ | 1741 | */ |
| 1742 | if (!t->exit_state && | 1742 | if (!(t->flags & PF_EXITING) && |
| 1743 | !task_is_stopped_or_traced(t)) { | 1743 | !task_is_stopped_or_traced(t)) { |
| 1744 | stop_count++; | 1744 | stop_count++; |
| 1745 | signal_wake_up(t, 0); | 1745 | signal_wake_up(t, 0); |
| @@ -1900,6 +1900,31 @@ relock: | |||
| 1900 | return signr; | 1900 | return signr; |
| 1901 | } | 1901 | } |
| 1902 | 1902 | ||
| 1903 | void exit_signals(struct task_struct *tsk) | ||
| 1904 | { | ||
| 1905 | int group_stop = 0; | ||
| 1906 | |||
| 1907 | spin_lock_irq(&tsk->sighand->siglock); | ||
| 1908 | if (unlikely(tsk->signal->group_stop_count) && | ||
| 1909 | !--tsk->signal->group_stop_count) { | ||
| 1910 | tsk->signal->flags = SIGNAL_STOP_STOPPED; | ||
| 1911 | group_stop = 1; | ||
| 1912 | } | ||
| 1913 | |||
| 1914 | /* | ||
| 1915 | * From now this task is not visible for group-wide signals, | ||
| 1916 | * see wants_signal(), do_signal_stop(). | ||
| 1917 | */ | ||
| 1918 | tsk->flags |= PF_EXITING; | ||
| 1919 | spin_unlock_irq(&tsk->sighand->siglock); | ||
| 1920 | |||
| 1921 | if (unlikely(group_stop)) { | ||
| 1922 | read_lock(&tasklist_lock); | ||
| 1923 | do_notify_parent_cldstop(tsk, CLD_STOPPED); | ||
| 1924 | read_unlock(&tasklist_lock); | ||
| 1925 | } | ||
| 1926 | } | ||
| 1927 | |||
| 1903 | EXPORT_SYMBOL(recalc_sigpending); | 1928 | EXPORT_SYMBOL(recalc_sigpending); |
| 1904 | EXPORT_SYMBOL_GPL(dequeue_signal); | 1929 | EXPORT_SYMBOL_GPL(dequeue_signal); |
| 1905 | EXPORT_SYMBOL(flush_signals); | 1930 | EXPORT_SYMBOL(flush_signals); |
