ia32_ioctl.c 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  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/ia32.h>
  15. #define CODE
  16. #include "compat_ioctl.c"
  17. #ifndef TIOCGDEV
  18. #define TIOCGDEV _IOR('T',0x32, unsigned int)
  19. #endif
  20. static int tiocgdev(unsigned fd, unsigned cmd, unsigned int __user *ptr)
  21. {
  22. struct file *file;
  23. struct tty_struct *real_tty;
  24. int fput_needed, ret;
  25. file = fget_light(fd, &fput_needed);
  26. if (!file)
  27. return -EBADF;
  28. ret = -EINVAL;
  29. if (file->f_op->ioctl != tty_ioctl)
  30. goto out;
  31. real_tty = (struct tty_struct *)file->private_data;
  32. if (!real_tty)
  33. goto out;
  34. ret = put_user(new_encode_dev(tty_devnum(real_tty)), ptr);
  35. out:
  36. fput_light(file, fput_needed);
  37. return ret;
  38. }
  39. #define RTC_IRQP_READ32 _IOR('p', 0x0b, unsigned int) /* Read IRQ rate */
  40. #define RTC_IRQP_SET32 _IOW('p', 0x0c, unsigned int) /* Set IRQ rate */
  41. #define RTC_EPOCH_READ32 _IOR('p', 0x0d, unsigned) /* Read epoch */
  42. #define RTC_EPOCH_SET32 _IOW('p', 0x0e, unsigned) /* Set epoch */
  43. static int rtc32_ioctl(unsigned fd, unsigned cmd, unsigned long arg)
  44. {
  45. unsigned long val;
  46. mm_segment_t oldfs = get_fs();
  47. int ret;
  48. switch (cmd) {
  49. case RTC_IRQP_READ32:
  50. set_fs(KERNEL_DS);
  51. ret = sys_ioctl(fd, RTC_IRQP_READ, (unsigned long)&val);
  52. set_fs(oldfs);
  53. if (!ret)
  54. ret = put_user(val, (unsigned int __user *) arg);
  55. return ret;
  56. case RTC_IRQP_SET32:
  57. cmd = RTC_IRQP_SET;
  58. break;
  59. case RTC_EPOCH_READ32:
  60. set_fs(KERNEL_DS);
  61. ret = sys_ioctl(fd, RTC_EPOCH_READ, (unsigned long) &val);
  62. set_fs(oldfs);
  63. if (!ret)
  64. ret = put_user(val, (unsigned int __user *) arg);
  65. return ret;
  66. case RTC_EPOCH_SET32:
  67. cmd = RTC_EPOCH_SET;
  68. break;
  69. }
  70. return sys_ioctl(fd,cmd,arg);
  71. }
  72. #define HANDLE_IOCTL(cmd,handler) { (cmd), (ioctl_trans_handler_t)(handler) },
  73. #define COMPATIBLE_IOCTL(cmd) HANDLE_IOCTL(cmd,sys_ioctl)
  74. struct ioctl_trans ioctl_start[] = {
  75. #include <linux/compat_ioctl.h>
  76. #define DECLARES
  77. #include "compat_ioctl.c"
  78. COMPATIBLE_IOCTL(HDIO_SET_KEEPSETTINGS)
  79. COMPATIBLE_IOCTL(HDIO_SCAN_HWIF)
  80. COMPATIBLE_IOCTL(BLKRASET)
  81. COMPATIBLE_IOCTL(0x4B50) /* KDGHWCLK - not in the kernel, but don't complain */
  82. COMPATIBLE_IOCTL(0x4B51) /* KDSHWCLK - not in the kernel, but don't complain */
  83. COMPATIBLE_IOCTL(FIOQSIZE)
  84. /* And these ioctls need translation */
  85. HANDLE_IOCTL(TIOCGDEV, tiocgdev)
  86. /* realtime device */
  87. HANDLE_IOCTL(RTC_IRQP_READ, rtc32_ioctl)
  88. HANDLE_IOCTL(RTC_IRQP_READ32,rtc32_ioctl)
  89. HANDLE_IOCTL(RTC_IRQP_SET32, rtc32_ioctl)
  90. HANDLE_IOCTL(RTC_EPOCH_READ32, rtc32_ioctl)
  91. HANDLE_IOCTL(RTC_EPOCH_SET32, rtc32_ioctl)
  92. /* take care of sizeof(sizeof()) breakage */
  93. };
  94. int ioctl_table_size = ARRAY_SIZE(ioctl_start);