diff options
author | Dmitry Stogov <dmitry@zend.com> | 2014-08-06 15:31:07 +0400 |
---|---|---|
committer | Dmitry Stogov <dmitry@zend.com> | 2014-08-06 15:31:07 +0400 |
commit | 414762fc12acef00733121ae515c9aba32e8bce3 (patch) | |
tree | e0df6dc84284f39c7a32314ae80b86f4beb0b47f | |
parent | e677e8cc87b2a6ae63587c1a3b3e7c6820bb448f (diff) | |
download | php-git-414762fc12acef00733121ae515c9aba32e8bce3.tar.gz |
Fixed GC (Zend/tests/bug63635.phpt)
-rw-r--r-- | Zend/zend_gc.c | 19 |
1 files changed, 13 insertions, 6 deletions
diff --git a/Zend/zend_gc.c b/Zend/zend_gc.c index eea4de8653..1d1f0b3058 100644 --- a/Zend/zend_gc.c +++ b/Zend/zend_gc.c @@ -472,7 +472,7 @@ static int gc_collect_white(zend_refcounted *ref TSRMLS_DC) Bucket *p; tail_call: - if (GC_INFO(ref) == GC_WHITE) { + if (GC_GET_COLOR(GC_INFO(ref)) == GC_WHITE) { ht = NULL; GC_SET_BLACK(GC_INFO(ref)); @@ -568,16 +568,23 @@ static int gc_collect_roots(TSRMLS_D) int count = 0; gc_root_buffer *current = GC_G(roots).next; + /* remove non-garbage from the list */ while (current != &GC_G(roots)) { - GC_SET_ADDRESS(GC_INFO(current->ref), 0); - if (GC_INFO(current->ref) == GC_WHITE) { - count += gc_collect_white(current->ref TSRMLS_CC); - GC_SET_ADDRESS(GC_INFO(current->ref), current - GC_G(buf)); - } else { + if (GC_GET_COLOR(GC_INFO(current->ref)) != GC_WHITE) { + GC_SET_ADDRESS(GC_INFO(current->ref), 0); GC_REMOVE_FROM_ROOTS(current); } current = current->next; } + + current = GC_G(roots).next; + while (current != &GC_G(roots)) { + if (GC_GET_COLOR(GC_INFO(current->ref)) == GC_WHITE) { + GC_REFCOUNT(current->ref)++; + count += gc_collect_white(current->ref TSRMLS_CC); + } + current = current->next; + } /* relink remaining roots into list to free */ if (GC_G(roots).next != &GC_G(roots)) { if (GC_G(to_free).next == &GC_G(to_free)) { |