1 // SPDX-License-Identifier: GPL-2.0
3 * sys_ia32.c: Conversion between 32bit and 64bit native syscalls. Based on
6 * Copyright (C) 2000 VA Linux Co
7 * Copyright (C) 2000 Don Dugger <n0ano@valinux.com>
8 * Copyright (C) 1999 Arun Sharma <arun.sharma@intel.com>
9 * Copyright (C) 1997,1998 Jakub Jelinek (jj@sunsite.mff.cuni.cz)
10 * Copyright (C) 1997 David S. Miller (davem@caip.rutgers.edu)
11 * Copyright (C) 2000 Hewlett-Packard Co.
12 * Copyright (C) 2000 David Mosberger-Tang <davidm@hpl.hp.com>
13 * Copyright (C) 2000,2001,2002 Andi Kleen, SuSE Labs (x86-64 port)
15 * These routines maintain argument size conversion between 32bit and 64bit
16 * environment. In 2.5 most of this should be moved to a generic directory.
18 * This file assumes that there is a hole at the end of user address space.
20 * Some of the functions are LE specific currently. These are
21 * hopefully all marked. This should be fixed.
24 #include <linux/kernel.h>
25 #include <linux/sched.h>
27 #include <linux/file.h>
28 #include <linux/signal.h>
29 #include <linux/syscalls.h>
30 #include <linux/times.h>
31 #include <linux/utsname.h>
33 #include <linux/uio.h>
34 #include <linux/poll.h>
35 #include <linux/personality.h>
36 #include <linux/stat.h>
37 #include <linux/rwsem.h>
38 #include <linux/compat.h>
39 #include <linux/vfs.h>
40 #include <linux/ptrace.h>
41 #include <linux/highuid.h>
42 #include <linux/sysctl.h>
43 #include <linux/slab.h>
44 #include <linux/sched/task.h>
46 #include <asm/types.h>
47 #include <linux/uaccess.h>
48 #include <linux/atomic.h>
49 #include <asm/vgtod.h>
52 #define AA(__x) ((unsigned long)(__x))
54 SYSCALL_DEFINE3(ia32_truncate64
, const char __user
*, filename
,
55 unsigned long, offset_low
, unsigned long, offset_high
)
57 return ksys_truncate(filename
,
58 ((loff_t
) offset_high
<< 32) | offset_low
);
61 SYSCALL_DEFINE3(ia32_ftruncate64
, unsigned int, fd
,
62 unsigned long, offset_low
, unsigned long, offset_high
)
64 return ksys_ftruncate(fd
, ((loff_t
) offset_high
<< 32) | offset_low
);
67 /* warning: next two assume little endian */
68 SYSCALL_DEFINE5(ia32_pread64
, unsigned int, fd
, char __user
*, ubuf
,
69 u32
, count
, u32
, poslo
, u32
, poshi
)
71 return ksys_pread64(fd
, ubuf
, count
,
72 ((loff_t
)AA(poshi
) << 32) | AA(poslo
));
75 SYSCALL_DEFINE5(ia32_pwrite64
, unsigned int, fd
, const char __user
*, ubuf
,
76 u32
, count
, u32
, poslo
, u32
, poshi
)
78 return ksys_pwrite64(fd
, ubuf
, count
,
79 ((loff_t
)AA(poshi
) << 32) | AA(poslo
));
84 * Some system calls that need sign extended arguments. This could be
85 * done by a generic wrapper.
87 SYSCALL_DEFINE6(ia32_fadvise64_64
, int, fd
, __u32
, offset_low
,
88 __u32
, offset_high
, __u32
, len_low
, __u32
, len_high
,
91 return ksys_fadvise64_64(fd
,
92 (((u64
)offset_high
)<<32) | offset_low
,
93 (((u64
)len_high
)<<32) | len_low
,
97 SYSCALL_DEFINE4(ia32_readahead
, int, fd
, unsigned int, off_lo
,
98 unsigned int, off_hi
, size_t, count
)
100 return ksys_readahead(fd
, ((u64
)off_hi
<< 32) | off_lo
, count
);
103 SYSCALL_DEFINE6(ia32_sync_file_range
, int, fd
, unsigned int, off_low
,
104 unsigned int, off_hi
, unsigned int, n_low
,
105 unsigned int, n_hi
, int, flags
)
107 return ksys_sync_file_range(fd
,
108 ((u64
)off_hi
<< 32) | off_low
,
109 ((u64
)n_hi
<< 32) | n_low
, flags
);
112 SYSCALL_DEFINE5(ia32_fadvise64
, int, fd
, unsigned int, offset_lo
,
113 unsigned int, offset_hi
, size_t, len
, int, advice
)
115 return ksys_fadvise64_64(fd
, ((u64
)offset_hi
<< 32) | offset_lo
,
119 SYSCALL_DEFINE6(ia32_fallocate
, int, fd
, int, mode
,
120 unsigned int, offset_lo
, unsigned int, offset_hi
,
121 unsigned int, len_lo
, unsigned int, len_hi
)
123 return ksys_fallocate(fd
, mode
, ((u64
)offset_hi
<< 32) | offset_lo
,
124 ((u64
)len_hi
<< 32) | len_lo
);
127 #ifdef CONFIG_IA32_EMULATION
129 * Another set for IA32/LFS -- x86_64 struct stat is different due to
130 * support for 64bit inode numbers.
132 static int cp_stat64(struct stat64 __user
*ubuf
, struct kstat
*stat
)
134 typeof(ubuf
->st_uid
) uid
= 0;
135 typeof(ubuf
->st_gid
) gid
= 0;
136 SET_UID(uid
, from_kuid_munged(current_user_ns(), stat
->uid
));
137 SET_GID(gid
, from_kgid_munged(current_user_ns(), stat
->gid
));
138 if (!access_ok(ubuf
, sizeof(struct stat64
)) ||
139 __put_user(huge_encode_dev(stat
->dev
), &ubuf
->st_dev
) ||
140 __put_user(stat
->ino
, &ubuf
->__st_ino
) ||
141 __put_user(stat
->ino
, &ubuf
->st_ino
) ||
142 __put_user(stat
->mode
, &ubuf
->st_mode
) ||
143 __put_user(stat
->nlink
, &ubuf
->st_nlink
) ||
144 __put_user(uid
, &ubuf
->st_uid
) ||
145 __put_user(gid
, &ubuf
->st_gid
) ||
146 __put_user(huge_encode_dev(stat
->rdev
), &ubuf
->st_rdev
) ||
147 __put_user(stat
->size
, &ubuf
->st_size
) ||
148 __put_user(stat
->atime
.tv_sec
, &ubuf
->st_atime
) ||
149 __put_user(stat
->atime
.tv_nsec
, &ubuf
->st_atime_nsec
) ||
150 __put_user(stat
->mtime
.tv_sec
, &ubuf
->st_mtime
) ||
151 __put_user(stat
->mtime
.tv_nsec
, &ubuf
->st_mtime_nsec
) ||
152 __put_user(stat
->ctime
.tv_sec
, &ubuf
->st_ctime
) ||
153 __put_user(stat
->ctime
.tv_nsec
, &ubuf
->st_ctime_nsec
) ||
154 __put_user(stat
->blksize
, &ubuf
->st_blksize
) ||
155 __put_user(stat
->blocks
, &ubuf
->st_blocks
))
160 COMPAT_SYSCALL_DEFINE2(ia32_stat64
, const char __user
*, filename
,
161 struct stat64 __user
*, statbuf
)
164 int ret
= vfs_stat(filename
, &stat
);
167 ret
= cp_stat64(statbuf
, &stat
);
171 COMPAT_SYSCALL_DEFINE2(ia32_lstat64
, const char __user
*, filename
,
172 struct stat64 __user
*, statbuf
)
175 int ret
= vfs_lstat(filename
, &stat
);
177 ret
= cp_stat64(statbuf
, &stat
);
181 COMPAT_SYSCALL_DEFINE2(ia32_fstat64
, unsigned int, fd
,
182 struct stat64 __user
*, statbuf
)
185 int ret
= vfs_fstat(fd
, &stat
);
187 ret
= cp_stat64(statbuf
, &stat
);
191 COMPAT_SYSCALL_DEFINE4(ia32_fstatat64
, unsigned int, dfd
,
192 const char __user
*, filename
,
193 struct stat64 __user
*, statbuf
, int, flag
)
198 error
= vfs_fstatat(dfd
, filename
, &stat
, flag
);
201 return cp_stat64(statbuf
, &stat
);
205 * Linux/i386 didn't use to be able to handle more than
206 * 4 system call parameters, so these system calls used a memory
207 * block for parameter passing..
210 struct mmap_arg_struct32
{
219 COMPAT_SYSCALL_DEFINE1(ia32_mmap
, struct mmap_arg_struct32 __user
*, arg
)
221 struct mmap_arg_struct32 a
;
223 if (copy_from_user(&a
, arg
, sizeof(a
)))
226 if (a
.offset
& ~PAGE_MASK
)
229 return ksys_mmap_pgoff(a
.addr
, a
.len
, a
.prot
, a
.flags
, a
.fd
,
230 a
.offset
>>PAGE_SHIFT
);
234 * The 32-bit clone ABI is CONFIG_CLONE_BACKWARDS
236 COMPAT_SYSCALL_DEFINE5(ia32_clone
, unsigned long, clone_flags
,
237 unsigned long, newsp
, int __user
*, parent_tidptr
,
238 unsigned long, tls_val
, int __user
*, child_tidptr
)
240 struct kernel_clone_args args
= {
241 .flags
= (clone_flags
& ~CSIGNAL
),
242 .pidfd
= parent_tidptr
,
243 .child_tid
= child_tidptr
,
244 .parent_tid
= parent_tidptr
,
245 .exit_signal
= (clone_flags
& CSIGNAL
),
250 if (!legacy_clone_args_valid(&args
))
253 return _do_fork(&args
);
255 #endif /* CONFIG_IA32_EMULATION */