summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDmitry Stogov <dmitry@php.net>2008-04-11 17:32:18 +0000
committerDmitry Stogov <dmitry@php.net>2008-04-11 17:32:18 +0000
commit111954d7f1aa4d9ed385f35b123e00aa1c47f559 (patch)
tree7a3c8a64d309c665d1877258a265a14a52ad4c3d
parent30fd5ba5173501af78a6bc1866381873e041c90a (diff)
downloadphp-git-111954d7f1aa4d9ed385f35b123e00aa1c47f559.tar.gz
Speedup GC initialization
-rw-r--r--Zend/zend_gc.c37
-rw-r--r--Zend/zend_gc.h2
2 files changed, 24 insertions, 15 deletions
diff --git a/Zend/zend_gc.c b/Zend/zend_gc.c
index 659ec55ec6..2f3910c2f2 100644
--- a/Zend/zend_gc.c
+++ b/Zend/zend_gc.c
@@ -111,20 +111,18 @@ ZEND_API void gc_reset(TSRMLS_D)
GC_G(zobj_marked_grey) = 0;
#endif
- if (GC_G(buf) &&
- (GC_G(roots).next != &GC_G(roots) ||
- GC_G(roots).prev != &GC_G(roots))) {
-
+ if (GC_G(buf)) {
GC_G(roots).next = &GC_G(roots);
GC_G(roots).prev = &GC_G(roots);
- GC_G(unused) = &GC_G(buf)[0];
- for (i = 0; i < GC_ROOT_BUFFER_MAX_ENTRIES-1; i++) {
- GC_G(buf)[i].prev = &GC_G(buf)[i+1];
- }
- GC_G(buf)[GC_ROOT_BUFFER_MAX_ENTRIES-1].prev = NULL;
+ GC_G(unused) = NULL;
+ GC_G(first_unused) = GC_G(buf);
GC_G(zval_to_free) = NULL;
+ } else {
+ GC_G(unused) = NULL;
+ GC_G(first_unused) = NULL;
+ GC_G(last_unused) = NULL;
}
}
@@ -132,6 +130,7 @@ ZEND_API void gc_init(TSRMLS_D)
{
if (GC_G(buf) == NULL && GC_G(gc_enabled)) {
GC_G(buf) = (gc_root_buffer*) malloc(sizeof(gc_root_buffer) * GC_ROOT_BUFFER_MAX_ENTRIES);
+ GC_G(last_unused) = &GC_G(buf)[GC_ROOT_BUFFER_MAX_ENTRIES];
gc_reset(TSRMLS_C);
}
}
@@ -164,7 +163,12 @@ ZEND_API void gc_zval_possible_root(zval *zv TSRMLS_DC)
if (!GC_ZVAL_ADDRESS(zv)) {
gc_root_buffer *newRoot = GC_G(unused);
- if (!newRoot) {
+ if (newRoot) {
+ GC_G(unused) = newRoot->prev;
+ } else if (GC_G(first_unused) != GC_G(last_unused)) {
+ newRoot = GC_G(first_unused);
+ GC_G(first_unused)++;
+ } else {
if (!GC_G(gc_enabled)) {
GC_ZVAL_SET_BLACK(zv);
return;
@@ -177,10 +181,9 @@ ZEND_API void gc_zval_possible_root(zval *zv TSRMLS_DC)
return;
}
GC_ZVAL_SET_PURPLE(zv);
+ GC_G(unused) = newRoot->prev;
}
- GC_G(unused) = newRoot->prev;
-
newRoot->next = GC_G(roots).next;
newRoot->prev = &GC_G(roots);
GC_G(roots).next->prev = newRoot;
@@ -215,7 +218,12 @@ ZEND_API void gc_zobj_possible_root(zval *zv TSRMLS_DC)
if (!GC_ADDRESS(obj->buffered)) {
gc_root_buffer *newRoot = GC_G(unused);
- if (!newRoot) {
+ if (newRoot) {
+ GC_G(unused) = newRoot->prev;
+ } else if (GC_G(first_unused) != GC_G(last_unused)) {
+ newRoot = GC_G(first_unused);
+ GC_G(first_unused)++;
+ } else {
if (!GC_G(gc_enabled)) {
GC_ZVAL_SET_BLACK(zv);
return;
@@ -229,10 +237,9 @@ ZEND_API void gc_zobj_possible_root(zval *zv TSRMLS_DC)
}
obj = &EG(objects_store).object_buckets[Z_OBJ_HANDLE_P(zv)].bucket.obj;
GC_SET_PURPLE(obj->buffered);
+ GC_G(unused) = newRoot->prev;
}
- GC_G(unused) = newRoot->prev;
-
newRoot->next = GC_G(roots).next;
newRoot->prev = &GC_G(roots);
GC_G(roots).next->prev = newRoot;
diff --git a/Zend/zend_gc.h b/Zend/zend_gc.h
index 45fb3a7d68..d72ef72ab7 100644
--- a/Zend/zend_gc.h
+++ b/Zend/zend_gc.h
@@ -103,6 +103,8 @@ typedef struct _zend_gc_globals {
gc_root_buffer *buf; /* preallocated arrays of buffers */
gc_root_buffer roots; /* list of possible roots of cycles */
gc_root_buffer *unused; /* list of unused buffers */
+ gc_root_buffer *first_unused; /* pointer to first unused buffer */
+ gc_root_buffer *last_unused; /* pointer to last unused buffer */
zval_gc_info *zval_to_free; /* temporaryt list of zvals to free */
zval_gc_info *free_list;