summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--ChangeLog18
-rw-r--r--variable.c12
2 files changed, 24 insertions, 6 deletions
diff --git a/ChangeLog b/ChangeLog
index 7d27cbc23a..7aeea8e265 100644
--- a/ChangeLog
+++ b/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>
* Re-apply r33078, thread-safe autoload which is reverted at r33093.
diff --git a/variable.c b/variable.c
index 6b3f814a71..3ee4627a2b 100644
--- a/variable.c
+++ b/variable.c
@@ -1524,17 +1524,17 @@ rb_autoload(VALUE mod, ID id, const char *file)
st_add_direct(tbl, (st_data_t)autoload, av);
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);
FL_UNSET(fn, FL_TAINT);
OBJ_FREEZE(fn);
+
+ ele = ALLOC(struct autoload_data_i);
ele->feature = fn;
ele->safe_level = rb_safe_level();
ele->thread = Qnil;
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
@@ -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) {
return 0;
}
- return 1;
+ return !rb_autoloading_value(mod, id, NULL);
}
int
@@ -1674,7 +1674,6 @@ rb_autoload_load(VALUE mod, ID id)
ele->thread = rb_thread_current();
}
/* 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);
if (ele->thread == rb_thread_current()) {
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_GC_GUARD(load);
return result;
}