]> granicus.if.org Git - php/commitdiff
Fixed bug #49269 (Ternary operator fails on Iterator object when used inside foreach...
authorDmitry Stogov <dmitry@php.net>
Mon, 17 Aug 2009 07:40:43 +0000 (07:40 +0000)
committerDmitry Stogov <dmitry@php.net>
Mon, 17 Aug 2009 07:40:43 +0000 (07:40 +0000)
NEWS
Zend/tests/bug49269.phpt [new file with mode: 0644]
Zend/zend_vm_def.h
Zend/zend_vm_execute.h

diff --git a/NEWS b/NEWS
index dfa92760d7f3cc954d1cdcdb9e730ec8aeb643fd..bede318ecbf373ed401d943c82aa6782be8db8a1 100644 (file)
--- a/NEWS
+++ b/NEWS
@@ -22,6 +22,8 @@ PHP                                                                        NEWS
 - Fixed memory leak in stream_is_local(). (Felipe, Tony)
 - Fixed BC break in mime_content_type(), removes the content encoding. (Scott) 
 
+- Fixed bug #49269 (Ternary operator fails on Iterator object when used inside
+  foreach declaration). (Etienne, Dmitry)
 - Fixed bug #49193 (gdJpegGetVersionString() inside gd_compact identifies
   wrong type in declaration). (Ilia)
 - Fixed bug #49183 (dns_get_record does not return NAPTR records). (Pierre)
diff --git a/Zend/tests/bug49269.phpt b/Zend/tests/bug49269.phpt
new file mode 100644 (file)
index 0000000..2de29d8
--- /dev/null
@@ -0,0 +1,26 @@
+--TEST--
+Bug #49269 (Ternary operator fails on Iterator object when used inside foreach declaration).
+--FILE--
+<?php
+class TestObject implements Iterator
+{
+    public $n = 0;
+    function valid()
+    {
+        return ($this->n < 3);
+    }
+    function current() {return $this->n;}
+    function next() {$this->n++;}
+    function key() { }
+    function rewind() {$this->n = 0;}
+}
+
+
+$array_object = new TestObject();
+
+foreach ((true ? $array_object : $array_object) as $item) echo "$item\n";
+?>
+--EXPECT--
+0
+1
+2
index dbf5e3af696e5024bf7fac5725ea2aa1cfd83546..96b4d7f1a1d877e4d2a826ce6c50b673b48364fa 100644 (file)
@@ -3573,28 +3573,33 @@ ZEND_VM_HANDLER(77, ZEND_FE_RESET, CONST|TMP|VAR|CV, ANY)
                        ALLOC_ZVAL(tmp);
                        INIT_PZVAL_COPY(tmp, array_ptr);
                        array_ptr = tmp;
+                       if (Z_TYPE_P(array_ptr) == IS_OBJECT) {
+                               ce = Z_OBJCE_P(array_ptr);
+                               if (ce && ce->get_iterator) {
+                                       Z_DELREF_P(array_ptr);
+                               }
+                       }
                } else if (Z_TYPE_P(array_ptr) == IS_OBJECT) {
                        ce = Z_OBJCE_P(array_ptr);
                        if (!ce || !ce->get_iterator) {
                                Z_ADDREF_P(array_ptr);
                        }
+               } else if (OP1_TYPE == IS_CONST || 
+                          ((OP1_TYPE == IS_CV || OP1_TYPE == IS_VAR) &&
+                           !Z_ISREF_P(array_ptr) &&
+                           Z_REFCOUNT_P(array_ptr) > 1)) {
+                       zval *tmp;
+
+                       ALLOC_ZVAL(tmp);
+                       INIT_PZVAL_COPY(tmp, array_ptr);
+                       zval_copy_ctor(tmp);
+                       array_ptr = tmp;
                } else {
-                       if ((OP1_TYPE == IS_CV || OP1_TYPE == IS_VAR) &&
-                           !Z_ISREF_P(array_ptr) &&
-                           Z_REFCOUNT_P(array_ptr) > 1) {
-                               zval *tmp;
-
-                               ALLOC_ZVAL(tmp);
-                               INIT_PZVAL_COPY(tmp, array_ptr);
-                               zval_copy_ctor(tmp);
-                               array_ptr = tmp;
-                       } else {
-                               Z_ADDREF_P(array_ptr);
-                       }
+                       Z_ADDREF_P(array_ptr);
                }
        }
 
