2008-10-23 09:26:29 +04:00
# ifndef _ASM_X86_FUTEX_H
# define _ASM_X86_FUTEX_H
2008-01-30 15:30:20 +03:00
# ifdef __KERNEL__
# include <linux/futex.h>
2008-04-30 11:54:49 +04:00
# include <linux/uaccess.h>
2008-01-30 15:30:20 +03:00
# include <asm/asm.h>
# include <asm/errno.h>
# include <asm/processor.h>
2012-09-21 23:43:12 +04:00
# include <asm/smap.h>
2008-01-30 15:30:20 +03:00
# define __futex_atomic_op1(insn, ret, oldval, uaddr, oparg) \
2012-09-21 23:43:12 +04:00
asm volatile ( " \t " ASM_STAC " \n " \
" 1: \t " insn " \n " \
" 2: \t " ASM_CLAC " \n " \
" \t .section .fixup, \" ax \" \n " \
2008-03-23 11:02:12 +03:00
" 3: \t mov \t %3, %1 \n " \
" \t jmp \t 2b \n " \
" \t .previous \n " \
_ASM_EXTABLE ( 1 b , 3 b ) \
: " =r " ( oldval ) , " =r " ( ret ) , " +m " ( * uaddr ) \
: " i " ( - EFAULT ) , " 0 " ( oparg ) , " 1 " ( 0 ) )
2008-01-30 15:30:20 +03:00
# define __futex_atomic_op2(insn, ret, oldval, uaddr, oparg) \
2012-09-21 23:43:12 +04:00
asm volatile ( " \t " ASM_STAC " \n " \
" 1: \t movl %2, %0 \n " \
2008-03-23 11:02:12 +03:00
" \t movl \t %0, %3 \n " \
" \t " insn " \n " \
2008-08-15 20:45:09 +04:00
" 2: \t " LOCK_PREFIX " cmpxchgl %3, %2 \n " \
2008-03-23 11:02:12 +03:00
" \t jnz \t 1b \n " \
2012-09-21 23:43:12 +04:00
" 3: \t " ASM_CLAC " \n " \
" \t .section .fixup, \" ax \" \n " \
2008-03-23 11:02:12 +03:00
" 4: \t mov \t %5, %1 \n " \
" \t jmp \t 3b \n " \
" \t .previous \n " \
_ASM_EXTABLE ( 1 b , 4 b ) \
_ASM_EXTABLE ( 2 b , 4 b ) \
: " =&a " ( oldval ) , " =&r " ( ret ) , \
" +m " ( * uaddr ) , " =&r " ( tem ) \
: " r " ( oparg ) , " i " ( - EFAULT ) , " 1 " ( 0 ) )
2011-03-11 05:50:58 +03:00
static inline int futex_atomic_op_inuser ( int encoded_op , u32 __user * uaddr )
2008-01-30 15:30:20 +03:00
{
int op = ( encoded_op > > 28 ) & 7 ;
int cmp = ( encoded_op > > 24 ) & 15 ;
int oparg = ( encoded_op < < 8 ) > > 20 ;
int cmparg = ( encoded_op < < 20 ) > > 20 ;
int oldval = 0 , ret , tem ;
if ( encoded_op & ( FUTEX_OP_OPARG_SHIFT < < 28 ) )
oparg = 1 < < oparg ;
2011-03-11 05:50:58 +03:00
if ( ! access_ok ( VERIFY_WRITE , uaddr , sizeof ( u32 ) ) )
2008-01-30 15:30:20 +03:00
return - EFAULT ;
pagefault_disable ( ) ;
switch ( op ) {
case FUTEX_OP_SET :
__futex_atomic_op1 ( " xchgl %0, %2 " , ret , oldval , uaddr , oparg ) ;
break ;
case FUTEX_OP_ADD :
2008-08-15 20:45:09 +04:00
__futex_atomic_op1 ( LOCK_PREFIX " xaddl %0, %2 " , ret , oldval ,
2008-01-30 15:30:20 +03:00
uaddr , oparg ) ;
break ;
case FUTEX_OP_OR :
__futex_atomic_op2 ( " orl %4, %3 " , ret , oldval , uaddr , oparg ) ;
break ;
case FUTEX_OP_ANDN :
__futex_atomic_op2 ( " andl %4, %3 " , ret , oldval , uaddr , ~ oparg ) ;
break ;
case FUTEX_OP_XOR :
__futex_atomic_op2 ( " xorl %4, %3 " , ret , oldval , uaddr , oparg ) ;
break ;
default :
ret = - ENOSYS ;
}
pagefault_enable ( ) ;
if ( ! ret ) {
switch ( cmp ) {
2008-03-23 11:02:12 +03:00
case FUTEX_OP_CMP_EQ :
ret = ( oldval = = cmparg ) ;
break ;
case FUTEX_OP_CMP_NE :
ret = ( oldval ! = cmparg ) ;
break ;
case FUTEX_OP_CMP_LT :
ret = ( oldval < cmparg ) ;
break ;
case FUTEX_OP_CMP_GE :
ret = ( oldval > = cmparg ) ;
break ;
case FUTEX_OP_CMP_LE :
ret = ( oldval < = cmparg ) ;
break ;
case FUTEX_OP_CMP_GT :
ret = ( oldval > cmparg ) ;
break ;
default :
ret = - ENOSYS ;
2008-01-30 15:30:20 +03:00
}
}
return ret ;
}
2011-03-11 05:50:58 +03:00
static inline int futex_atomic_cmpxchg_inatomic ( u32 * uval , u32 __user * uaddr ,
u32 oldval , u32 newval )
2008-01-30 15:30:20 +03:00
{
2013-12-14 10:25:03 +04:00
return user_atomic_cmpxchg_inatomic ( uval , uaddr , oldval , newval ) ;
2008-01-30 15:30:20 +03:00
}
# endif
2008-10-23 09:26:29 +04:00
# endif /* _ASM_X86_FUTEX_H */