diff options
-rw-r--r-- | ChangeLog | 8 | ||||
-rw-r--r-- | test/ruby/test_thread.rb | 12 | ||||
-rw-r--r-- | thread.c | 15 |
3 files changed, 35 insertions, 0 deletions
@@ -1,3 +1,11 @@ +Sat Dec 1 02:11:47 2012 KOSAKI Motohiro <kosaki.motohiro@gmail.com> + + * thread.c (rb_threadptr_interrupt_mask): add argument check. + * thread.c (async_interrupt_timing_arg_check_i): helper function + for the above. + * test/ruby/test_thread.rb (test_async_interrupt_timing_invalid_argument): + test for the above. + Sat Dec 1 01:19:34 2012 KOSAKI Motohiro <kosaki.motohiro@gmail.com> * lib/thread.rb (ConditionVariable#broadcast): protect from diff --git a/test/ruby/test_thread.rb b/test/ruby/test_thread.rb index 53111d9458..beee4c4be5 100644 --- a/test/ruby/test_thread.rb +++ b/test/ruby/test_thread.rb @@ -714,6 +714,18 @@ class TestThread < Test::Unit::TestCase # TODO: complex cases are needed. end + def test_async_interrupt_timing_invalid_argument + assert_raise(ArgumentError) { + Thread.async_interrupt_timing(RuntimeError => :immediate) # no block + } + assert_raise(ArgumentError) { + Thread.async_interrupt_timing(RuntimeError => :never) {} # never? + } + assert_raise(TypeError) { + Thread.async_interrupt_timing([]) {} # array + } + end + def test_async_interrupted? q = Queue.new Thread.async_interrupt_timing(RuntimeError => :defer){ @@ -1545,12 +1545,27 @@ rb_threadptr_async_errinfo_active_p(rb_thread_t *th) return 1; } +static int +async_interrupt_timing_arg_check_i(VALUE key, VALUE val) +{ + VALUE immediate = ID2SYM(rb_intern("immediate")); + VALUE on_blocking = ID2SYM(rb_intern("on_blocking")); + VALUE defer = ID2SYM(rb_intern("defer")); + + if (val != immediate && val != on_blocking && val != defer) { + rb_raise(rb_eArgError, "unknown mask signature"); + } + + return ST_CONTINUE; +} + static VALUE rb_threadptr_interrupt_mask(rb_thread_t *th, VALUE mask, VALUE (*func)(rb_thread_t *th)) { VALUE r = Qnil; int state; + rb_hash_foreach(mask, async_interrupt_timing_arg_check_i, 0); rb_ary_push(th->async_errinfo_mask_stack, mask); if (!rb_threadptr_async_errinfo_empty_p(th)) { th->async_errinfo_queue_checked = 0; |