mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
* io.c (io_setstrbuf): defer resizing buffer string until data is
read actually. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@34580 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
23ebb5ea6c
commit
df77202b37
3 changed files with 50 additions and 38 deletions
|
@ -1,3 +1,8 @@
|
||||||
|
Mon Feb 13 18:30:32 2012 Nobuyoshi Nakada <nobu@ruby-lang.org>
|
||||||
|
|
||||||
|
* io.c (io_setstrbuf): defer resizing buffer string until data is
|
||||||
|
read actually.
|
||||||
|
|
||||||
Mon Feb 13 10:24:39 2012 Loren Segal <lsegal@soen.ca>
|
Mon Feb 13 10:24:39 2012 Loren Segal <lsegal@soen.ca>
|
||||||
|
|
||||||
* io.c (Init_IO): use directive hack to make ARGF documentable
|
* io.c (Init_IO): use directive hack to make ARGF documentable
|
||||||
|
|
53
io.c
53
io.c
|
@ -1871,13 +1871,12 @@ io_bufread(char *ptr, long len, rb_io_t *fptr)
|
||||||
}
|
}
|
||||||
|
|
||||||
static long
|
static long
|
||||||
io_fread(VALUE str, long offset, rb_io_t *fptr)
|
io_fread(VALUE str, long offset, long size, rb_io_t *fptr)
|
||||||
{
|
{
|
||||||
long len;
|
long len;
|
||||||
|
|
||||||
rb_str_locktmp(str);
|
rb_str_locktmp(str);
|
||||||
len = io_bufread(RSTRING_PTR(str) + offset, RSTRING_LEN(str) - offset,
|
len = io_bufread(RSTRING_PTR(str) + offset, size, fptr);
|
||||||
fptr);
|
|
||||||
rb_str_unlocktmp(str);
|
rb_str_unlocktmp(str);
|
||||||
if (len < 0) rb_sys_fail_path(fptr->pathv);
|
if (len < 0) rb_sys_fail_path(fptr->pathv);
|
||||||
return len;
|
return len;
|
||||||
|
@ -2076,26 +2075,16 @@ static void
|
||||||
io_setstrbuf(VALUE *str,long len)
|
io_setstrbuf(VALUE *str,long len)
|
||||||
{
|
{
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
if (NIL_P(*str)) {
|
len = (len + 1) & ~1L; /* round up for wide char */
|
||||||
*str = rb_str_new(0, len+1);
|
|
||||||
rb_str_set_len(*str,len);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
StringValue(*str);
|
|
||||||
rb_str_modify(*str);
|
|
||||||
rb_str_resize(*str, len+1);
|
|
||||||
rb_str_set_len(*str,len);
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
if (NIL_P(*str)) {
|
|
||||||
*str = rb_str_new(0, len);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
StringValue(*str);
|
|
||||||
rb_str_modify(*str);
|
|
||||||
rb_str_resize(*str, len);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
if (NIL_P(*str)) {
|
||||||
|
*str = rb_str_new(0, 0);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
StringValue(*str);
|
||||||
|
len -= RSTRING_LEN(*str);
|
||||||
|
}
|
||||||
|
rb_str_modify_expand(*str, len);
|
||||||
}
|
}
|
||||||
|
|
||||||
static VALUE
|
static VALUE
|
||||||
|
@ -2141,18 +2130,19 @@ read_all(rb_io_t *fptr, long siz, VALUE str)
|
||||||
io_setstrbuf(&str,siz);
|
io_setstrbuf(&str,siz);
|
||||||
for (;;) {
|
for (;;) {
|
||||||
READ_CHECK(fptr);
|
READ_CHECK(fptr);
|
||||||
n = io_fread(str, bytes, fptr);
|
n = io_fread(str, bytes, siz - bytes, fptr);
|
||||||
if (n == 0 && bytes == 0) {
|
if (n == 0 && bytes == 0) {
|
||||||
|
rb_str_set_len(str, 0);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
bytes += n;
|
bytes += n;
|
||||||
|
rb_str_set_len(str, bytes);
|
||||||
if (cr != ENC_CODERANGE_BROKEN)
|
if (cr != ENC_CODERANGE_BROKEN)
|
||||||
pos += rb_str_coderange_scan_restartable(RSTRING_PTR(str) + pos, RSTRING_PTR(str) + bytes, enc, &cr);
|
pos += rb_str_coderange_scan_restartable(RSTRING_PTR(str) + pos, RSTRING_PTR(str) + bytes, enc, &cr);
|
||||||
if (bytes < siz) break;
|
if (bytes < siz) break;
|
||||||
siz += BUFSIZ;
|
siz += BUFSIZ;
|
||||||
rb_str_resize(str, siz);
|
rb_str_modify_expand(str, BUFSIZ);
|
||||||
}
|
}
|
||||||
if (bytes != siz) rb_str_resize(str, bytes);
|
|
||||||
str = io_enc_str(str, fptr);
|
str = io_enc_str(str, fptr);
|
||||||
ENC_CODERANGE_SET(str, cr);
|
ENC_CODERANGE_SET(str, cr);
|
||||||
return str;
|
return str;
|
||||||
|
@ -2219,7 +2209,7 @@ io_getpartial(int argc, VALUE *argv, VALUE io, int nonblock)
|
||||||
rb_sys_fail_path(fptr->pathv);
|
rb_sys_fail_path(fptr->pathv);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
rb_str_resize(str, n);
|
rb_str_set_len(str, n);
|
||||||
|
|
||||||
if (n == 0)
|
if (n == 0)
|
||||||
return Qnil;
|
return Qnil;
|
||||||
|
@ -2535,18 +2525,14 @@ io_read(int argc, VALUE *argv, VALUE io)
|
||||||
#if defined(RUBY_TEST_CRLF_ENVIRONMENT) || defined(_WIN32)
|
#if defined(RUBY_TEST_CRLF_ENVIRONMENT) || defined(_WIN32)
|
||||||
previous_mode = set_binary_mode_with_seek_cur(fptr);
|
previous_mode = set_binary_mode_with_seek_cur(fptr);
|
||||||
#endif
|
#endif
|
||||||
n = io_fread(str, 0, fptr);
|
n = io_fread(str, 0, len, fptr);
|
||||||
|
rb_str_set_len(str, n);
|
||||||
#if defined(RUBY_TEST_CRLF_ENVIRONMENT) || defined(_WIN32)
|
#if defined(RUBY_TEST_CRLF_ENVIRONMENT) || defined(_WIN32)
|
||||||
if (previous_mode == O_TEXT) {
|
if (previous_mode == O_TEXT) {
|
||||||
setmode(fptr->fd, O_TEXT);
|
setmode(fptr->fd, O_TEXT);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
if (n == 0) {
|
if (n == 0) return Qnil;
|
||||||
if (fptr->fd < 0) return Qnil;
|
|
||||||
rb_str_resize(str, 0);
|
|
||||||
return Qnil;
|
|
||||||
}
|
|
||||||
rb_str_resize(str, n);
|
|
||||||
OBJ_TAINT(str);
|
OBJ_TAINT(str);
|
||||||
|
|
||||||
return str;
|
return str;
|
||||||
|
@ -4284,7 +4270,6 @@ rb_io_sysread(int argc, VALUE *argv, VALUE io)
|
||||||
if (n == 0 && ilen > 0) {
|
if (n == 0 && ilen > 0) {
|
||||||
rb_eof_error();
|
rb_eof_error();
|
||||||
}
|
}
|
||||||
rb_str_resize(str, n);
|
|
||||||
OBJ_TAINT(str);
|
OBJ_TAINT(str);
|
||||||
|
|
||||||
return str;
|
return str;
|
||||||
|
|
|
@ -972,7 +972,7 @@ class TestIO < Test::Unit::TestCase
|
||||||
with_pipe do |r, w|
|
with_pipe do |r, w|
|
||||||
s = ""
|
s = ""
|
||||||
t = Thread.new { r.readpartial(5, s) }
|
t = Thread.new { r.readpartial(5, s) }
|
||||||
Thread.pass until s.size == 5
|
Thread.pass until t.stop?
|
||||||
assert_raise(RuntimeError) { s.clear }
|
assert_raise(RuntimeError) { s.clear }
|
||||||
w.write "foobarbaz"
|
w.write "foobarbaz"
|
||||||
w.close
|
w.close
|
||||||
|
@ -991,6 +991,17 @@ class TestIO < Test::Unit::TestCase
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_readpartial_buffer_error
|
||||||
|
with_pipe do |r, w|
|
||||||
|
s = ""
|
||||||
|
t = Thread.new { r.readpartial(5, s) }
|
||||||
|
Thread.pass until t.stop?
|
||||||
|
t.kill
|
||||||
|
t.value
|
||||||
|
assert_equal("", s)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def test_read
|
def test_read
|
||||||
pipe(proc do |w|
|
pipe(proc do |w|
|
||||||
w.write "foobarbaz"
|
w.write "foobarbaz"
|
||||||
|
@ -1007,7 +1018,7 @@ class TestIO < Test::Unit::TestCase
|
||||||
with_pipe do |r, w|
|
with_pipe do |r, w|
|
||||||
s = ""
|
s = ""
|
||||||
t = Thread.new { r.read(5, s) }
|
t = Thread.new { r.read(5, s) }
|
||||||
Thread.pass until s.size == 5
|
Thread.pass until t.stop?
|
||||||
assert_raise(RuntimeError) { s.clear }
|
assert_raise(RuntimeError) { s.clear }
|
||||||
w.write "foobarbaz"
|
w.write "foobarbaz"
|
||||||
w.close
|
w.close
|
||||||
|
@ -1015,6 +1026,17 @@ class TestIO < Test::Unit::TestCase
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_read_buffer_error
|
||||||
|
with_pipe do |r, w|
|
||||||
|
s = ""
|
||||||
|
t = Thread.new { r.read(5, s) }
|
||||||
|
Thread.pass until t.stop?
|
||||||
|
t.kill
|
||||||
|
t.value
|
||||||
|
assert_equal("", s)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def test_write_nonblock
|
def test_write_nonblock
|
||||||
skip "IO#write_nonblock is not supported on file/pipe." if /mswin|bccwin|mingw/ =~ RUBY_PLATFORM
|
skip "IO#write_nonblock is not supported on file/pipe." if /mswin|bccwin|mingw/ =~ RUBY_PLATFORM
|
||||||
pipe(proc do |w|
|
pipe(proc do |w|
|
||||||
|
@ -2117,8 +2139,8 @@ End
|
||||||
end
|
end
|
||||||
}
|
}
|
||||||
IO.pipe {|r,w|
|
IO.pipe {|r,w|
|
||||||
assert(r.close_on_exec?)
|
assert(r.close_on_exec?)
|
||||||
assert(w.close_on_exec?)
|
assert(w.close_on_exec?)
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue