diff options
Diffstat (limited to 'kernel/sys.c')
| -rw-r--r-- | kernel/sys.c | 57 |
1 files changed, 27 insertions, 30 deletions
diff --git a/kernel/sys.c b/kernel/sys.c index 2d39a84cd857..241507f23eca 100644 --- a/kernel/sys.c +++ b/kernel/sys.c | |||
| @@ -2015,7 +2015,6 @@ SYSCALL_DEFINE5(prctl, int, option, unsigned long, arg2, unsigned long, arg3, | |||
| 2015 | break; | 2015 | break; |
| 2016 | } | 2016 | } |
| 2017 | me->pdeath_signal = arg2; | 2017 | me->pdeath_signal = arg2; |
| 2018 | error = 0; | ||
| 2019 | break; | 2018 | break; |
| 2020 | case PR_GET_PDEATHSIG: | 2019 | case PR_GET_PDEATHSIG: |
| 2021 | error = put_user(me->pdeath_signal, (int __user *)arg2); | 2020 | error = put_user(me->pdeath_signal, (int __user *)arg2); |
| @@ -2029,7 +2028,6 @@ SYSCALL_DEFINE5(prctl, int, option, unsigned long, arg2, unsigned long, arg3, | |||
| 2029 | break; | 2028 | break; |
| 2030 | } | 2029 | } |
| 2031 | set_dumpable(me->mm, arg2); | 2030 | set_dumpable(me->mm, arg2); |
| 2032 | error = 0; | ||
| 2033 | break; | 2031 | break; |
| 2034 | 2032 | ||
| 2035 | case PR_SET_UNALIGN: | 2033 | case PR_SET_UNALIGN: |
| @@ -2056,10 +2054,7 @@ SYSCALL_DEFINE5(prctl, int, option, unsigned long, arg2, unsigned long, arg3, | |||
| 2056 | case PR_SET_TIMING: | 2054 | case PR_SET_TIMING: |
| 2057 | if (arg2 != PR_TIMING_STATISTICAL) | 2055 | if (arg2 != PR_TIMING_STATISTICAL) |
| 2058 | error = -EINVAL; | 2056 | error = -EINVAL; |
| 2059 | else | ||
| 2060 | error = 0; | ||
| 2061 | break; | 2057 | break; |
| 2062 | |||
| 2063 | case PR_SET_NAME: | 2058 | case PR_SET_NAME: |
| 2064 | comm[sizeof(me->comm)-1] = 0; | 2059 | comm[sizeof(me->comm)-1] = 0; |
| 2065 | if (strncpy_from_user(comm, (char __user *)arg2, | 2060 | if (strncpy_from_user(comm, (char __user *)arg2, |
| @@ -2067,20 +2062,19 @@ SYSCALL_DEFINE5(prctl, int, option, unsigned long, arg2, unsigned long, arg3, | |||
| 2067 | return -EFAULT; | 2062 | return -EFAULT; |
| 2068 | set_task_comm(me, comm); | 2063 | set_task_comm(me, comm); |
| 2069 | proc_comm_connector(me); | 2064 | proc_comm_connector(me); |
| 2070 | return 0; | 2065 | break; |
| 2071 | case PR_GET_NAME: | 2066 | case PR_GET_NAME: |
| 2072 | get_task_comm(comm, me); | 2067 | get_task_comm(comm, me); |
| 2073 | if (copy_to_user((char __user *)arg2, comm, | 2068 | if (copy_to_user((char __user *)arg2, comm, |
| 2074 | sizeof(comm))) | 2069 | sizeof(comm))) |
| 2075 | return -EFAULT; | 2070 | return -EFAULT; |
| 2076 | return 0; | 2071 | break; |
| 2077 | case PR_GET_ENDIAN: | 2072 | case PR_GET_ENDIAN: |
| 2078 | error = GET_ENDIAN(me, arg2); | 2073 | error = GET_ENDIAN(me, arg2); |
| 2079 | break; | 2074 | break; |
| 2080 | case PR_SET_ENDIAN: | 2075 | case PR_SET_ENDIAN: |
| 2081 | error = SET_ENDIAN(me, arg2); | 2076 | error = SET_ENDIAN(me, arg2); |
| 2082 | break; | 2077 | break; |
| 2083 | |||
| 2084 | case PR_GET_SECCOMP: | 2078 | case PR_GET_SECCOMP: |
| 2085 | error = prctl_get_seccomp(); | 2079 | error = prctl_get_seccomp(); |
| 2086 | break; | 2080 | break; |
| @@ -2108,7 +2102,6 @@ SYSCALL_DEFINE5(prctl, int, option, unsigned long, arg2, unsigned long, arg3, | |||
| 2108 | current->default_timer_slack_ns; | 2102 | current->default_timer_slack_ns; |
| 2109 | else | 2103 | else |
| 2110 | current->timer_slack_ns = arg2; | 2104 | current->timer_slack_ns = arg2; |
| 2111 | error = 0; | ||
| 2112 | break; | 2105 | break; |
| 2113 | case PR_MCE_KILL: | 2106 | case PR_MCE_KILL: |
| 2114 | if (arg4 | arg5) | 2107 | if (arg4 | arg5) |
| @@ -2134,7 +2127,6 @@ SYSCALL_DEFINE5(prctl, int, option, unsigned long, arg2, unsigned long, arg3, | |||
| 2134 | default: | 2127 | default: |
| 2135 | return -EINVAL; | 2128 | return -EINVAL; |
| 2136 | } | 2129 | } |
| 2137 | error = 0; | ||
| 2138 | break; | 2130 | break; |
| 2139 | case PR_MCE_KILL_GET: | 2131 | case PR_MCE_KILL_GET: |
| 2140 | if (arg2 | arg3 | arg4 | arg5) | 2132 | if (arg2 | arg3 | arg4 | arg5) |
| @@ -2153,7 +2145,6 @@ SYSCALL_DEFINE5(prctl, int, option, unsigned long, arg2, unsigned long, arg3, | |||
| 2153 | break; | 2145 | break; |
| 2154 | case PR_SET_CHILD_SUBREAPER: | 2146 | case PR_SET_CHILD_SUBREAPER: |
| 2155 | me->signal->is_child_subreaper = !!arg2; | 2147 | me->signal->is_child_subreaper = !!arg2; |
| 2156 | error = 0; | ||
| 2157 | break; | 2148 | break; |
| 2158 | case PR_GET_CHILD_SUBREAPER: | 2149 | case PR_GET_CHILD_SUBREAPER: |
| 2159 | error = put_user(me->signal->is_child_subreaper, | 2150 | error = put_user(me->signal->is_child_subreaper, |
| @@ -2195,46 +2186,52 @@ static void argv_cleanup(struct subprocess_info *info) | |||
| 2195 | argv_free(info->argv); | 2186 | argv_free(info->argv); |
| 2196 | } | 2187 | } |
| 2197 | 2188 | ||
| 2198 | /** | 2189 | static int __orderly_poweroff(void) |
| 2199 | * orderly_poweroff - Trigger an orderly system poweroff | ||
| 2200 | * @force: force poweroff if command execution fails | ||
| 2201 | * | ||
| 2202 | * This may be called from any context to trigger a system shutdown. | ||
| 2203 | * If the orderly shutdown fails, it will force an immediate shutdown. | ||
| 2204 | */ | ||
| 2205 | int orderly_poweroff(bool force) | ||
| 2206 | { | 2190 | { |
| 2207 | int argc; | 2191 | int argc; |
| 2208 | char **argv = argv_split(GFP_ATOMIC, poweroff_cmd, &argc); | 2192 | char **argv; |
| 2209 | static char *envp[] = { | 2193 | static char *envp[] = { |
| 2210 | "HOME=/", | 2194 | "HOME=/", |
| 2211 | "PATH=/sbin:/bin:/usr/sbin:/usr/bin", | 2195 | "PATH=/sbin:/bin:/usr/sbin:/usr/bin", |
| 2212 | NULL | 2196 | NULL |
| 2213 | }; | 2197 | }; |
| 2214 | int ret = -ENOMEM; | 2198 | int ret; |
| 2215 | 2199 | ||
| 2200 | argv = argv_split(GFP_ATOMIC, poweroff_cmd, &argc); | ||
| 2216 | if (argv == NULL) { | 2201 | if (argv == NULL) { |
| 2217 | printk(KERN_WARNING "%s failed to allocate memory for \"%s\"\n", | 2202 | printk(KERN_WARNING "%s failed to allocate memory for \"%s\"\n", |
| 2218 | __func__, poweroff_cmd); | 2203 | __func__, poweroff_cmd); |
| 2219 | goto out; | 2204 | return -ENOMEM; |
| 2220 | } | 2205 | } |
| 2221 | 2206 | ||
| 2222 | ret = call_usermodehelper_fns(argv[0], argv, envp, UMH_NO_WAIT, | 2207 | ret = call_usermodehelper_fns(argv[0], argv, envp, UMH_NO_WAIT, |
| 2223 | NULL, argv_cleanup, NULL); | 2208 | NULL, argv_cleanup, NULL); |
| 2224 | out: | ||
| 2225 | if (likely(!ret)) | ||
| 2226 | return 0; | ||
| 2227 | |||
| 2228 | if (ret == -ENOMEM) | 2209 | if (ret == -ENOMEM) |
| 2229 | argv_free(argv); | 2210 | argv_free(argv); |
| 2230 | 2211 | ||
| 2231 | if (force) { | 2212 | return ret; |
| 2213 | } | ||
| 2214 | |||
| 2215 | /** | ||
| 2216 | * orderly_poweroff - Trigger an orderly system poweroff | ||
| 2217 | * @force: force poweroff if command execution fails | ||
| 2218 | * | ||
| 2219 | * This may be called from any context to trigger a system shutdown. | ||
| 2220 | * If the orderly shutdown fails, it will force an immediate shutdown. | ||
| 2221 | */ | ||
| 2222 | int orderly_poweroff(bool force) | ||
| 2223 | { | ||
| 2224 | int ret = __orderly_poweroff(); | ||
| 2225 | |||
| 2226 | if (ret && force) { | ||
| 2232 | printk(KERN_WARNING "Failed to start orderly shutdown: " | 2227 | printk(KERN_WARNING "Failed to start orderly shutdown: " |
| 2233 | "forcing the issue\n"); | 2228 | "forcing the issue\n"); |
| 2234 | 2229 | ||
| 2235 | /* I guess this should try to kick off some daemon to | 2230 | /* |
| 2236 | sync and poweroff asap. Or not even bother syncing | 2231 | * I guess this should try to kick off some daemon to sync and |
| 2237 | if we're doing an emergency shutdown? */ | 2232 | * poweroff asap. Or not even bother syncing if we're doing an |
| 2233 | * emergency shutdown? | ||
| 2234 | */ | ||
| 2238 | emergency_sync(); | 2235 | emergency_sync(); |
| 2239 | kernel_power_off(); | 2236 | kernel_power_off(); |
| 2240 | } | 2237 | } |
