--- embedaddon/php/Zend/zend_objects.c 2012/02/21 23:47:52 1.1.1.1 +++ embedaddon/php/Zend/zend_objects.c 2012/05/29 12:34:36 1.1.1.2 @@ -17,7 +17,7 @@ +----------------------------------------------------------------------+ */ -/* $Id: zend_objects.c,v 1.1.1.1 2012/02/21 23:47:52 misho Exp $ */ +/* $Id: zend_objects.c,v 1.1.1.2 2012/05/29 12:34:36 misho Exp $ */ #include "zend.h" #include "zend_globals.h" @@ -28,10 +28,9 @@ ZEND_API void zend_object_std_init(zend_object *object, zend_class_entry *ce TSRMLS_DC) { - ALLOC_HASHTABLE(object->properties); - zend_hash_init(object->properties, 0, NULL, ZVAL_PTR_DTOR, 0); - object->ce = ce; + object->properties = NULL; + object->properties_table = NULL; object->guards = NULL; } @@ -44,6 +43,18 @@ ZEND_API void zend_object_std_dtor(zend_object *object if (object->properties) { zend_hash_destroy(object->properties); FREE_HASHTABLE(object->properties); + if (object->properties_table) { + efree(object->properties_table); + } + } else if (object->properties_table) { + int i; + + for (i = 0; i < object->ce->default_properties_count; i++) { + if (object->properties_table[i]) { + zval_ptr_dtor(&object->properties_table[i]); + } + } + efree(object->properties_table); } } @@ -133,9 +144,11 @@ ZEND_API zend_object_value zend_objects_new(zend_objec *object = emalloc(sizeof(zend_object)); (*object)->ce = class_type; + (*object)->properties = NULL; + (*object)->properties_table = NULL; + (*object)->guards = NULL; retval.handle = zend_objects_store_put(*object, (zend_objects_store_dtor_t) zend_objects_destroy_object, (zend_objects_free_object_storage_t) zend_objects_free_object_storage, NULL TSRMLS_CC); retval.handlers = &std_object_handlers; - (*object)->guards = NULL; return retval; } @@ -146,8 +159,48 @@ ZEND_API zend_object *zend_objects_get_address(const z ZEND_API void zend_objects_clone_members(zend_object *new_object, zend_object_value new_obj_val, zend_object *old_object, zend_object_handle handle TSRMLS_DC) { - zend_hash_copy(new_object->properties, old_object->properties, zval_copy_property_ctor(old_object->ce), (void *) NULL /* Not used anymore */, sizeof(zval *)); + int i; + if (old_object->properties_table) { + if (!new_object->properties_table) { + new_object->properties_table = emalloc(sizeof(zval*) * old_object->ce->default_properties_count); + memset(new_object->properties_table, 0, sizeof(zval*) * old_object->ce->default_properties_count); + } + for (i = 0; i < old_object->ce->default_properties_count; i++) { + if (!new_object->properties) { + if (new_object->properties_table[i]) { + zval_ptr_dtor(&new_object->properties_table[i]); + } + } + if (!old_object->properties) { + new_object->properties_table[i] = old_object->properties_table[i]; + if (new_object->properties_table[i]) { + Z_ADDREF_P(new_object->properties_table[i]); + } + } + } + } + if (old_object->properties) { + if (!new_object->properties) { + ALLOC_HASHTABLE(new_object->properties); + zend_hash_init(new_object->properties, 0, NULL, ZVAL_PTR_DTOR, 0); + } + zend_hash_copy(new_object->properties, old_object->properties, (copy_ctor_func_t) zval_add_ref, (void *) NULL /* Not used anymore */, sizeof(zval *)); + if (old_object->properties_table) { + HashPosition pos; + zend_property_info *prop_info; + for (zend_hash_internal_pointer_reset_ex(&old_object->ce->properties_info, &pos); + zend_hash_get_current_data_ex(&old_object->ce->properties_info, (void**)&prop_info, &pos) == SUCCESS; + zend_hash_move_forward_ex(&old_object->ce->properties_info, &pos)) { + if ((prop_info->flags & ZEND_ACC_STATIC) == 0) { + if (zend_hash_quick_find(new_object->properties, prop_info->name, prop_info->name_length+1, prop_info->h, (void**)&new_object->properties_table[prop_info->offset]) == FAILURE) { + new_object->properties_table[prop_info->offset] = NULL; + } + } + } + } + } + if (old_object->ce->clone) { zval *new_obj; @@ -173,9 +226,6 @@ ZEND_API zend_object_value zend_objects_clone_obj(zval * overwritten one then it must itself be overwritten */ old_object = zend_objects_get_address(zobject TSRMLS_CC); new_obj_val = zend_objects_new(&new_object, old_object->ce TSRMLS_CC); - - ALLOC_HASHTABLE(new_object->properties); - zend_hash_init(new_object->properties, 0, NULL, ZVAL_PTR_DTOR, 0); zend_objects_clone_members(new_object, new_obj_val, old_object, handle TSRMLS_CC);