Commit b772434b authored by Oleg Nesterov's avatar Oleg Nesterov Committed by Linus Torvalds

signal: simplify set_user_sigmask/restore_user_sigmask

task->saved_sigmask and ->restore_sigmask are only used in the ret-from-
syscall paths.  This means that set_user_sigmask() can save ->blocked in
->saved_sigmask and do set_restore_sigmask() to indicate that ->blocked
was modified.

This way the callers do not need 2 sigset_t's passed to set/restore and
restore_user_sigmask() renamed to restore_saved_sigmask_unless() turns
into the trivial helper which just calls restore_saved_sigmask().

Link: http://lkml.kernel.org/r/20190606113206.GA9464@redhat.comSigned-off-by: default avatarOleg Nesterov <oleg@redhat.com>
Cc: Deepa Dinamani <deepa.kernel@gmail.com>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: Jens Axboe <axboe@kernel.dk>
Cc: Davidlohr Bueso <dave@stgolabs.net>
Cc: Eric Wong <e@80x24.org>
Cc: Jason Baron <jbaron@akamai.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Al Viro <viro@ZenIV.linux.org.uk>
Cc: Eric W. Biederman <ebiederm@xmission.com>
Cc: David Laight <David.Laight@aculab.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent e2d9018e
...@@ -2094,7 +2094,6 @@ SYSCALL_DEFINE6(io_pgetevents, ...@@ -2094,7 +2094,6 @@ SYSCALL_DEFINE6(io_pgetevents,
const struct __aio_sigset __user *, usig) const struct __aio_sigset __user *, usig)
{ {
struct __aio_sigset ksig = { NULL, }; struct __aio_sigset ksig = { NULL, };
sigset_t ksigmask, sigsaved;
struct timespec64 ts; struct timespec64 ts;
bool interrupted; bool interrupted;
int ret; int ret;
...@@ -2105,14 +2104,14 @@ SYSCALL_DEFINE6(io_pgetevents, ...@@ -2105,14 +2104,14 @@ SYSCALL_DEFINE6(io_pgetevents,
if (usig && copy_from_user(&ksig, usig, sizeof(ksig))) if (usig && copy_from_user(&ksig, usig, sizeof(ksig)))
return -EFAULT; return -EFAULT;
ret = set_user_sigmask(ksig.sigmask, &ksigmask, &sigsaved, ksig.sigsetsize); ret = set_user_sigmask(ksig.sigmask, ksig.sigsetsize);
if (ret) if (ret)
return ret; return ret;
ret = do_io_getevents(ctx_id, min_nr, nr, events, timeout ? &ts : NULL); ret = do_io_getevents(ctx_id, min_nr, nr, events, timeout ? &ts : NULL);
interrupted = signal_pending(current); interrupted = signal_pending(current);
restore_user_sigmask(ksig.sigmask, &sigsaved, interrupted); restore_saved_sigmask_unless(interrupted);
if (interrupted && !ret) if (interrupted && !ret)
ret = -ERESTARTNOHAND; ret = -ERESTARTNOHAND;
...@@ -2130,7 +2129,6 @@ SYSCALL_DEFINE6(io_pgetevents_time32, ...@@ -2130,7 +2129,6 @@ SYSCALL_DEFINE6(io_pgetevents_time32,
const struct __aio_sigset __user *, usig) const struct __aio_sigset __user *, usig)
{ {
struct __aio_sigset ksig = { NULL, }; struct __aio_sigset ksig = { NULL, };
sigset_t ksigmask, sigsaved;
struct timespec64 ts; struct timespec64 ts;
bool interrupted; bool interrupted;
int ret; int ret;
...@@ -2142,14 +2140,14 @@ SYSCALL_DEFINE6(io_pgetevents_time32, ...@@ -2142,14 +2140,14 @@ SYSCALL_DEFINE6(io_pgetevents_time32,
return -EFAULT; return -EFAULT;
ret = set_user_sigmask(ksig.sigmask, &ksigmask, &sigsaved, ksig.sigsetsize); ret = set_user_sigmask(ksig.sigmask, ksig.sigsetsize);
if (ret) if (ret)
return ret; return ret;
ret = do_io_getevents(ctx_id, min_nr, nr, events, timeout ? &ts : NULL); ret = do_io_getevents(ctx_id, min_nr, nr, events, timeout ? &ts : NULL);
interrupted = signal_pending(current); interrupted = signal_pending(current);
restore_user_sigmask(ksig.sigmask, &sigsaved, interrupted); restore_saved_sigmask_unless(interrupted);
if (interrupted && !ret) if (interrupted && !ret)
ret = -ERESTARTNOHAND; ret = -ERESTARTNOHAND;
...@@ -2198,7 +2196,6 @@ COMPAT_SYSCALL_DEFINE6(io_pgetevents, ...@@ -2198,7 +2196,6 @@ COMPAT_SYSCALL_DEFINE6(io_pgetevents,
const struct __compat_aio_sigset __user *, usig) const struct __compat_aio_sigset __user *, usig)
{ {
struct __compat_aio_sigset ksig = { NULL, }; struct __compat_aio_sigset ksig = { NULL, };
sigset_t ksigmask, sigsaved;
struct timespec64 t; struct timespec64 t;
bool interrupted; bool interrupted;
int ret; int ret;
...@@ -2209,14 +2206,14 @@ COMPAT_SYSCALL_DEFINE6(io_pgetevents, ...@@ -2209,14 +2206,14 @@ COMPAT_SYSCALL_DEFINE6(io_pgetevents,
if (usig && copy_from_user(&ksig, usig, sizeof(ksig))) if (usig && copy_from_user(&ksig, usig, sizeof(ksig)))
return -EFAULT; return -EFAULT;
ret = set_compat_user_sigmask(ksig.sigmask, &ksigmask, &sigsaved, ksig.sigsetsize); ret = set_compat_user_sigmask(ksig.sigmask, ksig.sigsetsize);
if (ret) if (ret)
return ret; return ret;
ret = do_io_getevents(ctx_id, min_nr, nr, events, timeout ? &t : NULL); ret = do_io_getevents(ctx_id, min_nr, nr, events, timeout ? &t : NULL);
interrupted = signal_pending(current); interrupted = signal_pending(current);
restore_user_sigmask(ksig.sigmask, &sigsaved, interrupted); restore_saved_sigmask_unless(interrupted);
if (interrupted && !ret) if (interrupted && !ret)
ret = -ERESTARTNOHAND; ret = -ERESTARTNOHAND;
...@@ -2234,7 +2231,6 @@ COMPAT_SYSCALL_DEFINE6(io_pgetevents_time64, ...@@ -2234,7 +2231,6 @@ COMPAT_SYSCALL_DEFINE6(io_pgetevents_time64,
const struct __compat_aio_sigset __user *, usig) const struct __compat_aio_sigset __user *, usig)
{ {
struct __compat_aio_sigset ksig = { NULL, }; struct __compat_aio_sigset ksig = { NULL, };
sigset_t ksigmask, sigsaved;
struct timespec64 t; struct timespec64 t;
bool interrupted; bool interrupted;
int ret; int ret;
...@@ -2245,14 +2241,14 @@ COMPAT_SYSCALL_DEFINE6(io_pgetevents_time64, ...@@ -2245,14 +2241,14 @@ COMPAT_SYSCALL_DEFINE6(io_pgetevents_time64,
if (usig && copy_from_user(&ksig, usig, sizeof(ksig))) if (usig && copy_from_user(&ksig, usig, sizeof(ksig)))
return -EFAULT; return -EFAULT;
ret = set_compat_user_sigmask(ksig.sigmask, &ksigmask, &sigsaved, ksig.sigsetsize); ret = set_compat_user_sigmask(ksig.sigmask, ksig.sigsetsize);
if (ret) if (ret)
return ret; return ret;
ret = do_io_getevents(ctx_id, min_nr, nr, events, timeout ? &t : NULL); ret = do_io_getevents(ctx_id, min_nr, nr, events, timeout ? &t : NULL);
interrupted = signal_pending(current); interrupted = signal_pending(current);
restore_user_sigmask(ksig.sigmask, &sigsaved, interrupted); restore_saved_sigmask_unless(interrupted);
if (interrupted && !ret) if (interrupted && !ret)
ret = -ERESTARTNOHAND; ret = -ERESTARTNOHAND;
......
...@@ -2313,19 +2313,17 @@ SYSCALL_DEFINE6(epoll_pwait, int, epfd, struct epoll_event __user *, events, ...@@ -2313,19 +2313,17 @@ SYSCALL_DEFINE6(epoll_pwait, int, epfd, struct epoll_event __user *, events,
size_t, sigsetsize) size_t, sigsetsize)
{ {
int error; int error;
sigset_t ksigmask, sigsaved;
/* /*
* If the caller wants a certain signal mask to be set during the wait, * If the caller wants a certain signal mask to be set during the wait,
* we apply it here. * we apply it here.
*/ */
error = set_user_sigmask(sigmask, &ksigmask, &sigsaved, sigsetsize); error = set_user_sigmask(sigmask, sigsetsize);
if (error) if (error)
return error; return error;
error = do_epoll_wait(epfd, events, maxevents, timeout); error = do_epoll_wait(epfd, events, maxevents, timeout);
restore_saved_sigmask_unless(error == -EINTR);
restore_user_sigmask(sigmask, &sigsaved, error == -EINTR);
return error; return error;
} }
...@@ -2338,19 +2336,17 @@ COMPAT_SYSCALL_DEFINE6(epoll_pwait, int, epfd, ...@@ -2338,19 +2336,17 @@ COMPAT_SYSCALL_DEFINE6(epoll_pwait, int, epfd,
compat_size_t, sigsetsize) compat_size_t, sigsetsize)
{ {
long err; long err;
sigset_t ksigmask, sigsaved;
/* /*
* If the caller wants a certain signal mask to be set during the wait, * If the caller wants a certain signal mask to be set during the wait,
* we apply it here. * we apply it here.
*/ */
err = set_compat_user_sigmask(sigmask, &ksigmask, &sigsaved, sigsetsize); err = set_compat_user_sigmask(sigmask, sigsetsize);
if (err) if (err)
return err; return err;
err = do_epoll_wait(epfd, events, maxevents, timeout); err = do_epoll_wait(epfd, events, maxevents, timeout);
restore_saved_sigmask_unless(err == -EINTR);
restore_user_sigmask(sigmask, &sigsaved, err == -EINTR);
return err; return err;
} }
......
...@@ -2400,7 +2400,6 @@ static int io_cqring_wait(struct io_ring_ctx *ctx, int min_events, ...@@ -2400,7 +2400,6 @@ static int io_cqring_wait(struct io_ring_ctx *ctx, int min_events,
const sigset_t __user *sig, size_t sigsz) const sigset_t __user *sig, size_t sigsz)
{ {
struct io_cq_ring *ring = ctx->cq_ring; struct io_cq_ring *ring = ctx->cq_ring;
sigset_t ksigmask, sigsaved;
int ret; int ret;
if (io_cqring_events(ring) >= min_events) if (io_cqring_events(ring) >= min_events)
...@@ -2410,21 +2409,17 @@ static int io_cqring_wait(struct io_ring_ctx *ctx, int min_events, ...@@ -2410,21 +2409,17 @@ static int io_cqring_wait(struct io_ring_ctx *ctx, int min_events,
#ifdef CONFIG_COMPAT #ifdef CONFIG_COMPAT
if (in_compat_syscall()) if (in_compat_syscall())
ret = set_compat_user_sigmask((const compat_sigset_t __user *)sig, ret = set_compat_user_sigmask((const compat_sigset_t __user *)sig,
&ksigmask, &sigsaved, sigsz); sigsz);
else else
#endif #endif
ret = set_user_sigmask(sig, &ksigmask, ret = set_user_sigmask(sig, sigsz);
&sigsaved, sigsz);
if (ret) if (ret)
return ret; return ret;
} }
ret = wait_event_interruptible(ctx->wait, io_cqring_events(ring) >= min_events); ret = wait_event_interruptible(ctx->wait, io_cqring_events(ring) >= min_events);
restore_saved_sigmask_unless(ret == -ERESTARTSYS);
if (sig)
restore_user_sigmask(sig, &sigsaved, ret == -ERESTARTSYS);
if (ret == -ERESTARTSYS) if (ret == -ERESTARTSYS)
ret = -EINTR; ret = -EINTR;
......
...@@ -730,7 +730,6 @@ static long do_pselect(int n, fd_set __user *inp, fd_set __user *outp, ...@@ -730,7 +730,6 @@ static long do_pselect(int n, fd_set __user *inp, fd_set __user *outp,
const sigset_t __user *sigmask, size_t sigsetsize, const sigset_t __user *sigmask, size_t sigsetsize,
enum poll_time_type type) enum poll_time_type type)
{ {
sigset_t ksigmask, sigsaved;
struct timespec64 ts, end_time, *to = NULL; struct timespec64 ts, end_time, *to = NULL;
int ret; int ret;
...@@ -753,12 +752,12 @@ static long do_pselect(int n, fd_set __user *inp, fd_set __user *outp, ...@@ -753,12 +752,12 @@ static long do_pselect(int n, fd_set __user *inp, fd_set __user *outp,
return -EINVAL; return -EINVAL;
} }
ret = set_user_sigmask(sigmask, &ksigmask, &sigsaved, sigsetsize); ret = set_user_sigmask(sigmask, sigsetsize);
if (ret) if (ret)
return ret; return ret;
ret = core_sys_select(n, inp, outp, exp, to); ret = core_sys_select(n, inp, outp, exp, to);
restore_user_sigmask(sigmask, &sigsaved, ret == -ERESTARTNOHAND); restore_saved_sigmask_unless(ret == -ERESTARTNOHAND);
ret = poll_select_copy_remaining(&end_time, tsp, type, ret); ret = poll_select_copy_remaining(&end_time, tsp, type, ret);
return ret; return ret;
...@@ -1086,7 +1085,6 @@ SYSCALL_DEFINE5(ppoll, struct pollfd __user *, ufds, unsigned int, nfds, ...@@ -1086,7 +1085,6 @@ SYSCALL_DEFINE5(ppoll, struct pollfd __user *, ufds, unsigned int, nfds,
struct __kernel_timespec __user *, tsp, const sigset_t __user *, sigmask, struct __kernel_timespec __user *, tsp, const sigset_t __user *, sigmask,
size_t, sigsetsize) size_t, sigsetsize)
{ {
sigset_t ksigmask, sigsaved;
struct timespec64 ts, end_time, *to = NULL; struct timespec64 ts, end_time, *to = NULL;
int ret; int ret;
...@@ -1099,17 +1097,16 @@ SYSCALL_DEFINE5(ppoll, struct pollfd __user *, ufds, unsigned int, nfds, ...@@ -1099,17 +1097,16 @@ SYSCALL_DEFINE5(ppoll, struct pollfd __user *, ufds, unsigned int, nfds,
return -EINVAL; return -EINVAL;
} }
ret = set_user_sigmask(sigmask, &ksigmask, &sigsaved, sigsetsize); ret = set_user_sigmask(sigmask, sigsetsize);
if (ret) if (ret)
return ret; return ret;
ret = do_sys_poll(ufds, nfds, to); ret = do_sys_poll(ufds, nfds, to);
restore_user_sigmask(sigmask, &sigsaved, ret == -EINTR); restore_saved_sigmask_unless(ret == -EINTR);
/* We can restart this syscall, usually */ /* We can restart this syscall, usually */
if (ret == -EINTR) if (ret == -EINTR)
ret = -ERESTARTNOHAND; ret = -ERESTARTNOHAND;
ret = poll_select_copy_remaining(&end_time, tsp, PT_TIMESPEC, ret); ret = poll_select_copy_remaining(&end_time, tsp, PT_TIMESPEC, ret);
return ret; return ret;
...@@ -1121,7 +1118,6 @@ SYSCALL_DEFINE5(ppoll_time32, struct pollfd __user *, ufds, unsigned int, nfds, ...@@ -1121,7 +1118,6 @@ SYSCALL_DEFINE5(ppoll_time32, struct pollfd __user *, ufds, unsigned int, nfds,
struct old_timespec32 __user *, tsp, const sigset_t __user *, sigmask, struct old_timespec32 __user *, tsp, const sigset_t __user *, sigmask,
size_t, sigsetsize) size_t, sigsetsize)
{ {
sigset_t ksigmask, sigsaved;
struct timespec64 ts, end_time, *to = NULL; struct timespec64 ts, end_time, *to = NULL;
int ret; int ret;
...@@ -1134,17 +1130,16 @@ SYSCALL_DEFINE5(ppoll_time32, struct pollfd __user *, ufds, unsigned int, nfds, ...@@ -1134,17 +1130,16 @@ SYSCALL_DEFINE5(ppoll_time32, struct pollfd __user *, ufds, unsigned int, nfds,
return -EINVAL; return -EINVAL;
} }
ret = set_user_sigmask(sigmask, &ksigmask, &sigsaved, sigsetsize); ret = set_user_sigmask(sigmask, sigsetsize);
if (ret) if (ret)
return ret; return ret;
ret = do_sys_poll(ufds, nfds, to); ret = do_sys_poll(ufds, nfds, to);
restore_user_sigmask(sigmask, &sigsaved, ret == -EINTR); restore_saved_sigmask_unless(ret == -EINTR);
/* We can restart this syscall, usually */ /* We can restart this syscall, usually */
if (ret == -EINTR) if (ret == -EINTR)
ret = -ERESTARTNOHAND; ret = -ERESTARTNOHAND;
ret = poll_select_copy_remaining(&end_time, tsp, PT_OLD_TIMESPEC, ret); ret = poll_select_copy_remaining(&end_time, tsp, PT_OLD_TIMESPEC, ret);
return ret; return ret;
...@@ -1319,7 +1314,6 @@ static long do_compat_pselect(int n, compat_ulong_t __user *inp, ...@@ -1319,7 +1314,6 @@ static long do_compat_pselect(int n, compat_ulong_t __user *inp,
void __user *tsp, compat_sigset_t __user *sigmask, void __user *tsp, compat_sigset_t __user *sigmask,
compat_size_t sigsetsize, enum poll_time_type type) compat_size_t sigsetsize, enum poll_time_type type)
{ {
sigset_t ksigmask, sigsaved;
struct timespec64 ts, end_time, *to = NULL; struct timespec64 ts, end_time, *to = NULL;
int ret; int ret;
...@@ -1342,12 +1336,12 @@ static long do_compat_pselect(int n, compat_ulong_t __user *inp, ...@@ -1342,12 +1336,12 @@ static long do_compat_pselect(int n, compat_ulong_t __user *inp,
return -EINVAL; return -EINVAL;
} }
ret = set_compat_user_sigmask(sigmask, &ksigmask, &sigsaved, sigsetsize); ret = set_compat_user_sigmask(sigmask, sigsetsize);
if (ret) if (ret)
return ret; return ret;
ret = compat_core_sys_select(n, inp, outp, exp, to); ret = compat_core_sys_select(n, inp, outp, exp, to);
restore_user_sigmask(sigmask, &sigsaved, ret == -ERESTARTNOHAND); restore_saved_sigmask_unless(ret == -ERESTARTNOHAND);
ret = poll_select_copy_remaining(&end_time, tsp, type, ret); ret = poll_select_copy_remaining(&end_time, tsp, type, ret);
return ret; return ret;
...@@ -1402,7 +1396,6 @@ COMPAT_SYSCALL_DEFINE5(ppoll_time32, struct pollfd __user *, ufds, ...@@ -1402,7 +1396,6 @@ COMPAT_SYSCALL_DEFINE5(ppoll_time32, struct pollfd __user *, ufds,
unsigned int, nfds, struct old_timespec32 __user *, tsp, unsigned int, nfds, struct old_timespec32 __user *, tsp,
const compat_sigset_t __user *, sigmask, compat_size_t, sigsetsize) const compat_sigset_t __user *, sigmask, compat_size_t, sigsetsize)
{ {
sigset_t ksigmask, sigsaved;
struct timespec64 ts, end_time, *to = NULL; struct timespec64 ts, end_time, *to = NULL;
int ret; int ret;
...@@ -1415,17 +1408,16 @@ COMPAT_SYSCALL_DEFINE5(ppoll_time32, struct pollfd __user *, ufds, ...@@ -1415,17 +1408,16 @@ COMPAT_SYSCALL_DEFINE5(ppoll_time32, struct pollfd __user *, ufds,
return -EINVAL; return -EINVAL;
} }
ret = set_compat_user_sigmask(sigmask, &ksigmask, &sigsaved, sigsetsize); ret = set_compat_user_sigmask(sigmask, sigsetsize);
if (ret) if (ret)
return ret; return ret;
ret = do_sys_poll(ufds, nfds, to); ret = do_sys_poll(ufds, nfds, to);
restore_user_sigmask(sigmask, &sigsaved, ret == -EINTR); restore_saved_sigmask_unless(ret == -EINTR);
/* We can restart this syscall, usually */ /* We can restart this syscall, usually */
if (ret == -EINTR) if (ret == -EINTR)
ret = -ERESTARTNOHAND; ret = -ERESTARTNOHAND;
ret = poll_select_copy_remaining(&end_time, tsp, PT_OLD_TIMESPEC, ret); ret = poll_select_copy_remaining(&end_time, tsp, PT_OLD_TIMESPEC, ret);
return ret; return ret;
...@@ -1437,7 +1429,6 @@ COMPAT_SYSCALL_DEFINE5(ppoll_time64, struct pollfd __user *, ufds, ...@@ -1437,7 +1429,6 @@ COMPAT_SYSCALL_DEFINE5(ppoll_time64, struct pollfd __user *, ufds,
unsigned int, nfds, struct __kernel_timespec __user *, tsp, unsigned int, nfds, struct __kernel_timespec __user *, tsp,
const compat_sigset_t __user *, sigmask, compat_size_t, sigsetsize) const compat_sigset_t __user *, sigmask, compat_size_t, sigsetsize)
{ {
sigset_t ksigmask, sigsaved;
struct timespec64 ts, end_time, *to = NULL; struct timespec64 ts, end_time, *to = NULL;
int ret; int ret;
...@@ -1450,17 +1441,16 @@ COMPAT_SYSCALL_DEFINE5(ppoll_time64, struct pollfd __user *, ufds, ...@@ -1450,17 +1441,16 @@ COMPAT_SYSCALL_DEFINE5(ppoll_time64, struct pollfd __user *, ufds,
return -EINVAL; return -EINVAL;
} }
ret = set_compat_user_sigmask(sigmask, &ksigmask, &sigsaved, sigsetsize); ret = set_compat_user_sigmask(sigmask, sigsetsize);
if (ret) if (ret)
return ret; return ret;
ret = do_sys_poll(ufds, nfds, to); ret = do_sys_poll(ufds, nfds, to);
restore_user_sigmask(sigmask, &sigsaved, ret == -EINTR); restore_saved_sigmask_unless(ret == -EINTR);
/* We can restart this syscall, usually */ /* We can restart this syscall, usually */
if (ret == -EINTR) if (ret == -EINTR)
ret = -ERESTARTNOHAND; ret = -ERESTARTNOHAND;
ret = poll_select_copy_remaining(&end_time, tsp, PT_TIMESPEC, ret); ret = poll_select_copy_remaining(&end_time, tsp, PT_TIMESPEC, ret);
return ret; return ret;
......
...@@ -138,8 +138,7 @@ typedef struct { ...@@ -138,8 +138,7 @@ typedef struct {
compat_sigset_word sig[_COMPAT_NSIG_WORDS]; compat_sigset_word sig[_COMPAT_NSIG_WORDS];
} compat_sigset_t; } compat_sigset_t;
int set_compat_user_sigmask(const compat_sigset_t __user *usigmask, int set_compat_user_sigmask(const compat_sigset_t __user *umask,
sigset_t *set, sigset_t *oldset,
size_t sigsetsize); size_t sigsetsize);
struct compat_sigaction { struct compat_sigaction {
......
...@@ -420,7 +420,6 @@ void task_join_group_stop(struct task_struct *task); ...@@ -420,7 +420,6 @@ void task_join_group_stop(struct task_struct *task);
static inline void set_restore_sigmask(void) static inline void set_restore_sigmask(void)
{ {
set_thread_flag(TIF_RESTORE_SIGMASK); set_thread_flag(TIF_RESTORE_SIGMASK);
WARN_ON(!test_thread_flag(TIF_SIGPENDING));
} }
static inline void clear_tsk_restore_sigmask(struct task_struct *task) static inline void clear_tsk_restore_sigmask(struct task_struct *task)
...@@ -451,7 +450,6 @@ static inline bool test_and_clear_restore_sigmask(void) ...@@ -451,7 +450,6 @@ static inline bool test_and_clear_restore_sigmask(void)
static inline void set_restore_sigmask(void) static inline void set_restore_sigmask(void)
{ {
current->restore_sigmask = true; current->restore_sigmask = true;
WARN_ON(!test_thread_flag(TIF_SIGPENDING));
} }
static inline void clear_tsk_restore_sigmask(struct task_struct *task) static inline void clear_tsk_restore_sigmask(struct task_struct *task)
{ {
...@@ -484,6 +482,16 @@ static inline void restore_saved_sigmask(void) ...@@ -484,6 +482,16 @@ static inline void restore_saved_sigmask(void)
__set_current_blocked(&current->saved_sigmask); __set_current_blocked(&current->saved_sigmask);
} }
extern int set_user_sigmask(const sigset_t __user *umask, size_t sigsetsize);
static inline void restore_saved_sigmask_unless(bool interrupted)
{
if (interrupted)
WARN_ON(!test_thread_flag(TIF_SIGPENDING));
else
restore_saved_sigmask();
}
static inline sigset_t *sigmask_to_save(void) static inline sigset_t *sigmask_to_save(void)
{ {
sigset_t *res = &current->blocked; sigset_t *res = &current->blocked;
......
...@@ -273,10 +273,6 @@ extern int group_send_sig_info(int sig, struct kernel_siginfo *info, ...@@ -273,10 +273,6 @@ extern int group_send_sig_info(int sig, struct kernel_siginfo *info,
struct task_struct *p, enum pid_type type); struct task_struct *p, enum pid_type type);
extern int __group_send_sig_info(int, struct kernel_siginfo *, struct task_struct *); extern int __group_send_sig_info(int, struct kernel_siginfo *, struct task_struct *);
extern int sigprocmask(int, sigset_t *, sigset_t *); extern int sigprocmask(int, sigset_t *, sigset_t *);
extern int set_user_sigmask(const sigset_t __user *usigmask, sigset_t *set,
sigset_t *oldset, size_t sigsetsize);
extern void restore_user_sigmask(const void __user *usigmask,
sigset_t *sigsaved, bool interrupted);
extern void set_current_blocked(sigset_t *); extern void set_current_blocked(sigset_t *);
extern void __set_current_blocked(const sigset_t *); extern void __set_current_blocked(const sigset_t *);
extern int show_unhandled_signals; extern int show_unhandled_signals;
......
...@@ -2951,80 +2951,49 @@ EXPORT_SYMBOL(sigprocmask); ...@@ -2951,80 +2951,49 @@ EXPORT_SYMBOL(sigprocmask);
* *
* This is useful for syscalls such as ppoll, pselect, io_pgetevents and * This is useful for syscalls such as ppoll, pselect, io_pgetevents and
* epoll_pwait where a new sigmask is passed from userland for the syscalls. * epoll_pwait where a new sigmask is passed from userland for the syscalls.
*
* Note that it does set_restore_sigmask() in advance, so it must be always
* paired with restore_saved_sigmask_unless() before return from syscall.
*/ */
int set_user_sigmask(const sigset_t __user *usigmask, sigset_t *set, int set_user_sigmask(const sigset_t __user *umask, size_t sigsetsize)
sigset_t *oldset, size_t sigsetsize)
{ {
if (!usigmask) sigset_t kmask;
return 0;
if (!umask)
return 0;
if (sigsetsize != sizeof(sigset_t)) if (sigsetsize != sizeof(sigset_t))
return -EINVAL; return -EINVAL;
if (copy_from_user(set, usigmask, sizeof(sigset_t))) if (copy_from_user(&kmask, umask, sizeof(sigset_t)))
return -EFAULT; return -EFAULT;
*oldset = current->blocked; set_restore_sigmask();
set_current_blocked(set); current->saved_sigmask = current->blocked;
set_current_blocked(&kmask);
return 0; return 0;
} }
EXPORT_SYMBOL(set_user_sigmask);
#ifdef CONFIG_COMPAT #ifdef CONFIG_COMPAT
int set_compat_user_sigmask(const compat_sigset_t __user *usigmask, int set_compat_user_sigmask(const compat_sigset_t __user *umask,
sigset_t *set, sigset_t *oldset,
size_t sigsetsize) size_t sigsetsize)
{ {
if (!usigmask) sigset_t kmask;
return 0;
if (!umask)
return 0;
if (sigsetsize != sizeof(compat_sigset_t)) if (sigsetsize != sizeof(compat_sigset_t))
return -EINVAL; return -EINVAL;
if (get_compat_sigset(set, usigmask)) if (get_compat_sigset(&kmask, umask))
return -EFAULT; return -EFAULT;
*oldset = current->blocked; set_restore_sigmask();
set_current_blocked(set); current->saved_sigmask = current->blocked;
set_current_blocked(&kmask);
return 0; return 0;
} }
EXPORT_SYMBOL(set_compat_user_sigmask);
#endif #endif
/*
* restore_user_sigmask:
* usigmask: sigmask passed in from userland.
* sigsaved: saved sigmask when the syscall started and changed the sigmask to
* usigmask.
*
* This is useful for syscalls such as ppoll, pselect, io_pgetevents and
* epoll_pwait where a new sigmask is passed in from userland for the syscalls.
*/
void restore_user_sigmask(const void __user *usigmask, sigset_t *sigsaved,
bool interrupted)
{
if (!usigmask)
return;
/*
* When signals are pending, do not restore them here.
* Restoring sigmask here can lead to delivering signals that the above
* syscalls are intended to block because of the sigmask passed in.
*/
if (interrupted) {
current->saved_sigmask = *sigsaved;
set_restore_sigmask();
return;
}
/*
* This is needed because the fast syscall return path does not restore
* saved_sigmask when signals are not pending.
*/
set_current_blocked(sigsaved);
}
EXPORT_SYMBOL(restore_user_sigmask);
/** /**
* sys_rt_sigprocmask - change the list of currently blocked signals * sys_rt_sigprocmask - change the list of currently blocked signals
* @how: whether to add, remove, or set signals * @how: whether to add, remove, or set signals
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment