2005-04-16 15:20:36 -07:00
/*
* Conversion between 32 - bit and 64 - bit native system calls .
*
* Copyright ( C ) 2000 Silicon Graphics , Inc .
* Written by Ulf Carlsson ( ulfc @ engr . sgi . com )
* sys32_execve from ia64 / ia32 code , Feb 2000 , Kanoj Sarcar ( kanoj @ sgi . com )
*/
# include <linux/compiler.h>
# include <linux/mm.h>
# include <linux/errno.h>
# include <linux/file.h>
# include <linux/smp_lock.h>
# include <linux/highuid.h>
# include <linux/dirent.h>
# include <linux/resource.h>
# include <linux/highmem.h>
# include <linux/time.h>
# include <linux/times.h>
# include <linux/poll.h>
# include <linux/slab.h>
# include <linux/skbuff.h>
# include <linux/filter.h>
# include <linux/shm.h>
# include <linux/sem.h>
# include <linux/msg.h>
# include <linux/icmpv6.h>
# include <linux/syscalls.h>
# include <linux/sysctl.h>
# include <linux/utime.h>
# include <linux/utsname.h>
# include <linux/personality.h>
# include <linux/dnotify.h>
# include <linux/module.h>
# include <linux/binfmts.h>
# include <linux/security.h>
# include <linux/compat.h>
# include <linux/vfs.h>
# include <net/sock.h>
# include <net/scm.h>
2007-02-13 00:05:11 +00:00
# include <asm/compat-signal.h>
2005-04-16 15:20:36 -07:00
# include <asm/ipc.h>
# include <asm/sim.h>
# include <asm/uaccess.h>
# include <asm/mmu_context.h>
# include <asm/mman.h>
/* Use this to get at 32-bit user passed pointers. */
/* A() macro should be used for places where you e.g.
have some internal variable u32 and just want to get
rid of a compiler warning . AA ( ) has to be used in
places where you want to convert a function argument
to 32 bit pointer or when you e . g . access pt_regs
structure and want to consider 32 bit registers only .
*/
# define A(__x) ((unsigned long)(__x))
# define AA(__x) ((unsigned long)((int)__x))
# ifdef __MIPSEB__
# define merge_64(r1,r2) ((((r1) & 0xffffffffUL) << 32) + ((r2) & 0xffffffffUL))
# endif
# ifdef __MIPSEL__
# define merge_64(r1,r2) ((((r2) & 0xffffffffUL) << 32) + ((r1) & 0xffffffffUL))
# endif
/*
* Revalidate the inode . This is required for proper NFS attribute caching .
*/
2006-02-21 16:05:11 +09:00
int cp_compat_stat ( struct kstat * stat , struct compat_stat __user * statbuf )
2005-04-16 15:20:36 -07:00
{
struct compat_stat tmp ;
if ( ! new_valid_dev ( stat - > dev ) | | ! new_valid_dev ( stat - > rdev ) )
return - EOVERFLOW ;
memset ( & tmp , 0 , sizeof ( tmp ) ) ;
tmp . st_dev = new_encode_dev ( stat - > dev ) ;
tmp . st_ino = stat - > ino ;
2006-10-03 01:13:46 -07:00
if ( sizeof ( tmp . st_ino ) < sizeof ( stat - > ino ) & & tmp . st_ino ! = stat - > ino )
return - EOVERFLOW ;
2005-04-16 15:20:36 -07:00
tmp . st_mode = stat - > mode ;
tmp . st_nlink = stat - > nlink ;
SET_UID ( tmp . st_uid , stat - > uid ) ;
SET_GID ( tmp . st_gid , stat - > gid ) ;
tmp . st_rdev = new_encode_dev ( stat - > rdev ) ;
tmp . st_size = stat - > size ;
tmp . st_atime = stat - > atime . tv_sec ;
tmp . st_mtime = stat - > mtime . tv_sec ;
tmp . st_ctime = stat - > ctime . tv_sec ;
# ifdef STAT_HAVE_NSEC
tmp . st_atime_nsec = stat - > atime . tv_nsec ;
tmp . st_mtime_nsec = stat - > mtime . tv_nsec ;
tmp . st_ctime_nsec = stat - > ctime . tv_nsec ;
# endif
tmp . st_blocks = stat - > blocks ;
tmp . st_blksize = stat - > blksize ;
return copy_to_user ( statbuf , & tmp , sizeof ( tmp ) ) ? - EFAULT : 0 ;
}
asmlinkage unsigned long
sys32_mmap2 ( unsigned long addr , unsigned long len , unsigned long prot ,
unsigned long flags , unsigned long fd , unsigned long pgoff )
{
struct file * file = NULL ;
unsigned long error ;
error = - EINVAL ;
2006-02-24 21:20:29 -08:00
if ( pgoff & ( ~ PAGE_MASK > > 12 ) )
goto out ;
pgoff > > = PAGE_SHIFT - 12 ;
2005-04-16 15:20:36 -07:00
if ( ! ( flags & MAP_ANONYMOUS ) ) {
error = - EBADF ;
file = fget ( fd ) ;
if ( ! file )
goto out ;
}
flags & = ~ ( MAP_EXECUTABLE | MAP_DENYWRITE ) ;
down_write ( & current - > mm - > mmap_sem ) ;
error = do_mmap_pgoff ( file , addr , len , prot , flags , pgoff ) ;
up_write ( & current - > mm - > mmap_sem ) ;
if ( file )
fput ( file ) ;
out :
return error ;
}
2006-02-21 16:05:11 +09:00
asmlinkage int sys_truncate64 ( const char __user * path , unsigned int high ,
2005-04-16 15:20:36 -07:00
unsigned int low )
{
if ( ( int ) high < 0 )
return - EINVAL ;
return sys_truncate ( path , ( ( long ) high < < 32 ) | low ) ;
}
asmlinkage int sys_ftruncate64 ( unsigned int fd , unsigned int high ,
unsigned int low )
{
if ( ( int ) high < 0 )
return - EINVAL ;
return sys_ftruncate ( fd , ( ( long ) high < < 32 ) | low ) ;
}
/*
* sys_execve ( ) executes a new program .
*/
asmlinkage int sys32_execve ( nabi_no_regargs struct pt_regs regs )
{
int error ;
char * filename ;
filename = getname ( compat_ptr ( regs . regs [ 4 ] ) ) ;
error = PTR_ERR ( filename ) ;
if ( IS_ERR ( filename ) )
goto out ;
error = compat_do_execve ( filename , compat_ptr ( regs . regs [ 5 ] ) ,
compat_ptr ( regs . regs [ 6 ] ) , & regs ) ;
putname ( filename ) ;
out :
return error ;
}
# define RLIM_INFINITY32 0x7fffffff
# define RESOURCE32(x) ((x > RLIM_INFINITY32) ? RLIM_INFINITY32 : x)
struct rlimit32 {
int rlim_cur ;
int rlim_max ;
} ;
# ifdef __MIPSEB__
2006-02-21 16:05:11 +09:00
asmlinkage long sys32_truncate64 ( const char __user * path , unsigned long __dummy ,
2005-04-16 15:20:36 -07:00
int length_hi , int length_lo )
# endif
# ifdef __MIPSEL__
2006-02-21 16:05:11 +09:00
asmlinkage long sys32_truncate64 ( const char __user * path , unsigned long __dummy ,
2005-04-16 15:20:36 -07:00
int length_lo , int length_hi )
# endif
{
loff_t length ;
length = ( ( unsigned long ) length_hi < < 32 ) | ( unsigned int ) length_lo ;
return sys_truncate ( path , length ) ;
}
# ifdef __MIPSEB__
asmlinkage long sys32_ftruncate64 ( unsigned int fd , unsigned long __dummy ,
int length_hi , int length_lo )
# endif
# ifdef __MIPSEL__
asmlinkage long sys32_ftruncate64 ( unsigned int fd , unsigned long __dummy ,
int length_lo , int length_hi )
# endif
{
loff_t length ;
length = ( ( unsigned long ) length_hi < < 32 ) | ( unsigned int ) length_lo ;
return sys_ftruncate ( fd , length ) ;
}
static inline long
2006-02-21 16:05:11 +09:00
get_tv32 ( struct timeval * o , struct compat_timeval __user * i )
2005-04-16 15:20:36 -07:00
{
return ( ! access_ok ( VERIFY_READ , i , sizeof ( * i ) ) | |
( __get_user ( o - > tv_sec , & i - > tv_sec ) |
__get_user ( o - > tv_usec , & i - > tv_usec ) ) ) ;
}
static inline long
2006-02-21 16:05:11 +09:00
put_tv32 ( struct compat_timeval __user * o , struct timeval * i )
2005-04-16 15:20:36 -07:00
{
return ( ! access_ok ( VERIFY_WRITE , o , sizeof ( * o ) ) | |
( __put_user ( i - > tv_sec , & o - > tv_sec ) |
__put_user ( i - > tv_usec , & o - > tv_usec ) ) ) ;
}
extern struct timezone sys_tz ;
asmlinkage int
2006-02-21 16:05:11 +09:00
sys32_gettimeofday ( struct compat_timeval __user * tv , struct timezone __user * tz )
2005-04-16 15:20:36 -07:00
{
if ( tv ) {
struct timeval ktv ;
do_gettimeofday ( & ktv ) ;
if ( put_tv32 ( tv , & ktv ) )
return - EFAULT ;
}
if ( tz ) {
if ( copy_to_user ( tz , & sys_tz , sizeof ( sys_tz ) ) )
return - EFAULT ;
}
return 0 ;
}
2006-02-21 16:05:11 +09:00
static inline long get_ts32 ( struct timespec * o , struct compat_timeval __user * i )
2005-04-16 15:20:36 -07:00
{
long usec ;
if ( ! access_ok ( VERIFY_READ , i , sizeof ( * i ) ) )
return - EFAULT ;
if ( __get_user ( o - > tv_sec , & i - > tv_sec ) )
return - EFAULT ;
if ( __get_user ( usec , & i - > tv_usec ) )
return - EFAULT ;
o - > tv_nsec = usec * 1000 ;
return 0 ;
}
asmlinkage int
2006-02-21 16:05:11 +09:00
sys32_settimeofday ( struct compat_timeval __user * tv , struct timezone __user * tz )
2005-04-16 15:20:36 -07:00
{
struct timespec kts ;
struct timezone ktz ;
if ( tv ) {
if ( get_ts32 ( & kts , tv ) )
return - EFAULT ;
}
if ( tz ) {
if ( copy_from_user ( & ktz , tz , sizeof ( ktz ) ) )
return - EFAULT ;
}
return do_sys_settimeofday ( tv ? & kts : NULL , tz ? & ktz : NULL ) ;
}
asmlinkage int sys32_llseek ( unsigned int fd , unsigned int offset_high ,
2006-02-21 16:05:11 +09:00
unsigned int offset_low , loff_t __user * result ,
2005-04-16 15:20:36 -07:00
unsigned int origin )
{
return sys_llseek ( fd , offset_high , offset_low , result , origin ) ;
}
/* From the Single Unix Spec: pread & pwrite act like lseek to pos + op +
lseek back to original location . They fail just like lseek does on
non - seekable files . */
2006-02-21 16:05:11 +09:00
asmlinkage ssize_t sys32_pread ( unsigned int fd , char __user * buf ,
2005-04-16 15:20:36 -07:00
size_t count , u32 unused , u64 a4 , u64 a5 )
{
2006-04-26 07:28:09 +01:00
return sys_pread64 ( fd , buf , count , merge_64 ( a4 , a5 ) ) ;
2005-04-16 15:20:36 -07:00
}
2006-02-21 16:05:11 +09:00
asmlinkage ssize_t sys32_pwrite ( unsigned int fd , const char __user * buf ,
2005-04-16 15:20:36 -07:00
size_t count , u32 unused , u64 a4 , u64 a5 )
{
2006-04-26 07:28:09 +01:00
return sys_pwrite64 ( fd , buf , count , merge_64 ( a4 , a5 ) ) ;
2005-04-16 15:20:36 -07:00
}
asmlinkage int sys32_sched_rr_get_interval ( compat_pid_t pid ,
2006-02-21 16:05:11 +09:00
struct compat_timespec __user * interval )
2005-04-16 15:20:36 -07:00
{
struct timespec t ;
int ret ;
mm_segment_t old_fs = get_fs ( ) ;
set_fs ( KERNEL_DS ) ;
2006-02-21 16:05:11 +09:00
ret = sys_sched_rr_get_interval ( pid , ( struct timespec __user * ) & t ) ;
2005-04-16 15:20:36 -07:00
set_fs ( old_fs ) ;
if ( put_user ( t . tv_sec , & interval - > tv_sec ) | |
__put_user ( t . tv_nsec , & interval - > tv_nsec ) )
return - EFAULT ;
return ret ;
}
asmlinkage long
sys32_ipc ( u32 call , int first , int second , int third , u32 ptr , u32 fifth )
{
int version , err ;
version = call > > 16 ; /* hack for backward compatibility */
call & = 0xffff ;
switch ( call ) {
case SEMOP :
/* struct sembuf is the same on 32 and 64bit :)) */
2006-11-07 18:02:44 +09:00
err = sys_semtimedop ( first , compat_ptr ( ptr ) , second , NULL ) ;
2005-04-16 15:20:36 -07:00
break ;
case SEMTIMEDOP :
2006-11-07 18:02:44 +09:00
err = compat_sys_semtimedop ( first , compat_ptr ( ptr ) , second ,
compat_ptr ( fifth ) ) ;
2005-04-16 15:20:36 -07:00
break ;
case SEMGET :
2006-11-07 18:02:44 +09:00
err = sys_semget ( first , second , third ) ;
2005-04-16 15:20:36 -07:00
break ;
case SEMCTL :
2006-11-07 18:02:44 +09:00
err = compat_sys_semctl ( first , second , third , compat_ptr ( ptr ) ) ;
2005-04-16 15:20:36 -07:00
break ;
case MSGSND :
2006-11-07 18:02:44 +09:00
err = compat_sys_msgsnd ( first , second , third , compat_ptr ( ptr ) ) ;
2005-04-16 15:20:36 -07:00
break ;
case MSGRCV :
2006-11-07 18:02:44 +09:00
err = compat_sys_msgrcv ( first , second , fifth , third ,
version , compat_ptr ( ptr ) ) ;
2005-04-16 15:20:36 -07:00
break ;
case MSGGET :
2006-11-07 18:02:44 +09:00
err = sys_msgget ( ( key_t ) first , second ) ;
2005-04-16 15:20:36 -07:00
break ;
case MSGCTL :
2006-11-07 18:02:44 +09:00
err = compat_sys_msgctl ( first , second , compat_ptr ( ptr ) ) ;
2005-04-16 15:20:36 -07:00
break ;
case SHMAT :
2006-11-07 18:02:44 +09:00
err = compat_sys_shmat ( first , second , third , version ,
compat_ptr ( ptr ) ) ;
2005-04-16 15:20:36 -07:00
break ;
case SHMDT :
2006-11-07 18:02:44 +09:00
err = sys_shmdt ( compat_ptr ( ptr ) ) ;
2005-04-16 15:20:36 -07:00
break ;
case SHMGET :
2006-11-07 18:02:44 +09:00
err = sys_shmget ( first , ( unsigned ) second , third ) ;
2005-04-16 15:20:36 -07:00
break ;
case SHMCTL :
2006-11-07 18:02:44 +09:00
err = compat_sys_shmctl ( first , second , compat_ptr ( ptr ) ) ;
2005-04-16 15:20:36 -07:00
break ;
default :
err = - EINVAL ;
break ;
}
return err ;
}
2006-11-07 18:02:44 +09:00
# ifdef CONFIG_MIPS32_N32
2007-01-10 18:53:33 +09:00
asmlinkage long sysn32_semctl ( int semid , int semnum , int cmd , u32 arg )
2005-04-16 15:20:36 -07:00
{
2006-11-07 18:02:44 +09:00
/* compat_sys_semctl expects a pointer to union semun */
u32 __user * uptr = compat_alloc_user_space ( sizeof ( u32 ) ) ;
2007-01-10 18:53:33 +09:00
if ( put_user ( arg , uptr ) )
2006-11-07 18:02:44 +09:00
return - EFAULT ;
return compat_sys_semctl ( semid , semnum , cmd , uptr ) ;
2005-04-16 15:20:36 -07:00
}
2007-01-10 18:53:33 +09:00
asmlinkage long sysn32_msgsnd ( int msqid , u32 msgp , unsigned msgsz , int msgflg )
{
return compat_sys_msgsnd ( msqid , msgsz , msgflg , compat_ptr ( msgp ) ) ;
}
asmlinkage long sysn32_msgrcv ( int msqid , u32 msgp , size_t msgsz , int msgtyp ,
int msgflg )
{
return compat_sys_msgrcv ( msqid , msgsz , msgtyp , msgflg , IPC_64 ,
compat_ptr ( msgp ) ) ;
}
2006-11-07 18:02:44 +09:00
# endif
2005-04-16 15:20:36 -07:00
struct sysctl_args32
{
compat_caddr_t name ;
int nlen ;
compat_caddr_t oldval ;
compat_caddr_t oldlenp ;
compat_caddr_t newval ;
compat_size_t newlen ;
unsigned int __unused [ 4 ] ;
} ;
2006-09-27 01:51:04 -07:00
# ifdef CONFIG_SYSCTL_SYSCALL
2005-04-16 15:20:36 -07:00
2006-02-21 16:05:11 +09:00
asmlinkage long sys32_sysctl ( struct sysctl_args32 __user * args )
2005-04-16 15:20:36 -07:00
{
struct sysctl_args32 tmp ;
int error ;
2006-02-21 16:05:11 +09:00
size_t oldlen ;
size_t __user * oldlenp = NULL ;
unsigned long addr = ( ( ( unsigned long ) & args - > __unused [ 0 ] ) + 7 ) & ~ 7 ;
2005-04-16 15:20:36 -07:00
if ( copy_from_user ( & tmp , args , sizeof ( tmp ) ) )
return - EFAULT ;
if ( tmp . oldval & & tmp . oldlenp ) {
/* Duh, this is ugly and might not work if sysctl_args
is in read - only memory , but do_sysctl does indirectly
a lot of uaccess in both directions and we ' d have to
basically copy the whole sysctl . c here , and
glibc ' s __sysctl uses rw memory for the structure
anyway . */
2006-02-21 16:05:11 +09:00
if ( get_user ( oldlen , ( u32 __user * ) A ( tmp . oldlenp ) ) | |
put_user ( oldlen , ( size_t __user * ) addr ) )
2005-04-16 15:20:36 -07:00
return - EFAULT ;
2006-02-21 16:05:11 +09:00
oldlenp = ( size_t __user * ) addr ;
2005-04-16 15:20:36 -07:00
}
lock_kernel ( ) ;
2006-02-21 16:05:11 +09:00
error = do_sysctl ( ( int __user * ) A ( tmp . name ) , tmp . nlen , ( void __user * ) A ( tmp . oldval ) ,
oldlenp , ( void __user * ) A ( tmp . newval ) , tmp . newlen ) ;
2005-04-16 15:20:36 -07:00
unlock_kernel ( ) ;
if ( oldlenp ) {
if ( ! error ) {
2006-02-21 16:05:11 +09:00
if ( get_user ( oldlen , ( size_t __user * ) addr ) | |
put_user ( oldlen , ( u32 __user * ) A ( tmp . oldlenp ) ) )
2005-04-16 15:20:36 -07:00
error = - EFAULT ;
}
copy_to_user ( args - > __unused , tmp . __unused , sizeof ( tmp . __unused ) ) ;
}
return error ;
}
2006-09-27 01:51:04 -07:00
# endif /* CONFIG_SYSCTL_SYSCALL */
2005-04-16 15:20:36 -07:00
2006-02-21 16:05:11 +09:00
asmlinkage long sys32_newuname ( struct new_utsname __user * name )
2005-04-16 15:20:36 -07:00
{
int ret = 0 ;
down_read ( & uts_sem ) ;
2006-10-02 02:18:11 -07:00
if ( copy_to_user ( name , utsname ( ) , sizeof * name ) )
2005-04-16 15:20:36 -07:00
ret = - EFAULT ;
up_read ( & uts_sem ) ;
if ( current - > personality = = PER_LINUX32 & & ! ret )
if ( copy_to_user ( name - > machine , " mips \0 \0 \0 " , 8 ) )
ret = - EFAULT ;
return ret ;
}
asmlinkage int sys32_personality ( unsigned long personality )
{
int ret ;
2006-08-13 00:53:29 +01:00
personality & = 0xffffffff ;
if ( personality ( current - > personality ) = = PER_LINUX32 & &
personality = = PER_LINUX )
2005-04-16 15:20:36 -07:00
personality = PER_LINUX32 ;
ret = sys_personality ( personality ) ;
if ( ret = = PER_LINUX32 )
ret = PER_LINUX ;
return ret ;
}
/* ustat compatibility */
struct ustat32 {
compat_daddr_t f_tfree ;
compat_ino_t f_tinode ;
char f_fname [ 6 ] ;
char f_fpack [ 6 ] ;
} ;
2006-02-21 16:05:11 +09:00
extern asmlinkage long sys_ustat ( dev_t dev , struct ustat __user * ubuf ) ;
2005-04-16 15:20:36 -07:00
2006-02-21 16:05:11 +09:00
asmlinkage int sys32_ustat ( dev_t dev , struct ustat32 __user * ubuf32 )
2005-04-16 15:20:36 -07:00
{
int err ;
2007-02-05 00:10:11 +00:00
struct ustat tmp ;
2005-04-16 15:20:36 -07:00
struct ustat32 tmp32 ;
mm_segment_t old_fs = get_fs ( ) ;
set_fs ( KERNEL_DS ) ;
2006-02-21 16:05:11 +09:00
err = sys_ustat ( dev , ( struct ustat __user * ) & tmp ) ;
2005-04-16 15:20:36 -07:00
set_fs ( old_fs ) ;
if ( err )
goto out ;
2007-02-05 00:10:11 +00:00
memset ( & tmp32 , 0 , sizeof ( struct ustat32 ) ) ;
tmp32 . f_tfree = tmp . f_tfree ;
tmp32 . f_tinode = tmp . f_tinode ;
2005-04-16 15:20:36 -07:00
2007-02-05 00:10:11 +00:00
err = copy_to_user ( ubuf32 , & tmp32 , sizeof ( struct ustat32 ) ) ? - EFAULT : 0 ;
2005-04-16 15:20:36 -07:00
out :
return err ;
}
2006-02-21 16:05:11 +09:00
asmlinkage int sys32_sendfile ( int out_fd , int in_fd , compat_off_t __user * offset ,
2005-04-16 15:20:36 -07:00
s32 count )
{
mm_segment_t old_fs = get_fs ( ) ;
int ret ;
off_t of ;
2005-09-03 15:56:17 -07:00
2005-04-16 15:20:36 -07:00
if ( offset & & get_user ( of , offset ) )
return - EFAULT ;
2005-09-03 15:56:17 -07:00
2005-04-16 15:20:36 -07:00
set_fs ( KERNEL_DS ) ;
2006-02-21 16:05:11 +09:00
ret = sys_sendfile ( out_fd , in_fd , offset ? ( off_t __user * ) & of : NULL , count ) ;
2005-04-16 15:20:36 -07:00
set_fs ( old_fs ) ;
2005-09-03 15:56:17 -07:00
2005-04-16 15:20:36 -07:00
if ( offset & & put_user ( of , offset ) )
return - EFAULT ;
2005-09-03 15:56:17 -07:00
2005-04-16 15:20:36 -07:00
return ret ;
}
asmlinkage ssize_t sys32_readahead ( int fd , u32 pad0 , u64 a2 , u64 a3 ,
size_t count )
{
return sys_readahead ( fd , merge_64 ( a2 , a3 ) , count ) ;
}
2006-04-01 07:49:21 +01:00
asmlinkage long sys32_sync_file_range ( int fd , int __pad ,
unsigned long a2 , unsigned long a3 ,
unsigned long a4 , unsigned long a5 ,
int flags )
{
return sys_sync_file_range ( fd ,
merge_64 ( a2 , a3 ) , merge_64 ( a4 , a5 ) ,
flags ) ;
}
2005-04-13 17:43:59 +00:00
save_static_function ( sys32_clone ) ;
__attribute_used__ noinline static int
_sys32_clone ( nabi_no_regargs struct pt_regs regs )
{
unsigned long clone_flags ;
unsigned long newsp ;
int __user * parent_tidptr , * child_tidptr ;
clone_flags = regs . regs [ 4 ] ;
newsp = regs . regs [ 5 ] ;
if ( ! newsp )
newsp = regs . regs [ 29 ] ;
2006-02-21 16:05:11 +09:00
parent_tidptr = ( int __user * ) regs . regs [ 6 ] ;
2005-04-13 17:43:59 +00:00
/* Use __dummy4 instead of getting it off the stack, so that
syscall ( ) works . */
child_tidptr = ( int __user * ) __dummy4 ;
return do_fork ( clone_flags , newsp , & regs , 0 ,
parent_tidptr , child_tidptr ) ;
}
2007-02-13 00:05:11 +00:00
/*
* Implement the event wait interface for the eventpoll file . It is the kernel
* part of the user space epoll_pwait ( 2 ) .
*/
asmlinkage long compat_sys_epoll_pwait ( int epfd ,
struct epoll_event __user * events , int maxevents , int timeout ,
const compat_sigset_t __user * sigmask , size_t sigsetsize )
{
int error ;
sigset_t ksigmask , sigsaved ;
/*
* If the caller wants a certain signal mask to be set during the wait ,
* we apply it here .
*/
if ( sigmask ) {
if ( sigsetsize ! = sizeof ( sigset_t ) )
return - EINVAL ;
if ( ! access_ok ( VERIFY_READ , sigmask , sizeof ( ksigmask ) ) )
return - EFAULT ;
if ( __copy_conv_sigset_from_user ( & ksigmask , sigmask ) )
return - EFAULT ;
sigdelsetmask ( & ksigmask , sigmask ( SIGKILL ) | sigmask ( SIGSTOP ) ) ;
sigprocmask ( SIG_SETMASK , & ksigmask , & sigsaved ) ;
}
error = sys_epoll_wait ( epfd , events , maxevents , timeout ) ;
/*
* If we changed the signal mask , we need to restore the original one .
* In case we ' ve got a signal while waiting , we do not restore the
* signal mask yet , and we allow do_signal ( ) to deliver the signal on
* the way back to userspace , before the signal mask is restored .
*/
if ( sigmask ) {
if ( error = = - EINTR ) {
memcpy ( & current - > saved_sigmask , & sigsaved ,
sizeof ( sigsaved ) ) ;
set_thread_flag ( TIF_RESTORE_SIGMASK ) ;
} else
sigprocmask ( SIG_SETMASK , & sigsaved , NULL ) ;
}
return error ;
}