]> granicus.if.org Git - php/commitdiff
Support for nested exceptions and fatal errors in destructors
authorDmitry Stogov <dmitry@php.net>
Wed, 31 May 2006 12:59:45 +0000 (12:59 +0000)
committerDmitry Stogov <dmitry@php.net>
Wed, 31 May 2006 12:59:45 +0000 (12:59 +0000)
Zend/zend_objects.c
Zend/zend_objects_API.c

index 2827a2892fa6a783ecbe45e7d082abad1f3ffc8c..8f4e59474ef5f71fa7d9925a5c51a3f498daad20 100644 (file)
@@ -103,9 +103,10 @@ ZEND_API void zend_objects_destroy_object(zend_object *object, zend_object_handl
                                zval *file = zend_read_property(default_exception_ce, old_exception, "file", sizeof("file")-1, 1 TSRMLS_CC);
                                zval *line = zend_read_property(default_exception_ce, old_exception, "line", sizeof("line")-1, 1 TSRMLS_CC);
 
+                               zval_ptr_dtor(&EG(exception));
+                               EG(exception) = old_exception;
                                zend_error(E_ERROR, "Ignoring exception from %v::__destruct() while an exception is already active (Uncaught %v in %R on line %ld)", 
                                        object->ce->name, Z_OBJCE_P(old_exception)->name, Z_TYPE_P(file), Z_UNIVAL_P(file), Z_LVAL_P(line));
-                               zval_ptr_dtor(&EG(exception));
                        }
                        EG(exception) = old_exception;
                }
index b40a6f845a24326bd65cab1b38edbe13b60db3e6..08b50a92601c0991cafb9df0559ba346c0f29833 100644 (file)
@@ -173,6 +173,7 @@ ZEND_API void zend_objects_store_del_ref(zval *zobject TSRMLS_DC)
 ZEND_API void zend_objects_store_del_ref_by_handle(zend_object_handle handle TSRMLS_DC)
 {
        struct _store_object *obj;
+       int failure = 0;
 
        if (!EG(objects_store).object_buckets) {
                return;
@@ -190,12 +191,20 @@ ZEND_API void zend_objects_store_del_ref_by_handle(zend_object_handle handle TSR
                                EG(objects_store).object_buckets[handle].destructor_called = 1;
 
                                if (obj->dtor) {
-                                       obj->dtor(obj->object, handle TSRMLS_CC);
+                                       zend_try {
+                                               obj->dtor(obj->object, handle TSRMLS_CC);
+                                       } zend_catch {
+                                               failure = 1;
+                                       } zend_end_try();
                                }
                        }
                        if (obj->refcount == 1) {
                                if (obj->free_storage) {
-                                       obj->free_storage(obj->object TSRMLS_CC);
+                                       zend_try {
+                                               obj->free_storage(obj->object TSRMLS_CC);
+                                       } zend_catch {
+                                               failure = 1;
+                                       } zend_end_try();
                                }
                                ZEND_OBJECTS_STORE_ADD_TO_FREE_LIST();
                        }
@@ -211,6 +220,9 @@ ZEND_API void zend_objects_store_del_ref_by_handle(zend_object_handle handle TSR
                fprintf(stderr, "Decreased refcount of object id #%d\n", handle);
        }
 #endif
+       if (failure) {
+               zend_bailout();
+       }
 }
 
 ZEND_API zend_object_value zend_objects_store_clone_obj(zval *zobject TSRMLS_DC)