2008-10-22 22:26:29 -07:00
# ifndef _ASM_X86_FUTEX_H
# define _ASM_X86_FUTEX_H
2008-01-30 13:30:20 +01:00
# ifdef __KERNEL__
# include <linux/futex.h>
2008-04-30 00:54:49 -07:00
# include <linux/uaccess.h>
2008-01-30 13:30:20 +01:00
# include <asm/asm.h>
# include <asm/errno.h>
# include <asm/processor.h>
# define __futex_atomic_op1(insn, ret, oldval, uaddr, oparg) \
2008-03-23 01:02:12 -07:00
asm volatile ( " 1: \t " insn " \n " \
" 2: \t .section .fixup, \" ax \" \n " \
" 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 13:30:20 +01:00
# define __futex_atomic_op2(insn, ret, oldval, uaddr, oparg) \
2008-03-23 01:02:12 -07:00
asm volatile ( " 1: \t movl %2, %0 \n " \
" \t movl \t %0, %3 \n " \
" \t " insn " \n " \
2008-08-15 12:45:09 -04:00
" 2: \t " LOCK_PREFIX " cmpxchgl %3, %2 \n " \
2008-03-23 01:02:12 -07:00
" \t jnz \t 1b \n " \
" 3: \t .section .fixup, \" ax \" \n " \
" 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-10 18:50:58 -08:00
static inline int futex_atomic_op_inuser ( int encoded_op , u32 __user * uaddr )
2008-01-30 13:30:20 +01: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-10 18:50:58 -08:00
if ( ! access_ok ( VERIFY_WRITE , uaddr , sizeof ( u32 ) ) )
2008-01-30 13:30:20 +01:00
return - EFAULT ;
# if defined(CONFIG_X86_32) && !defined(CONFIG_X86_BSWAP)
/* Real i386 machines can only support FUTEX_OP_SET */
if ( op ! = FUTEX_OP_SET & & boot_cpu_data . x86 = = 3 )
return - ENOSYS ;
# endif
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 12:45:09 -04:00
__futex_atomic_op1 ( LOCK_PREFIX " xaddl %0, %2 " , ret , oldval ,
2008-01-30 13:30:20 +01: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 01:02:12 -07: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 13:30:20 +01:00
}
}
return ret ;
}
2011-03-10 18:50:58 -08:00
static inline int futex_atomic_cmpxchg_inatomic ( u32 * uval , u32 __user * uaddr ,
u32 oldval , u32 newval )
2008-01-30 13:30:20 +01:00
{
2011-03-10 18:48:51 -08:00
int ret = 0 ;
2008-02-16 14:05:01 +01:00
# if defined(CONFIG_X86_32) && !defined(CONFIG_X86_BSWAP)
/* Real i386 machines have no cmpxchg instruction */
if ( boot_cpu_data . x86 = = 3 )
return - ENOSYS ;
# endif
2011-03-10 18:50:58 -08:00
if ( ! access_ok ( VERIFY_WRITE , uaddr , sizeof ( u32 ) ) )
2008-01-30 13:30:20 +01:00
return - EFAULT ;
2011-03-10 18:48:51 -08:00
asm volatile ( " 1: \t " LOCK_PREFIX " cmpxchgl %4, %2 \n "
2008-03-23 01:02:12 -07:00
" 2: \t .section .fixup, \" ax \" \n "
2011-03-10 18:48:51 -08:00
" 3: \t mov %3, %0 \n "
2008-03-23 01:02:12 -07:00
" \t jmp 2b \n "
" \t .previous \n "
_ASM_EXTABLE ( 1 b , 3 b )
2011-03-10 18:48:51 -08:00
: " +r " ( ret ) , " =a " ( oldval ) , " +m " ( * uaddr )
: " i " ( - EFAULT ) , " r " ( newval ) , " 1 " ( oldval )
2008-03-23 01:02:12 -07:00
: " memory "
2008-01-30 13:30:20 +01:00
) ;
2011-03-10 18:48:51 -08:00
* uval = oldval ;
return ret ;
2008-01-30 13:30:20 +01:00
}
# endif
2008-10-22 22:26:29 -07:00
# endif /* _ASM_X86_FUTEX_H */