summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authornobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2009-01-22 02:56:14 +0000
committernobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2009-01-22 02:56:14 +0000
commit547f4725fa115ce7754e8fd52f73a9c71cff6710 (patch)
tree512c45d0fa79a50ecdd5af1d0925b0cf53895eec
parentaf6f5a911edc63d8f7363768b9a4763790eeda41 (diff)
* gc.c (rb_gc_call_finalizer_at_exit): deffers IO finalization.
[ruby-dev:36646] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/branches/ruby_1_8@21726 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r--ChangeLog5
-rw-r--r--gc.c50
2 files changed, 42 insertions, 13 deletions
diff --git a/ChangeLog b/ChangeLog
index 63260bde9f..88c1306dc4 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,8 @@
+Thu Jan 22 11:57:28 2009 Nobuyoshi Nakada <nobu@ruby-lang.org>
+
+ * gc.c (rb_gc_call_finalizer_at_exit): deffers IO finalization.
+ [ruby-dev:36646]
+
Thu Jan 22 05:17:06 2009 Masatoshi SEKI <m_seki@mva.biglobe.ne.jp>
* lib/erb.rb (def_erb_method): pass the trim_mode [Feature #1032]
diff --git a/gc.c b/gc.c
index c1afba7a32..175c84386c 100644
--- a/gc.c
+++ b/gc.c
@@ -1254,6 +1254,15 @@ make_deferred(p)
p->as.basic.flags = (p->as.basic.flags & ~T_MASK) | T_ZOMBIE;
}
+static inline void
+make_io_deferred(RVALUE *p)
+{
+ struct rb_io_t *fptr = p->as.file.fptr;
+ make_deferred(p);
+ p->as.data.dfree = (void (*)(void*))rb_io_fptr_finalize;
+ p->as.data.data = fptr;
+}
+
static int
obj_free(obj)
VALUE obj;
@@ -1327,10 +1336,7 @@ obj_free(obj)
break;
case T_FILE:
if (RANY(obj)->as.file.fptr) {
- struct rb_io_t *fptr = RANY(obj)->as.file.fptr;
- make_deferred(RANY(obj));
- RDATA(obj)->dfree = (void (*)(void*))rb_io_fptr_finalize;
- RDATA(obj)->data = fptr;
+ make_io_deferred(RANY(obj));
return 1;
}
break;
@@ -1966,29 +1972,36 @@ run_final(obj)
rb_thread_critical = critical_save;
}
-void
-rb_gc_finalize_deferred()
+static int
+finalize_deferred()
{
RVALUE *p = deferred_final_list;
deferred_final_list = 0;
if (p) {
finalize_list(p);
- free_unused_heaps();
+ return 1;
}
+ return 0;
+}
+
+void
+rb_gc_finalize_deferred()
+{
+ if (finalize_deferred())
+ free_unused_heaps();
}
void
rb_gc_call_finalizer_at_exit()
{
RVALUE *p, *pend;
+ RVALUE *final_list = 0;
int i;
/* run finalizers */
if (need_call_final) {
- p = deferred_final_list;
- deferred_final_list = 0;
- finalize_list(p);
+ finalize_deferred();
for (i = 0; i < heaps_used; i++) {
p = heaps[i].slot; pend = p + heaps[i].limit;
while (p < pend) {
@@ -2004,6 +2017,8 @@ rb_gc_call_finalizer_at_exit()
finalizer_table = 0;
}
}
+ /* finalizers are part of garbage collection */
+ during_gc++;
/* run data object's finalizers */
for (i = 0; i < heaps_used; i++) {
p = heaps[i].slot; pend = p + heaps[i].limit;
@@ -2015,16 +2030,25 @@ rb_gc_call_finalizer_at_exit()
RUBY_CRITICAL(free(DATA_PTR(p)));
}
else if (RANY(p)->as.data.dfree) {
- (*RANY(p)->as.data.dfree)(DATA_PTR(p));
+ make_deferred(RANY(p));
+ RANY(p)->as.free.next = final_list;
+ final_list = p;
}
}
else if (BUILTIN_TYPE(p) == T_FILE) {
- p->as.free.flags = 0;
- rb_io_fptr_finalize(RANY(p)->as.file.fptr);
+ if (RANY(p)->as.file.fptr) {
+ make_io_deferred(RANY(p));
+ RANY(p)->as.free.next = final_list;
+ final_list = p;
+ }
}
p++;
}
}
+ during_gc = 0;
+ if (final_list) {
+ finalize_list(final_list);
+ }
}
/*