io/wait: add IO#wait_writable method

* ext/io/wait/wait.c (io_wait_writable): this is easier to use than
  IO.select for a single IO object and is immune to the
  limitations/innefficiency of select() on platforms where poll/ppoll
  is available.  patched by Eric Wong.  [Feature #4646]


git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@37785 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
nobu 2012-11-21 13:40:00 +00:00
Родитель 7d0cabb8f8
Коммит e786aa711d
3 изменённых файлов: 81 добавлений и 0 удалений

Просмотреть файл

@ -1,3 +1,10 @@
Wed Nov 21 22:39:28 2012 Nobuyoshi Nakada <nobu@ruby-lang.org>
* ext/io/wait/wait.c (io_wait_writable): this is easier to use than
IO.select for a single IO object and is immune to the
limitations/innefficiency of select() on platforms where poll/ppoll
is available. patched by Eric Wong. [Feature #4646]
Wed Nov 21 22:27:52 2012 Narihiro Nakamura <authornari@gmail.com>
* gc.c (garbage_collect): remove a duplicative probe.

Просмотреть файл

@ -132,6 +132,43 @@ io_wait(int argc, VALUE *argv, VALUE io)
return Qnil;
}
/*
* call-seq:
* io.wait_writable -> IO
* io.wait_writable(timeout) -> IO or nil
*
* Waits until IO writable is available or times out and returns self or
* nil when EOF is reached.
*/
static VALUE
io_wait_writable(int argc, VALUE *argv, VALUE io)
{
rb_io_t *fptr;
int i;
VALUE timeout;
struct timeval timerec;
struct timeval *tv;
GetOpenFile(io, fptr);
rb_io_check_writable(fptr);
rb_scan_args(argc, argv, "01", &timeout);
if (NIL_P(timeout)) {
tv = NULL;
}
else {
timerec = rb_time_interval(timeout);
tv = &timerec;
}
i = rb_wait_for_single_fd(fptr->fd, RB_WAITFD_OUT, tv);
if (i < 0)
rb_sys_fail(0);
rb_io_check_closed(fptr);
if (i & RB_WAITFD_OUT)
return io;
return Qnil;
}
/*
* IO wait methods
*/
@ -142,4 +179,5 @@ Init_wait()
rb_define_method(rb_cIO, "nread", io_nread, 0);
rb_define_method(rb_cIO, "ready?", io_ready_p, 0);
rb_define_method(rb_cIO, "wait", io_wait, -1);
rb_define_method(rb_cIO, "wait_writable", io_wait_writable, -1);
}

Просмотреть файл

@ -1,3 +1,4 @@
# -*- coding: us-ascii -*-
require 'test/unit'
require 'timeout'
require 'socket'
@ -69,4 +70,39 @@ class TestIOWait < Test::Unit::TestCase
Thread.new { sleep 0.01; @w.close }
assert_nil @r.wait
end
def test_wait_writable
assert_equal @w, @w.wait_writable
end
def test_wait_writable_timeout
assert_equal @w, @w.wait_writable(0.001)
written = fill_pipe
assert_nil @w.wait_writable(0.001)
@r.read(written)
assert_equal @w, @w.wait_writable(0.001)
end
def test_wait_writable_EPIPE
fill_pipe
@r.close
assert_equal @w, @w.wait_writable
end
def test_wait_writable_closed
@w.close
assert_raises(IOError) { @w.wait_writable }
end
private
def fill_pipe
written = 0
buf = " " * 4096
begin
written += @w.write_nonblock(buf)
rescue Errno::EAGAIN
return written
end while true
end
end if IO.method_defined?(:wait)