diff options
author | Xinchen Hui <laruence@php.net> | 2015-03-14 00:52:53 +0800 |
---|---|---|
committer | Xinchen Hui <laruence@php.net> | 2015-03-14 00:52:53 +0800 |
commit | ed59370f0a8a2c1ea8164afcde827c4fba74693d (patch) | |
tree | 0d8da4e814b965f62afdc2aa4e3349957083bbd9 /ext/spl | |
parent | 1b5ad03f668626e5e4df90630c9da9f5a8498b30 (diff) | |
download | php-git-ed59370f0a8a2c1ea8164afcde827c4fba74693d.tar.gz |
Little improvement, update NEWs, added test
Diffstat (limited to 'ext/spl')
-rw-r--r-- | ext/spl/spl_observer.c | 14 | ||||
-rw-r--r-- | ext/spl/tests/bug69227.phpt | 14 |
2 files changed, 19 insertions, 9 deletions
diff --git a/ext/spl/spl_observer.c b/ext/spl/spl_observer.c index 2b50319a62..733257f817 100644 --- a/ext/spl/spl_observer.c +++ b/ext/spl/spl_observer.c @@ -87,7 +87,7 @@ typedef struct _spl_SplObjectStorage { /* {{{ */ zend_function *fptr_get_hash; HashTable *debug_info; zval **gcdata; - long gcdata_len; + long gcdata_num; } spl_SplObjectStorage; /* }}} */ /* {{{ storage is an assoc aray of [zend_object_value]=>[zval *obj, zval *inf] */ @@ -269,9 +269,6 @@ static zend_object_value spl_object_storage_new_ex(zend_class_entry *class_type, zend_object_std_init(&intern->std, class_type TSRMLS_CC); object_properties_init(&intern->std, class_type); - intern->gcdata = NULL; - intern->gcdata_len = 0; - zend_hash_init(&intern->storage, 0, NULL, (void (*)(void *))spl_object_storage_dtor, 0); retval.handle = zend_objects_store_put(intern, (zend_objects_store_dtor_t)zend_objects_destroy_object, (zend_objects_free_object_storage_t) spl_SplOjectStorage_free_storage, NULL TSRMLS_CC); @@ -371,15 +368,14 @@ static HashTable* spl_object_storage_debug_info(zval *obj, int *is_temp TSRMLS_D /* overriden for garbage collection */ static HashTable *spl_object_storage_get_gc(zval *obj, zval ***table, int *n TSRMLS_DC) /* {{{ */ { + long i = 0; spl_SplObjectStorage *intern = (spl_SplObjectStorage*)zend_object_store_get_object(obj TSRMLS_CC); spl_SplObjectStorageElement *element; HashPosition pos; - long i = 0; - long requiredLength = intern->storage.nNumOfElements * 2; - if (requiredLength > intern->gcdata_len) { - intern->gcdata = (zval**)erealloc(intern->gcdata, sizeof(zval*) * requiredLength); - intern->gcdata_len = requiredLength; + if (intern->storage.nNumOfElements > intern->gcdata_num) { + intern->gcdata_num = intern->storage.nNumOfElements * 2; + intern->gcdata = (zval**)erealloc(intern->gcdata, sizeof(zval*) * intern->gcdata_num); } zend_hash_internal_pointer_reset_ex(&intern->storage, &pos); diff --git a/ext/spl/tests/bug69227.phpt b/ext/spl/tests/bug69227.phpt new file mode 100644 index 0000000000..812d8bafd8 --- /dev/null +++ b/ext/spl/tests/bug69227.phpt @@ -0,0 +1,14 @@ +--TEST-- +Bug #69227 (Use after free in zval_scan caused by spl_object_storage_get_gc) +--INI-- +zend.enable_gc=1 +--FILE-- +<?php + +$s = new SplObjectStorage(); +$s->attach($s); +gc_collect_cycles(); +echo "ok"; +?> +--EXPECT-- +ok |