Skip to content

Commit

Permalink
* vm_trace.c: clear and restore recursive checking thread local data
Browse files Browse the repository at this point in the history
  to avoid unexpected throw from TracePoint.
  [Bug ruby#9940]
* test/ruby/test_settracefunc.rb: add a test.
* thread.c: adde
  * rb_threadptr_reset_recursive_data(rb_thread_t *th);
  * rb_threadptr_restore_recursive_data(rb_thread_t *th, VALUE old);
* vm_core.h: ditto.



git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@46419 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
  • Loading branch information
ko1 committed Jun 13, 2014
1 parent 537b4a6 commit ddae6a0
Show file tree
Hide file tree
Showing 5 changed files with 73 additions and 16 deletions.
14 changes: 14 additions & 0 deletions ChangeLog
Original file line number Diff line number Diff line change
@@ -1,3 +1,17 @@
Fri Jun 13 17:46:31 2014 Koichi Sasada <ko1@atdot.net>

* vm_trace.c: clear and restore recursive checking thread local data
to avoid unexpected throw from TracePoint.
[Bug #9940]

* test/ruby/test_settracefunc.rb: add a test.

* thread.c: adde
* rb_threadptr_reset_recursive_data(rb_thread_t *th);
* rb_threadptr_restore_recursive_data(rb_thread_t *th, VALUE old);

* vm_core.h: ditto.

Fri Jun 13 17:33:14 2014 Nobuyoshi Nakada <nobu@ruby-lang.org>

* array.c (rb_ary_combination): iterate on a shared copy, and use
Expand Down
12 changes: 12 additions & 0 deletions test/ruby/test_settracefunc.rb
Original file line number Diff line number Diff line change
Expand Up @@ -1188,6 +1188,18 @@ def test_define_method_on_exception
assert_equal([['call', :foo], ['return', :foo]], events, 'Bug #9759')
ensure
end
end

def test_recursive
assert_ruby_status [], %q{
stack = []
TracePoint.new(:c_call){|tp|
p 2
stack << tp.method_id
}.enable{
p 1
}
raise if stack != [:p, :hash, :inspect]
}, '[Bug #9940]'
end
end
58 changes: 42 additions & 16 deletions thread.c
Original file line number Diff line number Diff line change
Expand Up @@ -2738,22 +2738,25 @@ rb_thread_inspect(VALUE thread)
return str;
}

VALUE
rb_thread_local_aref(VALUE thread, ID id)
static VALUE
threadptr_local_aref(rb_thread_t *th, ID id)
{
rb_thread_t *th;
st_data_t val;

GetThreadPtr(thread, th);
if (!th->local_storage) {
return Qnil;
}
if (st_lookup(th->local_storage, id, &val)) {
if (th->local_storage && st_lookup(th->local_storage, id, &val)) {
return (VALUE)val;
}
return Qnil;
}

VALUE
rb_thread_local_aref(VALUE thread, ID id)
{
rb_thread_t *th;
GetThreadPtr(thread, th);
return threadptr_local_aref(th, id);
}

/*
* call-seq:
* thr[sym] -> obj or nil
Expand Down Expand Up @@ -2822,26 +2825,35 @@ rb_thread_aref(VALUE thread, VALUE key)
return rb_thread_local_aref(thread, id);
}

VALUE
rb_thread_local_aset(VALUE thread, ID id, VALUE val)
static VALUE
threadptr_local_aset(rb_thread_t *th, ID id, VALUE val)
{
rb_thread_t *th;
GetThreadPtr(thread, th);

if (OBJ_FROZEN(thread)) {
rb_error_frozen("thread locals");
}
if (NIL_P(val)) {
if (!th->local_storage) return Qnil;
st_delete_wrap(th->local_storage, id);
return Qnil;
}
else {
if (!th->local_storage) {
th->local_storage = st_init_numtable();
}
st_insert(th->local_storage, id, val);
return val;
}
}

VALUE
rb_thread_local_aset(VALUE thread, ID id, VALUE val)
{
rb_thread_t *th;
GetThreadPtr(thread, th);

if (OBJ_FROZEN(thread)) {
rb_error_frozen("thread locals");
}

return threadptr_local_aset(th, id, val);
}

/*
* call-seq:
Expand Down Expand Up @@ -4674,6 +4686,20 @@ recursive_list_access(void)
return list;
}

VALUE
rb_threadptr_reset_recursive_data(rb_thread_t *th)
{
VALUE old = threadptr_local_aref(th, recursive_key);
threadptr_local_aset(th, recursive_key, Qnil);
return old;
}

void
rb_threadptr_restore_recursive_data(rb_thread_t *th, VALUE old)
{
threadptr_local_aset(th, recursive_key, old);
}

/*
* Returns Qtrue iff obj_id (or the pair <obj, paired_obj>) is already
* in the recursion list.
Expand Down
3 changes: 3 additions & 0 deletions vm_core.h
Original file line number Diff line number Diff line change
Expand Up @@ -1046,6 +1046,9 @@ void rb_threadptr_exec_event_hooks_and_pop_frame(struct rb_trace_arg_struct *tra
#define EXEC_EVENT_HOOK_AND_POP_FRAME(th_, flag_, self_, id_, klass_, data_) \
EXEC_EVENT_HOOK_ORIG(th_, flag_, self_, id_, klass_, data_, 1)

VALUE rb_threadptr_reset_recursive_data(rb_thread_t *th);
void rb_threadptr_restore_recursive_data(rb_thread_t *th, VALUE old);

RUBY_SYMBOL_EXPORT_BEGIN

int rb_thread_check_trap_pending(void);
Expand Down
2 changes: 2 additions & 0 deletions vm_trace.c
Original file line number Diff line number Diff line change
Expand Up @@ -331,6 +331,7 @@ rb_threadptr_exec_event_hooks_orig(rb_trace_arg_t *trace_arg, int pop_p)
trace_arg->self != rb_mRubyVMFrozenCore /* skip special methods. TODO: remove it. */) {
const VALUE errinfo = th->errinfo;
const int outer_state = th->state;
const VALUE old_recursive = rb_threadptr_reset_recursive_data(th);
int state = 0;
th->state = 0;
th->errinfo = Qnil;
Expand All @@ -351,6 +352,7 @@ rb_threadptr_exec_event_hooks_orig(rb_trace_arg_t *trace_arg, int pop_p)
terminate:
th->trace_arg = 0;
th->vm->trace_running--;
rb_threadptr_restore_recursive_data(th, old_recursive);

if (state) {
if (pop_p) {
Expand Down

0 comments on commit ddae6a0

Please sign in to comment.