2008-04-19 04:08:44 +04:00
/*
* Access to user system call parameters and results
*
2009-09-23 06:57:51 +04:00
* Copyright ( C ) 2008 - 2009 Red Hat , Inc . All rights reserved .
2008-04-19 04:08:44 +04:00
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
* of the GNU General Public License v .2 .
*
* See asm - generic / syscall . h for descriptions of what we must do here .
*/
2008-10-23 11:20:33 +04:00
# ifndef _ASM_X86_SYSCALL_H
# define _ASM_X86_SYSCALL_H
2008-04-19 04:08:44 +04:00
2014-03-11 20:55:42 +04:00
# include <uapi/linux/audit.h>
2008-04-19 04:08:44 +04:00
# include <linux/sched.h>
2008-09-03 15:31:42 +04:00
# include <linux/err.h>
2012-01-08 02:10:18 +04:00
# include <asm/asm-offsets.h> /* For NR_syscalls */
2012-04-13 01:47:56 +04:00
# include <asm/thread_info.h> /* for TS_COMPAT */
2012-02-19 19:56:26 +04:00
# include <asm/unistd.h>
2008-04-19 04:08:44 +04:00
2018-04-05 12:53:05 +03:00
# ifdef CONFIG_X86_64
syscalls/x86: Use 'struct pt_regs' based syscall calling convention for 64-bit syscalls
Let's make use of ARCH_HAS_SYSCALL_WRAPPER=y on pure 64-bit x86-64 systems:
Each syscall defines a stub which takes struct pt_regs as its only
argument. It decodes just those parameters it needs, e.g:
asmlinkage long sys_xyzzy(const struct pt_regs *regs)
{
return SyS_xyzzy(regs->di, regs->si, regs->dx);
}
This approach avoids leaking random user-provided register content down
the call chain.
For example, for sys_recv() which is a 4-parameter syscall, the assembly
now is (in slightly reordered fashion):
<sys_recv>:
callq <__fentry__>
/* decode regs->di, ->si, ->dx and ->r10 */
mov 0x70(%rdi),%rdi
mov 0x68(%rdi),%rsi
mov 0x60(%rdi),%rdx
mov 0x38(%rdi),%rcx
[ SyS_recv() is automatically inlined by the compiler,
as it is not [yet] used anywhere else ]
/* clear %r9 and %r8, the 5th and 6th args */
xor %r9d,%r9d
xor %r8d,%r8d
/* do the actual work */
callq __sys_recvfrom
/* cleanup and return */
cltq
retq
The only valid place in an x86-64 kernel which rightfully calls
a syscall function on its own -- vsyscall -- needs to be modified
to pass struct pt_regs onwards as well.
To keep the syscall table generation working independent of
SYSCALL_PTREGS being enabled, the stubs are named the same as the
"original" syscall stubs, i.e. sys_*().
This patch is based on an original proof-of-concept
| From: Linus Torvalds <torvalds@linux-foundation.org>
| Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
and was split up and heavily modified by me, in particular to base it on
ARCH_HAS_SYSCALL_WRAPPER, to limit it to 64-bit-only for the time being,
and to update the vsyscall to the new calling convention.
Signed-off-by: Dominik Brodowski <linux@dominikbrodowski.net>
Acked-by: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Al Viro <viro@zeniv.linux.org.uk>
Cc: Andrew Morton <akpm@linux-foundation.org>
Cc: Andy Lutomirski <luto@kernel.org>
Cc: Borislav Petkov <bp@alien8.de>
Cc: Brian Gerst <brgerst@gmail.com>
Cc: Denys Vlasenko <dvlasenk@redhat.com>
Cc: H. Peter Anvin <hpa@zytor.com>
Cc: Josh Poimboeuf <jpoimboe@redhat.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Link: http://lkml.kernel.org/r/20180405095307.3730-4-linux@dominikbrodowski.net
Signed-off-by: Ingo Molnar <mingo@kernel.org>
2018-04-05 12:53:02 +03:00
typedef asmlinkage long ( * sys_call_ptr_t ) ( const struct pt_regs * ) ;
# else
2015-10-06 03:48:07 +03:00
typedef asmlinkage long ( * sys_call_ptr_t ) ( unsigned long , unsigned long ,
unsigned long , unsigned long ,
unsigned long , unsigned long ) ;
2018-04-05 12:53:05 +03:00
# endif /* CONFIG_X86_64 */
2013-08-06 02:02:35 +04:00
extern const sys_call_ptr_t sys_call_table [ ] ;
2010-01-26 12:40:03 +03:00
2015-10-06 03:48:06 +03:00
# if defined(CONFIG_X86_32)
# define ia32_sys_call_table sys_call_table
# define __NR_syscall_compat_max __NR_syscall_max
# define IA32_NR_syscalls NR_syscalls
# endif
# if defined(CONFIG_IA32_EMULATION)
extern const sys_call_ptr_t ia32_sys_call_table [ ] ;
# endif
2009-09-23 06:57:51 +04:00
/*
* Only the low 32 bits of orig_ax are meaningful , so we return int .
* This importantly ignores the high bits on 64 - bit , so comparisons
* sign - extend the low 32 bits .
*/
static inline int syscall_get_nr ( struct task_struct * task , struct pt_regs * regs )
2008-04-19 04:08:44 +04:00
{
2013-02-15 21:21:43 +04:00
return regs - > orig_ax ;
2008-04-19 04:08:44 +04:00
}
static inline void syscall_rollback ( struct task_struct * task ,
struct pt_regs * regs )
{
2013-02-15 21:21:43 +04:00
regs - > ax = regs - > orig_ax ;
2008-04-19 04:08:44 +04:00
}
static inline long syscall_get_error ( struct task_struct * task ,
struct pt_regs * regs )
{
unsigned long error = regs - > ax ;
# ifdef CONFIG_IA32_EMULATION
/*
* TS_COMPAT is set for 32 - bit syscall entries and then
* remains set until we return to user mode .
*/
2018-01-28 21:38:50 +03:00
if ( task - > thread_info . status & ( TS_COMPAT | TS_I386_REGS_POKED ) )
2008-04-19 04:08:44 +04:00
/*
* Sign - extend the value so ( int ) - EFOO becomes ( long ) - EFOO
* and will match correctly in comparisons .
*/
error = ( long ) ( int ) error ;
# endif
2008-09-03 15:31:42 +04:00
return IS_ERR_VALUE ( error ) ? error : 0 ;
2008-04-19 04:08:44 +04:00
}
static inline long syscall_get_return_value ( struct task_struct * task ,
struct pt_regs * regs )
{
return regs - > ax ;
}
static inline void syscall_set_return_value ( struct task_struct * task ,
struct pt_regs * regs ,
int error , long val )
{
regs - > ax = ( long ) error ? : val ;
}
# ifdef CONFIG_X86_32
static inline void syscall_get_arguments ( struct task_struct * task ,
struct pt_regs * regs ,
unsigned int i , unsigned int n ,
unsigned long * args )
{
BUG_ON ( i + n > 6 ) ;
memcpy ( args , & regs - > bx + i , n * sizeof ( args [ 0 ] ) ) ;
}
static inline void syscall_set_arguments ( struct task_struct * task ,
struct pt_regs * regs ,
unsigned int i , unsigned int n ,
const unsigned long * args )
{
BUG_ON ( i + n > 6 ) ;
memcpy ( & regs - > bx + i , args , n * sizeof ( args [ 0 ] ) ) ;
}
2014-03-11 20:48:43 +04:00
static inline int syscall_get_arch ( void )
2012-04-13 01:47:56 +04:00
{
return AUDIT_ARCH_I386 ;
}
2008-04-19 04:08:44 +04:00
# else /* CONFIG_X86_64 */
static inline void syscall_get_arguments ( struct task_struct * task ,
struct pt_regs * regs ,
unsigned int i , unsigned int n ,
unsigned long * args )
{
# ifdef CONFIG_IA32_EMULATION
2018-01-28 21:38:50 +03:00
if ( task - > thread_info . status & TS_COMPAT )
2008-10-14 05:40:04 +04:00
switch ( i ) {
case 0 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > bx ;
case 1 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > cx ;
case 2 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > dx ;
2008-04-19 04:08:44 +04:00
case 3 :
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > si ;
case 4 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > di ;
case 5 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > bp ;
case 6 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
default :
BUG ( ) ;
break ;
}
else
# endif
2008-10-14 05:40:04 +04:00
switch ( i ) {
case 0 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > di ;
case 1 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > si ;
case 2 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > dx ;
2008-04-19 04:08:44 +04:00
case 3 :
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > r10 ;
case 4 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > r8 ;
case 5 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
* args + + = regs - > r9 ;
case 6 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
default :
BUG ( ) ;
break ;
}
}
static inline void syscall_set_arguments ( struct task_struct * task ,
struct pt_regs * regs ,
unsigned int i , unsigned int n ,
const unsigned long * args )
{
# ifdef CONFIG_IA32_EMULATION
2018-01-28 21:38:50 +03:00
if ( task - > thread_info . status & TS_COMPAT )
2008-10-14 05:40:04 +04:00
switch ( i ) {
case 0 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > bx = * args + + ;
case 1 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > cx = * args + + ;
case 2 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > dx = * args + + ;
2008-04-19 04:08:44 +04:00
case 3 :
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > si = * args + + ;
case 4 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > di = * args + + ;
case 5 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > bp = * args + + ;
case 6 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
default :
BUG ( ) ;
2008-10-14 05:40:04 +04:00
break ;
2008-04-19 04:08:44 +04:00
}
else
# endif
2008-10-14 05:40:04 +04:00
switch ( i ) {
case 0 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > di = * args + + ;
case 1 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > si = * args + + ;
case 2 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > dx = * args + + ;
2008-04-19 04:08:44 +04:00
case 3 :
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > r10 = * args + + ;
case 4 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > r8 = * args + + ;
case 5 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
2008-10-14 05:40:04 +04:00
regs - > r9 = * args + + ;
case 6 :
2008-04-19 04:08:44 +04:00
if ( ! n - - ) break ;
default :
BUG ( ) ;
2008-10-14 05:40:04 +04:00
break ;
2008-04-19 04:08:44 +04:00
}
}
2014-03-11 20:48:43 +04:00
static inline int syscall_get_arch ( void )
2012-04-13 01:47:56 +04:00
{
2016-09-14 00:29:21 +03:00
/* x32 tasks should be considered AUDIT_ARCH_X86_64. */
return in_ia32_syscall ( ) ? AUDIT_ARCH_I386 : AUDIT_ARCH_X86_64 ;
2012-04-13 01:47:56 +04:00
}
2008-04-19 04:08:44 +04:00
# endif /* CONFIG_X86_32 */
2008-10-23 11:20:33 +04:00
# endif /* _ASM_X86_SYSCALL_H */