summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorkosaki <kosaki@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2011-03-04 16:38:34 +0000
committerkosaki <kosaki@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2011-03-04 16:38:34 +0000
commitc0359f81764e30ad041b152aeea1a2c0a95c3ccb (patch)
treefa7f805dedc13b163e2eb926282700070636ed74
parent1d6bd86acd77ad4dde428ca53878ae0b24b487ed (diff)
* io.c (io_cntl, nogvl_io_cntl): IO.fcntl() and IO.ioctl()
release GVL during calling kernel interface. Suggested by Eric Wong. [ruby-core:35417][Bug #4463] * test/ruby/test_io.rb (TestIO#test_fcntl_lock): add new test for IO.fcntl(). git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@31025 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r--ChangeLog9
-rw-r--r--io.c45
-rw-r--r--test/ruby/test_io.rb34
3 files changed, 75 insertions, 13 deletions
diff --git a/ChangeLog b/ChangeLog
index 18aebb8fe6..fd0333e7fb 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,12 @@
+Sat Mar 5 01:33:46 2011 KOSAKI Motohiro <kosaki.motohiro@gmail.com>
+
+ * io.c (io_cntl, nogvl_io_cntl): IO.fcntl() and IO.ioctl()
+ release GVL during calling kernel interface.
+ Suggested by Eric Wong. [ruby-core:35417][Bug #4463]
+
+ * test/ruby/test_io.rb (TestIO#test_fcntl_lock): add new test for
+ IO.fcntl().
+
Fri Mar 4 23:09:12 2011 CHIKANAGA Tomoyuki <nagachika00@gmail.com>
* test/testunit/test_parallel.rb
diff --git a/io.c b/io.c
index f023382ae5..055262aa34 100644
--- a/io.c
+++ b/io.c
@@ -7649,28 +7649,47 @@ rb_f_select(int argc, VALUE *argv, VALUE obj)
}
+struct io_cntl_arg {
+ int fd;
+ int cmd;
+ long narg;
+ int io_p;
+};
+
+static VALUE nogvl_io_cntl(void *ptr)
+{
+ struct io_cntl_arg *arg = ptr;
+
+ if (arg->io_p)
+ return (VALUE)ioctl(arg->fd, arg->cmd, arg->narg);
+ else
+ return (VALUE)fcntl(arg->fd, arg->cmd, arg->narg);
+}
+
static int
io_cntl(int fd, int cmd, long narg, int io_p)
{
int retval;
+ struct io_cntl_arg arg;
-#ifdef HAVE_FCNTL
-# if defined(__CYGWIN__)
- retval = io_p?ioctl(fd, cmd, (void*)narg):fcntl(fd, cmd, narg);
-# else
- retval = io_p?ioctl(fd, cmd, narg):fcntl(fd, cmd, narg);
-# endif
-# if defined(F_DUPFD)
- if (!io_p && retval != -1 && cmd == F_DUPFD) {
- UPDATE_MAXFD(retval);
- }
-# endif
-#else
+#ifndef HAVE_FCNTL
if (!io_p) {
rb_notimplement();
}
- retval = ioctl(fd, cmd, narg);
#endif
+
+ arg.fd = fd;
+ arg.cmd = cmd;
+ arg.narg = narg;
+ arg.io_p = io_p;
+
+ retval = (int)rb_thread_blocking_region(nogvl_io_cntl, &arg, RUBY_UBF_IO, 0);
+#if defined(F_DUPFD)
+ if (!io_p && retval != -1 && cmd == F_DUPFD) {
+ UPDATE_MAXFD(retval);
+ }
+#endif
+
return retval;
}
diff --git a/test/ruby/test_io.rb b/test/ruby/test_io.rb
index 6cc2897f5f..a0f1569895 100644
--- a/test/ruby/test_io.rb
+++ b/test/ruby/test_io.rb
@@ -1776,4 +1776,38 @@ End
end
end
end
+
+
+ if /x86_64-linux/ =~ RUBY_PLATFORM # A binary form of struct flock depend on platform
+ F_WRLCK = 1
+ F_UNLCK = 2
+ SEEK_SET = 0
+
+ def test_fcntl_lock
+ pad = 0
+ flocktype = "s!s!s!s!L!L!i!"
+
+ Tempfile.open(self.class.name) do |f|
+ r, w = IO.pipe
+ pid = fork do
+ r.close
+ lock = [F_WRLCK, SEEK_SET, pad, pad, 0, 0, 0].pack(flocktype)
+ f.fcntl Fcntl::F_SETLKW, lock
+ w.syswrite "."
+ sleep
+ end
+ w.close
+ assert_equal ".", r.read(1)
+ r.close
+ pad = 0
+ getlock = [F_WRLCK, 0, pad, pad, 0, 0, 0].pack(flocktype)
+ f.fcntl Fcntl::F_GETLK, getlock
+
+ ptype, whence, pad, pad, start, len, lockpid = getlock.unpack(flocktype)
+ assert_equal(pid, lockpid)
+ Process.kill :TERM, pid
+ Process.waitpid2(pid)
+ end
+ end
+ end
end