зеркало из https://github.com/github/ruby.git
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:
Родитель
7d0cabb8f8
Коммит
e786aa711d
|
@ -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>
|
Wed Nov 21 22:27:52 2012 Narihiro Nakamura <authornari@gmail.com>
|
||||||
|
|
||||||
* gc.c (garbage_collect): remove a duplicative probe.
|
* gc.c (garbage_collect): remove a duplicative probe.
|
||||||
|
|
|
@ -132,6 +132,43 @@ io_wait(int argc, VALUE *argv, VALUE io)
|
||||||
return Qnil;
|
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
|
* IO wait methods
|
||||||
*/
|
*/
|
||||||
|
@ -142,4 +179,5 @@ Init_wait()
|
||||||
rb_define_method(rb_cIO, "nread", io_nread, 0);
|
rb_define_method(rb_cIO, "nread", io_nread, 0);
|
||||||
rb_define_method(rb_cIO, "ready?", io_ready_p, 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", 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 'test/unit'
|
||||||
require 'timeout'
|
require 'timeout'
|
||||||
require 'socket'
|
require 'socket'
|
||||||
|
@ -69,4 +70,39 @@ class TestIOWait < Test::Unit::TestCase
|
||||||
Thread.new { sleep 0.01; @w.close }
|
Thread.new { sleep 0.01; @w.close }
|
||||||
assert_nil @r.wait
|
assert_nil @r.wait
|
||||||
end
|
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)
|
end if IO.method_defined?(:wait)
|
||||||
|
|
Загрузка…
Ссылка в новой задаче