uid16.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. /*
  2. * Wrapper functions for 16bit uid back compatibility. All nicely tied
  3. * together in the faint hope we can take the out in five years time.
  4. */
  5. #include <linux/mm.h>
  6. #include <linux/utsname.h>
  7. #include <linux/mman.h>
  8. #include <linux/notifier.h>
  9. #include <linux/reboot.h>
  10. #include <linux/prctl.h>
  11. #include <linux/capability.h>
  12. #include <linux/init.h>
  13. #include <linux/highuid.h>
  14. #include <linux/security.h>
  15. #include <linux/syscalls.h>
  16. #include <asm/uaccess.h>
  17. SYSCALL_DEFINE3(chown16, const char __user *, filename, old_uid_t, user, old_gid_t, group)
  18. {
  19. long ret = sys_chown(filename, low2highuid(user), low2highgid(group));
  20. /* avoid REGPARM breakage on x86: */
  21. asmlinkage_protect(3, ret, filename, user, group);
  22. return ret;
  23. }
  24. SYSCALL_DEFINE3(lchown16, const char __user *, filename, old_uid_t, user, old_gid_t, group)
  25. {
  26. long ret = sys_lchown(filename, low2highuid(user), low2highgid(group));
  27. /* avoid REGPARM breakage on x86: */
  28. asmlinkage_protect(3, ret, filename, user, group);
  29. return ret;
  30. }
  31. SYSCALL_DEFINE3(fchown16, unsigned int, fd, old_uid_t, user, old_gid_t, group)
  32. {
  33. long ret = sys_fchown(fd, low2highuid(user), low2highgid(group));
  34. /* avoid REGPARM breakage on x86: */
  35. asmlinkage_protect(3, ret, fd, user, group);
  36. return ret;
  37. }
  38. SYSCALL_DEFINE2(setregid16, old_gid_t, rgid, old_gid_t, egid)
  39. {
  40. long ret = sys_setregid(low2highgid(rgid), low2highgid(egid));
  41. /* avoid REGPARM breakage on x86: */
  42. asmlinkage_protect(2, ret, rgid, egid);
  43. return ret;
  44. }
  45. SYSCALL_DEFINE1(setgid16, old_gid_t, gid)
  46. {
  47. long ret = sys_setgid(low2highgid(gid));
  48. /* avoid REGPARM breakage on x86: */
  49. asmlinkage_protect(1, ret, gid);
  50. return ret;
  51. }
  52. SYSCALL_DEFINE2(setreuid16, old_uid_t, ruid, old_uid_t, euid)
  53. {
  54. long ret = sys_setreuid(low2highuid(ruid), low2highuid(euid));
  55. /* avoid REGPARM breakage on x86: */
  56. asmlinkage_protect(2, ret, ruid, euid);
  57. return ret;
  58. }
  59. SYSCALL_DEFINE1(setuid16, old_uid_t, uid)
  60. {
  61. long ret = sys_setuid(low2highuid(uid));
  62. /* avoid REGPARM breakage on x86: */
  63. asmlinkage_protect(1, ret, uid);
  64. return ret;
  65. }
  66. SYSCALL_DEFINE3(setresuid16, old_uid_t, ruid, old_uid_t, euid, old_uid_t, suid)
  67. {
  68. long ret = sys_setresuid(low2highuid(ruid), low2highuid(euid),
  69. low2highuid(suid));
  70. /* avoid REGPARM breakage on x86: */
  71. asmlinkage_protect(3, ret, ruid, euid, suid);
  72. return ret;
  73. }
  74. SYSCALL_DEFINE3(getresuid16, old_uid_t __user *, ruid, old_uid_t __user *, euid, old_uid_t __user *, suid)
  75. {
  76. const struct cred *cred = current_cred();
  77. int retval;
  78. if (!(retval = put_user(high2lowuid(cred->uid), ruid)) &&
  79. !(retval = put_user(high2lowuid(cred->euid), euid)))
  80. retval = put_user(high2lowuid(cred->suid), suid);
  81. return retval;
  82. }
  83. SYSCALL_DEFINE3(setresgid16, old_gid_t, rgid, old_gid_t, egid, old_gid_t, sgid)
  84. {
  85. long ret = sys_setresgid(low2highgid(rgid), low2highgid(egid),
  86. low2highgid(sgid));
  87. /* avoid REGPARM breakage on x86: */
  88. asmlinkage_protect(3, ret, rgid, egid, sgid);
  89. return ret;
  90. }
  91. SYSCALL_DEFINE3(getresgid16, old_gid_t __user *, rgid, old_gid_t __user *, egid, old_gid_t __user *, sgid)
  92. {
  93. const struct cred *cred = current_cred();
  94. int retval;
  95. if (!(retval = put_user(high2lowgid(cred->gid), rgid)) &&
  96. !(retval = put_user(high2lowgid(cred->egid), egid)))
  97. retval = put_user(high2lowgid(cred->sgid), sgid);
  98. return retval;
  99. }
  100. SYSCALL_DEFINE1(setfsuid16, old_uid_t, uid)
  101. {
  102. long ret = sys_setfsuid(low2highuid(uid));
  103. /* avoid REGPARM breakage on x86: */
  104. asmlinkage_protect(1, ret, uid);
  105. return ret;
  106. }
  107. SYSCALL_DEFINE1(setfsgid16, old_gid_t, gid)
  108. {
  109. long ret = sys_setfsgid(low2highgid(gid));
  110. /* avoid REGPARM breakage on x86: */
  111. asmlinkage_protect(1, ret, gid);
  112. return ret;
  113. }
  114. static int groups16_to_user(old_gid_t __user *grouplist,
  115. struct group_info *group_info)
  116. {
  117. int i;
  118. old_gid_t group;
  119. for (i = 0; i < group_info->ngroups; i++) {
  120. group = high2lowgid(GROUP_AT(group_info, i));
  121. if (put_user(group, grouplist+i))
  122. return -EFAULT;
  123. }
  124. return 0;
  125. }
  126. static int groups16_from_user(struct group_info *group_info,
  127. old_gid_t __user *grouplist)
  128. {
  129. int i;
  130. old_gid_t group;
  131. for (i = 0; i < group_info->ngroups; i++) {
  132. if (get_user(group, grouplist+i))
  133. return -EFAULT;
  134. GROUP_AT(group_info, i) = low2highgid(group);
  135. }
  136. return 0;
  137. }
  138. SYSCALL_DEFINE2(getgroups16, int, gidsetsize, old_gid_t __user *, grouplist)
  139. {
  140. const struct cred *cred = current_cred();
  141. int i;
  142. if (gidsetsize < 0)
  143. return -EINVAL;
  144. i = cred->group_info->ngroups;
  145. if (gidsetsize) {
  146. if (i > gidsetsize) {
  147. i = -EINVAL;
  148. goto out;
  149. }
  150. if (groups16_to_user(grouplist, cred->group_info)) {
  151. i = -EFAULT;
  152. goto out;
  153. }
  154. }
  155. out:
  156. return i;
  157. }
  158. SYSCALL_DEFINE2(setgroups16, int, gidsetsize, old_gid_t __user *, grouplist)
  159. {
  160. struct group_info *group_info;
  161. int retval;
  162. if (!capable(CAP_SETGID))
  163. return -EPERM;
  164. if ((unsigned)gidsetsize > NGROUPS_MAX)
  165. return -EINVAL;
  166. group_info = groups_alloc(gidsetsize);
  167. if (!group_info)
  168. return -ENOMEM;
  169. retval = groups16_from_user(group_info, grouplist);
  170. if (retval) {
  171. put_group_info(group_info);
  172. return retval;
  173. }
  174. retval = set_current_groups(group_info);
  175. put_group_info(group_info);
  176. return retval;
  177. }
  178. SYSCALL_DEFINE0(getuid16)
  179. {
  180. return high2lowuid(current_uid());
  181. }
  182. SYSCALL_DEFINE0(geteuid16)
  183. {
  184. return high2lowuid(current_euid());
  185. }
  186. SYSCALL_DEFINE0(getgid16)
  187. {
  188. return high2lowgid(current_gid());
  189. }
  190. SYSCALL_DEFINE0(getegid16)
  191. {
  192. return high2lowgid(current_egid());
  193. }