* variable.c (rb_autoload): There was a chance to run GC (from
rb_str_new2()) before finishing autoload_data_i construction. It caused SEGV at rb_gc_mark() at autoload_i_mark. * variable.c (rb_autoload_load): Move RB_GC_GUARD() to proper position based on suggestion by CHIKANAGA Tomoyuki at http://d.hatena.ne.jp/nagachika/20110826/ruby_trunk_changes_33070_33078 * variable.c (autoload_defined_p): Fix incompatible autoload behavior that causes Rails crash. Class deifnition instruction defined in 'defineclass' in insns.def always invokes rb_autoload_load for a constant. It's invoked for every class definition regardless of existence of autoload definition. rb_autoload_load checkes if a constant is defined as autoloaded, but new thread-safe autoload returned different value if the constant is under autoloading. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@33147 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
cfe8ed5cbc
commit
0d475da805
18
ChangeLog
18
ChangeLog
@ -1,3 +1,21 @@
|
|||||||
|
Wed Aug 31 17:28:23 2011 Hiroshi Nakamura <nahi@ruby-lang.org>
|
||||||
|
|
||||||
|
* variable.c (rb_autoload): There was a chance to run GC (from
|
||||||
|
rb_str_new2()) before finishing autoload_data_i construction. It
|
||||||
|
caused SEGV at rb_gc_mark() at autoload_i_mark.
|
||||||
|
|
||||||
|
* variable.c (rb_autoload_load): Move RB_GC_GUARD() to proper
|
||||||
|
position based on suggestion by CHIKANAGA Tomoyuki at
|
||||||
|
http://d.hatena.ne.jp/nagachika/20110826/ruby_trunk_changes_33070_33078
|
||||||
|
|
||||||
|
* variable.c (autoload_defined_p): Fix incompatible autoload behavior
|
||||||
|
that causes Rails crash. Class deifnition instruction defined in
|
||||||
|
'defineclass' in insns.def always invokes rb_autoload_load for a
|
||||||
|
constant. It's invoked for every class definition regardless of
|
||||||
|
existence of autoload definition. rb_autoload_load checkes if a
|
||||||
|
constant is defined as autoloaded, but new thread-safe autoload
|
||||||
|
returned different value if the constant is under autoloading.
|
||||||
|
|
||||||
Wed Aug 31 17:20:56 2011 Hiroshi Nakamura <nahi@ruby-lang.org>
|
Wed Aug 31 17:20:56 2011 Hiroshi Nakamura <nahi@ruby-lang.org>
|
||||||
|
|
||||||
* Re-apply r33078, thread-safe autoload which is reverted at r33093.
|
* Re-apply r33078, thread-safe autoload which is reverted at r33093.
|
||||||
|
12
variable.c
12
variable.c
@ -1524,17 +1524,17 @@ rb_autoload(VALUE mod, ID id, const char *file)
|
|||||||
st_add_direct(tbl, (st_data_t)autoload, av);
|
st_add_direct(tbl, (st_data_t)autoload, av);
|
||||||
DATA_PTR(av) = tbl = st_init_numtable();
|
DATA_PTR(av) = tbl = st_init_numtable();
|
||||||
}
|
}
|
||||||
ad = TypedData_Wrap_Struct(0, &autoload_data_i_type, 0);
|
|
||||||
st_insert(tbl, (st_data_t)id, (st_data_t)ad);
|
|
||||||
DATA_PTR(ad) = ele = ALLOC(struct autoload_data_i);
|
|
||||||
|
|
||||||
fn = rb_str_new2(file);
|
fn = rb_str_new2(file);
|
||||||
FL_UNSET(fn, FL_TAINT);
|
FL_UNSET(fn, FL_TAINT);
|
||||||
OBJ_FREEZE(fn);
|
OBJ_FREEZE(fn);
|
||||||
|
|
||||||
|
ele = ALLOC(struct autoload_data_i);
|
||||||
ele->feature = fn;
|
ele->feature = fn;
|
||||||
ele->safe_level = rb_safe_level();
|
ele->safe_level = rb_safe_level();
|
||||||
ele->thread = Qnil;
|
ele->thread = Qnil;
|
||||||
ele->value = Qundef;
|
ele->value = Qundef;
|
||||||
|
ad = TypedData_Wrap_Struct(0, &autoload_data_i_type, ele);
|
||||||
|
st_insert(tbl, (st_data_t)id, (st_data_t)ad);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -1610,7 +1610,7 @@ autoload_defined_p(VALUE mod, ID id)
|
|||||||
if (!tbl || !st_lookup(tbl, (st_data_t)id, &val) || ((rb_const_entry_t*)val)->value != Qundef) {
|
if (!tbl || !st_lookup(tbl, (st_data_t)id, &val) || ((rb_const_entry_t*)val)->value != Qundef) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
return 1;
|
return !rb_autoloading_value(mod, id, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
@ -1674,7 +1674,6 @@ rb_autoload_load(VALUE mod, ID id)
|
|||||||
ele->thread = rb_thread_current();
|
ele->thread = rb_thread_current();
|
||||||
}
|
}
|
||||||
/* autoload_data_i can be deleted by another thread while require */
|
/* autoload_data_i can be deleted by another thread while require */
|
||||||
RB_GC_GUARD(load);
|
|
||||||
result = rb_protect((VALUE(*)(VALUE))autoload_require, (VALUE)ele, &state);
|
result = rb_protect((VALUE(*)(VALUE))autoload_require, (VALUE)ele, &state);
|
||||||
if (ele->thread == rb_thread_current()) {
|
if (ele->thread == rb_thread_current()) {
|
||||||
ele->thread = Qnil;
|
ele->thread = Qnil;
|
||||||
@ -1694,6 +1693,7 @@ rb_autoload_load(VALUE mod, ID id)
|
|||||||
rb_ensure((VALUE(*)(VALUE))autoload_const_set, (VALUE)&args, reset_safe, (VALUE)safe_backup);
|
rb_ensure((VALUE(*)(VALUE))autoload_const_set, (VALUE)&args, reset_safe, (VALUE)safe_backup);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
RB_GC_GUARD(load);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user