ia32_ioctl.c 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. /* $Id: ia32_ioctl.c,v 1.25 2002/10/11 07:17:06 ak Exp $
  2. * ioctl32.c: Conversion between 32bit and 64bit native ioctls.
  3. *
  4. * Copyright (C) 1997-2000 Jakub Jelinek (jakub@redhat.com)
  5. * Copyright (C) 1998 Eddie C. Dost (ecd@skynet.be)
  6. * Copyright (C) 2001,2002 Andi Kleen, SuSE Labs
  7. *
  8. * These routines maintain argument size conversion between 32bit and 64bit
  9. * ioctls.
  10. */
  11. #define INCLUDES
  12. #include <linux/syscalls.h>
  13. #include "compat_ioctl.c"
  14. #include <asm/mtrr.h>
  15. #include <asm/ia32.h>
  16. #define CODE
  17. #include "compat_ioctl.c"
  18. #ifndef TIOCGDEV
  19. #define TIOCGDEV _IOR('T',0x32, unsigned int)
  20. #endif
  21. static int tiocgdev(unsigned fd, unsigned cmd, unsigned int __user *ptr)
  22. {
  23. struct file *file;
  24. struct tty_struct *real_tty;
  25. int fput_needed, ret;
  26. file = fget_light(fd, &fput_needed);
  27. if (!file)
  28. return -EBADF;
  29. ret = -EINVAL;
  30. if (file->f_op->ioctl != tty_ioctl)
  31. goto out;
  32. real_tty = (struct tty_struct *)file->private_data;
  33. if (!real_tty)
  34. goto out;
  35. ret = put_user(new_encode_dev(tty_devnum(real_tty)), ptr);
  36. out:
  37. fput_light(file, fput_needed);
  38. return ret;
  39. }
  40. #define RTC_IRQP_READ32 _IOR('p', 0x0b, unsigned int) /* Read IRQ rate */
  41. #define RTC_IRQP_SET32 _IOW('p', 0x0c, unsigned int) /* Set IRQ rate */
  42. #define RTC_EPOCH_READ32 _IOR('p', 0x0d, unsigned) /* Read epoch */
  43. #define RTC_EPOCH_SET32 _IOW('p', 0x0e, unsigned) /* Set epoch */
  44. static int rtc32_ioctl(unsigned fd, unsigned cmd, unsigned long arg)
  45. {
  46. unsigned long val;
  47. mm_segment_t oldfs = get_fs();
  48. int ret;
  49. switch (cmd) {
  50. case RTC_IRQP_READ32:
  51. set_fs(KERNEL_DS);
  52. ret = sys_ioctl(fd, RTC_IRQP_READ, (unsigned long)&val);
  53. set_fs(oldfs);
  54. if (!ret)
  55. ret = put_user(val, (unsigned int __user *) arg);
  56. return ret;
  57. case RTC_IRQP_SET32:
  58. cmd = RTC_IRQP_SET;
  59. break;
  60. case RTC_EPOCH_READ32:
  61. set_fs(KERNEL_DS);
  62. ret = sys_ioctl(fd, RTC_EPOCH_READ, (unsigned long) &val);
  63. set_fs(oldfs);
  64. if (!ret)
  65. ret = put_user(val, (unsigned int __user *) arg);
  66. return ret;
  67. case RTC_EPOCH_SET32:
  68. cmd = RTC_EPOCH_SET;
  69. break;
  70. }
  71. return sys_ioctl(fd,cmd,arg);
  72. }
  73. /* /proc/mtrr ioctls */
  74. struct mtrr_sentry32
  75. {
  76. compat_ulong_t base; /* Base address */
  77. compat_uint_t size; /* Size of region */
  78. compat_uint_t type; /* Type of region */
  79. };
  80. struct mtrr_gentry32
  81. {
  82. compat_ulong_t regnum; /* Register number */
  83. compat_uint_t base; /* Base address */
  84. compat_uint_t size; /* Size of region */
  85. compat_uint_t type; /* Type of region */
  86. };
  87. #define MTRR_IOCTL_BASE 'M'
  88. #define MTRRIOC32_ADD_ENTRY _IOW(MTRR_IOCTL_BASE, 0, struct mtrr_sentry32)
  89. #define MTRRIOC32_SET_ENTRY _IOW(MTRR_IOCTL_BASE, 1, struct mtrr_sentry32)
  90. #define MTRRIOC32_DEL_ENTRY _IOW(MTRR_IOCTL_BASE, 2, struct mtrr_sentry32)
  91. #define MTRRIOC32_GET_ENTRY _IOWR(MTRR_IOCTL_BASE, 3, struct mtrr_gentry32)
  92. #define MTRRIOC32_KILL_ENTRY _IOW(MTRR_IOCTL_BASE, 4, struct mtrr_sentry32)
  93. #define MTRRIOC32_ADD_PAGE_ENTRY _IOW(MTRR_IOCTL_BASE, 5, struct mtrr_sentry32)
  94. #define MTRRIOC32_SET_PAGE_ENTRY _IOW(MTRR_IOCTL_BASE, 6, struct mtrr_sentry32)
  95. #define MTRRIOC32_DEL_PAGE_ENTRY _IOW(MTRR_IOCTL_BASE, 7, struct mtrr_sentry32)
  96. #define MTRRIOC32_GET_PAGE_ENTRY _IOWR(MTRR_IOCTL_BASE, 8, struct mtrr_gentry32)
  97. #define MTRRIOC32_KILL_PAGE_ENTRY _IOW(MTRR_IOCTL_BASE, 9, struct mtrr_sentry32)
  98. static int mtrr_ioctl32(unsigned int fd, unsigned int cmd, unsigned long arg)
  99. {
  100. struct mtrr_gentry g;
  101. struct mtrr_sentry s;
  102. int get = 0, err = 0;
  103. struct mtrr_gentry32 __user *g32 = (struct mtrr_gentry32 __user *)arg;
  104. mm_segment_t oldfs = get_fs();
  105. switch (cmd) {
  106. #define SET(x) case MTRRIOC32_ ## x ## _ENTRY: cmd = MTRRIOC_ ## x ## _ENTRY; break
  107. #define GET(x) case MTRRIOC32_ ## x ## _ENTRY: cmd = MTRRIOC_ ## x ## _ENTRY; get=1; break
  108. SET(ADD);
  109. SET(SET);
  110. SET(DEL);
  111. GET(GET);
  112. SET(KILL);
  113. SET(ADD_PAGE);
  114. SET(SET_PAGE);
  115. SET(DEL_PAGE);
  116. GET(GET_PAGE);
  117. SET(KILL_PAGE);
  118. }
  119. if (get) {
  120. err = get_user(g.regnum, &g32->regnum);
  121. err |= get_user(g.base, &g32->base);
  122. err |= get_user(g.size, &g32->size);
  123. err |= get_user(g.type, &g32->type);
  124. arg = (unsigned long)&g;
  125. } else {
  126. struct mtrr_sentry32 __user *s32 = (struct mtrr_sentry32 __user *)arg;
  127. err = get_user(s.base, &s32->base);
  128. err |= get_user(s.size, &s32->size);
  129. err |= get_user(s.type, &s32->type);
  130. arg = (unsigned long)&s;
  131. }
  132. if (err) return err;
  133. set_fs(KERNEL_DS);
  134. err = sys_ioctl(fd, cmd, arg);
  135. set_fs(oldfs);
  136. if (!err && get) {
  137. err = put_user(g.base, &g32->base);
  138. err |= put_user(g.size, &g32->size);
  139. err |= put_user(g.regnum, &g32->regnum);
  140. err |= put_user(g.type, &g32->type);
  141. }
  142. return err;
  143. }
  144. #define HANDLE_IOCTL(cmd,handler) { (cmd), (ioctl_trans_handler_t)(handler) },
  145. #define COMPATIBLE_IOCTL(cmd) HANDLE_IOCTL(cmd,sys_ioctl)
  146. struct ioctl_trans ioctl_start[] = {
  147. #include <linux/compat_ioctl.h>
  148. #define DECLARES
  149. #include "compat_ioctl.c"
  150. COMPATIBLE_IOCTL(HDIO_SET_KEEPSETTINGS)
  151. COMPATIBLE_IOCTL(HDIO_SCAN_HWIF)
  152. COMPATIBLE_IOCTL(BLKRASET)
  153. COMPATIBLE_IOCTL(0x4B50) /* KDGHWCLK - not in the kernel, but don't complain */
  154. COMPATIBLE_IOCTL(0x4B51) /* KDSHWCLK - not in the kernel, but don't complain */
  155. COMPATIBLE_IOCTL(FIOQSIZE)
  156. /* And these ioctls need translation */
  157. HANDLE_IOCTL(TIOCGDEV, tiocgdev)
  158. /* realtime device */
  159. HANDLE_IOCTL(RTC_IRQP_READ, rtc32_ioctl)
  160. HANDLE_IOCTL(RTC_IRQP_READ32,rtc32_ioctl)
  161. HANDLE_IOCTL(RTC_IRQP_SET32, rtc32_ioctl)
  162. HANDLE_IOCTL(RTC_EPOCH_READ32, rtc32_ioctl)
  163. HANDLE_IOCTL(RTC_EPOCH_SET32, rtc32_ioctl)
  164. /* take care of sizeof(sizeof()) breakage */
  165. /* mtrr */
  166. HANDLE_IOCTL(MTRRIOC32_ADD_ENTRY, mtrr_ioctl32)
  167. HANDLE_IOCTL(MTRRIOC32_SET_ENTRY, mtrr_ioctl32)
  168. HANDLE_IOCTL(MTRRIOC32_DEL_ENTRY, mtrr_ioctl32)
  169. HANDLE_IOCTL(MTRRIOC32_GET_ENTRY, mtrr_ioctl32)
  170. HANDLE_IOCTL(MTRRIOC32_KILL_ENTRY, mtrr_ioctl32)
  171. HANDLE_IOCTL(MTRRIOC32_ADD_PAGE_ENTRY, mtrr_ioctl32)
  172. HANDLE_IOCTL(MTRRIOC32_SET_PAGE_ENTRY, mtrr_ioctl32)
  173. HANDLE_IOCTL(MTRRIOC32_DEL_PAGE_ENTRY, mtrr_ioctl32)
  174. HANDLE_IOCTL(MTRRIOC32_GET_PAGE_ENTRY, mtrr_ioctl32)
  175. HANDLE_IOCTL(MTRRIOC32_KILL_PAGE_ENTRY, mtrr_ioctl32)
  176. };
  177. int ioctl_table_size = ARRAY_SIZE(ioctl_start);