1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0
2a88b5ba8SSam Ravnborg /* sys_sparc32.c: Conversion between 32bit and 64bit native syscalls.
3a88b5ba8SSam Ravnborg *
4a88b5ba8SSam Ravnborg * Copyright (C) 1997,1998 Jakub Jelinek (jj@sunsite.mff.cuni.cz)
5a88b5ba8SSam Ravnborg * Copyright (C) 1997, 2007 David S. Miller (davem@davemloft.net)
6a88b5ba8SSam Ravnborg *
7a88b5ba8SSam Ravnborg * These routines maintain argument size conversion between 32bit and 64bit
8a88b5ba8SSam Ravnborg * environment.
9a88b5ba8SSam Ravnborg */
10a88b5ba8SSam Ravnborg
11a88b5ba8SSam Ravnborg #include <linux/kernel.h>
12a88b5ba8SSam Ravnborg #include <linux/sched.h>
13a88b5ba8SSam Ravnborg #include <linux/capability.h>
14a88b5ba8SSam Ravnborg #include <linux/fs.h>
15a88b5ba8SSam Ravnborg #include <linux/mm.h>
16a88b5ba8SSam Ravnborg #include <linux/file.h>
17a88b5ba8SSam Ravnborg #include <linux/signal.h>
18a88b5ba8SSam Ravnborg #include <linux/resource.h>
19a88b5ba8SSam Ravnborg #include <linux/times.h>
20a88b5ba8SSam Ravnborg #include <linux/smp.h>
21a88b5ba8SSam Ravnborg #include <linux/sem.h>
22a88b5ba8SSam Ravnborg #include <linux/msg.h>
23a88b5ba8SSam Ravnborg #include <linux/shm.h>
24a88b5ba8SSam Ravnborg #include <linux/uio.h>
25a88b5ba8SSam Ravnborg #include <linux/quota.h>
26a88b5ba8SSam Ravnborg #include <linux/poll.h>
27a88b5ba8SSam Ravnborg #include <linux/personality.h>
28a88b5ba8SSam Ravnborg #include <linux/stat.h>
29a88b5ba8SSam Ravnborg #include <linux/filter.h>
30a88b5ba8SSam Ravnborg #include <linux/highmem.h>
31a88b5ba8SSam Ravnborg #include <linux/highuid.h>
32a88b5ba8SSam Ravnborg #include <linux/mman.h>
33a88b5ba8SSam Ravnborg #include <linux/ipv6.h>
34a88b5ba8SSam Ravnborg #include <linux/in.h>
35a88b5ba8SSam Ravnborg #include <linux/icmpv6.h>
36a88b5ba8SSam Ravnborg #include <linux/syscalls.h>
37a88b5ba8SSam Ravnborg #include <linux/sysctl.h>
38a88b5ba8SSam Ravnborg #include <linux/binfmts.h>
39a88b5ba8SSam Ravnborg #include <linux/dnotify.h>
40a88b5ba8SSam Ravnborg #include <linux/security.h>
41a88b5ba8SSam Ravnborg #include <linux/compat.h>
42a88b5ba8SSam Ravnborg #include <linux/vfs.h>
43a88b5ba8SSam Ravnborg #include <linux/ptrace.h>
445a0e3ad6STejun Heo #include <linux/slab.h>
45a88b5ba8SSam Ravnborg
46a88b5ba8SSam Ravnborg #include <asm/types.h>
477c0f6ba6SLinus Torvalds #include <linux/uaccess.h>
48a88b5ba8SSam Ravnborg #include <asm/fpumacro.h>
49a88b5ba8SSam Ravnborg #include <asm/mmu_context.h>
50a88b5ba8SSam Ravnborg #include <asm/compat_signal.h>
51a88b5ba8SSam Ravnborg
52ed8eb755SSam Ravnborg #include "systbls.h"
53ed8eb755SSam Ravnborg
COMPAT_SYSCALL_DEFINE3(truncate64,const char __user *,path,u32,high,u32,low)543d0e354eSAl Viro COMPAT_SYSCALL_DEFINE3(truncate64, const char __user *, path, u32, high, u32, low)
55a88b5ba8SSam Ravnborg {
56*8844a509SLinus Torvalds return ksys_truncate(path, ((u64)high << 32) | low);
57a88b5ba8SSam Ravnborg }
58a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE3(ftruncate64,unsigned int,fd,u32,high,u32,low)593d0e354eSAl Viro COMPAT_SYSCALL_DEFINE3(ftruncate64, unsigned int, fd, u32, high, u32, low)
60a88b5ba8SSam Ravnborg {
61*8844a509SLinus Torvalds return ksys_ftruncate(fd, ((u64)high << 32) | low);
62a88b5ba8SSam Ravnborg }
63a88b5ba8SSam Ravnborg
cp_compat_stat64(struct kstat * stat,struct compat_stat64 __user * statbuf)64a88b5ba8SSam Ravnborg static int cp_compat_stat64(struct kstat *stat,
65a88b5ba8SSam Ravnborg struct compat_stat64 __user *statbuf)
66a88b5ba8SSam Ravnborg {
67a88b5ba8SSam Ravnborg int err;
68a88b5ba8SSam Ravnborg
69a88b5ba8SSam Ravnborg err = put_user(huge_encode_dev(stat->dev), &statbuf->st_dev);
70a88b5ba8SSam Ravnborg err |= put_user(stat->ino, &statbuf->st_ino);
71a88b5ba8SSam Ravnborg err |= put_user(stat->mode, &statbuf->st_mode);
72a88b5ba8SSam Ravnborg err |= put_user(stat->nlink, &statbuf->st_nlink);
73a7c1938eSEric W. Biederman err |= put_user(from_kuid_munged(current_user_ns(), stat->uid), &statbuf->st_uid);
74a7c1938eSEric W. Biederman err |= put_user(from_kgid_munged(current_user_ns(), stat->gid), &statbuf->st_gid);
75a88b5ba8SSam Ravnborg err |= put_user(huge_encode_dev(stat->rdev), &statbuf->st_rdev);
76a88b5ba8SSam Ravnborg err |= put_user(0, (unsigned long __user *) &statbuf->__pad3[0]);
77a88b5ba8SSam Ravnborg err |= put_user(stat->size, &statbuf->st_size);
78a88b5ba8SSam Ravnborg err |= put_user(stat->blksize, &statbuf->st_blksize);
79a88b5ba8SSam Ravnborg err |= put_user(0, (unsigned int __user *) &statbuf->__pad4[0]);
80a88b5ba8SSam Ravnborg err |= put_user(0, (unsigned int __user *) &statbuf->__pad4[4]);
81a88b5ba8SSam Ravnborg err |= put_user(stat->blocks, &statbuf->st_blocks);
82a88b5ba8SSam Ravnborg err |= put_user(stat->atime.tv_sec, &statbuf->st_atime);
83a88b5ba8SSam Ravnborg err |= put_user(stat->atime.tv_nsec, &statbuf->st_atime_nsec);
84a88b5ba8SSam Ravnborg err |= put_user(stat->mtime.tv_sec, &statbuf->st_mtime);
85a88b5ba8SSam Ravnborg err |= put_user(stat->mtime.tv_nsec, &statbuf->st_mtime_nsec);
86a88b5ba8SSam Ravnborg err |= put_user(stat->ctime.tv_sec, &statbuf->st_ctime);
87a88b5ba8SSam Ravnborg err |= put_user(stat->ctime.tv_nsec, &statbuf->st_ctime_nsec);
88a88b5ba8SSam Ravnborg err |= put_user(0, &statbuf->__unused4);
89a88b5ba8SSam Ravnborg err |= put_user(0, &statbuf->__unused5);
90a88b5ba8SSam Ravnborg
91a88b5ba8SSam Ravnborg return err;
92a88b5ba8SSam Ravnborg }
93a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE2(stat64,const char __user *,filename,struct compat_stat64 __user *,statbuf)94ee076e81SAl Viro COMPAT_SYSCALL_DEFINE2(stat64, const char __user *, filename,
95ee076e81SAl Viro struct compat_stat64 __user *, statbuf)
96a88b5ba8SSam Ravnborg {
97a88b5ba8SSam Ravnborg struct kstat stat;
98a88b5ba8SSam Ravnborg int error = vfs_stat(filename, &stat);
99a88b5ba8SSam Ravnborg
100a88b5ba8SSam Ravnborg if (!error)
101a88b5ba8SSam Ravnborg error = cp_compat_stat64(&stat, statbuf);
102a88b5ba8SSam Ravnborg return error;
103a88b5ba8SSam Ravnborg }
104a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE2(lstat64,const char __user *,filename,struct compat_stat64 __user *,statbuf)105ee076e81SAl Viro COMPAT_SYSCALL_DEFINE2(lstat64, const char __user *, filename,
106ee076e81SAl Viro struct compat_stat64 __user *, statbuf)
107a88b5ba8SSam Ravnborg {
108a88b5ba8SSam Ravnborg struct kstat stat;
109a88b5ba8SSam Ravnborg int error = vfs_lstat(filename, &stat);
110a88b5ba8SSam Ravnborg
111a88b5ba8SSam Ravnborg if (!error)
112a88b5ba8SSam Ravnborg error = cp_compat_stat64(&stat, statbuf);
113a88b5ba8SSam Ravnborg return error;
114a88b5ba8SSam Ravnborg }
115a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE2(fstat64,unsigned int,fd,struct compat_stat64 __user *,statbuf)116ee076e81SAl Viro COMPAT_SYSCALL_DEFINE2(fstat64, unsigned int, fd,
117ee076e81SAl Viro struct compat_stat64 __user *, statbuf)
118a88b5ba8SSam Ravnborg {
119a88b5ba8SSam Ravnborg struct kstat stat;
120a88b5ba8SSam Ravnborg int error = vfs_fstat(fd, &stat);
121a88b5ba8SSam Ravnborg
122a88b5ba8SSam Ravnborg if (!error)
123a88b5ba8SSam Ravnborg error = cp_compat_stat64(&stat, statbuf);
124a88b5ba8SSam Ravnborg return error;
125a88b5ba8SSam Ravnborg }
126a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE4(fstatat64,unsigned int,dfd,const char __user *,filename,struct compat_stat64 __user *,statbuf,int,flag)127ee076e81SAl Viro COMPAT_SYSCALL_DEFINE4(fstatat64, unsigned int, dfd,
128ee076e81SAl Viro const char __user *, filename,
129ee076e81SAl Viro struct compat_stat64 __user *, statbuf, int, flag)
130a88b5ba8SSam Ravnborg {
131a88b5ba8SSam Ravnborg struct kstat stat;
1320112fc22SOleg Drokin int error;
133a88b5ba8SSam Ravnborg
1340112fc22SOleg Drokin error = vfs_fstatat(dfd, filename, &stat, flag);
1350112fc22SOleg Drokin if (error)
136a88b5ba8SSam Ravnborg return error;
1370112fc22SOleg Drokin return cp_compat_stat64(&stat, statbuf);
138a88b5ba8SSam Ravnborg }
139a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE3(sparc_sigaction,int,sig,struct compat_old_sigaction __user *,act,struct compat_old_sigaction __user *,oact)140a274bd49SAl Viro COMPAT_SYSCALL_DEFINE3(sparc_sigaction, int, sig,
141a274bd49SAl Viro struct compat_old_sigaction __user *,act,
142a274bd49SAl Viro struct compat_old_sigaction __user *,oact)
143a88b5ba8SSam Ravnborg {
144a88b5ba8SSam Ravnborg WARN_ON_ONCE(sig >= 0);
145a274bd49SAl Viro return compat_sys_sigaction(-sig, act, oact);
146a88b5ba8SSam Ravnborg }
147a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE5(rt_sigaction,int,sig,struct compat_sigaction __user *,act,struct compat_sigaction __user *,oact,void __user *,restorer,compat_size_t,sigsetsize)148bdc40abfSAl Viro COMPAT_SYSCALL_DEFINE5(rt_sigaction, int, sig,
149bdc40abfSAl Viro struct compat_sigaction __user *,act,
150bdc40abfSAl Viro struct compat_sigaction __user *,oact,
151bdc40abfSAl Viro void __user *,restorer,
152bdc40abfSAl Viro compat_size_t,sigsetsize)
153a88b5ba8SSam Ravnborg {
154a88b5ba8SSam Ravnborg struct k_sigaction new_ka, old_ka;
155a88b5ba8SSam Ravnborg int ret;
156a88b5ba8SSam Ravnborg
157a88b5ba8SSam Ravnborg /* XXX: Don't preclude handling different sized sigset_t's. */
158a88b5ba8SSam Ravnborg if (sigsetsize != sizeof(compat_sigset_t))
159a88b5ba8SSam Ravnborg return -EINVAL;
160a88b5ba8SSam Ravnborg
161a88b5ba8SSam Ravnborg if (act) {
162a88b5ba8SSam Ravnborg u32 u_handler, u_restorer;
163a88b5ba8SSam Ravnborg
164a88b5ba8SSam Ravnborg new_ka.ka_restorer = restorer;
165a88b5ba8SSam Ravnborg ret = get_user(u_handler, &act->sa_handler);
166a88b5ba8SSam Ravnborg new_ka.sa.sa_handler = compat_ptr(u_handler);
1673968cf62SAl Viro ret |= get_compat_sigset(&new_ka.sa.sa_mask, &act->sa_mask);
1683ddc5b46SMathieu Desnoyers ret |= get_user(new_ka.sa.sa_flags, &act->sa_flags);
1693ddc5b46SMathieu Desnoyers ret |= get_user(u_restorer, &act->sa_restorer);
170a88b5ba8SSam Ravnborg new_ka.sa.sa_restorer = compat_ptr(u_restorer);
171a88b5ba8SSam Ravnborg if (ret)
172a88b5ba8SSam Ravnborg return -EFAULT;
173a88b5ba8SSam Ravnborg }
174a88b5ba8SSam Ravnborg
175a88b5ba8SSam Ravnborg ret = do_sigaction(sig, act ? &new_ka : NULL, oact ? &old_ka : NULL);
176a88b5ba8SSam Ravnborg
177a88b5ba8SSam Ravnborg if (!ret && oact) {
178a88b5ba8SSam Ravnborg ret = put_user(ptr_to_compat(old_ka.sa.sa_handler), &oact->sa_handler);
179f454322eSDmitry V. Levin ret |= put_compat_sigset(&oact->sa_mask, &old_ka.sa.sa_mask,
180f454322eSDmitry V. Levin sizeof(oact->sa_mask));
1813ddc5b46SMathieu Desnoyers ret |= put_user(old_ka.sa.sa_flags, &oact->sa_flags);
1823ddc5b46SMathieu Desnoyers ret |= put_user(ptr_to_compat(old_ka.sa.sa_restorer), &oact->sa_restorer);
183a88b5ba8SSam Ravnborg if (ret)
184a88b5ba8SSam Ravnborg ret = -EFAULT;
185a88b5ba8SSam Ravnborg }
186a88b5ba8SSam Ravnborg
187a88b5ba8SSam Ravnborg return ret;
188a88b5ba8SSam Ravnborg }
189a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE5(pread64,unsigned int,fd,char __user *,ubuf,compat_size_t,count,u32,poshi,u32,poslo)1908c82ccd6SAl Viro COMPAT_SYSCALL_DEFINE5(pread64, unsigned int, fd, char __user *, ubuf,
1918c82ccd6SAl Viro compat_size_t, count, u32, poshi, u32, poslo)
192a88b5ba8SSam Ravnborg {
193*8844a509SLinus Torvalds return ksys_pread64(fd, ubuf, count, ((u64)poshi << 32) | poslo);
194a88b5ba8SSam Ravnborg }
195a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE5(pwrite64,unsigned int,fd,char __user *,ubuf,compat_size_t,count,u32,poshi,u32,poslo)1968c82ccd6SAl Viro COMPAT_SYSCALL_DEFINE5(pwrite64, unsigned int, fd, char __user *, ubuf,
1978c82ccd6SAl Viro compat_size_t, count, u32, poshi, u32, poslo)
198a88b5ba8SSam Ravnborg {
199*8844a509SLinus Torvalds return ksys_pwrite64(fd, ubuf, count, ((u64)poshi << 32) | poslo);
200a88b5ba8SSam Ravnborg }
201a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE4(readahead,int,fd,u32,offhi,u32,offlo,compat_size_t,count)202a00a700bSAl Viro COMPAT_SYSCALL_DEFINE4(readahead, int, fd, u32, offhi, u32, offlo,
203a00a700bSAl Viro compat_size_t, count)
204a88b5ba8SSam Ravnborg {
205*8844a509SLinus Torvalds return ksys_readahead(fd, ((u64)offhi << 32) | offlo, count);
206a88b5ba8SSam Ravnborg }
207a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE5(fadvise64,int,fd,u32,offhi,u32,offlo,compat_size_t,len,int,advice)208a00a700bSAl Viro COMPAT_SYSCALL_DEFINE5(fadvise64, int, fd, u32, offhi, u32, offlo,
209a00a700bSAl Viro compat_size_t, len, int, advice)
210a88b5ba8SSam Ravnborg {
211*8844a509SLinus Torvalds return ksys_fadvise64_64(fd, ((u64)offhi << 32) | offlo, len, advice);
212a88b5ba8SSam Ravnborg }
213a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE6(fadvise64_64,int,fd,u32,offhi,u32,offlo,u32,lenhi,u32,lenlo,int,advice)214a00a700bSAl Viro COMPAT_SYSCALL_DEFINE6(fadvise64_64, int, fd, u32, offhi, u32, offlo,
215a00a700bSAl Viro u32, lenhi, u32, lenlo, int, advice)
216a88b5ba8SSam Ravnborg {
2179d5b7c95SDominik Brodowski return ksys_fadvise64_64(fd,
218a00a700bSAl Viro ((u64)offhi << 32) | offlo,
219a00a700bSAl Viro ((u64)lenhi << 32) | lenlo,
220a88b5ba8SSam Ravnborg advice);
221a88b5ba8SSam Ravnborg }
222a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE6(sync_file_range,unsigned int,fd,u32,off_high,u32,off_low,u32,nb_high,u32,nb_low,unsigned int,flags)2238ccb0046SAl Viro COMPAT_SYSCALL_DEFINE6(sync_file_range, unsigned int, fd, u32, off_high, u32, off_low,
2248ccb0046SAl Viro u32, nb_high, u32, nb_low, unsigned int, flags)
225a88b5ba8SSam Ravnborg {
226806cbae1SDominik Brodowski return ksys_sync_file_range(fd,
2278ccb0046SAl Viro ((u64)off_high << 32) | off_low,
2288ccb0046SAl Viro ((u64)nb_high << 32) | nb_low,
229a88b5ba8SSam Ravnborg flags);
230a88b5ba8SSam Ravnborg }
231a88b5ba8SSam Ravnborg
COMPAT_SYSCALL_DEFINE6(fallocate,int,fd,int,mode,u32,offhi,u32,offlo,u32,lenhi,u32,lenlo)232ee076e81SAl Viro COMPAT_SYSCALL_DEFINE6(fallocate, int, fd, int, mode, u32, offhi, u32, offlo,
233ee076e81SAl Viro u32, lenhi, u32, lenlo)
234a88b5ba8SSam Ravnborg {
235edf292c7SDominik Brodowski return ksys_fallocate(fd, mode, ((loff_t)offhi << 32) | offlo,
236a88b5ba8SSam Ravnborg ((loff_t)lenhi << 32) | lenlo);
237a88b5ba8SSam Ravnborg }
238