From eff05531058813e423a3ea77d97f5c4aa7a78dae Mon Sep 17 00:00:00 2001 From: nobu Date: Fri, 9 Sep 2016 08:59:48 +0000 Subject: thread.c: set cause by Thread#raise * thread.c (rb_threadptr_raise): set cause from the called thread, but not from the thread to be interrupted. [ruby-core:77222] [Bug #12741] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@56125 b2dd03c8-39d4-4d8f-98ff-823fe69b080e --- ChangeLog | 6 ++++++ eval.c | 11 +++++++++++ test/ruby/test_exception.rb | 46 +++++++++++++++++++++++++++++++++++++++++++++ thread.c | 3 +++ 4 files changed, 66 insertions(+) diff --git a/ChangeLog b/ChangeLog index 2ed78e9c76..29cca7c0b2 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,9 @@ +Fri Sep 9 17:59:46 2016 Nobuyoshi Nakada + + * thread.c (rb_threadptr_raise): set cause from the called thread, + but not from the thread to be interrupted. + [ruby-core:77222] [Bug #12741] + Fri Sep 9 13:50:05 2016 Nobuyoshi Nakada * doc/extension.rdoc, doc/extension.ja.rdoc: fix file name. diff --git a/eval.c b/eval.c index bad3c83714..1baf914a3a 100644 --- a/eval.c +++ b/eval.c @@ -567,6 +567,17 @@ setup_exception(rb_thread_t *th, int tag, volatile VALUE mesg, VALUE cause) } } +void +rb_threadptr_setup_exception(rb_thread_t *th, VALUE mesg, VALUE cause) +{ + if (cause == Qundef) { + cause = get_thread_errinfo(th); + } + if (cause != mesg) { + rb_ivar_set(mesg, id_cause, cause); + } +} + static void rb_longjmp(int tag, volatile VALUE mesg, VALUE cause) { diff --git a/test/ruby/test_exception.rb b/test/ruby/test_exception.rb index 5023262e5d..03c0c819f5 100644 --- a/test/ruby/test_exception.rb +++ b/test/ruby/test_exception.rb @@ -698,6 +698,52 @@ end.join assert_nil(e.cause.cause) end + def test_cause_thread_no_cause + bug12741 = '[ruby-core:77222] [Bug #12741]' + + x = Thread.current + a = false + y = Thread.start do + Thread.pass until a + x.raise "stop" + end + + begin + raise bug12741 + rescue + e = assert_raise_with_message(RuntimeError, "stop") do + a = true + sleep 1 + end + end + assert_nil(e.cause) + end + + def test_cause_thread_with_cause + bug12741 = '[ruby-core:77222] [Bug #12741]' + + x = Thread.current + a = false + y = Thread.start do + Thread.pass until a + begin + raise "caller's cause" + rescue + x.raise "stop" + end + end + + begin + raise bug12741 + rescue + e = assert_raise_with_message(RuntimeError, "stop") do + a = true + sleep 1 + end + end + assert_equal("caller's cause", e.cause.message) + end + def test_unknown_option bug = '[ruby-core:63203] [Feature #8257] should pass unknown options' diff --git a/thread.c b/thread.c index 6404bd27d7..46077f6ea4 100644 --- a/thread.c +++ b/thread.c @@ -2084,6 +2084,8 @@ rb_threadptr_ready(rb_thread_t *th) rb_threadptr_interrupt(th); } +void rb_threadptr_setup_exception(rb_thread_t *th, VALUE mesg, VALUE cause); + static VALUE rb_threadptr_raise(rb_thread_t *th, int argc, VALUE *argv) { @@ -2099,6 +2101,7 @@ rb_threadptr_raise(rb_thread_t *th, int argc, VALUE *argv) else { exc = rb_make_exception(argc, argv); } + rb_threadptr_setup_exception(GET_THREAD(), exc, Qundef); rb_threadptr_pending_interrupt_enque(th, exc); rb_threadptr_interrupt(th); return Qnil; -- cgit v1.2.3