summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBob Weinand <bobwei9@hotmail.com>2016-01-24 19:55:16 +0100
committerBob Weinand <bobwei9@hotmail.com>2016-01-24 19:55:16 +0100
commit001ce475ee158cbf020755f1f65aade28caafd73 (patch)
tree19f246fd423a01b60da9aa6e546b2ec9024858d1
parentae122bf7adf4f8b2df6b5acf316d16112587d921 (diff)
downloadphp-git-001ce475ee158cbf020755f1f65aade28caafd73.tar.gz
Fixed bug #71441 (Typehinted Generator with return in try/finally crashes)
-rw-r--r--NEWS4
-rw-r--r--Zend/tests/generators/bug71441.phpt29
-rw-r--r--Zend/zend_opcode.c8
3 files changed, 38 insertions, 3 deletions
diff --git a/NEWS b/NEWS
index 0010f31bb3..65bd694d7b 100644
--- a/NEWS
+++ b/NEWS
@@ -2,7 +2,9 @@ PHP NEWS
|||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
?? ??? 2016 PHP 7.0.4
-
+- Core:
+ . Fixed bug #71441 (Typehinted Generator with return in try/finally crashes).
+ (Bob)
04 Feb 2016 PHP 7.0.3
diff --git a/Zend/tests/generators/bug71441.phpt b/Zend/tests/generators/bug71441.phpt
new file mode 100644
index 0000000000..3a103888b0
--- /dev/null
+++ b/Zend/tests/generators/bug71441.phpt
@@ -0,0 +1,29 @@
+--TEST--
+Bug #71441 (Typehinted Generator with return in try/finally crashes)
+--FILE--
+<?php
+
+$num = 2000; /* to be sure to be in wild memory */
+$add = str_repeat("1 +", $num);
+$gen = (eval(<<<PHP
+return function (): \Generator {
+ try {
+ \$a = 1;
+ \$foo = \$a + $add \$a;
+ return yield \$foo;
+ } finally {
+ print "Ok\n";
+ }
+};
+PHP
+))();
+var_dump($gen->current());
+$gen->send("Success");
+var_dump($gen->getReturn());
+
+?>
+--EXPECT--
+int(2002)
+Ok
+string(7) "Success"
+
diff --git a/Zend/zend_opcode.c b/Zend/zend_opcode.c
index dee54ba14c..2cf32b9c0a 100644
--- a/Zend/zend_opcode.c
+++ b/Zend/zend_opcode.c
@@ -668,9 +668,13 @@ ZEND_API int pass_two(zend_op_array *op_array)
case ZEND_VERIFY_RETURN_TYPE:
if (op_array->fn_flags & ZEND_ACC_GENERATOR) {
if (opline->op1_type != IS_UNUSED) {
- (opline + 1)->op1 = opline->op1;
- (opline + 1)->op1_type = opline->op1_type;
+ zend_op *ret = opline;
+ do ret++; while (ret->opcode != ZEND_RETURN);
+
+ ret->op1 = opline->op1;
+ ret->op1_type = opline->op1_type;
}
+
MAKE_NOP(opline);
}
break;