2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* Wrapper functions for 16bit uid back compatibility. All nicely tied
|
|
|
|
* together in the faint hope we can take the out in five years time.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/mm.h>
|
|
|
|
#include <linux/mman.h>
|
|
|
|
#include <linux/notifier.h>
|
|
|
|
#include <linux/reboot.h>
|
|
|
|
#include <linux/prctl.h>
|
2006-01-12 04:17:46 +08:00
|
|
|
#include <linux/capability.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/highuid.h>
|
|
|
|
#include <linux/security.h>
|
|
|
|
#include <linux/syscalls.h>
|
|
|
|
|
|
|
|
#include <asm/uaccess.h>
|
|
|
|
|
2009-01-14 21:14:19 +08:00
|
|
|
SYSCALL_DEFINE3(chown16, const char __user *, filename, old_uid_t, user, old_gid_t, group)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_chown(filename, low2highuid(user), low2highgid(group));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(3, ret, filename, user, group);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:19 +08:00
|
|
|
SYSCALL_DEFINE3(lchown16, const char __user *, filename, old_uid_t, user, old_gid_t, group)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_lchown(filename, low2highuid(user), low2highgid(group));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(3, ret, filename, user, group);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:19 +08:00
|
|
|
SYSCALL_DEFINE3(fchown16, unsigned int, fd, old_uid_t, user, old_gid_t, group)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_fchown(fd, low2highuid(user), low2highgid(group));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(3, ret, fd, user, group);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:20 +08:00
|
|
|
SYSCALL_DEFINE2(setregid16, old_gid_t, rgid, old_gid_t, egid)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_setregid(low2highgid(rgid), low2highgid(egid));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(2, ret, rgid, egid);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:20 +08:00
|
|
|
SYSCALL_DEFINE1(setgid16, old_gid_t, gid)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_setgid(low2highgid(gid));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(1, ret, gid);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:20 +08:00
|
|
|
SYSCALL_DEFINE2(setreuid16, old_uid_t, ruid, old_uid_t, euid)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_setreuid(low2highuid(ruid), low2highuid(euid));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(2, ret, ruid, euid);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:20 +08:00
|
|
|
SYSCALL_DEFINE1(setuid16, old_uid_t, uid)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_setuid(low2highuid(uid));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(1, ret, uid);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:20 +08:00
|
|
|
SYSCALL_DEFINE3(setresuid16, old_uid_t, ruid, old_uid_t, euid, old_uid_t, suid)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_setresuid(low2highuid(ruid), low2highuid(euid),
|
|
|
|
low2highuid(suid));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(3, ret, ruid, euid, suid);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
userns: Convert setting and getting uid and gid system calls to use kuid and kgid
Convert setregid, setgid, setreuid, setuid,
setresuid, getresuid, setresgid, getresgid, setfsuid, setfsgid,
getuid, geteuid, getgid, getegid,
waitpid, waitid, wait4.
Convert userspace uids and gids into kuids and kgids before
being placed on struct cred. Convert struct cred kuids and
kgids into userspace uids and gids when returning them.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
2012-02-08 10:51:01 +08:00
|
|
|
SYSCALL_DEFINE3(getresuid16, old_uid_t __user *, ruidp, old_uid_t __user *, euidp, old_uid_t __user *, suidp)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2008-11-14 07:39:18 +08:00
|
|
|
const struct cred *cred = current_cred();
|
2005-04-17 06:20:36 +08:00
|
|
|
int retval;
|
userns: Convert setting and getting uid and gid system calls to use kuid and kgid
Convert setregid, setgid, setreuid, setuid,
setresuid, getresuid, setresgid, getresgid, setfsuid, setfsgid,
getuid, geteuid, getgid, getegid,
waitpid, waitid, wait4.
Convert userspace uids and gids into kuids and kgids before
being placed on struct cred. Convert struct cred kuids and
kgids into userspace uids and gids when returning them.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
2012-02-08 10:51:01 +08:00
|
|
|
old_uid_t ruid, euid, suid;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
userns: Convert setting and getting uid and gid system calls to use kuid and kgid
Convert setregid, setgid, setreuid, setuid,
setresuid, getresuid, setresgid, getresgid, setfsuid, setfsgid,
getuid, geteuid, getgid, getegid,
waitpid, waitid, wait4.
Convert userspace uids and gids into kuids and kgids before
being placed on struct cred. Convert struct cred kuids and
kgids into userspace uids and gids when returning them.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
2012-02-08 10:51:01 +08:00
|
|
|
ruid = high2lowuid(from_kuid_munged(cred->user_ns, cred->uid));
|
|
|
|
euid = high2lowuid(from_kuid_munged(cred->user_ns, cred->euid));
|
|
|
|
suid = high2lowuid(from_kuid_munged(cred->user_ns, cred->suid));
|
|
|
|
|
|
|
|
if (!(retval = put_user(ruid, ruidp)) &&
|
|
|
|
!(retval = put_user(euid, euidp)))
|
|
|
|
retval = put_user(suid, suidp);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:20 +08:00
|
|
|
SYSCALL_DEFINE3(setresgid16, old_gid_t, rgid, old_gid_t, egid, old_gid_t, sgid)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_setresgid(low2highgid(rgid), low2highgid(egid),
|
|
|
|
low2highgid(sgid));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(3, ret, rgid, egid, sgid);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:20 +08:00
|
|
|
|
userns: Convert setting and getting uid and gid system calls to use kuid and kgid
Convert setregid, setgid, setreuid, setuid,
setresuid, getresuid, setresgid, getresgid, setfsuid, setfsgid,
getuid, geteuid, getgid, getegid,
waitpid, waitid, wait4.
Convert userspace uids and gids into kuids and kgids before
being placed on struct cred. Convert struct cred kuids and
kgids into userspace uids and gids when returning them.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
2012-02-08 10:51:01 +08:00
|
|
|
SYSCALL_DEFINE3(getresgid16, old_gid_t __user *, rgidp, old_gid_t __user *, egidp, old_gid_t __user *, sgidp)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2008-11-14 07:39:18 +08:00
|
|
|
const struct cred *cred = current_cred();
|
2005-04-17 06:20:36 +08:00
|
|
|
int retval;
|
userns: Convert setting and getting uid and gid system calls to use kuid and kgid
Convert setregid, setgid, setreuid, setuid,
setresuid, getresuid, setresgid, getresgid, setfsuid, setfsgid,
getuid, geteuid, getgid, getegid,
waitpid, waitid, wait4.
Convert userspace uids and gids into kuids and kgids before
being placed on struct cred. Convert struct cred kuids and
kgids into userspace uids and gids when returning them.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
2012-02-08 10:51:01 +08:00
|
|
|
old_gid_t rgid, egid, sgid;
|
|
|
|
|
|
|
|
rgid = high2lowgid(from_kgid_munged(cred->user_ns, cred->gid));
|
|
|
|
egid = high2lowgid(from_kgid_munged(cred->user_ns, cred->egid));
|
|
|
|
sgid = high2lowgid(from_kgid_munged(cred->user_ns, cred->sgid));
|
2005-04-17 06:20:36 +08:00
|
|
|
|
userns: Convert setting and getting uid and gid system calls to use kuid and kgid
Convert setregid, setgid, setreuid, setuid,
setresuid, getresuid, setresgid, getresgid, setfsuid, setfsgid,
getuid, geteuid, getgid, getegid,
waitpid, waitid, wait4.
Convert userspace uids and gids into kuids and kgids before
being placed on struct cred. Convert struct cred kuids and
kgids into userspace uids and gids when returning them.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
2012-02-08 10:51:01 +08:00
|
|
|
if (!(retval = put_user(rgid, rgidp)) &&
|
|
|
|
!(retval = put_user(egid, egidp)))
|
|
|
|
retval = put_user(sgid, sgidp);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:20 +08:00
|
|
|
SYSCALL_DEFINE1(setfsuid16, old_uid_t, uid)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_setfsuid(low2highuid(uid));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(1, ret, uid);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:20 +08:00
|
|
|
SYSCALL_DEFINE1(setfsgid16, old_gid_t, gid)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2006-04-20 05:41:39 +08:00
|
|
|
long ret = sys_setfsgid(low2highgid(gid));
|
|
|
|
/* avoid REGPARM breakage on x86: */
|
2008-04-11 06:37:38 +08:00
|
|
|
asmlinkage_protect(1, ret, gid);
|
2006-04-20 05:41:39 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int groups16_to_user(old_gid_t __user *grouplist,
|
|
|
|
struct group_info *group_info)
|
|
|
|
{
|
2011-11-15 07:56:38 +08:00
|
|
|
struct user_namespace *user_ns = current_user_ns();
|
2005-04-17 06:20:36 +08:00
|
|
|
int i;
|
|
|
|
old_gid_t group;
|
2011-11-15 07:56:38 +08:00
|
|
|
kgid_t kgid;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
for (i = 0; i < group_info->ngroups; i++) {
|
2011-11-15 07:56:38 +08:00
|
|
|
kgid = GROUP_AT(group_info, i);
|
|
|
|
group = high2lowgid(from_kgid_munged(user_ns, kgid));
|
2005-04-17 06:20:36 +08:00
|
|
|
if (put_user(group, grouplist+i))
|
|
|
|
return -EFAULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int groups16_from_user(struct group_info *group_info,
|
|
|
|
old_gid_t __user *grouplist)
|
|
|
|
{
|
2011-11-15 07:56:38 +08:00
|
|
|
struct user_namespace *user_ns = current_user_ns();
|
2005-04-17 06:20:36 +08:00
|
|
|
int i;
|
|
|
|
old_gid_t group;
|
2011-11-15 07:56:38 +08:00
|
|
|
kgid_t kgid;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
for (i = 0; i < group_info->ngroups; i++) {
|
|
|
|
if (get_user(group, grouplist+i))
|
|
|
|
return -EFAULT;
|
2011-11-15 07:56:38 +08:00
|
|
|
|
|
|
|
kgid = make_kgid(user_ns, low2highgid(group));
|
|
|
|
if (!gid_valid(kgid))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
GROUP_AT(group_info, i) = kgid;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:21 +08:00
|
|
|
SYSCALL_DEFINE2(getgroups16, int, gidsetsize, old_gid_t __user *, grouplist)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2008-11-14 07:39:18 +08:00
|
|
|
const struct cred *cred = current_cred();
|
|
|
|
int i;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
if (gidsetsize < 0)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2008-11-14 07:39:18 +08:00
|
|
|
i = cred->group_info->ngroups;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (gidsetsize) {
|
|
|
|
if (i > gidsetsize) {
|
|
|
|
i = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2008-11-14 07:39:18 +08:00
|
|
|
if (groups16_to_user(grouplist, cred->group_info)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
i = -EFAULT;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:21 +08:00
|
|
|
SYSCALL_DEFINE2(setgroups16, int, gidsetsize, old_gid_t __user *, grouplist)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct group_info *group_info;
|
|
|
|
int retval;
|
|
|
|
|
2011-03-24 07:43:24 +08:00
|
|
|
if (!nsown_capable(CAP_SETGID))
|
2005-04-17 06:20:36 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:21 +08:00
|
|
|
SYSCALL_DEFINE0(getuid16)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
userns: Convert setting and getting uid and gid system calls to use kuid and kgid
Convert setregid, setgid, setreuid, setuid,
setresuid, getresuid, setresgid, getresgid, setfsuid, setfsgid,
getuid, geteuid, getgid, getegid,
waitpid, waitid, wait4.
Convert userspace uids and gids into kuids and kgids before
being placed on struct cred. Convert struct cred kuids and
kgids into userspace uids and gids when returning them.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
2012-02-08 10:51:01 +08:00
|
|
|
return high2lowuid(from_kuid_munged(current_user_ns(), current_uid()));
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:21 +08:00
|
|
|
SYSCALL_DEFINE0(geteuid16)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
userns: Convert setting and getting uid and gid system calls to use kuid and kgid
Convert setregid, setgid, setreuid, setuid,
setresuid, getresuid, setresgid, getresgid, setfsuid, setfsgid,
getuid, geteuid, getgid, getegid,
waitpid, waitid, wait4.
Convert userspace uids and gids into kuids and kgids before
being placed on struct cred. Convert struct cred kuids and
kgids into userspace uids and gids when returning them.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
2012-02-08 10:51:01 +08:00
|
|
|
return high2lowuid(from_kuid_munged(current_user_ns(), current_euid()));
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:21 +08:00
|
|
|
SYSCALL_DEFINE0(getgid16)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
userns: Convert setting and getting uid and gid system calls to use kuid and kgid
Convert setregid, setgid, setreuid, setuid,
setresuid, getresuid, setresgid, getresgid, setfsuid, setfsgid,
getuid, geteuid, getgid, getegid,
waitpid, waitid, wait4.
Convert userspace uids and gids into kuids and kgids before
being placed on struct cred. Convert struct cred kuids and
kgids into userspace uids and gids when returning them.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
2012-02-08 10:51:01 +08:00
|
|
|
return high2lowgid(from_kgid_munged(current_user_ns(), current_gid()));
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-01-14 21:14:21 +08:00
|
|
|
SYSCALL_DEFINE0(getegid16)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
userns: Convert setting and getting uid and gid system calls to use kuid and kgid
Convert setregid, setgid, setreuid, setuid,
setresuid, getresuid, setresgid, getresgid, setfsuid, setfsgid,
getuid, geteuid, getgid, getegid,
waitpid, waitid, wait4.
Convert userspace uids and gids into kuids and kgids before
being placed on struct cred. Convert struct cred kuids and
kgids into userspace uids and gids when returning them.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
2012-02-08 10:51:01 +08:00
|
|
|
return high2lowgid(from_kgid_munged(current_user_ns(), current_egid()));
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|