Bug #5229 ยป 0001-thread.c-rb_thread_select-implement-using-rb_thread_.patch
ext/-test-/old_thread_select/depend | ||
---|---|---|
old_thread_select.o: $(top_srcdir)/thread.c \
|
||
$(hdrdir)/ruby/ruby.h $(hdrdir)/ruby/io.h
|
ext/-test-/old_thread_select/extconf.rb | ||
---|---|---|
create_makefile("-test-/old_thread_select/old_thread_select")
|
ext/-test-/old_thread_select/old_thread_select.c | ||
---|---|---|
/* test case for deprecated C API */
|
||
#include "ruby/ruby.h"
|
||
#include "ruby/io.h"
|
||
static fd_set * array2fdset(fd_set *fds, VALUE ary, int *max)
|
||
{
|
||
long i;
|
||
if (NIL_P(ary))
|
||
return NULL;
|
||
FD_ZERO(fds);
|
||
Check_Type(ary, T_ARRAY);
|
||
for (i = 0; i < RARRAY_LEN(ary); i++) {
|
||
VALUE val = RARRAY_PTR(ary)[i];
|
||
int fd;
|
||
Check_Type(val, T_FIXNUM);
|
||
fd = FIX2INT(val);
|
||
if (fd >= *max)
|
||
*max = fd + 1;
|
||
FD_SET(fd, fds);
|
||
}
|
||
return fds;
|
||
}
|
||
static VALUE
|
||
old_thread_select(VALUE klass, VALUE r, VALUE w, VALUE e, VALUE timeout)
|
||
{
|
||
struct timeval tv;
|
||
struct timeval *tvp = NULL;
|
||
fd_set rfds, wfds, efds;
|
||
fd_set *rp, *wp, *ep;
|
||
int rc;
|
||
int max = 0;
|
||
if (!NIL_P(timeout)) {
|
||
tv = rb_time_timeval(timeout);
|
||
tvp = &tv;
|
||
}
|
||
rp = array2fdset(&rfds, r, &max);
|
||
wp = array2fdset(&wfds, w, &max);
|
||
ep = array2fdset(&efds, w, &max);
|
||
rc = rb_thread_select(max, rp, wp, ep, tvp);
|
||
if (rc == -1)
|
||
rb_sys_fail("rb_wait_for_single_fd");
|
||
return INT2NUM(rc);
|
||
}
|
||
void
|
||
Init_old_thread_select(void)
|
||
{
|
||
rb_define_singleton_method(rb_cIO, "old_thread_select",
|
||
old_thread_select, 4);
|
||
}
|
test/-ext-/old_thread_select/test_old_thread_select.rb | ||
---|---|---|
require 'test/unit'
|
||
class TestOldThreadSelect < Test::Unit::TestCase
|
||
require '-test-/old_thread_select/old_thread_select'
|
||
def with_pipe
|
||
r, w = IO.pipe
|
||
begin
|
||
yield r, w
|
||
ensure
|
||
r.close unless r.closed?
|
||
w.close unless w.closed?
|
||
end
|
||
end
|
||
def test_old_select_read_timeout
|
||
with_pipe do |r, w|
|
||
t0 = Time.now
|
||
rc = IO.old_thread_select([r.fileno], nil, nil, 0.001)
|
||
diff = Time.now - t0
|
||
assert_equal 0, rc
|
||
assert diff > 0.001, "returned too early"
|
||
end
|
||
end
|
||
def test_old_select_read_write_check
|
||
with_pipe do |r, w|
|
||
w.syswrite('.')
|
||
rc = IO.old_thread_select([r.fileno], nil, nil, nil)
|
||
assert_equal 1, rc
|
||
rc = IO.old_thread_select([r.fileno], [w.fileno], nil, nil)
|
||
assert_equal 2, rc
|
||
assert_equal '.', r.read(1)
|
||
rc = IO.old_thread_select([r.fileno], [w.fileno], nil, nil)
|
||
assert_equal 1, rc
|
||
end
|
||
end
|
||
def test_old_select_signal_safe
|
||
return unless Process.respond_to?(:kill)
|
||
usr1 = false
|
||
trap(:USR1) { usr1 = true }
|
||
main = Thread.current
|
||
thr = Thread.new do
|
||
Thread.pass until main.stop?
|
||
Process.kill(:USR1, $$)
|
||
true
|
||
end
|
||
rc = nil
|
||
t0 = Time.now
|
||
with_pipe do |r,w|
|
||
assert_nothing_raised do
|
||
rc = IO.old_thread_select([r.fileno], nil, nil, 1)
|
||
end
|
||
end
|
||
diff = Time.now - t0
|
||
assert diff >= 1.0, "interrupted or short wait"
|
||
assert_equal 0, rc
|
||
assert_equal true, thr.value
|
||
assert usr1, "USR1 not received"
|
||
ensure
|
||
trap(:USR1, "DEFAULT")
|
||
end
|
||
end
|
thread.c | ||
---|---|---|
rb_thread_select(int max, fd_set * read, fd_set * write, fd_set * except,
|
||
struct timeval *timeout)
|
||
{
|
||
if (!read && !write && !except) {
|
||
if (!timeout) {
|
||
rb_thread_sleep_forever();
|
||
return 0;
|
||
}
|
||
rb_thread_wait_for(*timeout);
|
||
return 0;
|
||
rb_fdset_t fdsets[3] = { 0 };
|
||
rb_fdset_t *rfds = NULL;
|
||
rb_fdset_t *wfds = NULL;
|
||
rb_fdset_t *efds = NULL;
|
||
int retval;
|
||
if (read) {
|
||
rfds = &fdsets[0];
|
||
rb_fd_copy(rfds, read, max);
|
||
}
|
||
if (write) {
|
||
wfds = &fdsets[1];
|
||
rb_fd_copy(wfds, write, max);
|
||
}
|
||
if (except) {
|
||
efds = &fdsets[2];
|
||
rb_fd_copy(efds, except, max);
|
||
}
|
||
else {
|
||
int lerrno = errno;
|
||
int result;
|
||
BLOCKING_REGION({
|
||
result = select(max, read, write, except, timeout);
|
||
if (result < 0)
|
||
lerrno = errno;
|
||
}, ubf_select, GET_THREAD());
|
||
errno = lerrno;
|
||
retval = rb_thread_fd_select(max, rfds, efds, wfds, timeout);
|
||
return result;
|
||
}
|
||
}
|
||
if (rfds)
|
||
rb_fd_term(rfds);
|
||
if (wfds)
|
||
rb_fd_term(wfds);
|
||
if (efds)
|
||
rb_fd_term(efds);
|
||
return retval;
|
||
}
|
||
int
|
||
rb_thread_fd_select(int max, rb_fdset_t * read, rb_fdset_t * write, rb_fdset_t * except,
|