ruby-changes:22531
From: nobu <ko1@a...>
Date: Mon, 13 Feb 2012 18:30:52 +0900 (JST)
Subject: [ruby-changes:22531] nobu:r34580 (trunk): * io.c (io_setstrbuf): defer resizing buffer string until data is
nobu 2012-02-13 18:30:40 +0900 (Mon, 13 Feb 2012) New Revision: 34580 http://svn.ruby-lang.org/cgi-bin/viewvc.cgi?view=rev&revision=34580 Log: * io.c (io_setstrbuf): defer resizing buffer string until data is read actually. Modified files: trunk/ChangeLog trunk/io.c trunk/test/ruby/test_io.rb Index: ChangeLog =================================================================== --- ChangeLog (revision 34579) +++ ChangeLog (revision 34580) @@ -1,3 +1,8 @@ +Mon Feb 13 18:30:32 2012 Nobuyoshi Nakada <nobu@r...> + + * io.c (io_setstrbuf): defer resizing buffer string until data is + read actually. + Mon Feb 13 10:24:39 2012 Loren Segal <lsegal@s...> * io.c (Init_IO): use directive hack to make ARGF documentable Index: io.c =================================================================== --- io.c (revision 34579) +++ io.c (revision 34580) @@ -1871,13 +1871,12 @@ } 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; rb_str_locktmp(str); - len = io_bufread(RSTRING_PTR(str) + offset, RSTRING_LEN(str) - offset, - fptr); + len = io_bufread(RSTRING_PTR(str) + offset, size, fptr); rb_str_unlocktmp(str); if (len < 0) rb_sys_fail_path(fptr->pathv); return len; @@ -2076,26 +2075,16 @@ io_setstrbuf(VALUE *str,long len) { #ifdef _WIN32 + len = (len + 1) & ~1L; /* round up for wide char */ +#endif if (NIL_P(*str)) { - *str = rb_str_new(0, len+1); - rb_str_set_len(*str,len); + *str = rb_str_new(0, 0); } else { StringValue(*str); - rb_str_modify(*str); - rb_str_resize(*str, len+1); - rb_str_set_len(*str,len); + len -= RSTRING_LEN(*str); } -#else - if (NIL_P(*str)) { - *str = rb_str_new(0, len); - } - else { - StringValue(*str); - rb_str_modify(*str); - rb_str_resize(*str, len); - } -#endif + rb_str_modify_expand(*str, len); } static VALUE @@ -2141,18 +2130,19 @@ io_setstrbuf(&str,siz); for (;;) { READ_CHECK(fptr); - n = io_fread(str, bytes, fptr); + n = io_fread(str, bytes, siz - bytes, fptr); if (n == 0 && bytes == 0) { + rb_str_set_len(str, 0); break; } bytes += n; + rb_str_set_len(str, bytes); if (cr != ENC_CODERANGE_BROKEN) pos += rb_str_coderange_scan_restartable(RSTRING_PTR(str) + pos, RSTRING_PTR(str) + bytes, enc, &cr); if (bytes < siz) break; 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); ENC_CODERANGE_SET(str, cr); return str; @@ -2219,7 +2209,7 @@ rb_sys_fail_path(fptr->pathv); } } - rb_str_resize(str, n); + rb_str_set_len(str, n); if (n == 0) return Qnil; @@ -2535,18 +2525,14 @@ #if defined(RUBY_TEST_CRLF_ENVIRONMENT) || defined(_WIN32) previous_mode = set_binary_mode_with_seek_cur(fptr); #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 (previous_mode == O_TEXT) { setmode(fptr->fd, O_TEXT); } #endif - if (n == 0) { - if (fptr->fd < 0) return Qnil; - rb_str_resize(str, 0); - return Qnil; - } - rb_str_resize(str, n); + if (n == 0) return Qnil; OBJ_TAINT(str); return str; @@ -4284,7 +4270,6 @@ if (n == 0 && ilen > 0) { rb_eof_error(); } - rb_str_resize(str, n); OBJ_TAINT(str); return str; Index: test/ruby/test_io.rb =================================================================== --- test/ruby/test_io.rb (revision 34579) +++ test/ruby/test_io.rb (revision 34580) @@ -972,7 +972,7 @@ with_pipe do |r, w| s = "" t = Thread.new { r.readpartial(5, s) } - Thread.pass until s.size == 5 + Thread.pass until t.stop? assert_raise(RuntimeError) { s.clear } w.write "foobarbaz" w.close @@ -991,6 +991,17 @@ } 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 pipe(proc do |w| w.write "foobarbaz" @@ -1007,7 +1018,7 @@ with_pipe do |r, w| s = "" t = Thread.new { r.read(5, s) } - Thread.pass until s.size == 5 + Thread.pass until t.stop? assert_raise(RuntimeError) { s.clear } w.write "foobarbaz" w.close @@ -1015,6 +1026,17 @@ 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 skip "IO#write_nonblock is not supported on file/pipe." if /mswin|bccwin|mingw/ =~ RUBY_PLATFORM pipe(proc do |w| @@ -2117,8 +2139,8 @@ end } IO.pipe {|r,w| - assert(r.close_on_exec?) - assert(w.close_on_exec?) + assert(r.close_on_exec?) + assert(w.close_on_exec?) } end -- ML: ruby-changes@q... Info: http://www.atdot.net/~ko1/quickml/