summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--ChangeLog5
-rw-r--r--ext/stringio/stringio.c33
-rw-r--r--test/ruby/test_io.rb16
-rw-r--r--test/stringio/test_stringio.rb15
-rw-r--r--version.h2
5 files changed, 68 insertions, 3 deletions
diff --git a/ChangeLog b/ChangeLog
index 7e74a8937a..9e22c7e792 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,8 @@
+Thu May 14 16:07:48 2009 Nobuyoshi Nakada <nobu@ruby-lang.org>
+
+ * ext/stringio/stringio.c (strio_ungetbyte): encoding should no
+ be effective.
+
Sun May 10 11:36:11 2009 Nobuyoshi Nakada <nobu@ruby-lang.org>
* ext/dl/cfunc.c (rb_dlcfunc_instance_p): new function to check if
diff --git a/ext/stringio/stringio.c b/ext/stringio/stringio.c
index ce1d55fd46..4e24a4723a 100644
--- a/ext/stringio/stringio.c
+++ b/ext/stringio/stringio.c
@@ -744,8 +744,37 @@ strio_ungetc(VALUE self, VALUE c)
static VALUE
strio_ungetbyte(VALUE self, VALUE c)
{
- NUM2INT(c);
- return strio_ungetc(self, c);
+ struct StringIO *ptr = readable(StringIO(self));
+ char buf[1], *cp = buf;
+ long pos = ptr->pos, cl = 1;
+ VALUE str = ptr->string;
+
+ if (NIL_P(c)) return Qnil;
+ if (FIXNUM_P(c)) {
+ buf[0] = (char)FIX2INT(c);
+ }
+ else {
+ SafeStringValue(c);
+ cp = RSTRING_PTR(c);
+ cl = RSTRING_LEN(c);
+ if (cl == 0) return Qnil;
+ }
+ rb_str_modify(str);
+ if (cl > pos) {
+ char *s;
+ long rest = RSTRING_LEN(str) - pos;
+ rb_str_resize(str, rest + cl);
+ s = RSTRING_PTR(str);
+ memmove(s + cl, s + pos, rest);
+ pos = 0;
+ }
+ else {
+ pos -= cl;
+ }
+ memcpy(RSTRING_PTR(str) + pos, cp, cl);
+ ptr->pos = pos;
+ RB_GC_GUARD(c);
+ return Qnil;
}
/*
diff --git a/test/ruby/test_io.rb b/test/ruby/test_io.rb
index 9082b76b61..776cf34147 100644
--- a/test/ruby/test_io.rb
+++ b/test/ruby/test_io.rb
@@ -77,6 +77,22 @@ class TestIO < Test::Unit::TestCase
r.close
end
+ def test_ungetbyte
+ t = make_tempfile
+ t.open
+ t.ungetbyte(0x41)
+ assert_equal(0x41, t.getbyte)
+ t.rewind
+ t.ungetbyte("qux")
+ assert_equal("quxfoo\n", t.gets)
+ t.set_encoding("utf-8")
+ t.ungetbyte(0x89)
+ t.ungetbyte(0x8e)
+ t.ungetbyte("\xe7")
+ t.ungetbyte("\xe7\xb4\x85")
+ assert_equal("\u7d05\u7389bar\n", t.gets)
+ end
+
def test_each_byte
r, w = IO.pipe
w << "abc def"
diff --git a/test/stringio/test_stringio.rb b/test/stringio/test_stringio.rb
index 98cf84e00f..4d1fdf28d1 100644
--- a/test/stringio/test_stringio.rb
+++ b/test/stringio/test_stringio.rb
@@ -293,6 +293,21 @@ class TestStringIO < Test::Unit::TestCase
f.close unless f.closed?
end
+ def test_ungetbyte
+ s = "foo\nbar\n"
+ t = StringIO.new(s, "r")
+ t.ungetbyte(0x41)
+ assert_equal(0x41, t.getbyte)
+ t.ungetbyte("qux")
+ assert_equal("quxfoo\n", t.gets)
+ t.set_encoding("utf-8")
+ t.ungetbyte(0x89)
+ t.ungetbyte(0x8e)
+ t.ungetbyte("\xe7")
+ t.ungetbyte("\xe7\xb4\x85")
+ assert_equal("\u7d05\u7389bar\n", t.gets)
+ end
+
def test_ungetc
s = "1234"
f = StringIO.new(s, "r")
diff --git a/version.h b/version.h
index af7491e7de..bd1f11b142 100644
--- a/version.h
+++ b/version.h
@@ -1,6 +1,6 @@
#define RUBY_VERSION "1.9.1"
#define RUBY_RELEASE_DATE "2009-05-12"
-#define RUBY_PATCHLEVEL 137
+#define RUBY_PATCHLEVEL 138
#define RUBY_VERSION_MAJOR 1
#define RUBY_VERSION_MINOR 9
#define RUBY_VERSION_TEENY 1