Commit a4412fc9 authored by Andy Lutomirski's avatar Andy Lutomirski Committed by Kees Cook

seccomp,x86,arm,mips,s390: Remove nr parameter from secure_computing

The secure_computing function took a syscall number parameter, but
it only paid any attention to that parameter if seccomp mode 1 was
enabled.  Rather than coming up with a kludge to get the parameter
to work in mode 2, just remove the parameter.

To avoid churn in arches that don't have seccomp filters (and may
not even support syscall_get_nr right now), this leaves the
parameter in secure_computing_strict, which is now a real function.

For ARM, this is a bit ugly due to the fact that ARM conditionally
supports seccomp filters.  Fixing that would probably only be a
couple of lines of code, but it should be coordinated with the audit
maintainers.

This will be a slight slowdown on some arches.  The right fix is to
pass in all of seccomp_data instead of trying to make just the
syscall nr part be fast.

This is a prerequisite for making two-phase seccomp work cleanly.

Cc: Russell King <linux@arm.linux.org.uk>
Cc: linux-arm-kernel@lists.infradead.org
Cc: Ralf Baechle <ralf@linux-mips.org>
Cc: linux-mips@linux-mips.org
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: Heiko Carstens <heiko.carstens@de.ibm.com>
Cc: linux-s390@vger.kernel.org
Cc: x86@kernel.org
Cc: Kees Cook <keescook@chromium.org>
Signed-off-by: default avatarAndy Lutomirski <luto@amacapital.net>
Signed-off-by: default avatarKees Cook <keescook@chromium.org>
parent 70c8038d
...@@ -933,8 +933,13 @@ asmlinkage int syscall_trace_enter(struct pt_regs *regs, int scno) ...@@ -933,8 +933,13 @@ asmlinkage int syscall_trace_enter(struct pt_regs *regs, int scno)
current_thread_info()->syscall = scno; current_thread_info()->syscall = scno;
/* Do the secure computing check first; failures should be fast. */ /* Do the secure computing check first; failures should be fast. */
if (secure_computing(scno) == -1) #ifdef CONFIG_HAVE_ARCH_SECCOMP_FILTER
if (secure_computing() == -1)
return -1; return -1;
#else
/* XXX: remove this once OABI gets fixed */
secure_computing_strict(scno);
#endif
if (test_thread_flag(TIF_SYSCALL_TRACE)) if (test_thread_flag(TIF_SYSCALL_TRACE))
tracehook_report_syscall(regs, PTRACE_SYSCALL_ENTER); tracehook_report_syscall(regs, PTRACE_SYSCALL_ENTER);
......
...@@ -770,7 +770,7 @@ asmlinkage long syscall_trace_enter(struct pt_regs *regs, long syscall) ...@@ -770,7 +770,7 @@ asmlinkage long syscall_trace_enter(struct pt_regs *regs, long syscall)
long ret = 0; long ret = 0;
user_exit(); user_exit();
if (secure_computing(syscall) == -1) if (secure_computing() == -1)
return -1; return -1;
if (test_thread_flag(TIF_SYSCALL_TRACE) && if (test_thread_flag(TIF_SYSCALL_TRACE) &&
......
...@@ -803,7 +803,7 @@ asmlinkage long do_syscall_trace_enter(struct pt_regs *regs) ...@@ -803,7 +803,7 @@ asmlinkage long do_syscall_trace_enter(struct pt_regs *regs)
long ret = 0; long ret = 0;
/* Do the secure computing check first. */ /* Do the secure computing check first. */
if (secure_computing(regs->gprs[2])) { if (secure_computing()) {
/* seccomp failures shouldn't expose any additional code. */ /* seccomp failures shouldn't expose any additional code. */
ret = -1; ret = -1;
goto out; goto out;
......
...@@ -1471,7 +1471,7 @@ long syscall_trace_enter(struct pt_regs *regs) ...@@ -1471,7 +1471,7 @@ long syscall_trace_enter(struct pt_regs *regs)
regs->flags |= X86_EFLAGS_TF; regs->flags |= X86_EFLAGS_TF;
/* do the secure computing check first */ /* do the secure computing check first */
if (secure_computing(regs->orig_ax)) { if (secure_computing()) {
/* seccomp failures shouldn't expose any additional code. */ /* seccomp failures shouldn't expose any additional code. */
ret = -1L; ret = -1L;
goto out; goto out;
......
...@@ -216,7 +216,7 @@ bool emulate_vsyscall(struct pt_regs *regs, unsigned long address) ...@@ -216,7 +216,7 @@ bool emulate_vsyscall(struct pt_regs *regs, unsigned long address)
*/ */
regs->orig_ax = syscall_nr; regs->orig_ax = syscall_nr;
regs->ax = -ENOSYS; regs->ax = -ENOSYS;
tmp = secure_computing(syscall_nr); tmp = secure_computing();
if ((!tmp && regs->orig_ax != syscall_nr) || regs->ip != address) { if ((!tmp && regs->orig_ax != syscall_nr) || regs->ip != address) {
warn_bad_vsyscall(KERN_DEBUG, regs, warn_bad_vsyscall(KERN_DEBUG, regs,
"seccomp tried to change syscall nr or ip"); "seccomp tried to change syscall nr or ip");
......
...@@ -27,19 +27,17 @@ struct seccomp { ...@@ -27,19 +27,17 @@ struct seccomp {
struct seccomp_filter *filter; struct seccomp_filter *filter;
}; };
extern int __secure_computing(int); #ifdef CONFIG_HAVE_ARCH_SECCOMP_FILTER
static inline int secure_computing(int this_syscall) extern int __secure_computing(void);
static inline int secure_computing(void)
{ {
if (unlikely(test_thread_flag(TIF_SECCOMP))) if (unlikely(test_thread_flag(TIF_SECCOMP)))
return __secure_computing(this_syscall); return __secure_computing();
return 0; return 0;
} }
#else
/* A wrapper for architectures supporting only SECCOMP_MODE_STRICT. */ extern void secure_computing_strict(int this_syscall);
static inline void secure_computing_strict(int this_syscall) #endif
{
BUG_ON(secure_computing(this_syscall) != 0);
}
extern long prctl_get_seccomp(void); extern long prctl_get_seccomp(void);
extern long prctl_set_seccomp(unsigned long, char __user *); extern long prctl_set_seccomp(unsigned long, char __user *);
...@@ -56,8 +54,11 @@ static inline int seccomp_mode(struct seccomp *s) ...@@ -56,8 +54,11 @@ static inline int seccomp_mode(struct seccomp *s)
struct seccomp { }; struct seccomp { };
struct seccomp_filter { }; struct seccomp_filter { };
static inline int secure_computing(int this_syscall) { return 0; } #ifdef CONFIG_HAVE_ARCH_SECCOMP_FILTER
static inline int secure_computing(void) { return 0; }
#else
static inline void secure_computing_strict(int this_syscall) { return; } static inline void secure_computing_strict(int this_syscall) { return; }
#endif
static inline long prctl_get_seccomp(void) static inline long prctl_get_seccomp(void)
{ {
......
...@@ -23,8 +23,11 @@ ...@@ -23,8 +23,11 @@
/* #define SECCOMP_DEBUG 1 */ /* #define SECCOMP_DEBUG 1 */
#ifdef CONFIG_SECCOMP_FILTER #ifdef CONFIG_HAVE_ARCH_SECCOMP_FILTER
#include <asm/syscall.h> #include <asm/syscall.h>
#endif
#ifdef CONFIG_SECCOMP_FILTER
#include <linux/filter.h> #include <linux/filter.h>
#include <linux/pid.h> #include <linux/pid.h>
#include <linux/ptrace.h> #include <linux/ptrace.h>
...@@ -172,7 +175,7 @@ static int seccomp_check_filter(struct sock_filter *filter, unsigned int flen) ...@@ -172,7 +175,7 @@ static int seccomp_check_filter(struct sock_filter *filter, unsigned int flen)
* *
* Returns valid seccomp BPF response codes. * Returns valid seccomp BPF response codes.
*/ */
static u32 seccomp_run_filters(int syscall) static u32 seccomp_run_filters(void)
{ {
struct seccomp_filter *f = ACCESS_ONCE(current->seccomp.filter); struct seccomp_filter *f = ACCESS_ONCE(current->seccomp.filter);
struct seccomp_data sd; struct seccomp_data sd;
...@@ -564,10 +567,43 @@ static int mode1_syscalls_32[] = { ...@@ -564,10 +567,43 @@ static int mode1_syscalls_32[] = {
}; };
#endif #endif
int __secure_computing(int this_syscall) static void __secure_computing_strict(int this_syscall)
{
int *syscall_whitelist = mode1_syscalls;
#ifdef CONFIG_COMPAT
if (is_compat_task())
syscall_whitelist = mode1_syscalls_32;
#endif
do {
if (*syscall_whitelist == this_syscall)
return;
} while (*++syscall_whitelist);
#ifdef SECCOMP_DEBUG
dump_stack();
#endif
audit_seccomp(this_syscall, SIGKILL, SECCOMP_RET_KILL);
do_exit(SIGKILL);
}
#ifndef CONFIG_HAVE_ARCH_SECCOMP_FILTER
void secure_computing_strict(int this_syscall)
{
int mode = current->seccomp.mode;
if (mode == 0)
return;
else if (mode == SECCOMP_MODE_STRICT)
__secure_computing_strict(this_syscall);
else
BUG();
}
#else
int __secure_computing(void)
{ {
struct pt_regs *regs = task_pt_regs(current);
int this_syscall = syscall_get_nr(current, regs);
int exit_sig = 0; int exit_sig = 0;
int *syscall;
u32 ret; u32 ret;
/* /*
...@@ -578,23 +614,12 @@ int __secure_computing(int this_syscall) ...@@ -578,23 +614,12 @@ int __secure_computing(int this_syscall)
switch (current->seccomp.mode) { switch (current->seccomp.mode) {
case SECCOMP_MODE_STRICT: case SECCOMP_MODE_STRICT:
syscall = mode1_syscalls; __secure_computing_strict(this_syscall);
#ifdef CONFIG_COMPAT return 0;
if (is_compat_task())
syscall = mode1_syscalls_32;
#endif
do {
if (*syscall == this_syscall)
return 0;
} while (*++syscall);
exit_sig = SIGKILL;
ret = SECCOMP_RET_KILL;
break;
#ifdef CONFIG_SECCOMP_FILTER #ifdef CONFIG_SECCOMP_FILTER
case SECCOMP_MODE_FILTER: { case SECCOMP_MODE_FILTER: {
int data; int data;
struct pt_regs *regs = task_pt_regs(current); ret = seccomp_run_filters();
ret = seccomp_run_filters(this_syscall);
data = ret & SECCOMP_RET_DATA; data = ret & SECCOMP_RET_DATA;
ret &= SECCOMP_RET_ACTION; ret &= SECCOMP_RET_ACTION;
switch (ret) { switch (ret) {
...@@ -652,9 +677,10 @@ int __secure_computing(int this_syscall) ...@@ -652,9 +677,10 @@ int __secure_computing(int this_syscall)
#ifdef CONFIG_SECCOMP_FILTER #ifdef CONFIG_SECCOMP_FILTER
skip: skip:
audit_seccomp(this_syscall, exit_sig, ret); audit_seccomp(this_syscall, exit_sig, ret);
#endif
return -1; return -1;
#endif
} }
#endif /* CONFIG_HAVE_ARCH_SECCOMP_FILTER */
long prctl_get_seccomp(void) long prctl_get_seccomp(void)
{ {
......
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