diff options
Diffstat (limited to 'kernel')
-rw-r--r-- | kernel/fork.c | 2 | ||||
-rw-r--r-- | kernel/kmod.c | 6 | ||||
-rw-r--r-- | kernel/signal.c | 74 |
3 files changed, 77 insertions, 5 deletions
diff --git a/kernel/fork.c b/kernel/fork.c index 85f6d536608d..a31b823b3c2d 100644 --- a/kernel/fork.c +++ b/kernel/fork.c | |||
@@ -1613,7 +1613,6 @@ long do_fork(unsigned long clone_flags, | |||
1613 | return nr; | 1613 | return nr; |
1614 | } | 1614 | } |
1615 | 1615 | ||
1616 | #ifdef CONFIG_GENERIC_KERNEL_THREAD | ||
1617 | /* | 1616 | /* |
1618 | * Create a kernel thread. | 1617 | * Create a kernel thread. |
1619 | */ | 1618 | */ |
@@ -1622,7 +1621,6 @@ pid_t kernel_thread(int (*fn)(void *), void *arg, unsigned long flags) | |||
1622 | return do_fork(flags|CLONE_VM|CLONE_UNTRACED, (unsigned long)fn, | 1621 | return do_fork(flags|CLONE_VM|CLONE_UNTRACED, (unsigned long)fn, |
1623 | (unsigned long)arg, NULL, NULL); | 1622 | (unsigned long)arg, NULL, NULL); |
1624 | } | 1623 | } |
1625 | #endif | ||
1626 | 1624 | ||
1627 | #ifdef __ARCH_WANT_SYS_FORK | 1625 | #ifdef __ARCH_WANT_SYS_FORK |
1628 | SYSCALL_DEFINE0(fork) | 1626 | SYSCALL_DEFINE0(fork) |
diff --git a/kernel/kmod.c b/kernel/kmod.c index 1c317e386831..0023a87e8de6 100644 --- a/kernel/kmod.c +++ b/kernel/kmod.c | |||
@@ -219,9 +219,9 @@ static int ____call_usermodehelper(void *data) | |||
219 | 219 | ||
220 | commit_creds(new); | 220 | commit_creds(new); |
221 | 221 | ||
222 | retval = kernel_execve(sub_info->path, | 222 | retval = do_execve(sub_info->path, |
223 | (const char *const *)sub_info->argv, | 223 | (const char __user *const __user *)sub_info->argv, |
224 | (const char *const *)sub_info->envp); | 224 | (const char __user *const __user *)sub_info->envp); |
225 | if (!retval) | 225 | if (!retval) |
226 | return 0; | 226 | return 0; |
227 | 227 | ||
diff --git a/kernel/signal.c b/kernel/signal.c index 580a91e63471..7aaa51d8e5b8 100644 --- a/kernel/signal.c +++ b/kernel/signal.c | |||
@@ -31,6 +31,7 @@ | |||
31 | #include <linux/nsproxy.h> | 31 | #include <linux/nsproxy.h> |
32 | #include <linux/user_namespace.h> | 32 | #include <linux/user_namespace.h> |
33 | #include <linux/uprobes.h> | 33 | #include <linux/uprobes.h> |
34 | #include <linux/compat.h> | ||
34 | #define CREATE_TRACE_POINTS | 35 | #define CREATE_TRACE_POINTS |
35 | #include <trace/events/signal.h> | 36 | #include <trace/events/signal.h> |
36 | 37 | ||
@@ -3094,6 +3095,79 @@ do_sigaltstack (const stack_t __user *uss, stack_t __user *uoss, unsigned long s | |||
3094 | out: | 3095 | out: |
3095 | return error; | 3096 | return error; |
3096 | } | 3097 | } |
3098 | #ifdef CONFIG_GENERIC_SIGALTSTACK | ||
3099 | SYSCALL_DEFINE2(sigaltstack,const stack_t __user *,uss, stack_t __user *,uoss) | ||
3100 | { | ||
3101 | return do_sigaltstack(uss, uoss, current_user_stack_pointer()); | ||
3102 | } | ||
3103 | #endif | ||
3104 | |||
3105 | int restore_altstack(const stack_t __user *uss) | ||
3106 | { | ||
3107 | int err = do_sigaltstack(uss, NULL, current_user_stack_pointer()); | ||
3108 | /* squash all but EFAULT for now */ | ||
3109 | return err == -EFAULT ? err : 0; | ||
3110 | } | ||
3111 | |||
3112 | int __save_altstack(stack_t __user *uss, unsigned long sp) | ||
3113 | { | ||
3114 | struct task_struct *t = current; | ||
3115 | return __put_user((void __user *)t->sas_ss_sp, &uss->ss_sp) | | ||
3116 | __put_user(sas_ss_flags(sp), &uss->ss_flags) | | ||
3117 | __put_user(t->sas_ss_size, &uss->ss_size); | ||
3118 | } | ||
3119 | |||
3120 | #ifdef CONFIG_COMPAT | ||
3121 | #ifdef CONFIG_GENERIC_SIGALTSTACK | ||
3122 | asmlinkage long compat_sys_sigaltstack(const compat_stack_t __user *uss_ptr, | ||
3123 | compat_stack_t __user *uoss_ptr) | ||
3124 | { | ||
3125 | stack_t uss, uoss; | ||
3126 | int ret; | ||
3127 | mm_segment_t seg; | ||
3128 | |||
3129 | if (uss_ptr) { | ||
3130 | compat_stack_t uss32; | ||
3131 | |||
3132 | memset(&uss, 0, sizeof(stack_t)); | ||
3133 | if (copy_from_user(&uss32, uss_ptr, sizeof(compat_stack_t))) | ||
3134 | return -EFAULT; | ||
3135 | uss.ss_sp = compat_ptr(uss32.ss_sp); | ||
3136 | uss.ss_flags = uss32.ss_flags; | ||
3137 | uss.ss_size = uss32.ss_size; | ||
3138 | } | ||
3139 | seg = get_fs(); | ||
3140 | set_fs(KERNEL_DS); | ||
3141 | ret = do_sigaltstack((stack_t __force __user *) (uss_ptr ? &uss : NULL), | ||
3142 | (stack_t __force __user *) &uoss, | ||
3143 | compat_user_stack_pointer()); | ||
3144 | set_fs(seg); | ||
3145 | if (ret >= 0 && uoss_ptr) { | ||
3146 | if (!access_ok(VERIFY_WRITE, uoss_ptr, sizeof(compat_stack_t)) || | ||
3147 | __put_user(ptr_to_compat(uoss.ss_sp), &uoss_ptr->ss_sp) || | ||
3148 | __put_user(uoss.ss_flags, &uoss_ptr->ss_flags) || | ||
3149 | __put_user(uoss.ss_size, &uoss_ptr->ss_size)) | ||
3150 | ret = -EFAULT; | ||
3151 | } | ||
3152 | return ret; | ||
3153 | } | ||
3154 | |||
3155 | int compat_restore_altstack(const compat_stack_t __user *uss) | ||
3156 | { | ||
3157 | int err = compat_sys_sigaltstack(uss, NULL); | ||
3158 | /* squash all but -EFAULT for now */ | ||
3159 | return err == -EFAULT ? err : 0; | ||
3160 | } | ||
3161 | |||
3162 | int __compat_save_altstack(compat_stack_t __user *uss, unsigned long sp) | ||
3163 | { | ||
3164 | struct task_struct *t = current; | ||
3165 | return __put_user(ptr_to_compat((void __user *)t->sas_ss_sp), &uss->ss_sp) | | ||
3166 | __put_user(sas_ss_flags(sp), &uss->ss_flags) | | ||
3167 | __put_user(t->sas_ss_size, &uss->ss_size); | ||
3168 | } | ||
3169 | #endif | ||
3170 | #endif | ||
3097 | 3171 | ||
3098 | #ifdef __ARCH_WANT_SYS_SIGPENDING | 3172 | #ifdef __ARCH_WANT_SYS_SIGPENDING |
3099 | 3173 | ||