mirror of
https://github.com/php/php-src.git
synced 2026-03-24 16:22:37 +01:00
The aim of this PR is twofold:
- Reduce the number of highly similar TMP|VAR handlers
- Avoid ZVAL_DEREF in most of these cases
This is achieved by guaranteeing that all zend_compile_expr() calls, as well as
all other compile calls with BP_VAR_{R,IS}, will result in a TMP variable. This
implies that the result will not contain an IS_INDIRECT or IS_REFERENCE value,
which was mostly already the case, with two exceptions:
- Calls to return-by-reference functions. Because return-by-reference functions
are quite rare, this is solved by delegating the DEREF to the RETURN_BY_REF
handler, which will examine the stack to check whether the caller expects a
VAR or TMP to understand whether the DEREF is needed. Internal functions will
also need to adjust by calling the zend_return_unwrap_ref() function.
- By-reference assignments, including both $a = &$b, as well as $a = [&$b]. When
the result of these expressions is used in a BP_VAR_R context, the reference
is unwrapped via a ZEND_QM_ASSIGN opcode beforehand. This is exceptionally
rare.
Closes GH-20628
74 lines
1.2 KiB
PHP
74 lines
1.2 KiB
PHP
--TEST--
|
|
array_map(): foreach optimization - const array
|
|
--EXTENSIONS--
|
|
opcache
|
|
--INI--
|
|
opcache.enable=1
|
|
opcache.enable_cli=1
|
|
opcache.opt_debug_level=0x20000
|
|
--FILE--
|
|
<?php
|
|
|
|
function plus1($x) {
|
|
return $x + 1;
|
|
}
|
|
|
|
$foo = array_map(plus1(...), [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
|
|
|
|
var_dump($foo);
|
|
|
|
?>
|
|
--EXPECTF--
|
|
$_main:
|
|
; (lines=%d, args=0, vars=%d, tmps=%d)
|
|
; (after optimizer)
|
|
; %s
|
|
0000 TYPE_ASSERT 131079 string("array_map") array(...)
|
|
0001 T1 = INIT_ARRAY 0 (packed) NEXT
|
|
0002 V2 = FE_RESET_R array(...) 0009
|
|
0003 T4 = FE_FETCH_R V2 T3 0009
|
|
0004 INIT_FCALL 1 %d string("plus1")
|
|
0005 SEND_VAL T3 1
|
|
0006 T3 = DO_UCALL
|
|
0007 T1 = ADD_ARRAY_ELEMENT T3 T4
|
|
0008 JMP 0003
|
|
0009 FE_FREE V2
|
|
0010 ASSIGN CV0($foo) T1
|
|
0011 INIT_FCALL 1 %d string("var_dump")
|
|
0012 SEND_VAR CV0($foo) 1
|
|
0013 DO_ICALL
|
|
0014 RETURN int(1)
|
|
LIVE RANGES:
|
|
1: 0002 - 0010 (tmp/var)
|
|
2: 0003 - 0009 (loop)
|
|
|
|
plus1:
|
|
; (lines=3, args=1, vars=1, tmps=%d)
|
|
; (after optimizer)
|
|
; %s
|
|
0000 CV0($x) = RECV 1
|
|
0001 T1 = ADD CV0($x) int(1)
|
|
0002 RETURN T1
|
|
array(10) {
|
|
[0]=>
|
|
int(2)
|
|
[1]=>
|
|
int(3)
|
|
[2]=>
|
|
int(4)
|
|
[3]=>
|
|
int(5)
|
|
[4]=>
|
|
int(6)
|
|
[5]=>
|
|
int(7)
|
|
[6]=>
|
|
int(8)
|
|
[7]=>
|
|
int(9)
|
|
[8]=>
|
|
int(10)
|
|
[9]=>
|
|
int(11)
|
|
}
|