]> granicus.if.org Git - php/commitdiff
Fixed bug #38808 ("maybe ref" issue for current() and others)
authorDmitry Stogov <dmitry@php.net>
Tue, 26 Sep 2006 10:31:04 +0000 (10:31 +0000)
committerDmitry Stogov <dmitry@php.net>
Tue, 26 Sep 2006 10:31:04 +0000 (10:31 +0000)
Zend/tests/bug38808.phpt [new file with mode: 0755]
Zend/zend_compile.h
Zend/zend_vm_def.h
Zend/zend_vm_execute.h

diff --git a/Zend/tests/bug38808.phpt b/Zend/tests/bug38808.phpt
new file mode 100755 (executable)
index 0000000..0fc4bfe
--- /dev/null
@@ -0,0 +1,17 @@
+--TEST--
+Bug #38808 ("maybe ref" issue for current() and others)
+--FILE--
+<?php
+$current = "current";
+$next = "next";
+
+$b = array(1=>'one', 2=>'two');
+$a =& $b;
+
+echo $current($a)."\n";
+$next($a);
+echo $current($a)."\n";
+?>
+--EXPECT--
+one
+two
\ No newline at end of file
index 6dc24704ef955c2afb8def758e284dbcc58dc386..df0bf24c6a1a4dd7ba4d76d8b20e08d8161effb0 100644 (file)
@@ -691,40 +691,23 @@ int zendlex(znode *zendlval TSRMLS_DC);
 #define ZEND_SEND_BY_REF     1
 #define ZEND_SEND_PREFER_REF 2
 
-/* Lost In Stupid Parentheses */
-#define ARG_SHOULD_BE_SENT_BY_REF(zf, arg_num)                                                                                 \
-       (                                                                                                                                                                       \
-               zf                                                                                                                                                              \
-               && ((zend_function *) zf)->common.arg_info                                                                              \
-               &&                                                                                                                                                              \
-               (                                                                                                                                                               \
-                       (                                                                                                                                                       \
-                               arg_num<=((zend_function *) zf)->common.num_args                                                \
-                               && ((zend_function *) zf)->common.arg_info[arg_num-1].pass_by_reference == ZEND_SEND_BY_REF \
-                       )                                                                                                                                                       \
-               ||      (                                                                                                                                                       \
-                               arg_num>((zend_function *) zf)->common.num_args                                                 \
-                               && ((zend_function *) zf)->common.pass_rest_by_reference == ZEND_SEND_BY_REF                            \
-                       )                                                                                                                                                       \
-               )                                                                                                                                                               \
-       )
-
-#define ARG_MAY_BE_SENT_BY_REF(zf, arg_num)                                                                                    \
-       (                                                                                                                                                                       \
-               zf                                                                                                                                                              \
-               && ((zend_function *) zf)->common.arg_info                                                                              \
-               &&                                                                                                                                                              \
-               (                                                                                                                                                               \
-                       (                                                                                                                                                       \
-                               arg_num<=((zend_function *) zf)->common.num_args                                                \
-                               && ((zend_function *) zf)->common.arg_info[arg_num-1].pass_by_reference == ZEND_SEND_PREFER_REF \
-                       )                                                                                                                                                       \
-               ||      (                                                                                                                                                       \
-                               arg_num>((zend_function *) zf)->common.num_args                                                 \
-                               && ((zend_function *) zf)->common.pass_rest_by_reference == ZEND_SEND_PREFER_REF                                \
-                       )                                                                                                                                                       \
-               )                                                                                                                                                               \
-       )
+#define ARG_SEND_TYPE(zf, arg_num)                                                                                             \
+       ((zf) ?                                                                     \
+        ((((zend_function*)(zf))->common.arg_info &&                               \
+          arg_num<=((zend_function*)(zf))->common.num_args) ?                      \
+         ((zend_function *)(zf))->common.arg_info[arg_num-1].pass_by_reference :   \
+         ((zend_function *)(zf))->common.pass_rest_by_reference) :                 \
+        ZEND_SEND_BY_VAL)      
+
+#define ARG_MUST_BE_SENT_BY_REF(zf, arg_num) \
+       (ARG_SEND_TYPE(zf, arg_num) == ZEND_SEND_BY_REF)
+
+#define ARG_SHOULD_BE_SENT_BY_REF(zf, arg_num) \
+       (ARG_SEND_TYPE(zf, arg_num) & (ZEND_SEND_BY_REF|ZEND_SEND_PREFER_REF))
+
+#define ARG_MAY_BE_SENT_BY_REF(zf, arg_num) \
+       (ARG_SEND_TYPE(zf, arg_num) == ZEND_SEND_PREFER_REF)
+
 
 #define ZEND_RETURN_VAL 0
 #define ZEND_RETURN_REF 1
