summaryrefslogtreecommitdiff
path: root/io.c
diff options
context:
space:
mode:
authornaruse <naruse@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2012-10-11 15:34:34 +0000
committernaruse <naruse@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2012-10-11 15:34:34 +0000
commit3036104a6a052622af49ec6c92d1b269aee114b7 (patch)
treeaf7fd82469cdc30564a26fa8777168c251b2dc3e /io.c
parenta9f1e5dafb17e8015e201cc03725f55ab681e4e6 (diff)
merge revision(s) 33819,33839:
* io.c (ioctl_narg_len): don't use _IOC_SIZE macro on Linux. On Linux some constants for ioctl(2) doesn't include the size of its return value and 16bit value; for example FIONREAD 0x541B. Moreover the manual, ioctl_list(2), says "Note that the size bits are very unreliable: in lots of cases they are wrong, either because of buggy macros using sizeof(sizeof(struct)), or because of legacy values." So we shouldn't use it. * io.c (ioctl_narg_len, linux_iocparm_len): reinstantiate linux specific narg length calculation. * test/ruby/test_io.rb (test_ioctl_linux2): add new test for old and unstructured ioctl. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/branches/ruby_1_9_3@37144 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 'io.c')
-rw-r--r--io.c30
1 files changed, 27 insertions, 3 deletions
diff --git a/io.c b/io.c
index 4526d4590a..5821b4a434 100644
--- a/io.c
+++ b/io.c
@@ -8019,6 +8019,29 @@ do_ioctl(int fd, ioctl_req_t cmd, long narg)
return retval;
}
+#define DEFULT_IOCTL_NARG_LEN (256)
+
+#ifdef __linux__
+static long
+linux_iocparm_len(ioctl_req_t cmd)
+{
+ long len;
+
+ if ((cmd & 0xFFFF0000) == 0) {
+ /* legacy and unstructured ioctl number. */
+ return DEFULT_IOCTL_NARG_LEN;
+ }
+
+ len = _IOC_SIZE(cmd);
+
+ /* paranoia check for silly drivers which don't keep ioctl convention */
+ if (len < DEFULT_IOCTL_NARG_LEN)
+ len = DEFULT_IOCTL_NARG_LEN;
+
+ return len;
+}
+#endif
+
static long
ioctl_narg_len(ioctl_req_t cmd)
{
@@ -8031,10 +8054,11 @@ ioctl_narg_len(ioctl_req_t cmd)
#endif
#ifdef IOCPARM_LEN
len = IOCPARM_LEN(cmd); /* on BSDish systems we're safe */
-#elif defined(_IOC_SIZE)
- len = _IOC_SIZE(cmd);
+#elif defined(__linux__)
+ len = linux_iocparm_len(cmd);
#else
- len = 256; /* otherwise guess at what's safe */
+ /* otherwise guess at what's safe */
+ len = DEFULT_IOCTL_NARG_LEN;
#endif
return len;