aboutsummaryrefslogtreecommitdiffstats
path: root/kernel
diff options
context:
space:
mode:
Diffstat (limited to 'kernel')
-rw-r--r--kernel/signal.c38
1 files changed, 24 insertions, 14 deletions
diff --git a/kernel/signal.c b/kernel/signal.c
index b0b1ca9daa33..819fa49aa70a 100644
--- a/kernel/signal.c
+++ b/kernel/signal.c
@@ -1120,27 +1120,37 @@ void zap_other_threads(struct task_struct *p)
1120/* 1120/*
1121 * Must be called under rcu_read_lock() or with tasklist_lock read-held. 1121 * Must be called under rcu_read_lock() or with tasklist_lock read-held.
1122 */ 1122 */
1123struct sighand_struct *lock_task_sighand(struct task_struct *tsk, unsigned long *flags)
1124{
1125 struct sighand_struct *sighand;
1126
1127 for (;;) {
1128 sighand = rcu_dereference(tsk->sighand);
1129 if (unlikely(sighand == NULL))
1130 break;
1131
1132 spin_lock_irqsave(&sighand->siglock, *flags);
1133 if (likely(sighand == tsk->sighand))
1134 break;
1135 spin_unlock_irqrestore(&sighand->siglock, *flags);
1136 }
1137
1138 return sighand;
1139}
1140
1123int group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p) 1141int group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p)
1124{ 1142{
1125 unsigned long flags; 1143 unsigned long flags;
1126 struct sighand_struct *sp;
1127 int ret; 1144 int ret;
1128 1145
1129retry:
1130 ret = check_kill_permission(sig, info, p); 1146 ret = check_kill_permission(sig, info, p);
1131 if (!ret && sig && (sp = rcu_dereference(p->sighand))) { 1147
1132 spin_lock_irqsave(&sp->siglock, flags); 1148 if (!ret && sig) {
1133 if (p->sighand != sp) { 1149 ret = -ESRCH;
1134 spin_unlock_irqrestore(&sp->siglock, flags); 1150 if (lock_task_sighand(p, &flags)) {
1135 goto retry; 1151 ret = __group_send_sig_info(sig, info, p);
1136 } 1152 unlock_task_sighand(p, &flags);
1137 if ((atomic_read(&sp->count) == 0) ||
1138 (atomic_read(&p->usage) == 0)) {
1139 spin_unlock_irqrestore(&sp->siglock, flags);
1140 return -ESRCH;
1141 } 1153 }
1142 ret = __group_send_sig_info(sig, info, p);
1143 spin_unlock_irqrestore(&sp->siglock, flags);
1144 } 1154 }
1145 1155
1146 return ret; 1156 return ret;