diff options
author | Oleg Nesterov <oleg@tv-sign.ru> | 2006-03-28 19:11:13 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@g5.osdl.org> | 2006-03-28 21:36:42 -0500 |
commit | f63ee72e0fb82e504a0489490babc7612c7cd6c2 (patch) | |
tree | 85167f8016d1f746135bf1777646a58c44376af1 /kernel/signal.c | |
parent | aa1757f90bea3f598b6e5d04d922a6a60200f1da (diff) |
[PATCH] introduce lock_task_sighand() helper
Add lock_task_sighand() helper and converts group_send_sig_info() to use
it. Hopefully we will have more users soon.
This patch also removes '!sighand->count' and '!p->usage' checks, I think
they both are bogus, racy and unneeded (but probably it makes sense to
restore them as BUG_ON()s).
->sighand is cleared and it's ->count is decremented in release_task() with
sighand->siglock held, so it is a bug to have '!p->usage || !->count' after
we already locked and verified it is the same. On the other hand, an
already dead task without ->sighand can have a non-zero ->usage due to
ptrace, for example.
If we read the stale value of ->sighand we must see the change after
spin_lock(), because that change was done while holding that same old
->sighand.siglock.
Signed-off-by: Oleg Nesterov <oleg@tv-sign.ru>
Cc: "Eric W. Biederman" <ebiederm@xmission.com>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'kernel/signal.c')
-rw-r--r-- | kernel/signal.c | 38 |
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 | */ |
1123 | struct 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 | |||
1123 | int group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p) | 1141 | int 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 | ||
1129 | retry: | ||
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; |