mirror of
https://github.com/torvalds/linux.git
synced 2024-12-31 23:31:29 +00:00
83b942bd34
If we use 64bit kernel on ia64/x86_64/s390 architecture, and we run 32bit binary on 32bit compatibility mode, sendfile system call seems be not set offset argument. This is because sendfile's return value is not zero but the code regards the result by return value is zero or not. This problem will be affect to ia64/x86_64/s390 and not affect to other architecture does not affect other architecture (mips/parisc/ppc64/sparc64). Signed-off-by: Andi Kleen <ak@suse.de> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
1084 lines
27 KiB
C
1084 lines
27 KiB
C
/*
|
|
* arch/s390x/kernel/linux32.c
|
|
*
|
|
* S390 version
|
|
* Copyright (C) 2000 IBM Deutschland Entwicklung GmbH, IBM Corporation
|
|
* Author(s): Martin Schwidefsky (schwidefsky@de.ibm.com),
|
|
* Gerhard Tonn (ton@de.ibm.com)
|
|
* Thomas Spatzier (tspat@de.ibm.com)
|
|
*
|
|
* Conversion between 31bit and 64bit native syscalls.
|
|
*
|
|
* Heavily inspired by the 32-bit Sparc compat code which is
|
|
* Copyright (C) 1997,1998 Jakub Jelinek (jj@sunsite.mff.cuni.cz)
|
|
* Copyright (C) 1997 David S. Miller (davem@caip.rutgers.edu)
|
|
*
|
|
*/
|
|
|
|
|
|
#include <linux/config.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/mm.h>
|
|
#include <linux/file.h>
|
|
#include <linux/signal.h>
|
|
#include <linux/resource.h>
|
|
#include <linux/times.h>
|
|
#include <linux/utsname.h>
|
|
#include <linux/timex.h>
|
|
#include <linux/smp.h>
|
|
#include <linux/smp_lock.h>
|
|
#include <linux/sem.h>
|
|
#include <linux/msg.h>
|
|
#include <linux/shm.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/uio.h>
|
|
#include <linux/nfs_fs.h>
|
|
#include <linux/quota.h>
|
|
#include <linux/module.h>
|
|
#include <linux/sunrpc/svc.h>
|
|
#include <linux/nfsd/nfsd.h>
|
|
#include <linux/nfsd/cache.h>
|
|
#include <linux/nfsd/xdr.h>
|
|
#include <linux/nfsd/syscall.h>
|
|
#include <linux/poll.h>
|
|
#include <linux/personality.h>
|
|
#include <linux/stat.h>
|
|
#include <linux/filter.h>
|
|
#include <linux/highmem.h>
|
|
#include <linux/highuid.h>
|
|
#include <linux/mman.h>
|
|
#include <linux/ipv6.h>
|
|
#include <linux/in.h>
|
|
#include <linux/icmpv6.h>
|
|
#include <linux/syscalls.h>
|
|
#include <linux/sysctl.h>
|
|
#include <linux/binfmts.h>
|
|
#include <linux/compat.h>
|
|
#include <linux/vfs.h>
|
|
#include <linux/ptrace.h>
|
|
#include <linux/fadvise.h>
|
|
|
|
#include <asm/types.h>
|
|
#include <asm/ipc.h>
|
|
#include <asm/uaccess.h>
|
|
#include <asm/semaphore.h>
|
|
|
|
#include <net/scm.h>
|
|
#include <net/sock.h>
|
|
|
|
#include "compat_linux.h"
|
|
|
|
|
|
/* For this source file, we want overflow handling. */
|
|
|
|
#undef high2lowuid
|
|
#undef high2lowgid
|
|
#undef low2highuid
|
|
#undef low2highgid
|
|
#undef SET_UID16
|
|
#undef SET_GID16
|
|
#undef NEW_TO_OLD_UID
|
|
#undef NEW_TO_OLD_GID
|
|
#undef SET_OLDSTAT_UID
|
|
#undef SET_OLDSTAT_GID
|
|
#undef SET_STAT_UID
|
|
#undef SET_STAT_GID
|
|
|
|
#define high2lowuid(uid) ((uid) > 65535) ? (u16)overflowuid : (u16)(uid)
|
|
#define high2lowgid(gid) ((gid) > 65535) ? (u16)overflowgid : (u16)(gid)
|
|
#define low2highuid(uid) ((uid) == (u16)-1) ? (uid_t)-1 : (uid_t)(uid)
|
|
#define low2highgid(gid) ((gid) == (u16)-1) ? (gid_t)-1 : (gid_t)(gid)
|
|
#define SET_UID16(var, uid) var = high2lowuid(uid)
|
|
#define SET_GID16(var, gid) var = high2lowgid(gid)
|
|
#define NEW_TO_OLD_UID(uid) high2lowuid(uid)
|
|
#define NEW_TO_OLD_GID(gid) high2lowgid(gid)
|
|
#define SET_OLDSTAT_UID(stat, uid) (stat).st_uid = high2lowuid(uid)
|
|
#define SET_OLDSTAT_GID(stat, gid) (stat).st_gid = high2lowgid(gid)
|
|
#define SET_STAT_UID(stat, uid) (stat).st_uid = high2lowuid(uid)
|
|
#define SET_STAT_GID(stat, gid) (stat).st_gid = high2lowgid(gid)
|
|
|
|
asmlinkage long sys32_chown16(const char * filename, u16 user, u16 group)
|
|
{
|
|
return sys_chown(filename, low2highuid(user), low2highgid(group));
|
|
}
|
|
|
|
asmlinkage long sys32_lchown16(const char * filename, u16 user, u16 group)
|
|
{
|
|
return sys_lchown(filename, low2highuid(user), low2highgid(group));
|
|
}
|
|
|
|
asmlinkage long sys32_fchown16(unsigned int fd, u16 user, u16 group)
|
|
{
|
|
return sys_fchown(fd, low2highuid(user), low2highgid(group));
|
|
}
|
|
|
|
asmlinkage long sys32_setregid16(u16 rgid, u16 egid)
|
|
{
|
|
return sys_setregid(low2highgid(rgid), low2highgid(egid));
|
|
}
|
|
|
|
asmlinkage long sys32_setgid16(u16 gid)
|
|
{
|
|
return sys_setgid((gid_t)gid);
|
|
}
|
|
|
|
asmlinkage long sys32_setreuid16(u16 ruid, u16 euid)
|
|
{
|
|
return sys_setreuid(low2highuid(ruid), low2highuid(euid));
|
|
}
|
|
|
|
asmlinkage long sys32_setuid16(u16 uid)
|
|
{
|
|
return sys_setuid((uid_t)uid);
|
|
}
|
|
|
|
asmlinkage long sys32_setresuid16(u16 ruid, u16 euid, u16 suid)
|
|
{
|
|
return sys_setresuid(low2highuid(ruid), low2highuid(euid),
|
|
low2highuid(suid));
|
|
}
|
|
|
|
asmlinkage long sys32_getresuid16(u16 *ruid, u16 *euid, u16 *suid)
|
|
{
|
|
int retval;
|
|
|
|
if (!(retval = put_user(high2lowuid(current->uid), ruid)) &&
|
|
!(retval = put_user(high2lowuid(current->euid), euid)))
|
|
retval = put_user(high2lowuid(current->suid), suid);
|
|
|
|
return retval;
|
|
}
|
|
|
|
asmlinkage long sys32_setresgid16(u16 rgid, u16 egid, u16 sgid)
|
|
{
|
|
return sys_setresgid(low2highgid(rgid), low2highgid(egid),
|
|
low2highgid(sgid));
|
|
}
|
|
|
|
asmlinkage long sys32_getresgid16(u16 *rgid, u16 *egid, u16 *sgid)
|
|
{
|
|
int retval;
|
|
|
|
if (!(retval = put_user(high2lowgid(current->gid), rgid)) &&
|
|
!(retval = put_user(high2lowgid(current->egid), egid)))
|
|
retval = put_user(high2lowgid(current->sgid), sgid);
|
|
|
|
return retval;
|
|
}
|
|
|
|
asmlinkage long sys32_setfsuid16(u16 uid)
|
|
{
|
|
return sys_setfsuid((uid_t)uid);
|
|
}
|
|
|
|
asmlinkage long sys32_setfsgid16(u16 gid)
|
|
{
|
|
return sys_setfsgid((gid_t)gid);
|
|
}
|
|
|
|
static int groups16_to_user(u16 *grouplist, struct group_info *group_info)
|
|
{
|
|
int i;
|
|
u16 group;
|
|
|
|
for (i = 0; i < group_info->ngroups; i++) {
|
|
group = (u16)GROUP_AT(group_info, i);
|
|
if (put_user(group, grouplist+i))
|
|
return -EFAULT;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int groups16_from_user(struct group_info *group_info, u16 *grouplist)
|
|
{
|
|
int i;
|
|
u16 group;
|
|
|
|
for (i = 0; i < group_info->ngroups; i++) {
|
|
if (get_user(group, grouplist+i))
|
|
return -EFAULT;
|
|
GROUP_AT(group_info, i) = (gid_t)group;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
asmlinkage long sys32_getgroups16(int gidsetsize, u16 *grouplist)
|
|
{
|
|
int i;
|
|
|
|
if (gidsetsize < 0)
|
|
return -EINVAL;
|
|
|
|
get_group_info(current->group_info);
|
|
i = current->group_info->ngroups;
|
|
if (gidsetsize) {
|
|
if (i > gidsetsize) {
|
|
i = -EINVAL;
|
|
goto out;
|
|
}
|
|
if (groups16_to_user(grouplist, current->group_info)) {
|
|
i = -EFAULT;
|
|
goto out;
|
|
}
|
|
}
|
|
out:
|
|
put_group_info(current->group_info);
|
|
return i;
|
|
}
|
|
|
|
asmlinkage long sys32_setgroups16(int gidsetsize, u16 *grouplist)
|
|
{
|
|
struct group_info *group_info;
|
|
int retval;
|
|
|
|
if (!capable(CAP_SETGID))
|
|
return -EPERM;
|
|
if ((unsigned)gidsetsize > NGROUPS_MAX)
|
|
return -EINVAL;
|
|
|
|
group_info = groups_alloc(gidsetsize);
|
|
if (!group_info)
|
|
return -ENOMEM;
|
|
retval = groups16_from_user(group_info, grouplist);
|
|
if (retval) {
|
|
put_group_info(group_info);
|
|
return retval;
|
|
}
|
|
|
|
retval = set_current_groups(group_info);
|
|
put_group_info(group_info);
|
|
|
|
return retval;
|
|
}
|
|
|
|
asmlinkage long sys32_getuid16(void)
|
|
{
|
|
return high2lowuid(current->uid);
|
|
}
|
|
|
|
asmlinkage long sys32_geteuid16(void)
|
|
{
|
|
return high2lowuid(current->euid);
|
|
}
|
|
|
|
asmlinkage long sys32_getgid16(void)
|
|
{
|
|
return high2lowgid(current->gid);
|
|
}
|
|
|
|
asmlinkage long sys32_getegid16(void)
|
|
{
|
|
return high2lowgid(current->egid);
|
|
}
|
|
|
|
/* 32-bit timeval and related flotsam. */
|
|
|
|
static inline long get_tv32(struct timeval *o, struct compat_timeval *i)
|
|
{
|
|
return (!access_ok(VERIFY_READ, tv32, sizeof(*tv32)) ||
|
|
(__get_user(o->tv_sec, &i->tv_sec) ||
|
|
__get_user(o->tv_usec, &i->tv_usec)));
|
|
}
|
|
|
|
static inline long put_tv32(struct compat_timeval *o, struct timeval *i)
|
|
{
|
|
return (!access_ok(VERIFY_WRITE, o, sizeof(*o)) ||
|
|
(__put_user(i->tv_sec, &o->tv_sec) ||
|
|
__put_user(i->tv_usec, &o->tv_usec)));
|
|
}
|
|
|
|
/*
|
|
* sys32_ipc() is the de-multiplexer for the SysV IPC calls in 32bit emulation.
|
|
*
|
|
* This is really horribly ugly.
|
|
*/
|
|
asmlinkage long sys32_ipc(u32 call, int first, int second, int third, u32 ptr)
|
|
{
|
|
if (call >> 16) /* hack for backward compatibility */
|
|
return -EINVAL;
|
|
|
|
call &= 0xffff;
|
|
|
|
switch (call) {
|
|
case SEMTIMEDOP:
|
|
return compat_sys_semtimedop(first, compat_ptr(ptr),
|
|
second, compat_ptr(third));
|
|
case SEMOP:
|
|
/* struct sembuf is the same on 32 and 64bit :)) */
|
|
return sys_semtimedop(first, compat_ptr(ptr),
|
|
second, NULL);
|
|
case SEMGET:
|
|
return sys_semget(first, second, third);
|
|
case SEMCTL:
|
|
return compat_sys_semctl(first, second, third,
|
|
compat_ptr(ptr));
|
|
case MSGSND:
|
|
return compat_sys_msgsnd(first, second, third,
|
|
compat_ptr(ptr));
|
|
case MSGRCV:
|
|
return compat_sys_msgrcv(first, second, 0, third,
|
|
0, compat_ptr(ptr));
|
|
case MSGGET:
|
|
return sys_msgget((key_t) first, second);
|
|
case MSGCTL:
|
|
return compat_sys_msgctl(first, second, compat_ptr(ptr));
|
|
case SHMAT:
|
|
return compat_sys_shmat(first, second, third,
|
|
0, compat_ptr(ptr));
|
|
case SHMDT:
|
|
return sys_shmdt(compat_ptr(ptr));
|
|
case SHMGET:
|
|
return sys_shmget(first, (unsigned)second, third);
|
|
case SHMCTL:
|
|
return compat_sys_shmctl(first, second, compat_ptr(ptr));
|
|
}
|
|
|
|
return -ENOSYS;
|
|
}
|
|
|
|
asmlinkage long sys32_truncate64(const char * path, unsigned long high, unsigned long low)
|
|
{
|
|
if ((int)high < 0)
|
|
return -EINVAL;
|
|
else
|
|
return sys_truncate(path, (high << 32) | low);
|
|
}
|
|
|
|
asmlinkage long sys32_ftruncate64(unsigned int fd, unsigned long high, unsigned long low)
|
|
{
|
|
if ((int)high < 0)
|
|
return -EINVAL;
|
|
else
|
|
return sys_ftruncate(fd, (high << 32) | low);
|
|
}
|
|
|
|
int cp_compat_stat(struct kstat *stat, struct compat_stat *statbuf)
|
|
{
|
|
int err;
|
|
|
|
if (!old_valid_dev(stat->dev) || !old_valid_dev(stat->rdev))
|
|
return -EOVERFLOW;
|
|
|
|
err = put_user(old_encode_dev(stat->dev), &statbuf->st_dev);
|
|
err |= put_user(stat->ino, &statbuf->st_ino);
|
|
err |= put_user(stat->mode, &statbuf->st_mode);
|
|
err |= put_user(stat->nlink, &statbuf->st_nlink);
|
|
err |= put_user(high2lowuid(stat->uid), &statbuf->st_uid);
|
|
err |= put_user(high2lowgid(stat->gid), &statbuf->st_gid);
|
|
err |= put_user(old_encode_dev(stat->rdev), &statbuf->st_rdev);
|
|
err |= put_user(stat->size, &statbuf->st_size);
|
|
err |= put_user(stat->atime.tv_sec, &statbuf->st_atime);
|
|
err |= put_user(stat->atime.tv_nsec, &statbuf->st_atime_nsec);
|
|
err |= put_user(stat->mtime.tv_sec, &statbuf->st_mtime);
|
|
err |= put_user(stat->mtime.tv_nsec, &statbuf->st_mtime_nsec);
|
|
err |= put_user(stat->ctime.tv_sec, &statbuf->st_ctime);
|
|
err |= put_user(stat->ctime.tv_nsec, &statbuf->st_ctime_nsec);
|
|
err |= put_user(stat->blksize, &statbuf->st_blksize);
|
|
err |= put_user(stat->blocks, &statbuf->st_blocks);
|
|
/* fixme
|
|
err |= put_user(0, &statbuf->__unused4[0]);
|
|
err |= put_user(0, &statbuf->__unused4[1]);
|
|
*/
|
|
return err;
|
|
}
|
|
|
|
struct sysinfo32 {
|
|
s32 uptime;
|
|
u32 loads[3];
|
|
u32 totalram;
|
|
u32 freeram;
|
|
u32 sharedram;
|
|
u32 bufferram;
|
|
u32 totalswap;
|
|
u32 freeswap;
|
|
unsigned short procs;
|
|
unsigned short pads;
|
|
u32 totalhigh;
|
|
u32 freehigh;
|
|
unsigned int mem_unit;
|
|
char _f[8];
|
|
};
|
|
|
|
asmlinkage long sys32_sysinfo(struct sysinfo32 __user *info)
|
|
{
|
|
struct sysinfo s;
|
|
int ret, err;
|
|
mm_segment_t old_fs = get_fs ();
|
|
|
|
set_fs (KERNEL_DS);
|
|
ret = sys_sysinfo(&s);
|
|
set_fs (old_fs);
|
|
err = put_user (s.uptime, &info->uptime);
|
|
err |= __put_user (s.loads[0], &info->loads[0]);
|
|
err |= __put_user (s.loads[1], &info->loads[1]);
|
|
err |= __put_user (s.loads[2], &info->loads[2]);
|
|
err |= __put_user (s.totalram, &info->totalram);
|
|
err |= __put_user (s.freeram, &info->freeram);
|
|
err |= __put_user (s.sharedram, &info->sharedram);
|
|
err |= __put_user (s.bufferram, &info->bufferram);
|
|
err |= __put_user (s.totalswap, &info->totalswap);
|
|
err |= __put_user (s.freeswap, &info->freeswap);
|
|
err |= __put_user (s.procs, &info->procs);
|
|
err |= __put_user (s.totalhigh, &info->totalhigh);
|
|
err |= __put_user (s.freehigh, &info->freehigh);
|
|
err |= __put_user (s.mem_unit, &info->mem_unit);
|
|
if (err)
|
|
return -EFAULT;
|
|
return ret;
|
|
}
|
|
|
|
asmlinkage long sys32_sched_rr_get_interval(compat_pid_t pid,
|
|
struct compat_timespec __user *interval)
|
|
{
|
|
struct timespec t;
|
|
int ret;
|
|
mm_segment_t old_fs = get_fs ();
|
|
|
|
set_fs (KERNEL_DS);
|
|
ret = sys_sched_rr_get_interval(pid, &t);
|
|
set_fs (old_fs);
|
|
if (put_compat_timespec(&t, interval))
|
|
return -EFAULT;
|
|
return ret;
|
|
}
|
|
|
|
asmlinkage long sys32_rt_sigprocmask(int how, compat_sigset_t __user *set,
|
|
compat_sigset_t __user *oset, size_t sigsetsize)
|
|
{
|
|
sigset_t s;
|
|
compat_sigset_t s32;
|
|
int ret;
|
|
mm_segment_t old_fs = get_fs();
|
|
|
|
if (set) {
|
|
if (copy_from_user (&s32, set, sizeof(compat_sigset_t)))
|
|
return -EFAULT;
|
|
switch (_NSIG_WORDS) {
|
|
case 4: s.sig[3] = s32.sig[6] | (((long)s32.sig[7]) << 32);
|
|
case 3: s.sig[2] = s32.sig[4] | (((long)s32.sig[5]) << 32);
|
|
case 2: s.sig[1] = s32.sig[2] | (((long)s32.sig[3]) << 32);
|
|
case 1: s.sig[0] = s32.sig[0] | (((long)s32.sig[1]) << 32);
|
|
}
|
|
}
|
|
set_fs (KERNEL_DS);
|
|
ret = sys_rt_sigprocmask(how, set ? &s : NULL, oset ? &s : NULL, sigsetsize);
|
|
set_fs (old_fs);
|
|
if (ret) return ret;
|
|
if (oset) {
|
|
switch (_NSIG_WORDS) {
|
|
case 4: s32.sig[7] = (s.sig[3] >> 32); s32.sig[6] = s.sig[3];
|
|
case 3: s32.sig[5] = (s.sig[2] >> 32); s32.sig[4] = s.sig[2];
|
|
case 2: s32.sig[3] = (s.sig[1] >> 32); s32.sig[2] = s.sig[1];
|
|
case 1: s32.sig[1] = (s.sig[0] >> 32); s32.sig[0] = s.sig[0];
|
|
}
|
|
if (copy_to_user (oset, &s32, sizeof(compat_sigset_t)))
|
|
return -EFAULT;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
asmlinkage long sys32_rt_sigpending(compat_sigset_t __user *set,
|
|
size_t sigsetsize)
|
|
{
|
|
sigset_t s;
|
|
compat_sigset_t s32;
|
|
int ret;
|
|
mm_segment_t old_fs = get_fs();
|
|
|
|
set_fs (KERNEL_DS);
|
|
ret = sys_rt_sigpending(&s, sigsetsize);
|
|
set_fs (old_fs);
|
|
if (!ret) {
|
|
switch (_NSIG_WORDS) {
|
|
case 4: s32.sig[7] = (s.sig[3] >> 32); s32.sig[6] = s.sig[3];
|
|
case 3: s32.sig[5] = (s.sig[2] >> 32); s32.sig[4] = s.sig[2];
|
|
case 2: s32.sig[3] = (s.sig[1] >> 32); s32.sig[2] = s.sig[1];
|
|
case 1: s32.sig[1] = (s.sig[0] >> 32); s32.sig[0] = s.sig[0];
|
|
}
|
|
if (copy_to_user (set, &s32, sizeof(compat_sigset_t)))
|
|
return -EFAULT;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
asmlinkage long
|
|
sys32_rt_sigqueueinfo(int pid, int sig, compat_siginfo_t __user *uinfo)
|
|
{
|
|
siginfo_t info;
|
|
int ret;
|
|
mm_segment_t old_fs = get_fs();
|
|
|
|
if (copy_siginfo_from_user32(&info, uinfo))
|
|
return -EFAULT;
|
|
set_fs (KERNEL_DS);
|
|
ret = sys_rt_sigqueueinfo(pid, sig, &info);
|
|
set_fs (old_fs);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* sys32_execve() executes a new program after the asm stub has set
|
|
* things up for us. This should basically do what I want it to.
|
|
*/
|
|
asmlinkage long
|
|
sys32_execve(struct pt_regs regs)
|
|
{
|
|
int error;
|
|
char * filename;
|
|
|
|
filename = getname(compat_ptr(regs.orig_gpr2));
|
|
error = PTR_ERR(filename);
|
|
if (IS_ERR(filename))
|
|
goto out;
|
|
error = compat_do_execve(filename, compat_ptr(regs.gprs[3]),
|
|
compat_ptr(regs.gprs[4]), ®s);
|
|
if (error == 0)
|
|
{
|
|
task_lock(current);
|
|
current->ptrace &= ~PT_DTRACE;
|
|
task_unlock(current);
|
|
current->thread.fp_regs.fpc=0;
|
|
__asm__ __volatile__
|
|
("sr 0,0\n\t"
|
|
"sfpc 0,0\n\t"
|
|
: : :"0");
|
|
}
|
|
putname(filename);
|
|
out:
|
|
return error;
|
|
}
|
|
|
|
|
|
#ifdef CONFIG_MODULES
|
|
|
|
asmlinkage long
|
|
sys32_init_module(void __user *umod, unsigned long len,
|
|
const char __user *uargs)
|
|
{
|
|
return sys_init_module(umod, len, uargs);
|
|
}
|
|
|
|
asmlinkage long
|
|
sys32_delete_module(const char __user *name_user, unsigned int flags)
|
|
{
|
|
return sys_delete_module(name_user, flags);
|
|
}
|
|
|
|
#else /* CONFIG_MODULES */
|
|
|
|
asmlinkage long
|
|
sys32_init_module(void __user *umod, unsigned long len,
|
|
const char __user *uargs)
|
|
{
|
|
return -ENOSYS;
|
|
}
|
|
|
|
asmlinkage long
|
|
sys32_delete_module(const char __user *name_user, unsigned int flags)
|
|
{
|
|
return -ENOSYS;
|
|
}
|
|
|
|
#endif /* CONFIG_MODULES */
|
|
|
|
/* Translations due to time_t size differences. Which affects all
|
|
sorts of things, like timeval and itimerval. */
|
|
|
|
extern struct timezone sys_tz;
|
|
|
|
asmlinkage long sys32_gettimeofday(struct compat_timeval *tv, struct timezone *tz)
|
|
{
|
|
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;
|
|
}
|
|
|
|
static inline long get_ts32(struct timespec *o, struct compat_timeval *i)
|
|
{
|
|
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 long sys32_settimeofday(struct compat_timeval *tv, struct timezone *tz)
|
|
{
|
|
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);
|
|
}
|
|
|
|
/* These are here just in case some old sparc32 binary calls it. */
|
|
asmlinkage long sys32_pause(void)
|
|
{
|
|
current->state = TASK_INTERRUPTIBLE;
|
|
schedule();
|
|
return -ERESTARTNOHAND;
|
|
}
|
|
|
|
asmlinkage long sys32_pread64(unsigned int fd, char *ubuf,
|
|
size_t count, u32 poshi, u32 poslo)
|
|
{
|
|
if ((compat_ssize_t) count < 0)
|
|
return -EINVAL;
|
|
return sys_pread64(fd, ubuf, count, ((loff_t)AA(poshi) << 32) | AA(poslo));
|
|
}
|
|
|
|
asmlinkage long sys32_pwrite64(unsigned int fd, const char *ubuf,
|
|
size_t count, u32 poshi, u32 poslo)
|
|
{
|
|
if ((compat_ssize_t) count < 0)
|
|
return -EINVAL;
|
|
return sys_pwrite64(fd, ubuf, count, ((loff_t)AA(poshi) << 32) | AA(poslo));
|
|
}
|
|
|
|
asmlinkage compat_ssize_t sys32_readahead(int fd, u32 offhi, u32 offlo, s32 count)
|
|
{
|
|
return sys_readahead(fd, ((loff_t)AA(offhi) << 32) | AA(offlo), count);
|
|
}
|
|
|
|
asmlinkage long sys32_sendfile(int out_fd, int in_fd, compat_off_t *offset, size_t count)
|
|
{
|
|
mm_segment_t old_fs = get_fs();
|
|
int ret;
|
|
off_t of;
|
|
|
|
if (offset && get_user(of, offset))
|
|
return -EFAULT;
|
|
|
|
set_fs(KERNEL_DS);
|
|
ret = sys_sendfile(out_fd, in_fd, offset ? &of : NULL, count);
|
|
set_fs(old_fs);
|
|
|
|
if (offset && put_user(of, offset))
|
|
return -EFAULT;
|
|
|
|
return ret;
|
|
}
|
|
|
|
asmlinkage long sys32_sendfile64(int out_fd, int in_fd,
|
|
compat_loff_t *offset, s32 count)
|
|
{
|
|
mm_segment_t old_fs = get_fs();
|
|
int ret;
|
|
loff_t lof;
|
|
|
|
if (offset && get_user(lof, offset))
|
|
return -EFAULT;
|
|
|
|
set_fs(KERNEL_DS);
|
|
ret = sys_sendfile64(out_fd, in_fd, offset ? &lof : NULL, count);
|
|
set_fs(old_fs);
|
|
|
|
if (offset && put_user(lof, offset))
|
|
return -EFAULT;
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* Handle adjtimex compatibility. */
|
|
|
|
struct timex32 {
|
|
u32 modes;
|
|
s32 offset, freq, maxerror, esterror;
|
|
s32 status, constant, precision, tolerance;
|
|
struct compat_timeval time;
|
|
s32 tick;
|
|
s32 ppsfreq, jitter, shift, stabil;
|
|
s32 jitcnt, calcnt, errcnt, stbcnt;
|
|
s32 :32; s32 :32; s32 :32; s32 :32;
|
|
s32 :32; s32 :32; s32 :32; s32 :32;
|
|
s32 :32; s32 :32; s32 :32; s32 :32;
|
|
};
|
|
|
|
extern int do_adjtimex(struct timex *);
|
|
|
|
asmlinkage long sys32_adjtimex(struct timex32 *utp)
|
|
{
|
|
struct timex txc;
|
|
int ret;
|
|
|
|
memset(&txc, 0, sizeof(struct timex));
|
|
|
|
if(get_user(txc.modes, &utp->modes) ||
|
|
__get_user(txc.offset, &utp->offset) ||
|
|
__get_user(txc.freq, &utp->freq) ||
|
|
__get_user(txc.maxerror, &utp->maxerror) ||
|
|
__get_user(txc.esterror, &utp->esterror) ||
|
|
__get_user(txc.status, &utp->status) ||
|
|
__get_user(txc.constant, &utp->constant) ||
|
|
__get_user(txc.precision, &utp->precision) ||
|
|
__get_user(txc.tolerance, &utp->tolerance) ||
|
|
__get_user(txc.time.tv_sec, &utp->time.tv_sec) ||
|
|
__get_user(txc.time.tv_usec, &utp->time.tv_usec) ||
|
|
__get_user(txc.tick, &utp->tick) ||
|
|
__get_user(txc.ppsfreq, &utp->ppsfreq) ||
|
|
__get_user(txc.jitter, &utp->jitter) ||
|
|
__get_user(txc.shift, &utp->shift) ||
|
|
__get_user(txc.stabil, &utp->stabil) ||
|
|
__get_user(txc.jitcnt, &utp->jitcnt) ||
|
|
__get_user(txc.calcnt, &utp->calcnt) ||
|
|
__get_user(txc.errcnt, &utp->errcnt) ||
|
|
__get_user(txc.stbcnt, &utp->stbcnt))
|
|
return -EFAULT;
|
|
|
|
ret = do_adjtimex(&txc);
|
|
|
|
if(put_user(txc.modes, &utp->modes) ||
|
|
__put_user(txc.offset, &utp->offset) ||
|
|
__put_user(txc.freq, &utp->freq) ||
|
|
__put_user(txc.maxerror, &utp->maxerror) ||
|
|
__put_user(txc.esterror, &utp->esterror) ||
|
|
__put_user(txc.status, &utp->status) ||
|
|
__put_user(txc.constant, &utp->constant) ||
|
|
__put_user(txc.precision, &utp->precision) ||
|
|
__put_user(txc.tolerance, &utp->tolerance) ||
|
|
__put_user(txc.time.tv_sec, &utp->time.tv_sec) ||
|
|
__put_user(txc.time.tv_usec, &utp->time.tv_usec) ||
|
|
__put_user(txc.tick, &utp->tick) ||
|
|
__put_user(txc.ppsfreq, &utp->ppsfreq) ||
|
|
__put_user(txc.jitter, &utp->jitter) ||
|
|
__put_user(txc.shift, &utp->shift) ||
|
|
__put_user(txc.stabil, &utp->stabil) ||
|
|
__put_user(txc.jitcnt, &utp->jitcnt) ||
|
|
__put_user(txc.calcnt, &utp->calcnt) ||
|
|
__put_user(txc.errcnt, &utp->errcnt) ||
|
|
__put_user(txc.stbcnt, &utp->stbcnt))
|
|
ret = -EFAULT;
|
|
|
|
return ret;
|
|
}
|
|
|
|
#ifdef CONFIG_SYSCTL
|
|
struct __sysctl_args32 {
|
|
u32 name;
|
|
int nlen;
|
|
u32 oldval;
|
|
u32 oldlenp;
|
|
u32 newval;
|
|
u32 newlen;
|
|
u32 __unused[4];
|
|
};
|
|
|
|
asmlinkage long sys32_sysctl(struct __sysctl_args32 *args)
|
|
{
|
|
struct __sysctl_args32 tmp;
|
|
int error;
|
|
size_t oldlen, *oldlenp = NULL;
|
|
unsigned long addr = (((long)&args->__unused[0]) + 7) & ~7;
|
|
|
|
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. */
|
|
if (get_user(oldlen, (u32 *)A(tmp.oldlenp)) ||
|
|
put_user(oldlen, (size_t *)addr))
|
|
return -EFAULT;
|
|
oldlenp = (size_t *)addr;
|
|
}
|
|
|
|
lock_kernel();
|
|
error = do_sysctl((int *)A(tmp.name), tmp.nlen, (void *)A(tmp.oldval),
|
|
oldlenp, (void *)A(tmp.newval), tmp.newlen);
|
|
unlock_kernel();
|
|
if (oldlenp) {
|
|
if (!error) {
|
|
if (get_user(oldlen, (size_t *)addr) ||
|
|
put_user(oldlen, (u32 *)A(tmp.oldlenp)))
|
|
error = -EFAULT;
|
|
}
|
|
copy_to_user(args->__unused, tmp.__unused, sizeof(tmp.__unused));
|
|
}
|
|
return error;
|
|
}
|
|
#endif
|
|
|
|
struct stat64_emu31 {
|
|
unsigned long long st_dev;
|
|
unsigned int __pad1;
|
|
#define STAT64_HAS_BROKEN_ST_INO 1
|
|
u32 __st_ino;
|
|
unsigned int st_mode;
|
|
unsigned int st_nlink;
|
|
u32 st_uid;
|
|
u32 st_gid;
|
|
unsigned long long st_rdev;
|
|
unsigned int __pad3;
|
|
long st_size;
|
|
u32 st_blksize;
|
|
unsigned char __pad4[4];
|
|
u32 __pad5; /* future possible st_blocks high bits */
|
|
u32 st_blocks; /* Number 512-byte blocks allocated. */
|
|
u32 st_atime;
|
|
u32 __pad6;
|
|
u32 st_mtime;
|
|
u32 __pad7;
|
|
u32 st_ctime;
|
|
u32 __pad8; /* will be high 32 bits of ctime someday */
|
|
unsigned long st_ino;
|
|
};
|
|
|
|
static int cp_stat64(struct stat64_emu31 *ubuf, struct kstat *stat)
|
|
{
|
|
struct stat64_emu31 tmp;
|
|
|
|
memset(&tmp, 0, sizeof(tmp));
|
|
|
|
tmp.st_dev = huge_encode_dev(stat->dev);
|
|
tmp.st_ino = stat->ino;
|
|
tmp.__st_ino = (u32)stat->ino;
|
|
tmp.st_mode = stat->mode;
|
|
tmp.st_nlink = (unsigned int)stat->nlink;
|
|
tmp.st_uid = stat->uid;
|
|
tmp.st_gid = stat->gid;
|
|
tmp.st_rdev = huge_encode_dev(stat->rdev);
|
|
tmp.st_size = stat->size;
|
|
tmp.st_blksize = (u32)stat->blksize;
|
|
tmp.st_blocks = (u32)stat->blocks;
|
|
tmp.st_atime = (u32)stat->atime.tv_sec;
|
|
tmp.st_mtime = (u32)stat->mtime.tv_sec;
|
|
tmp.st_ctime = (u32)stat->ctime.tv_sec;
|
|
|
|
return copy_to_user(ubuf,&tmp,sizeof(tmp)) ? -EFAULT : 0;
|
|
}
|
|
|
|
asmlinkage long sys32_stat64(char * filename, struct stat64_emu31 * statbuf)
|
|
{
|
|
struct kstat stat;
|
|
int ret = vfs_stat(filename, &stat);
|
|
if (!ret)
|
|
ret = cp_stat64(statbuf, &stat);
|
|
return ret;
|
|
}
|
|
|
|
asmlinkage long sys32_lstat64(char * filename, struct stat64_emu31 * statbuf)
|
|
{
|
|
struct kstat stat;
|
|
int ret = vfs_lstat(filename, &stat);
|
|
if (!ret)
|
|
ret = cp_stat64(statbuf, &stat);
|
|
return ret;
|
|
}
|
|
|
|
asmlinkage long sys32_fstat64(unsigned long fd, struct stat64_emu31 * statbuf)
|
|
{
|
|
struct kstat stat;
|
|
int ret = vfs_fstat(fd, &stat);
|
|
if (!ret)
|
|
ret = cp_stat64(statbuf, &stat);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Linux/i386 didn't use to be able to handle more than
|
|
* 4 system call parameters, so these system calls used a memory
|
|
* block for parameter passing..
|
|
*/
|
|
|
|
struct mmap_arg_struct_emu31 {
|
|
u32 addr;
|
|
u32 len;
|
|
u32 prot;
|
|
u32 flags;
|
|
u32 fd;
|
|
u32 offset;
|
|
};
|
|
|
|
/* common code for old and new mmaps */
|
|
static inline long do_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 = -EBADF;
|
|
|
|
flags &= ~(MAP_EXECUTABLE | MAP_DENYWRITE);
|
|
if (!(flags & MAP_ANONYMOUS)) {
|
|
file = fget(fd);
|
|
if (!file)
|
|
goto out;
|
|
}
|
|
|
|
down_write(¤t->mm->mmap_sem);
|
|
error = do_mmap_pgoff(file, addr, len, prot, flags, pgoff);
|
|
if (!IS_ERR((void *) error) && error + len >= 0x80000000ULL) {
|
|
/* Result is out of bounds. */
|
|
do_munmap(current->mm, addr, len);
|
|
error = -ENOMEM;
|
|
}
|
|
up_write(¤t->mm->mmap_sem);
|
|
|
|
if (file)
|
|
fput(file);
|
|
out:
|
|
return error;
|
|
}
|
|
|
|
|
|
asmlinkage unsigned long
|
|
old32_mmap(struct mmap_arg_struct_emu31 *arg)
|
|
{
|
|
struct mmap_arg_struct_emu31 a;
|
|
int error = -EFAULT;
|
|
|
|
if (copy_from_user(&a, arg, sizeof(a)))
|
|
goto out;
|
|
|
|
error = -EINVAL;
|
|
if (a.offset & ~PAGE_MASK)
|
|
goto out;
|
|
|
|
error = do_mmap2(a.addr, a.len, a.prot, a.flags, a.fd, a.offset >> PAGE_SHIFT);
|
|
out:
|
|
return error;
|
|
}
|
|
|
|
asmlinkage long
|
|
sys32_mmap2(struct mmap_arg_struct_emu31 *arg)
|
|
{
|
|
struct mmap_arg_struct_emu31 a;
|
|
int error = -EFAULT;
|
|
|
|
if (copy_from_user(&a, arg, sizeof(a)))
|
|
goto out;
|
|
error = do_mmap2(a.addr, a.len, a.prot, a.flags, a.fd, a.offset);
|
|
out:
|
|
return error;
|
|
}
|
|
|
|
asmlinkage long sys32_read(unsigned int fd, char * buf, size_t count)
|
|
{
|
|
if ((compat_ssize_t) count < 0)
|
|
return -EINVAL;
|
|
|
|
return sys_read(fd, buf, count);
|
|
}
|
|
|
|
asmlinkage long sys32_write(unsigned int fd, char * buf, size_t count)
|
|
{
|
|
if ((compat_ssize_t) count < 0)
|
|
return -EINVAL;
|
|
|
|
return sys_write(fd, buf, count);
|
|
}
|
|
|
|
asmlinkage long sys32_clone(struct pt_regs regs)
|
|
{
|
|
unsigned long clone_flags;
|
|
unsigned long newsp;
|
|
int *parent_tidptr, *child_tidptr;
|
|
|
|
clone_flags = regs.gprs[3] & 0xffffffffUL;
|
|
newsp = regs.orig_gpr2 & 0x7fffffffUL;
|
|
parent_tidptr = (int *) (regs.gprs[4] & 0x7fffffffUL);
|
|
child_tidptr = (int *) (regs.gprs[5] & 0x7fffffffUL);
|
|
if (!newsp)
|
|
newsp = regs.gprs[15];
|
|
return do_fork(clone_flags, newsp, ®s, 0,
|
|
parent_tidptr, child_tidptr);
|
|
}
|
|
|
|
/*
|
|
* Wrapper function for sys_timer_create.
|
|
*/
|
|
extern asmlinkage long
|
|
sys_timer_create(clockid_t, struct sigevent *, timer_t *);
|
|
|
|
asmlinkage long
|
|
sys32_timer_create(clockid_t which_clock, struct compat_sigevent *se32,
|
|
timer_t *timer_id)
|
|
{
|
|
struct sigevent se;
|
|
timer_t ktimer_id;
|
|
mm_segment_t old_fs;
|
|
long ret;
|
|
|
|
if (se32 == NULL)
|
|
return sys_timer_create(which_clock, NULL, timer_id);
|
|
|
|
if (get_compat_sigevent(&se, se32))
|
|
return -EFAULT;
|
|
|
|
old_fs = get_fs();
|
|
set_fs(KERNEL_DS);
|
|
ret = sys_timer_create(which_clock, &se, &ktimer_id);
|
|
set_fs(old_fs);
|
|
|
|
if (!ret)
|
|
ret = put_user (ktimer_id, timer_id);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* 31 bit emulation wrapper functions for sys_fadvise64/fadvise64_64.
|
|
* These need to rewrite the advise values for POSIX_FADV_{DONTNEED,NOREUSE}
|
|
* because the 31 bit values differ from the 64 bit values.
|
|
*/
|
|
|
|
asmlinkage long
|
|
sys32_fadvise64(int fd, loff_t offset, size_t len, int advise)
|
|
{
|
|
if (advise == 4)
|
|
advise = POSIX_FADV_DONTNEED;
|
|
else if (advise == 5)
|
|
advise = POSIX_FADV_NOREUSE;
|
|
return sys_fadvise64(fd, offset, len, advise);
|
|
}
|
|
|
|
struct fadvise64_64_args {
|
|
int fd;
|
|
long long offset;
|
|
long long len;
|
|
int advice;
|
|
};
|
|
|
|
asmlinkage long
|
|
sys32_fadvise64_64(struct fadvise64_64_args __user *args)
|
|
{
|
|
struct fadvise64_64_args a;
|
|
|
|
if ( copy_from_user(&a, args, sizeof(a)) )
|
|
return -EFAULT;
|
|
if (a.advice == 4)
|
|
a.advice = POSIX_FADV_DONTNEED;
|
|
else if (a.advice == 5)
|
|
a.advice = POSIX_FADV_NOREUSE;
|
|
return sys_fadvise64_64(a.fd, a.offset, a.len, a.advice);
|
|
}
|