diff options
author | nobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2009-07-14 07:13:11 +0000 |
---|---|---|
committer | nobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2009-07-14 07:13:11 +0000 |
commit | 9681409e3691037c11c19052295616454f74c936 (patch) | |
tree | ebb308e156ad0b3b74c4f65a9ff54d7c0b1da15c | |
parent | ed586352292f331f2166205c0d7b99b69eab4ba4 (diff) | |
download | ruby-9681409e3691037c11c19052295616454f74c936.tar.gz |
* io.c (rb_io_initialize): check if the descriptor can be accessed
in the specified open mode. [ruby-dev:38571]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@24102 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r-- | ChangeLog | 5 | ||||
-rw-r--r-- | io.c | 22 | ||||
-rw-r--r-- | test/ruby/test_io.rb | 3 |
3 files changed, 25 insertions, 5 deletions
@@ -1,3 +1,8 @@ +Tue Jul 14 16:13:04 2009 Nobuyoshi Nakada <nobu@ruby-lang.org> + + * io.c (rb_io_initialize): check if the descriptor can be accessed + in the specified open mode. [ruby-dev:38571] + Tue Jul 14 09:26:14 2009 Hidetoshi NAGAI <nagai@ai.kyutech.ac.jp> * ext/tk/lib/multi-tk.rb: Long-term-callback support isn't stable yet. @@ -6342,7 +6342,11 @@ rb_io_initialize(int argc, VALUE *argv, VALUE io) int fd, fmode, oflags = O_RDONLY; convconfig_t convconfig; VALUE opt; +#if defined(HAVE_FCNTL) && defined(F_GETFL) + int ofmode; +#else struct stat st; +#endif rb_secure(4); @@ -6351,15 +6355,23 @@ rb_io_initialize(int argc, VALUE *argv, VALUE io) rb_io_extract_modeenc(&vmode, 0, opt, &oflags, &fmode, &convconfig); fd = NUM2INT(fnum); +#if defined(HAVE_FCNTL) && defined(F_GETFL) + oflags = fcntl(fd, F_GETFL); + if (oflags == -1) rb_sys_fail(0); +#else if (fstat(fd, &st) == -1) rb_sys_fail(0); +#endif UPDATE_MAXFD(fd); - if (NIL_P(vmode)) { #if defined(HAVE_FCNTL) && defined(F_GETFL) - oflags = fcntl(fd, F_GETFL); - if (oflags == -1) rb_sys_fail(0); - fmode = rb_io_oflags_fmode(oflags); -#endif + ofmode = rb_io_oflags_fmode(oflags); + if (NIL_P(vmode)) { + fmode = ofmode; + } + else if ((~ofmode & fmode) & FMODE_READWRITE) { + VALUE error = INT2FIX(EINVAL); + rb_exc_raise(rb_class_new_instance(1, &error, rb_eSystemCallError)); } +#endif MakeOpenFile(io, fp); fp->fd = fd; fp->mode = fmode; diff --git a/test/ruby/test_io.rb b/test/ruby/test_io.rb index 69e3666cff..8a5b878f3b 100644 --- a/test/ruby/test_io.rb +++ b/test/ruby/test_io.rb @@ -1379,6 +1379,9 @@ class TestIO < Test::Unit::TestCase fd = IO.sysopen(t.path, "w") assert_kind_of(Integer, fd) + %w[r r+ w+ a+].each do |mode| + assert_raise(Errno::EINVAL, '[ruby-dev:38571]') {IO.new(fd, mode)} + end f = IO.new(fd, "w") f.write("FOO\n") f.close |