diff options
Diffstat (limited to 'kernel')
-rw-r--r-- | kernel/signal.c | 18 |
1 files changed, 7 insertions, 11 deletions
diff --git a/kernel/signal.c b/kernel/signal.c index 1f7b2aaa4a39..27533b90c347 100644 --- a/kernel/signal.c +++ b/kernel/signal.c | |||
@@ -651,9 +651,7 @@ static int check_kill_permission(int sig, struct siginfo *info, | |||
651 | if (!valid_signal(sig)) | 651 | if (!valid_signal(sig)) |
652 | return error; | 652 | return error; |
653 | error = -EPERM; | 653 | error = -EPERM; |
654 | if ((info == SEND_SIG_NOINFO || | 654 | if ((info == SEND_SIG_NOINFO || (!is_si_special(info) && SI_FROMUSER(info))) |
655 | (info != SEND_SIG_PRIV && info != SEND_SIG_FORCED | ||
656 | && SI_FROMUSER(info))) | ||
657 | && ((sig != SIGCONT) || | 655 | && ((sig != SIGCONT) || |
658 | (current->signal->session != t->signal->session)) | 656 | (current->signal->session != t->signal->session)) |
659 | && (current->euid ^ t->suid) && (current->euid ^ t->uid) | 657 | && (current->euid ^ t->suid) && (current->euid ^ t->uid) |
@@ -802,7 +800,7 @@ static int send_signal(int sig, struct siginfo *info, struct task_struct *t, | |||
802 | pass on the info struct. */ | 800 | pass on the info struct. */ |
803 | 801 | ||
804 | q = __sigqueue_alloc(t, GFP_ATOMIC, (sig < SIGRTMIN && | 802 | q = __sigqueue_alloc(t, GFP_ATOMIC, (sig < SIGRTMIN && |
805 | (info < SEND_SIG_FORCED || | 803 | (is_si_special(info) || |
806 | info->si_code >= 0))); | 804 | info->si_code >= 0))); |
807 | if (q) { | 805 | if (q) { |
808 | list_add_tail(&q->list, &signals->list); | 806 | list_add_tail(&q->list, &signals->list); |
@@ -825,16 +823,14 @@ static int send_signal(int sig, struct siginfo *info, struct task_struct *t, | |||
825 | copy_siginfo(&q->info, info); | 823 | copy_siginfo(&q->info, info); |
826 | break; | 824 | break; |
827 | } | 825 | } |
828 | } else { | 826 | } else if (!is_si_special(info)) { |
829 | if (sig >= SIGRTMIN | 827 | if (sig >= SIGRTMIN && info->si_code != SI_USER) |
830 | && info != SEND_SIG_NOINFO && info != SEND_SIG_PRIV | ||
831 | && info->si_code != SI_USER) | ||
832 | /* | 828 | /* |
833 | * Queue overflow, abort. We may abort if the signal was rt | 829 | * Queue overflow, abort. We may abort if the signal was rt |
834 | * and sent by user using something other than kill(). | 830 | * and sent by user using something other than kill(). |
835 | */ | 831 | */ |
836 | return -EAGAIN; | 832 | return -EAGAIN; |
837 | if ((info > SEND_SIG_PRIV) && (info->si_code == SI_TIMER)) | 833 | if (info->si_code == SI_TIMER) |
838 | /* | 834 | /* |
839 | * Set up a return to indicate that we dropped | 835 | * Set up a return to indicate that we dropped |
840 | * the signal. | 836 | * the signal. |
@@ -860,7 +856,7 @@ specific_send_sig_info(int sig, struct siginfo *info, struct task_struct *t) | |||
860 | BUG(); | 856 | BUG(); |
861 | assert_spin_locked(&t->sighand->siglock); | 857 | assert_spin_locked(&t->sighand->siglock); |
862 | 858 | ||
863 | if ((info > SEND_SIG_FORCED) && (info->si_code == SI_TIMER)) | 859 | if (!is_si_special(info) && (info->si_code == SI_TIMER)) |
864 | /* | 860 | /* |
865 | * Set up a return to indicate that we dropped the signal. | 861 | * Set up a return to indicate that we dropped the signal. |
866 | */ | 862 | */ |
@@ -1052,7 +1048,7 @@ __group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p) | |||
1052 | assert_spin_locked(&p->sighand->siglock); | 1048 | assert_spin_locked(&p->sighand->siglock); |
1053 | handle_stop_signal(sig, p); | 1049 | handle_stop_signal(sig, p); |
1054 | 1050 | ||
1055 | if ((info > SEND_SIG_FORCED) && (info->si_code == SI_TIMER)) | 1051 | if (!is_si_special(info) && (info->si_code == SI_TIMER)) |
1056 | /* | 1052 | /* |
1057 | * Set up a return to indicate that we dropped the signal. | 1053 | * Set up a return to indicate that we dropped the signal. |
1058 | */ | 1054 | */ |