forked from Minki/linux
signal: introduce kernel_signal_stop() to fix jffs2_garbage_collect_thread()
jffs2_garbage_collect_thread() can race with SIGCONT and sleep in TASK_STOPPED state after it was already sent. Add the new helper, kernel_signal_stop(), which does this correctly. Signed-off-by: Oleg Nesterov <oleg@redhat.com> Reviewed-by: Tejun Heo <tj@kernel.org> Cc: David Woodhouse <dwmw2@infradead.org> Cc: Felipe Balbi <balbi@ti.com> Cc: Markus Pargmann <mpa@pengutronix.de> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
be0e6f290f
commit
9a13049e83
@ -132,8 +132,7 @@ static int jffs2_garbage_collect_thread(void *_c)
|
||||
case SIGSTOP:
|
||||
jffs2_dbg(1, "%s(): SIGSTOP received\n",
|
||||
__func__);
|
||||
set_current_state(TASK_STOPPED);
|
||||
schedule();
|
||||
kernel_signal_stop();
|
||||
break;
|
||||
|
||||
case SIGKILL:
|
||||
|
@ -2475,6 +2475,16 @@ static inline int kernel_dequeue_signal(siginfo_t *info)
|
||||
return ret;
|
||||
}
|
||||
|
||||
static inline void kernel_signal_stop(void)
|
||||
{
|
||||
spin_lock_irq(¤t->sighand->siglock);
|
||||
if (current->jobctl & JOBCTL_STOP_DEQUEUED)
|
||||
__set_current_state(TASK_STOPPED);
|
||||
spin_unlock_irq(¤t->sighand->siglock);
|
||||
|
||||
schedule();
|
||||
}
|
||||
|
||||
extern void release_task(struct task_struct * p);
|
||||
extern int send_sig_info(int, struct siginfo *, struct task_struct *);
|
||||
extern int force_sigsegv(int, struct task_struct *);
|
||||
|
Loading…
Reference in New Issue
Block a user