From def63c3466939161f2459f6489815b444bbde8a3 Mon Sep 17 00:00:00 2001 From: kosaki Date: Wed, 5 Dec 2012 19:37:37 +0000 Subject: [PATCH] * io.c (io_binwrite): check interrupt before io issue. * test/ruby/test_thread.rb (test_async_interrupt_and_io): test for the above. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@38224 b2dd03c8-39d4-4d8f-98ff-823fe69b080e --- ChangeLog | 6 ++++++ io.c | 3 +++ test/ruby/test_thread.rb | 20 ++++++++++++++++++++ 3 files changed, 29 insertions(+) diff --git a/ChangeLog b/ChangeLog index 82d9e59a5c..c5b43d44d6 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,9 @@ +Thu Dec 6 04:27:10 2012 KOSAKI Motohiro + + * io.c (io_binwrite): check interrupt before io issue. + * test/ruby/test_thread.rb (test_async_interrupt_and_io): + test for the above. + Thu Dec 6 01:10:36 2012 Nobuyoshi Nakada * vm_eval.c (rb_method_call_status): use Qundef as no self instead of diff --git a/io.c b/io.c index c678cc3b21..c8a5905630 100644 --- a/io.c +++ b/io.c @@ -1120,6 +1120,9 @@ io_binwrite(VALUE str, const char *ptr, long len, rb_io_t *fptr, int nosync) { long n, r, offset = 0; + /* don't write anything if current thread has a pending interrupt. */ + rb_thread_check_ints(); + if ((n = len) <= 0) return n; if (fptr->wbuf.ptr == NULL && !(!nosync && (fptr->mode & FMODE_SYNC))) { fptr->wbuf.off = 0; diff --git a/test/ruby/test_thread.rb b/test/ruby/test_thread.rb index 6bec7dfc34..b82cae4cb7 100644 --- a/test/ruby/test_thread.rb +++ b/test/ruby/test_thread.rb @@ -617,6 +617,26 @@ class TestThread < Test::Unit::TestCase assert_equal(:ok,r) end + def test_async_interrupt_and_io + assert_in_out_err([], <<-INPUT, %w(ok), []) + th_waiting = true + + t = Thread.new { + Thread.async_interrupt_timing(RuntimeError => :on_blocking) { + nil while th_waiting + # async interrupt should be raised _before_ writing puts arguments + puts "ng" + } + } + + sleep 0.1 + t.raise RuntimeError + th_waiting = false + t.join rescue nil + puts "ok" + INPUT + end + def test_async_interrupted? q = Queue.new Thread.async_interrupt_timing(RuntimeError => :defer){