📄 compat_linux.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 longsys32_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 longsys32_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;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -