Skip to content

Commit d0fad34

Browse files
committed
Fix circumvented type check with return by ref + finally
Fixes GH-18736 Closes GH-19172
1 parent a96b05e commit d0fad34

File tree

3 files changed

+38
-0
lines changed

3 files changed

+38
-0
lines changed

NEWS

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@ PHP NEWS
1717
delegated Generator). (Arnaud)
1818
. Fixed bug GH-19326 (Calling Generator::throw() on a running generator with
1919
a non-Generator delegate crashes). (Arnaud)
20+
. Fixed bug GH-18736 (Circumvented type check with return by ref + finally).
21+
(ilutov)
2022

2123
- FTP:
2224
. Fix theoretical issues with hrtime() not being available. (nielsdos)

Zend/tests/gh18736.phpt

Lines changed: 24 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
--TEST--
2+
GH-18736: Circumvented type check with return by ref + finally
3+
--FILE--
4+
<?php
5+
6+
function &test(): int {
7+
$x = 0;
8+
try {
9+
return $x;
10+
} finally {
11+
$x = 'test';
12+
}
13+
}
14+
15+
try {
16+
$x = &test();
17+
var_dump($x);
18+
} catch (Error $e) {
19+
echo $e->getMessage(), "\n";
20+
}
21+
22+
?>
23+
--EXPECT--
24+
test(): Return value must be of type int, string returned

Zend/zend_compile.c

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -5201,8 +5201,20 @@ static void zend_compile_return(zend_ast *ast) /* {{{ */
52015201
expr_ast ? &expr_node : NULL, CG(active_op_array)->arg_info - 1, 0);
52025202
}
52035203

5204+
uint32_t opnum_before_finally = get_next_op_number();
5205+
52045206
zend_handle_loops_and_finally((expr_node.op_type & (IS_TMP_VAR | IS_VAR)) ? &expr_node : NULL);
52055207

5208+
/* Content of reference might have changed in finally, repeat type check. */
5209+
if (by_ref
5210+
/* Check if any opcodes were emitted since the last return type check. */
5211+
&& opnum_before_finally != get_next_op_number()
5212+
&& !is_generator
5213+
&& (CG(active_op_array)->fn_flags & ZEND_ACC_HAS_RETURN_TYPE)) {
5214+
zend_emit_return_type_check(
5215+
expr_ast ? &expr_node : NULL, CG(active_op_array)->arg_info - 1, 0);
5216+
}
5217+
52065218
opline = zend_emit_op(NULL, by_ref ? ZEND_RETURN_BY_REF : ZEND_RETURN,
52075219
&expr_node, NULL);
52085220

0 commit comments

Comments
 (0)