index 9fcd1284aa92b04f37cf2c21a360fae96b0f8fd0..dca5b9c56ea86e0c0fced5977593f5085c47f97b 100644 (file)
@@ -2202,7 +2202,7 @@ ZEND_VM_HANDLER(65, ZEND_SEND_VAL, CONST|TMP|VAR|CV, ANY)
 {
        zend_op *opline = EX(opline);
        if (opline->extended_value==ZEND_DO_FCALL_BY_NAME
-               && ARG_SHOULD_BE_SENT_BY_REF(EX(fbc), opline->op2.u.opline_num)) {
+               && ARG_MUST_BE_SENT_BY_REF(EX(fbc), opline->op2.u.opline_num)) {
                        zend_error_noreturn(E_ERROR, "Cannot pass parameter %d by reference", opline->op2.u.opline_num);
        }
        {
index d2afe00271937b8fdc4d32dfbecaae56de30b7bd..99277f2b18f76d9a461dc176a5cb26c2904230fc 100644 (file)
@@ -1744,7 +1744,7 @@ static int ZEND_SEND_VAL_SPEC_CONST_HANDLER(ZEND_OPCODE_HANDLER_ARGS)
 {
        zend_op *opline = EX(opline);
        if (opline->extended_value==ZEND_DO_FCALL_BY_NAME
-               && ARG_SHOULD_BE_SENT_BY_REF(EX(fbc), opline->op2.u.opline_num)) {
+               && ARG_MUST_BE_SENT_BY_REF(EX(fbc), opline->op2.u.opline_num)) {
                        zend_error_noreturn(E_ERROR, "Cannot pass parameter %d by reference", opline->op2.u.opline_num);
        }
        {
@@ -4338,7 +4338,7 @@ static int ZEND_SEND_VAL_SPEC_TMP_HANDLER(ZEND_OPCODE_HANDLER_ARGS)
 {
        zend_op *opline = EX(opline);
        if (opline->extended_value==ZEND_DO_FCALL_BY_NAME
-               && ARG_SHOULD_BE_SENT_BY_REF(EX(fbc), opline->op2.u.opline_num)) {
+               && ARG_MUST_BE_SENT_BY_REF(EX(fbc), opline->op2.u.opline_num)) {
                        zend_error_noreturn(E_ERROR, "Cannot pass parameter %d by reference", opline->op2.u.opline_num);
        }
        {
@@ -7439,7 +7439,7 @@ static int ZEND_SEND_VAL_SPEC_VAR_HANDLER(ZEND_OPCODE_HANDLER_ARGS)
 {
        zend_op *opline = EX(opline);
        if (opline->extended_value==ZEND_DO_FCALL_BY_NAME
-               && ARG_SHOULD_BE_SENT_BY_REF(EX(fbc), opline->op2.u.opline_num)) {
+               && ARG_MUST_BE_SENT_BY_REF(EX(fbc), opline->op2.u.opline_num)) {
                        zend_error_noreturn(E_ERROR, "Cannot pass parameter %d by reference", opline->op2.u.opline_num);
        }
        {
@@ -19980,7 +19980,7 @@ static int ZEND_SEND_VAL_SPEC_CV_HANDLER(ZEND_OPCODE_HANDLER_ARGS)
 {
        zend_op *opline = EX(opline);
        if (opline->extended_value==ZEND_DO_FCALL_BY_NAME
-               && ARG_SHOULD_BE_SENT_BY_REF(EX(fbc), opline->op2.u.opline_num)) {
+               && ARG_MUST_BE_SENT_BY_REF(EX(fbc), opline->op2.u.opline_num)) {
                        zend_error_noreturn(E_ERROR, "Cannot pass parameter %d by reference", opline->op2.u.opline_num);
        }
        {