⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 sys_sparc32.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 2 页
字号:
/* $Id: sys_sparc32.c,v 1.184 2002/02/09 19:49:31 davem Exp $ * sys_sparc32.c: Conversion between 32bit and 64bit native syscalls. * * Copyright (C) 1997,1998 Jakub Jelinek (jj@sunsite.mff.cuni.cz) * Copyright (C) 1997 David S. Miller (davem@caip.rutgers.edu) * * These routines maintain argument size conversion between 32bit and 64bit * environment. */#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/dnotify.h>#include <linux/security.h>#include <linux/compat.h>#include <linux/vfs.h>#include <linux/netfilter_ipv4/ip_tables.h>#include <linux/ptrace.h>#include <linux/highuid.h>#include <asm/types.h>#include <asm/ipc.h>#include <asm/uaccess.h>#include <asm/fpumacro.h>#include <asm/semaphore.h>#include <asm/mmu_context.h>asmlinkage long sys32_chown16(const char __user * filename, u16 user, u16 group){	return sys_chown(filename, low2highuid(user), low2highgid(group));}asmlinkage long sys32_lchown16(const char __user * 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 __user *ruid, u16 __user *euid, u16 __user *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 __user *rgid, u16 __user *egid, u16 __user *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 __user *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 __user *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 __user *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 __user *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 long get_tv32(struct timeval *o, struct compat_timeval __user *i){	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 put_tv32(struct compat_timeval __user *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)));}#ifdef CONFIG_SYSVIPC                                                        asmlinkage long compat_sys_ipc(u32 call, u32 first, u32 second, u32 third, compat_uptr_t ptr, u32 fifth){	int version;	version = call >> 16; /* hack for backward compatibility */	call &= 0xffff;	switch (call) {	case SEMTIMEDOP:		if (fifth)			/* sign extend semid */			return compat_sys_semtimedop((int)first,						     compat_ptr(ptr), second,						     compat_ptr(fifth));		/* else fall through for normal semop() */	case SEMOP:		/* struct sembuf is the same on 32 and 64bit :)) */		/* sign extend semid */		return sys_semtimedop((int)first, compat_ptr(ptr), second,				      NULL);	case SEMGET:		/* sign extend key, nsems */		return sys_semget((int)first, (int)second, third);	case SEMCTL:		/* sign extend semid, semnum */		return compat_sys_semctl((int)first, (int)second, third,					 compat_ptr(ptr));	case MSGSND:		/* sign extend msqid */		return compat_sys_msgsnd((int)first, (int)second, third,					 compat_ptr(ptr));	case MSGRCV:		/* sign extend msqid, msgtyp */		return compat_sys_msgrcv((int)first, second, (int)fifth,					 third, version, compat_ptr(ptr));	case MSGGET:		/* sign extend key */		return sys_msgget((int)first, second);	case MSGCTL:		/* sign extend msqid */		return compat_sys_msgctl((int)first, second, compat_ptr(ptr));	case SHMAT:		/* sign extend shmid */		return compat_sys_shmat((int)first, second, third, version,					compat_ptr(ptr));	case SHMDT:		return sys_shmdt(compat_ptr(ptr));	case SHMGET:		/* sign extend key_t */		return sys_shmget((int)first, second, third);	case SHMCTL:		/* sign extend shmid */		return compat_sys_shmctl((int)first, second, compat_ptr(ptr));	default:		return -ENOSYS;	};	return -ENOSYS;}#endifasmlinkage long sys32_truncate64(const char __user * 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 __user *statbuf){	int err;	if (stat->size > MAX_NON_LFS || !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);	err |= put_user(0, &statbuf->__unused4[0]);	err |= put_user(0, &statbuf->__unused4[1]);	return err;}int cp_compat_stat64(struct kstat *stat, struct compat_stat64 __user *statbuf){	int err;	err  = put_user(huge_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(stat->uid, &statbuf->st_uid);	err |= put_user(stat->gid, &statbuf->st_gid);	err |= put_user(huge_encode_dev(stat->rdev), &statbuf->st_rdev);	err |= put_user(0, (unsigned long __user *) &statbuf->__pad3[0]);	err |= put_user(stat->size, &statbuf->st_size);	err |= put_user(stat->blksize, &statbuf->st_blksize);	err |= put_user(0, (unsigned int __user *) &statbuf->__pad4[0]);	err |= put_user(0, (unsigned int __user *) &statbuf->__pad4[4]);	err |= put_user(stat->blocks, &statbuf->st_blocks);	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(0, &statbuf->__unused4);	err |= put_user(0, &statbuf->__unused5);	return err;}asmlinkage long compat_sys_stat64(char __user * filename,		struct compat_stat64 __user *statbuf){	struct kstat stat;	int error = vfs_stat(filename, &stat);	if (!error)		error = cp_compat_stat64(&stat, statbuf);	return error;}asmlinkage long compat_sys_lstat64(char __user * filename,		struct compat_stat64 __user *statbuf){	struct kstat stat;	int error = vfs_lstat(filename, &stat);	if (!error)		error = cp_compat_stat64(&stat, statbuf);	return error;}asmlinkage long compat_sys_fstat64(unsigned int fd,		struct compat_stat64 __user * statbuf){	struct kstat stat;	int error = vfs_fstat(fd, &stat);	if (!error)		error = cp_compat_stat64(&stat, statbuf);	return error;}asmlinkage long compat_sys_sysfs(int option, u32 arg1, u32 arg2){	return sys_sysfs(option, arg1, arg2);}struct sysinfo32 {        s32 uptime;        u32 loads[3];        u32 totalram;        u32 freeram;        u32 sharedram;        u32 bufferram;        u32 totalswap;        u32 freeswap;        unsigned short procs;	unsigned short pad;	u32 totalhigh;	u32 freehigh;	u32 mem_unit;	char _f[20-2*sizeof(int)-sizeof(int)];};asmlinkage long sys32_sysinfo(struct sysinfo32 __user *info){	struct sysinfo s;	int ret, err;	int bitcount = 0;	mm_segment_t old_fs = get_fs ();		set_fs(KERNEL_DS);	ret = sys_sysinfo((struct sysinfo __user *) &s);	set_fs(old_fs);	/* Check to see if any memory value is too large for 32-bit and         * scale down if needed.         */	if ((s.totalram >> 32) || (s.totalswap >> 32)) {		while (s.mem_unit < PAGE_SIZE) {			s.mem_unit <<= 1;			bitcount++;		}		s.totalram >>= bitcount;		s.freeram >>= bitcount;		s.sharedram >>= bitcount;		s.bufferram >>= bitcount;		s.totalswap >>= bitcount;		s.freeswap >>= bitcount;		s.totalhigh >>= bitcount;		s.freehigh >>= bitcount;	}	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 compat_sys_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, (struct timespec __user *) &t);	set_fs (old_fs);	if (put_compat_timespec(&t, interval))		return -EFAULT;	return ret;}asmlinkage long compat_sys_rt_sigprocmask(int how,					  compat_sigset_t __user *set,					  compat_sigset_t __user *oset,					  compat_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 ? (sigset_t __user *) &s : NULL,				 oset ? (sigset_t __user *) &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,				    compat_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((sigset_t __user *) &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 compat_sys_rt_sigqueueinfo(int pid, int sig,					   struct compat_siginfo __user *uinfo){	siginfo_t info;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -