]> granicus.if.org Git - php/commitdiff
Generalize object storage and reference bookkeeping
authorStanislav Malyshev <stas@php.net>
Fri, 31 May 2002 12:09:19 +0000 (12:09 +0000)
committerStanislav Malyshev <stas@php.net>
Fri, 31 May 2002 12:09:19 +0000 (12:09 +0000)
Zend/Makefile.am
Zend/zend_execute_API.c
Zend/zend_globals.h
Zend/zend_object_handlers.c
Zend/zend_objects.c
Zend/zend_objects.h
Zend/zend_objects_API.c [new file with mode: 0644]
Zend/zend_objects_API.h [new file with mode: 0644]

index 4f73a2ac92ca1f689447a775a051daa9afe2dab5..2182442d5b6c05d98d7ea0c62baedb0a84617f89 100644 (file)
@@ -13,7 +13,8 @@ libZend_la_SOURCES=\
        zend_opcode.c zend_operators.c zend_ptr_stack.c zend_stack.c \
        zend_variables.c zend.c zend_API.c zend_extensions.c zend_hash.c \
        zend_list.c zend_indent.c zend_builtin_functions.c zend_sprintf.c \
-       zend_ini.c zend_qsort.c zend_objects.c zend_object_handlers.c
+       zend_ini.c zend_qsort.c zend_objects.c zend_object_handlers.c \
+       zend_object_API.c
 
 libZend_la_LDFLAGS =
 libZend_la_LIBADD = @ZEND_EXTRA_LIBS@
index e6ef99b114357ecf7124c0a127d8b987d78525c4..c007ee8bb5de7842df9fc5c531a80639f02a5569 100644 (file)
@@ -160,7 +160,7 @@ void init_executor(TSRMLS_D)
        zend_ptr_stack_init(&EG(user_error_handlers));
 
        EG(orig_error_reporting) = EG(error_reporting);
-       zend_objects_init(&EG(objects), 1024);
+       zend_objects_store_init(&EG(objects_store), 1024);
 
        EG(full_tables_cleanup) = 0;
 #ifdef ZEND_WIN32
@@ -183,7 +183,7 @@ void init_executor(TSRMLS_D)
 void shutdown_executor(TSRMLS_D)
 {
        zend_try {
-               zend_objects_call_destructors(&EG(objects) TSRMLS_CC);
+               zend_objects_store_call_destructors(&EG(objects_store) TSRMLS_CC);
 
                zend_ptr_stack_destroy(&EG(arg_types_stack));
                                
@@ -236,7 +236,7 @@ void shutdown_executor(TSRMLS_D)
                zend_ptr_stack_destroy(&EG(user_error_handlers));
 
                EG(error_reporting) = EG(orig_error_reporting);
-               zend_objects_destroy(&EG(objects));
+               zend_objects_store_destroy(&EG(objects_store));
        } zend_end_try();
 }
 
index 58b2c4894546c085640f8e8ecf77efe0d38eec13..b133dfe5c056f4b6e1f981ff88a628becbe88237 100644 (file)
@@ -32,6 +32,7 @@
 #include "zend_llist.h"
 #include "zend_fast_cache.h"
 #include "zend_objects.h"
+#include "zend_objects_API.h"
 
 /* Define ZTS if you want a thread-safe Zend */
 /*#undef ZTS*/
@@ -200,7 +201,7 @@ struct _zend_executor_globals {
        int lambda_count;
 
        HashTable ini_directives;
-       zend_objects objects;
+       zend_objects_store objects_store;
        zval *exception;
 
        struct _zend_execute_data *current_execute_data;
index eb95aca182dbf97ad42b456c5f1d1cbf9f292cd7..55a4073bf6b3c0e56172ac8a8989961bf1f2e899 100644 (file)
@@ -3,6 +3,7 @@
 #include "zend_variables.h"
 #include "zend_API.h"
 #include "zend_objects.h"
+#include "zend_objects_API.h"
 #include "zend_object_handlers.h"
 
 #define DEBUG_OBJECT_HANDLERS 0
@@ -224,9 +225,9 @@ zend_class_entry *zend_std_object_get_class(zval *object TSRMLS_DC)
 }
 
 zend_object_handlers std_object_handlers = {
-       zend_objects_add_ref,                    /* add_ref */
-       zend_objects_del_ref,                    /* del_ref */
-       zend_objects_delete_obj,                 /* delete_obj */
+       zend_objects_store_add_ref,              /* add_ref */
+       zend_objects_store_del_ref,              /* del_ref */
+       zend_objects_store_delete_obj,           /* delete_obj */
        zend_objects_clone_obj,                  /* clone_obj */
        
        zend_std_read_property,                  /* read_property */
index 3cf69b9498f025be976666cedd145f7f39bd226f..4e94ebfd5bf9be06a0e8743e39624e818d91f4ac 100644 (file)
@@ -3,21 +3,6 @@
 #include "zend_variables.h"
 #include "zend_API.h"
 
-#define ZEND_DEBUG_OBJECTS 0
-
-void zend_objects_init(zend_objects *objects, zend_uint init_size)
-{
-       objects->object_buckets = (zend_object_bucket *) emalloc(init_size * sizeof(zend_object_bucket));
-       objects->top = 1; /* Skip 0 so that handles are true */
-       objects->size = init_size;
-       objects->free_list_head = -1;
-}
-
-void zend_objects_destroy(zend_objects *objects)
-{
-       efree(objects->object_buckets);
-}
-
 static inline void zend_objects_call_destructor(zend_object *object, zend_object_handle handle TSRMLS_DC)
 {
        if (object->ce->destructor) {
@@ -59,24 +44,9 @@ static inline void zend_objects_destroy_object(zend_object *object, zend_object_
        /* Nuke the object */
        zend_hash_destroy(object->properties);
        efree(object->properties);
-       
+       efree(object);
 }
 
-
-void zend_objects_call_destructors(zend_objects *objects TSRMLS_DC)
-{
-       zend_uint i = 1;
-
-       for (i = 1; i < objects->top ; i++) {
-               if (EG(objects).object_buckets[i].valid) {
-                       EG(objects).object_buckets[i].destructor_called = 1;
-                       zend_objects_destroy_object(&EG(objects).object_buckets[i].bucket.obj.object, i TSRMLS_CC);
-                       EG(objects).object_buckets[i].valid = 0;
-               }
-       }
-}
-
-
 zend_object_value zend_objects_new(zend_object **object, zend_class_entry *class_type)
 {
        zend_object_handle handle;
@@ -84,116 +54,17 @@ zend_object_value zend_objects_new(zend_object **object, zend_class_entry *class
 
        TSRMLS_FETCH();
 
-       if (EG(objects).free_list_head != -1) {
-               handle = EG(objects).free_list_head;
-               EG(objects).free_list_head = EG(objects).object_buckets[handle].bucket.free_list.next;
-       } else {
-               if (EG(objects).top == EG(objects).size) {
-                       EG(objects).size <<= 1;
-                       EG(objects).object_buckets = (zend_object_bucket *) erealloc(EG(objects).object_buckets, EG(objects).size * sizeof(zend_object_bucket));
-               }
-               handle = EG(objects).top++;
-       }
-       EG(objects).object_buckets[handle].valid = 1;
-       EG(objects).object_buckets[handle].destructor_called = 0;
-       EG(objects).object_buckets[handle].bucket.obj.refcount = 1;
-       
-       *object = &EG(objects).object_buckets[handle].bucket.obj.object;
-
+       *object = emalloc(sizeof(zend_object));
        (*object)->ce = class_type;
-
-       retval.handle = handle;
+       retval.handle = zend_objects_store_put(*object, (zend_objects_store_dtor_t) zend_objects_destroy_object, NULL TSRMLS_DC);
        retval.handlers = &std_object_handlers;
-#if ZEND_DEBUG_OBJECTS
-       fprintf(stderr, "Allocated object id #%d\n", handle);
-#endif
        return retval;
 }
 
 zend_object *zend_objects_get_address(zval *zobject)
 {
-       zend_object_handle handle = Z_OBJ_HANDLE_P(zobject);
        TSRMLS_FETCH();
-       
-       if (!EG(objects).object_buckets[handle].valid) {
-               zend_error(E_ERROR, "Trying to access invalid object");
-       }
-       return &EG(objects).object_buckets[handle].bucket.obj.object;
-}
-
-void zend_objects_add_ref(zval *object TSRMLS_DC)
-{
-       zend_object_handle handle = Z_OBJ_HANDLE_P(object);
-
-       if (!EG(objects).object_buckets[handle].valid) {
-               zend_error(E_ERROR, "Trying to add reference to invalid object");
-       }
-
-       EG(objects).object_buckets[handle].bucket.obj.refcount++;
-#if ZEND_DEBUG_OBJECTS
-       fprintf(stderr, "Increased refcount of object id #%d\n", handle);
-#endif
-}
-
-void zend_objects_delete_obj(zval *zobject TSRMLS_DC)
-{
-       zend_object *object;
-       zend_object_handle handle = Z_OBJ_HANDLE_P(zobject);
-
-       if (!EG(objects).object_buckets[handle].valid) {
-               zend_error(E_ERROR, "Trying to delete invalid object");
-       }
-
-       object = &EG(objects).object_buckets[handle].bucket.obj.object;
-
-       if (!EG(objects).object_buckets[handle].destructor_called) {
-               EG(objects).object_buckets[handle].destructor_called = 1;
-               zend_objects_destroy_object(object, handle TSRMLS_CC);
-       }
-       
-       EG(objects).object_buckets[handle].valid = 0;
-       
-#if ZEND_DEBUG_OBJECTS
-       fprintf(stderr, "Deleted object id #%d\n", handle);
-#endif
-
-}
-
-#define ZEND_OBJECTS_ADD_TO_FREE_LIST()                                                                                                                                \
-                       EG(objects).object_buckets[handle].bucket.free_list.next = EG(objects).free_list_head;  \
-                       EG(objects).free_list_head = handle;                                                                                                    \
-                       EG(objects).object_buckets[handle].valid = 0;
-
-void zend_objects_del_ref(zval *zobject TSRMLS_DC)
-{
-       zend_object_handle handle = Z_OBJ_HANDLE_P(zobject);
-
-       if (--EG(objects).object_buckets[handle].bucket.obj.refcount == 0) {
-               zend_object *object;
-               do {
-                       if (EG(objects).object_buckets[handle].valid) {
-                               if (!EG(objects).object_buckets[handle].destructor_called) {
-                                       object = &EG(objects).object_buckets[handle].bucket.obj.object;
-                                       EG(objects).object_buckets[handle].destructor_called = 1;
-                                       zend_objects_destroy_object(object, handle TSRMLS_CC);
-                                       if (EG(objects).object_buckets[handle].bucket.obj.refcount == 0) {
-                                               ZEND_OBJECTS_ADD_TO_FREE_LIST();
-                                       }
-                                       break;
-                               }                       
-                       }
-                       ZEND_OBJECTS_ADD_TO_FREE_LIST();
-               } while (0);
-               
-#if ZEND_DEBUG_OBJECTS
-               fprintf(stderr, "Deallocated object id #%d\n", handle);
-#endif
-       }
-#if ZEND_DEBUG_OBJECTS
-       else {
-               fprintf(stderr, "Decreased refcount of object id #%d\n", handle);
-       }
-#endif
+       return (zend_object *)zend_object_store_get_object(zobject TSRMLS_DC);
 }
 
 zend_object_value zend_objects_clone_obj(zval *zobject TSRMLS_DC)
@@ -203,12 +74,7 @@ zend_object_value zend_objects_clone_obj(zval *zobject TSRMLS_DC)
        zend_object *new_object;
        zend_object_handle handle = Z_OBJ_HANDLE_P(zobject);
 
-
-       if (!EG(objects).object_buckets[handle].valid) {
-               zend_error(E_ERROR, "Trying to clone invalid object");
-       }
-
-       old_object = &EG(objects).object_buckets[handle].bucket.obj.object;
+       old_object = zend_objects_get_address(zobject);
        retval = zend_objects_new(&new_object, old_object->ce);
 
        if (old_object->ce->clone) {
index dcb718b0881360db611af36b0f1c7e981546a349..f173e1195d9b4c759129ee8884e204b53c4e096a 100644 (file)
@@ -24,16 +24,8 @@ typedef struct _zend_objects {
        int free_list_head;
 } zend_objects;
 
-void zend_objects_init(zend_objects *objects, zend_uint init_size);
-void zend_objects_call_destructors(zend_objects *objects TSRMLS_DC);
-void zend_objects_destroy(zend_objects *objects);
 zend_object_value zend_objects_new(zend_object **object, zend_class_entry *class_type);
-
 zend_object *zend_objects_get_address(zval *object);
-
-void zend_objects_add_ref(zval *object TSRMLS_DC);
-void zend_objects_del_ref(zval *object TSRMLS_DC);
-void zend_objects_delete_obj(zval *object TSRMLS_DC);
 zend_object_value zend_objects_clone_obj(zval *object TSRMLS_DC);
 
 #endif /* ZEND_OBJECTS_H */
diff --git a/Zend/zend_objects_API.c b/Zend/zend_objects_API.c
new file mode 100644 (file)
index 0000000..16b40c8
--- /dev/null
@@ -0,0 +1,258 @@
+#include "zend.h"
+#include "zend_globals.h"
+#include "zend_variables.h"
+#include "zend_API.h"
+#include "zend_objects_API.h"
+
+#define ZEND_DEBUG_OBJECTS 0
+
+void zend_objects_store_init(zend_objects_store *objects, zend_uint init_size)
+{
+       objects->object_buckets = (zend_object_store_bucket *) emalloc(init_size * sizeof(zend_object_bucket));
+       objects->top = 1; /* Skip 0 so that handles are true */
+       objects->size = init_size;
+       objects->free_list_head = -1;
+}
+
+void zend_objects_store_destroy(zend_objects_store *objects)
+{
+       efree(objects->object_buckets);
+}
+
+void zend_objects_store_call_destructors(zend_objects_store *objects)
+{
+       zend_uint i = 1;
+
+       for (i = 1; i < objects->top ; i++) {
+               if (objects->object_buckets[i].valid) {
+                       struct _store_object *obj = &objects->object_buckets[i].bucket.obj;
+                       if(obj->dtor) {
+                               objects->object_buckets[i].destructor_called = 1;
+                               obj->dtor(obj->object, i TSRMLS_CC);
+                       }
+                       objects->object_buckets[i].valid = 0;
+               }
+       }
+}
+
+/* Store objects API */
+
+zend_object_handle zend_objects_store_put(void *object, zend_objects_store_dtor_t dtor, zend_objects_store_clone_t clone TSRMLS_DC)
+{
+       zend_object_handle handle;
+       struct _store_object *obj;
+       
+       if (EG(objects_store).free_list_head != -1) {
+               handle = EG(objects_store).free_list_head;
+               EG(objects_store).free_list_head = EG(objects_store).object_buckets[handle].bucket.free_list.next;
+       } else {
+               if (EG(objects_store).top == EG(objects_store).size) {
+                       EG(objects_store).size <<= 1;
+                       EG(objects_store).object_buckets = (zend_object_store_bucket *) erealloc(EG(objects_store).object_buckets, EG(objects_store).size * sizeof(zend_object_store_bucket));
+               }
+               handle = EG(objects_store).top++;
+       }
+       obj = &EG(objects_store).object_buckets[handle].bucket.obj;
+       EG(objects_store).object_buckets[handle].valid = 1;
+       EG(objects_store).object_buckets[handle].destructor_called = 0;
+
+       obj->refcount = 1;
+       obj->object = object;
+       obj->dtor = dtor;
+       obj->clone = clone;
+
+#if ZEND_DEBUG_OBJECTS
+       fprintf(stderr, "Allocated object id #%d\n", handle);
+#endif
+       return handle;
+}
+
+void zend_objects_store_add_ref(zval *object TSRMLS_DC)
+{
+       zend_object_handle handle = Z_OBJ_HANDLE_P(object);
+
+       if (!EG(objects_store).object_buckets[handle].valid) {
+               zend_error(E_ERROR, "Trying to add reference to invalid object");
+       }
+
+       EG(objects_store).object_buckets[handle].bucket.obj.refcount++;
+#if ZEND_DEBUG_OBJECTS
+       fprintf(stderr, "Increased refcount of object id #%d\n", handle);
+#endif
+}
+
+void zend_objects_store_delete_obj(zval *zobject TSRMLS_DC)
+{
+       zend_object_handle handle = Z_OBJ_HANDLE_P(zobject);
+       struct _store_object *obj = &EG(objects_store).object_buckets[handle].bucket.obj;
+       
+       if (!EG(objects_store).object_buckets[handle].valid) {
+               zend_error(E_ERROR, "Trying to delete invalid object");
+       }
+
+
+       if(obj->dtor && !EG(objects_store).object_buckets[handle].destructor_called) {
+               EG(objects_store).object_buckets[handle].destructor_called = 1;
+               obj->dtor(obj->object, handle TSRMLS_CC);
+       }
+       EG(objects_store).object_buckets[handle].valid = 0;
+       
+#if ZEND_DEBUG_OBJECTS
+       fprintf(stderr, "Deleted object id #%d\n", handle);
+#endif
+
+}
+
+#define ZEND_OBJECTS_STORE_ADD_TO_FREE_LIST()                                                                                                                          \
+                       EG(objects_store).object_buckets[handle].bucket.free_list.next = EG(objects_store).free_list_head;      \
+                       EG(objects_store).free_list_head = handle;                                                                                                      \
+                       EG(objects_store).object_buckets[handle].valid = 0;
+
+void zend_objects_store_del_ref(zval *zobject TSRMLS_DC)
+{
+       zend_object_handle handle = Z_OBJ_HANDLE_P(zobject);
+       struct _store_object *obj = &EG(objects_store).object_buckets[handle].bucket.obj;
+       
+       if (--obj->refcount == 0) {
+               if (EG(objects_store).object_buckets[handle].valid) {
+                       if(obj->dtor && !EG(objects_store).object_buckets[handle].destructor_called) {
+                               EG(objects_store).object_buckets[handle].destructor_called = 1;
+                               obj->dtor(obj->object, handle TSRMLS_CC);
+                       }
+               }
+               ZEND_OBJECTS_STORE_ADD_TO_FREE_LIST();
+#if ZEND_DEBUG_OBJECTS
+               fprintf(stderr, "Deallocated object id #%d\n", handle);
+#endif
+       }
+#if ZEND_DEBUG_OBJECTS
+       else {
+               fprintf(stderr, "Decreased refcount of object id #%d\n", handle);
+       }
+#endif
+}
+
+zend_object_value zend_objects_store_clone_obj(zval *zobject TSRMLS_DC)
+{
+       zend_object_value retval;
+       void *new_object;
+       struct _store_object *obj;
+       zend_object_handle handle = Z_OBJ_HANDLE_P(zobject);
+
+       if (!EG(objects_store).object_buckets[handle].valid) {
+               zend_error(E_ERROR, "Trying to clone invalid object");
+       }
+       obj = &EG(objects_store).object_buckets[handle].bucket.obj;
+       
+       if(obj->clone == NULL) {
+               zend_error(E_ERROR, "Trying to clone uncloneable object");
+       }               
+
+       obj->clone(&obj->object, &new_object TSRMLS_CC);
+
+       retval.handle = zend_objects_store_put(new_object, obj->dtor, obj->clone TSRMLS_CC);
+       retval.handlers = Z_OBJ_HT_P(zobject);
+       
+       return retval;
+}
+
+void *zend_object_store_get_object(zval *zobject TSRMLS_DC)
+{
+       zend_object_handle handle = Z_OBJ_HANDLE_P(zobject);
+
+       if (!EG(objects_store).object_buckets[handle].valid) {
+               zend_error(E_ERROR, "Trying to access invalid object");
+               return NULL;
+       }
+
+       return EG(objects_store).object_buckets[handle].bucket.obj.object;
+}
+
+
+/* Proxy objects workings */
+typedef struct _zend_proxy_object {
+       zval *object;
+       zval *property;
+} zend_proxy_object;
+
+static zend_object_handlers zend_object_proxy_handlers;
+
+void zend_objects_proxy_dtor(zend_proxy_object *object, zend_object_handle handle TSRMLS_DC)
+{
+       zval_ptr_dtor(&object->object);
+       zval_ptr_dtor(&object->property);
+       efree(object);
+}
+
+void zend_objects_proxy_clone(zend_proxy_object *object, zend_proxy_object **object_clone TSRMLS_DC)
+{
+       *object_clone = emalloc(sizeof(zend_proxy_object));
+       (*object_clone)->object = object->object;
+       (*object_clone)->property = object->property;
+       zval_add_ref(&(*object_clone)->property);
+       zval_add_ref(&(*object_clone)->object);
+}
+
+zval **zend_object_create_proxy(zval *object, zval *member TSRMLS_DC)
+{
+       zend_proxy_object *pobj = emalloc(sizeof(zend_proxy_object));
+       zval *retval, **pretval;
+
+       pobj->object = object;
+       pobj->property = member;
+       zval_add_ref(&pobj->property);
+       zval_add_ref(&pobj->object);
+
+       MAKE_STD_ZVAL(retval);
+       retval->type = IS_OBJECT;
+       Z_OBJ_HANDLE_P(retval) = zend_objects_store_put(pobj, (zend_objects_store_dtor_t)zend_objects_proxy_dtor, (zend_objects_store_clone_t)zend_objects_proxy_clone TSRMLS_CC);
+       Z_OBJ_HT_P(retval) = &zend_object_proxy_handlers;
+       pretval = emalloc(sizeof(zval *));
+       *pretval = retval;
+       
+       return pretval;
+}
+
+void zend_object_proxy_set(zval **property, zval *value TSRMLS_DC)
+{
+       zend_proxy_object *probj = zend_object_store_get_object(*property TSRMLS_CC);
+
+       if(Z_OBJ_HT_P(probj->object) && Z_OBJ_HT_P(probj->object)->write_property) {
+               Z_OBJ_HT_P(probj->object)->write_property(probj->object, probj->property, value TSRMLS_CC);
+       } else {
+               zend_error(E_WARNING, "Cannot write property of object - no write handler defined");
+       }
+}
+
+zval* zend_object_proxy_get(zval *property TSRMLS_DC)
+{
+       zend_proxy_object *probj = zend_object_store_get_object(property TSRMLS_CC);
+
+       if(Z_OBJ_HT_P(probj->object) && Z_OBJ_HT_P(probj->object)->read_property) {
+               return Z_OBJ_HT_P(probj->object)->read_property(probj->object, probj->property, BP_VAR_R TSRMLS_CC);
+       } else {
+               zend_error(E_WARNING, "Cannot read property of object - no read handler defined");
+       }
+
+       return NULL;
+}
+
+static zend_object_handlers zend_object_proxy_handlers = {
+       ZEND_OBJECTS_STORE_HANDLERS,
+       
+       NULL,                  /* read_property */
+       NULL,                 /* write_property */
+       NULL,               /* get_property_ptr */
+       NULL,               /* get_property_zval_ptr */
+       zend_object_proxy_get,                   /* get */
+       zend_object_proxy_set,                   /* set */
+       NULL,                   /* has_property */
+       NULL,                   /* unset_property */
+       NULL,                   /* get_properties */
+       NULL,                   /* get_method */
+       NULL,                   /* call_method */
+       NULL,                /* get_constructor */
+       NULL,                /* get_class_entry */
+       NULL,                /* get_class_name */
+       NULL                 /* compare_objects */
+};
diff --git a/Zend/zend_objects_API.h b/Zend/zend_objects_API.h
new file mode 100644 (file)
index 0000000..18a5a09
--- /dev/null
@@ -0,0 +1,49 @@
+#ifndef ZEND_OBJECTS_API_H
+#define ZEND_OBJECTS_API_H
+
+#include "zend.h"
+
+typedef void (*zend_objects_store_dtor_t)(void *object, zend_object_handle handle TSRMLS_DC);
+typedef void (*zend_objects_store_clone_t)(void *object, void **object_clone TSRMLS_DC);
+
+typedef struct _zend_object_store_bucket {
+       zend_bool destructor_called;
+       zend_bool valid;
+       union _store_bucket {
+               struct _store_object {
+                       void *object;
+                       zend_objects_store_dtor_t dtor;
+                       zend_objects_store_clone_t clone;
+                       zend_uint refcount;
+               } obj;
+               struct {
+                       int next;
+               } free_list;
+       } bucket;
+} zend_object_store_bucket;
+
+typedef struct _zend_objects_store {
+       zend_object_store_bucket *object_buckets;
+       zend_uint top;
+       zend_uint size;
+       int free_list_head;
+} zend_objects_store;
+
+/* Global store handling functions */
+void zend_objects_store_init(zend_objects_store *objects, zend_uint init_size);
+void zend_objects_store_call_destructors(zend_objects_store *objects TSRMLS_DC);
+void zend_objects_store_destroy(zend_objects_store *objects);
+
+/* Store API functions */
+zend_object_handle zend_objects_store_put(void *object, zend_objects_store_dtor_t dtor, zend_objects_store_clone_t clone TSRMLS_DC);
+
+void zend_objects_store_add_ref(zval *object TSRMLS_DC);
+void zend_objects_store_del_ref(zval *object TSRMLS_DC);
+void zend_objects_store_delete_obj(zval *object TSRMLS_DC);
+zend_object_value zend_objects_store_clone_obj(zval *object TSRMLS_DC);
+void *zend_object_store_get_object(zval *object TSRMLS_DC);
+       
+#define ZEND_OBJECTS_STORE_HANDLERS zend_objects_store_add_ref, zend_objects_store_del_ref, zend_objects_store_delete_obj, zend_objects_store_clone_obj
+
+zval **zend_object_create_proxy(zval *object, zval *member TSRMLS_DC);
+#endif /* ZEND_OBJECTS_H */