2006-03-27 01:16:24 -08:00
/*
* linux / kernel / futex_compat . c
*
* Futex compatibililty routines .
*
* Copyright 2006 , Red Hat , Inc . , Ingo Molnar
*/
# include <linux/linkage.h>
# include <linux/compat.h>
2007-10-18 23:40:14 -07:00
# include <linux/nsproxy.h>
2006-03-27 01:16:24 -08:00
# include <linux/futex.h>
# include <asm/uaccess.h>
2006-07-29 05:17:57 +02:00
/*
* Fetch a robust - list pointer . Bit 0 signals PI futexes :
*/
static inline int
fetch_robust_entry ( compat_uptr_t * uentry , struct robust_list __user * * entry ,
2006-10-10 22:46:07 +01:00
compat_uptr_t __user * head , int * pi )
2006-07-29 05:17:57 +02:00
{
if ( get_user ( * uentry , head ) )
return - EFAULT ;
* entry = compat_ptr ( ( * uentry ) & ~ 1 ) ;
* pi = ( unsigned int ) ( * uentry ) & 1 ;
return 0 ;
}
2007-11-06 21:13:56 -08:00
static void __user * futex_uaddr ( struct robust_list * entry ,
compat_long_t futex_offset )
{
compat_uptr_t base = ptr_to_compat ( entry ) ;
void __user * uaddr = compat_ptr ( base + futex_offset ) ;
return uaddr ;
}
2006-03-27 01:16:24 -08:00
/*
* Walk curr - > robust_list ( very carefully , it ' s a userspace list ! )
* and mark any locks found there dead , and notify any waiters .
*
* We silently return on any sign of list - walking problem .
*/
void compat_exit_robust_list ( struct task_struct * curr )
{
struct compat_robust_list_head __user * head = curr - > compat_robust_list ;
2007-10-01 01:20:13 -07:00
struct robust_list __user * entry , * next_entry , * pending ;
unsigned int limit = ROBUST_LIST_LIMIT , pi , next_pi , pip ;
compat_uptr_t uentry , next_uentry , upending ;
2006-03-27 01:16:24 -08:00
compat_long_t futex_offset ;
2007-10-01 01:20:13 -07:00
int rc ;
2006-03-27 01:16:24 -08:00
/*
* Fetch the list head ( which was registered earlier , via
* sys_set_robust_list ( ) ) :
*/
2006-07-29 05:17:57 +02:00
if ( fetch_robust_entry ( & uentry , & entry , & head - > list . next , & pi ) )
2006-03-27 01:16:24 -08:00
return ;
/*
* Fetch the relative futex offset :
*/
if ( get_user ( futex_offset , & head - > futex_offset ) )
return ;
/*
* Fetch any possibly pending lock - add first , and handle it
* if it exists :
*/
2006-07-29 05:17:57 +02:00
if ( fetch_robust_entry ( & upending , & pending ,
2006-08-05 12:15:15 -07:00
& head - > list_op_pending , & pip ) )
2006-03-27 01:16:24 -08:00
return ;
2007-10-01 01:20:13 -07:00
next_entry = NULL ; /* avoid warning with gcc */
2007-09-11 15:23:49 -07:00
while ( entry ! = ( struct robust_list __user * ) & head - > list ) {
2007-10-01 01:20:13 -07:00
/*
* Fetch the next entry in the list before calling
* handle_futex_death :
*/
rc = fetch_robust_entry ( & next_uentry , & next_entry ,
( compat_uptr_t __user * ) & entry - > next , & next_pi ) ;
2006-03-27 01:16:24 -08:00
/*
* A pending lock might already be on the list , so
* dont process it twice :
*/
2007-11-06 21:13:56 -08:00
if ( entry ! = pending ) {
void __user * uaddr = futex_uaddr ( entry , futex_offset ) ;
2006-03-27 01:16:24 -08:00
2007-11-06 21:13:56 -08:00
if ( handle_futex_death ( uaddr , curr , pi ) )
return ;
}
2007-10-01 01:20:13 -07:00
if ( rc )
2006-03-27 01:16:24 -08:00
return ;
2007-10-01 01:20:13 -07:00
uentry = next_uentry ;
entry = next_entry ;
pi = next_pi ;
2006-03-27 01:16:24 -08:00
/*
* Avoid excessively long or circular lists :
*/
if ( ! - - limit )
break ;
cond_resched ( ) ;
}
2007-11-06 21:13:56 -08:00
if ( pending ) {
void __user * uaddr = futex_uaddr ( pending , futex_offset ) ;
handle_futex_death ( uaddr , curr , pip ) ;
}
2006-03-27 01:16:24 -08:00
}
asmlinkage long
compat_sys_set_robust_list ( struct compat_robust_list_head __user * head ,
compat_size_t len )
{
if ( unlikely ( len ! = sizeof ( * head ) ) )
return - EINVAL ;
current - > compat_robust_list = head ;
return 0 ;
}
asmlinkage long
2006-10-10 22:46:07 +01:00
compat_sys_get_robust_list ( int pid , compat_uptr_t __user * head_ptr ,
2006-03-27 01:16:24 -08:00
compat_size_t __user * len_ptr )
{
2006-10-10 22:46:07 +01:00
struct compat_robust_list_head __user * head ;
2006-03-27 01:16:24 -08:00
unsigned long ret ;
if ( ! pid )
head = current - > compat_robust_list ;
else {
struct task_struct * p ;
ret = - ESRCH ;
read_lock ( & tasklist_lock ) ;
2007-10-18 23:40:16 -07:00
p = find_task_by_vpid ( pid ) ;
2006-03-27 01:16:24 -08:00
if ( ! p )
goto err_unlock ;
ret = - EPERM ;
if ( ( current - > euid ! = p - > euid ) & & ( current - > euid ! = p - > uid ) & &
! capable ( CAP_SYS_PTRACE ) )
goto err_unlock ;
head = p - > compat_robust_list ;
read_unlock ( & tasklist_lock ) ;
}
if ( put_user ( sizeof ( * head ) , len_ptr ) )
return - EFAULT ;
return put_user ( ptr_to_compat ( head ) , head_ptr ) ;
err_unlock :
read_unlock ( & tasklist_lock ) ;
return ret ;
}
2006-03-27 01:16:27 -08:00
asmlinkage long compat_sys_futex ( u32 __user * uaddr , int op , u32 val ,
2006-03-27 01:16:24 -08:00
struct compat_timespec __user * utime , u32 __user * uaddr2 ,
2006-03-27 01:16:27 -08:00
u32 val3 )
2006-03-27 01:16:24 -08:00
{
2007-05-09 02:35:02 -07:00
struct timespec ts ;
ktime_t t , * tp = NULL ;
2006-03-27 01:16:24 -08:00
int val2 = 0 ;
2007-06-01 00:46:41 -07:00
int cmd = op & FUTEX_CMD_MASK ;
2006-03-27 01:16:24 -08:00
2007-06-01 00:46:41 -07:00
if ( utime & & ( cmd = = FUTEX_WAIT | | cmd = = FUTEX_LOCK_PI ) ) {
2007-05-09 02:35:02 -07:00
if ( get_compat_timespec ( & ts , utime ) )
2006-03-27 01:16:24 -08:00
return - EFAULT ;
2007-05-09 02:35:02 -07:00
if ( ! timespec_valid ( & ts ) )
2006-03-31 02:31:32 -08:00
return - EINVAL ;
2007-05-09 02:35:02 -07:00
t = timespec_to_ktime ( ts ) ;
2007-06-01 00:46:41 -07:00
if ( cmd = = FUTEX_WAIT )
2007-05-09 02:35:02 -07:00
t = ktime_add ( ktime_get ( ) , t ) ;
tp = & t ;
2006-03-27 01:16:24 -08:00
}
2007-06-17 21:11:10 +02:00
if ( cmd = = FUTEX_REQUEUE | | cmd = = FUTEX_CMP_REQUEUE )
2006-03-27 01:16:24 -08:00
val2 = ( int ) ( unsigned long ) utime ;
2007-05-09 02:35:02 -07:00
return do_futex ( uaddr , op , val , tp , uaddr2 , val2 , val3 ) ;
2006-03-27 01:16:24 -08:00
}