-       if (OP1_TYPE != IS_TMP_VAR && ce && ce->get_iterator) {
+       if (ce && ce->get_iterator) {
                iter = ce->get_iterator(ce, array_ptr, opline->extended_value & ZEND_FE_RESET_REFERENCE TSRMLS_CC);
 
                if (iter && !EG(exception)) {
index 43bc3abde5dc6541f478f999b009d79f03d0fa75..f809f4e740ed3dfe97a878ebb64e5cf57ed52a62 100644 (file)
@@ -2108,28 +2108,33 @@ static int ZEND_FASTCALL  ZEND_FE_RESET_SPEC_CONST_HANDLER(ZEND_OPCODE_HANDLER_A
                        ALLOC_ZVAL(tmp);
                        INIT_PZVAL_COPY(tmp, array_ptr);
                        array_ptr = tmp;
+                       if (Z_TYPE_P(array_ptr) == IS_OBJECT) {
+                               ce = Z_OBJCE_P(array_ptr);
+                               if (ce && ce->get_iterator) {
+                                       Z_DELREF_P(array_ptr);
+                               }
+                       }
                } else if (Z_TYPE_P(array_ptr) == IS_OBJECT) {
                        ce = Z_OBJCE_P(array_ptr);
                        if (!ce || !ce->get_iterator) {
                                Z_ADDREF_P(array_ptr);
                        }
+               } else if (IS_CONST == IS_CONST ||
+                          ((IS_CONST == IS_CV || IS_CONST == IS_VAR) &&
+                           !Z_ISREF_P(array_ptr) &&
+                           Z_REFCOUNT_P(array_ptr) > 1)) {
+                       zval *tmp;
+
+                       ALLOC_ZVAL(tmp);
+                       INIT_PZVAL_COPY(tmp, array_ptr);
+                       zval_copy_ctor(tmp);
+                       array_ptr = tmp;
                } else {
-                       if ((IS_CONST == IS_CV || IS_CONST == IS_VAR) &&
-                           !Z_ISREF_P(array_ptr) &&
-                           Z_REFCOUNT_P(array_ptr) > 1) {
-                               zval *tmp;
-
-                               ALLOC_ZVAL(tmp);
-                               INIT_PZVAL_COPY(tmp, array_ptr);
-                               zval_copy_ctor(tmp);
-                               array_ptr = tmp;
-                       } else {
-                               Z_ADDREF_P(array_ptr);
-                       }
+                       Z_ADDREF_P(array_ptr);
                }
        }
 
-       if (IS_CONST != IS_TMP_VAR && ce && ce->get_iterator) {
+       if (ce && ce->get_iterator) {
                iter = ce->get_iterator(ce, array_ptr, opline->extended_value & ZEND_FE_RESET_REFERENCE TSRMLS_CC);
 
                if (iter && !EG(exception)) {
@@ -5370,28 +5375,33 @@ static int ZEND_FASTCALL  ZEND_FE_RESET_SPEC_TMP_HANDLER(ZEND_OPCODE_HANDLER_ARG
                        ALLOC_ZVAL(tmp);
                        INIT_PZVAL_COPY(tmp, array_ptr);
                        array_ptr = tmp;
+                       if (Z_TYPE_P(array_ptr) == IS_OBJECT) {
+                               ce = Z_OBJCE_P(array_ptr);
+                               if (ce && ce->get_iterator) {
+                                       Z_DELREF_P(array_ptr);
+                               }
+                       }
                } else if (Z_TYPE_P(array_ptr) == IS_OBJECT) {
                        ce = Z_OBJCE_P(array_ptr);
                        if (!ce || !ce->get_iterator) {
                                Z_ADDREF_P(array_ptr);
                        }
+               } else if (IS_TMP_VAR == IS_CONST ||
+                          ((IS_TMP_VAR == IS_CV || IS_TMP_VAR == IS_VAR) &&
+                           !Z_ISREF_P(array_ptr) &&
+                           Z_REFCOUNT_P(array_ptr) > 1)) {
+                       zval *tmp;
+
+                       ALLOC_ZVAL(tmp);
+                       INIT_PZVAL_COPY(tmp, array_ptr);
+                       zval_copy_ctor(tmp);
+                       array_ptr = tmp;
                } else {
-                       if ((IS_TMP_VAR == IS_CV || IS_TMP_VAR == IS_VAR) &&
-                           !Z_ISREF_P(array_ptr) &&
-                           Z_REFCOUNT_P(array_ptr) > 1) {
-                               zval *tmp;
-
-                               ALLOC_ZVAL(tmp);
-                               INIT_PZVAL_COPY(tmp, array_ptr);
-                               zval_copy_ctor(tmp);
-                               array_ptr = tmp;
-                       } else {
-                               Z_ADDREF_P(array_ptr);
-                       }
+                       Z_ADDREF_P(array_ptr);
                }
        }
 
-       if (IS_TMP_VAR != IS_TMP_VAR && ce && ce->get_iterator) {
+       if (ce && ce->get_iterator) {
                iter = ce->get_iterator(ce, array_ptr, opline->extended_value & ZEND_FE_RESET_REFERENCE TSRMLS_CC);
 
                if (iter && !EG(exception)) {
@@ -8717,28 +8727,33 @@ static int ZEND_FASTCALL  ZEND_FE_RESET_SPEC_VAR_HANDLER(ZEND_OPCODE_HANDLER_ARG
                        ALLOC_ZVAL(tmp);
                        INIT_PZVAL_COPY(tmp, array_ptr);
                        array_ptr = tmp;
+                       if (Z_TYPE_P(array_ptr) == IS_OBJECT) {
+                               ce = Z_OBJCE_P(array_ptr);
+                               if (ce && ce->get_iterator) {
+                                       Z_DELREF_P(array_ptr);
+                               }
+                       }
                } else if (Z_TYPE_P(array_ptr) == IS_OBJECT) {
                        ce = Z_OBJCE_P(array_ptr);
                        if (!ce || !ce->get_iterator) {
                                Z_ADDREF_P(array_ptr);
                        }
+               } else if (IS_VAR == IS_CONST ||
+                          ((IS_VAR == IS_CV || IS_VAR == IS_VAR) &&
+                           !Z_ISREF_P(array_ptr) &&
+                           Z_REFCOUNT_P(array_ptr) > 1)) {
+                       zval *tmp;
+
+                       ALLOC_ZVAL(tmp);
+                       INIT_PZVAL_COPY(tmp, array_ptr);
+                       zval_copy_ctor(tmp);
+                       array_ptr = tmp;
                } else {
-                       if ((IS_VAR == IS_CV || IS_VAR == IS_VAR) &&
-                           !Z_ISREF_P(array_ptr) &&
-                           Z_REFCOUNT_P(array_ptr) > 1) {
-                               zval *tmp;
-
-                               ALLOC_ZVAL(tmp);
-                               INIT_PZVAL_COPY(tmp, array_ptr);
-                               zval_copy_ctor(tmp);
-                               array_ptr = tmp;
-                       } else {
-                               Z_ADDREF_P(array_ptr);
-                       }
+                       Z_ADDREF_P(array_ptr);
                }
        }
 
-       if (IS_VAR != IS_TMP_VAR && ce && ce->get_iterator) {
+       if (ce && ce->get_iterator) {
                iter = ce->get_iterator(ce, array_ptr, opline->extended_value & ZEND_FE_RESET_REFERENCE TSRMLS_CC);
 
                if (iter && !EG(exception)) {
@@ -22559,28 +22574,33 @@ static int ZEND_FASTCALL  ZEND_FE_RESET_SPEC_CV_HANDLER(ZEND_OPCODE_HANDLER_ARGS
                        ALLOC_ZVAL(tmp);
                        INIT_PZVAL_COPY(tmp, array_ptr);
                        array_ptr = tmp;
+                       if (Z_TYPE_P(array_ptr) == IS_OBJECT) {
+                               ce = Z_OBJCE_P(array_ptr);
+                               if (ce && ce->get_iterator) {
+                                       Z_DELREF_P(array_ptr);
+                               }
+                       }
                } else if (Z_TYPE_P(array_ptr) == IS_OBJECT) {
                        ce = Z_OBJCE_P(array_ptr);
                        if (!ce || !ce->get_iterator) {
                                Z_ADDREF_P(array_ptr);
                        }
+               } else if (IS_CV == IS_CONST ||
+                          ((IS_CV == IS_CV || IS_CV == IS_VAR) &&
+                           !Z_ISREF_P(array_ptr) &&
+                           Z_REFCOUNT_P(array_ptr) > 1)) {
+                       zval *tmp;
+
+                       ALLOC_ZVAL(tmp);
+                       INIT_PZVAL_COPY(tmp, array_ptr);
+                       zval_copy_ctor(tmp);
+                       array_ptr = tmp;
                } else {
-                       if ((IS_CV == IS_CV || IS_CV == IS_VAR) &&
-                           !Z_ISREF_P(array_ptr) &&
-                           Z_REFCOUNT_P(array_ptr) > 1) {
-                               zval *tmp;
-
-                               ALLOC_ZVAL(tmp);
-                               INIT_PZVAL_COPY(tmp, array_ptr);
-                               zval_copy_ctor(tmp);
-                               array_ptr = tmp;
-                       } else {
-                               Z_ADDREF_P(array_ptr);
-                       }
+                       Z_ADDREF_P(array_ptr);
                }
        }
 
-       if (IS_CV != IS_TMP_VAR && ce && ce->get_iterator) {
+       if (ce && ce->get_iterator) {
                iter = ce->get_iterator(ce, array_ptr, opline->extended_value & ZEND_FE_RESET_REFERENCE TSRMLS_CC);
 
                if (iter && !EG(exception)) {