Skip to content

Commit a262419

Browse files
committed
Merge branch 'PHP-8.4'
* PHP-8.4: Fix circumvented type check with return by ref + finally
2 parents 65b9cf1 + b3f4863 commit a262419

File tree

2 files changed

+36
-0
lines changed

2 files changed

+36
-0
lines changed

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
@@ -5699,8 +5699,20 @@ static void zend_compile_return(zend_ast *ast) /* {{{ */
56995699
expr_ast ? &expr_node : NULL, CG(active_op_array)->arg_info - 1, 0);
57005700
}
57015701

5702+
uint32_t opnum_before_finally = get_next_op_number();
5703+
57025704
zend_handle_loops_and_finally((expr_node.op_type & (IS_TMP_VAR | IS_VAR)) ? &expr_node : NULL);
57035705

5706+
/* Content of reference might have changed in finally, repeat type check. */
5707+
if (by_ref
5708+
/* Check if any opcodes were emitted since the last return type check. */
5709+
&& opnum_before_finally != get_next_op_number()
5710+
&& !is_generator
5711+
&& (CG(active_op_array)->fn_flags & ZEND_ACC_HAS_RETURN_TYPE)) {
5712+
zend_emit_return_type_check(
5713+
expr_ast ? &expr_node : NULL, CG(active_op_array)->arg_info - 1, 0);
5714+
}
5715+
57045716
opline = zend_emit_op(NULL, by_ref ? ZEND_RETURN_BY_REF : ZEND_RETURN,
57055717
&expr_node, NULL);
57065718

0 commit comments

Comments
 (0)