php-internal-docs 8.4.8
Unofficial docs for php/php-src
Loading...
Searching...
No Matches
zend_ssa.h
Go to the documentation of this file.
1/*
2 +----------------------------------------------------------------------+
3 | Zend Engine, SSA - Static Single Assignment Form |
4 +----------------------------------------------------------------------+
5 | Copyright (c) The PHP Group |
6 +----------------------------------------------------------------------+
7 | This source file is subject to version 3.01 of the PHP license, |
8 | that is bundled with this package in the file LICENSE, and is |
9 | available through the world-wide-web at the following url: |
10 | https://www.php.net/license/3_01.txt |
11 | If you did not receive a copy of the PHP license and are unable to |
12 | obtain it through the world-wide-web, please send a note to |
13 | license@php.net so we can mail you a copy immediately. |
14 +----------------------------------------------------------------------+
15 | Authors: Dmitry Stogov <dmitry@php.net> |
16 +----------------------------------------------------------------------+
17*/
18
19#ifndef ZEND_SSA_H
20#define ZEND_SSA_H
21
22#include "zend_optimizer.h"
23#include "zend_cfg.h"
24
31
40
41/* Special kind of SSA Phi function used in eSSA */
43 zend_ssa_range range; /* simple range constraint */
46 int min_ssa_var; /* ((min_var>0) ? MIN(ssa_var) : 0) + range.min */
47 int max_ssa_var; /* ((max_var>0) ? MAX(ssa_var) : 0) + range.max */
50
52 uint32_t type_mask; /* Type mask to intersect with */
53 zend_class_entry *ce; /* Class entry for instanceof constraints */
55
60
61/* SSA Phi - ssa_var = Phi(source0, source1, ...sourceN) */
64 zend_ssa_phi *next; /* next Phi in the same BB */
65 int pi; /* if >= 0 this is actually a e-SSA Pi */
66 zend_ssa_pi_constraint constraint; /* e-SSA Pi constraint */
67 int var; /* Original CV, VAR or TMP variable index */
68 int ssa_var; /* SSA variable index */
69 int block; /* current BB index */
73 int *sources; /* Array of SSA IDs that produce this var.
74 As many as this block has
75 predecessors. */
76};
77
81
93
99
106
107typedef struct _zend_ssa_var {
108 int var; /* original var number; op.var for CVs and following numbers for VARs and TMP_VARs */
109 int scc; /* strongly connected component */
110 int definition; /* opcode that defines this value */
111 int use_chain; /* uses of this value, linked through opN_use_chain */
112 zend_ssa_phi *definition_phi; /* phi that defines this value */
113 zend_ssa_phi *phi_use_chain; /* uses of this value in Phi, linked through use_chain */
114 zend_ssa_phi *sym_use_chain; /* uses of this value in Pi constraints */
115 bool no_val : 1; /* value doesn't matter (used as op1 in ZEND_ASSIGN) */
116 bool scc_entry : 1;
117 unsigned int alias : 2; /* value may be changed indirectly */
118 unsigned int escape_state : 2;
120
121typedef struct _zend_ssa_var_info {
122 uint32_t type; /* inferred type (see zend_inference.h) */
123 bool has_range : 1;
124 bool is_instanceof : 1; /* 0 - class == "ce", 1 - may be child of "ce" */
125 bool recursive : 1;
130 bool indirect_reference : 1; /* IS_INDIRECT returned by FETCH_DIM_W/FETCH_OBJ_W */
134
135typedef struct _zend_ssa {
136 zend_cfg cfg; /* control flow graph */
137 int vars_count; /* number of SSA variables */
138 int sccs; /* number of SCCs */
139 zend_ssa_block *blocks; /* array of SSA blocks */
140 zend_ssa_op *ops; /* array of SSA instructions */
141 zend_ssa_var *vars; /* use/def chain of SSA variables */
144
146
147ZEND_API zend_result zend_build_ssa(zend_arena **arena, const zend_script *script, const zend_op_array *op_array, uint32_t build_flags, zend_ssa *ssa);
149ZEND_API int zend_ssa_rename_op(const zend_op_array *op_array, const zend_op *opline, uint32_t k, uint32_t build_flags, int ssa_vars_count, zend_ssa_op *ssa_ops, int *var);
150void zend_ssa_unlink_use_chain(zend_ssa *ssa, int op, int var);
151void zend_ssa_replace_use_chain(zend_ssa *ssa, int op, int new_op, int var);
152
153void zend_ssa_remove_predecessor(zend_ssa *ssa, int from, int to);
155void zend_ssa_remove_instr(zend_ssa *ssa, zend_op *opline, zend_ssa_op *ssa_op);
157void zend_ssa_remove_uses_of_var(zend_ssa *ssa, int var_num);
158void zend_ssa_remove_block(zend_op_array *op_array, zend_ssa *ssa, int b);
159void zend_ssa_rename_var_uses(zend_ssa *ssa, int old_var, int new_var, bool update_types);
161
162static zend_always_inline void _zend_ssa_remove_def(zend_ssa_var *var)
163{
164 ZEND_ASSERT(var->definition >= 0);
165 ZEND_ASSERT(var->use_chain < 0);
166 ZEND_ASSERT(!var->phi_use_chain);
167 var->definition = -1;
168}
169
170static zend_always_inline void zend_ssa_remove_result_def(zend_ssa *ssa, zend_ssa_op *ssa_op)
171{
172 zend_ssa_var *var = &ssa->vars[ssa_op->result_def];
173 _zend_ssa_remove_def(var);
174 ssa_op->result_def = -1;
175}
176
177static zend_always_inline void zend_ssa_remove_op1_def(zend_ssa *ssa, zend_ssa_op *ssa_op)
178{
179 zend_ssa_var *var = &ssa->vars[ssa_op->op1_def];
180 _zend_ssa_remove_def(var);
181 ssa_op->op1_def = -1;
182}
183
184static zend_always_inline void zend_ssa_remove_op2_def(zend_ssa *ssa, zend_ssa_op *ssa_op)
185{
186 zend_ssa_var *var = &ssa->vars[ssa_op->op2_def];
187 _zend_ssa_remove_def(var);
188 ssa_op->op2_def = -1;
189}
190
192
193static zend_always_inline int zend_ssa_next_use(const zend_ssa_op *ssa_op, int var, int use)
194{
195 ssa_op += use;
196 if (ssa_op->op1_use == var) {
197 return ssa_op->op1_use_chain;
198 } else if (ssa_op->op2_use == var) {
199 return ssa_op->op2_use_chain;
200 } else {
201 return ssa_op->res_use_chain;
202 }
203}
204
205static zend_always_inline zend_ssa_phi* zend_ssa_next_use_phi(const zend_ssa *ssa, int var, const zend_ssa_phi *p)
206{
207 if (p->pi >= 0) {
208 return p->use_chains[0];
209 } else {
210 int j;
211 for (j = 0; j < ssa->cfg.blocks[p->block].predecessors_count; j++) {
212 if (p->sources[j] == var) {
213 return p->use_chains[j];
214 }
215 }
216 }
217 return NULL;
218}
219
220static zend_always_inline bool zend_ssa_is_no_val_use(const zend_op *opline, const zend_ssa_op *ssa_op, int var)
221{
222 if (opline->opcode == ZEND_ASSIGN
223 || opline->opcode == ZEND_UNSET_CV
224 || opline->opcode == ZEND_BIND_GLOBAL
225 || opline->opcode == ZEND_BIND_STATIC) {
226 return ssa_op->op1_use == var && ssa_op->op2_use != var;
227 }
228 if (opline->opcode == ZEND_FE_FETCH_R || opline->opcode == ZEND_FE_FETCH_RW) {
229 return ssa_op->op2_use == var && ssa_op->op1_use != var;
230 }
231 if (ssa_op->result_use == var
232 && opline->opcode != ZEND_ADD_ARRAY_ELEMENT
233 && opline->opcode != ZEND_ADD_ARRAY_UNPACK) {
234 return ssa_op->op1_use != var && ssa_op->op2_use != var;
235 }
236 return 0;
237}
238
239static zend_always_inline void zend_ssa_rename_defs_of_instr(zend_ssa *ssa, zend_ssa_op *ssa_op) {
240 /* Rename def to use if possible. Mark variable as not defined otherwise. */
241 if (ssa_op->op1_def >= 0) {
242 if (ssa_op->op1_use >= 0) {
243 zend_ssa_rename_var_uses(ssa, ssa_op->op1_def, ssa_op->op1_use, 1);
244 }
245 ssa->vars[ssa_op->op1_def].definition = -1;
246 ssa_op->op1_def = -1;
247 }
248 if (ssa_op->op2_def >= 0) {
249 if (ssa_op->op2_use >= 0) {
250 zend_ssa_rename_var_uses(ssa, ssa_op->op2_def, ssa_op->op2_use, 1);
251 }
252 ssa->vars[ssa_op->op2_def].definition = -1;
253 ssa_op->op2_def = -1;
254 }
255 if (ssa_op->result_def >= 0) {
256 if (ssa_op->result_use >= 0) {
257 zend_ssa_rename_var_uses(ssa, ssa_op->result_def, ssa_op->result_use, 1);
258 }
259 ssa->vars[ssa_op->result_def].definition = -1;
260 ssa_op->result_def = -1;
261 }
262}
263
264#define NUM_PHI_SOURCES(phi) \
265 ((phi)->pi >= 0 ? 1 : (ssa->cfg.blocks[(phi)->block].predecessors_count))
266
267/* FOREACH_USE and FOREACH_PHI_USE explicitly support "continue"
268 * and changing the use chain of the current element */
269#define FOREACH_USE(var, use) do { \
270 int _var_num = (var) - ssa->vars, next; \
271 for (use = (var)->use_chain; use >= 0; use = next) { \
272 next = zend_ssa_next_use(ssa->ops, _var_num, use);
273#define FOREACH_USE_END() \
274 } \
275} while (0)
276
277#define FOREACH_PHI_USE(var, phi) do { \
278 int _var_num = (var) - ssa->vars; \
279 zend_ssa_phi *next_phi; \
280 for (phi = (var)->phi_use_chain; phi; phi = next_phi) { \
281 next_phi = zend_ssa_next_use_phi(ssa, _var_num, phi);
282#define FOREACH_PHI_USE_END() \
283 } \
284} while (0)
285
286#define FOREACH_PHI_SOURCE(phi, source) do { \
287 zend_ssa_phi *_phi = (phi); \
288 int _i, _end = NUM_PHI_SOURCES(phi); \
289 for (_i = 0; _i < _end; _i++) { \
290 ZEND_ASSERT(_phi->sources[_i] >= 0); \
291 source = _phi->sources[_i];
292#define FOREACH_PHI_SOURCE_END() \
293 } \
294} while (0)
295
296#define FOREACH_PHI(phi) do { \
297 int _i; \
298 for (_i = 0; _i < ssa->cfg.blocks_count; _i++) { \
299 phi = ssa->blocks[_i].phis; \
300 for (; phi; phi = phi->next) {
301#define FOREACH_PHI_END() \
302 } \
303 } \
304} while (0)
305
306#define FOREACH_BLOCK(block) do { \
307 int _i; \
308 for (_i = 0; _i < ssa->cfg.blocks_count; _i++) { \
309 (block) = &ssa->cfg.blocks[_i]; \
310 if (!((block)->flags & ZEND_BB_REACHABLE)) { \
311 continue; \
312 }
313#define FOREACH_BLOCK_END() \
314 } \
315} while (0)
316
317/* Does not support "break" */
318#define FOREACH_INSTR_NUM(i) do { \
319 zend_basic_block *_block; \
320 FOREACH_BLOCK(_block) { \
321 uint32_t _end = _block->start + _block->len; \
322 for ((i) = _block->start; (i) < _end; (i)++) {
323#define FOREACH_INSTR_NUM_END() \
324 } \
325 } FOREACH_BLOCK_END(); \
326} while (0)
327
328#endif /* ZEND_SSA_H */
#define NULL
Definition gdcache.h:45
again j
foreach($dp as $el) foreach( $dp as $el) if( $pass2< 2) echo ""
char * arena
Definition php_bcmath.h:37
p
Definition session.c:1105
zend_basic_block * blocks
Definition zend_cfg.h:87
uint8_t opcode
zend_ssa_phi * phis
Definition zend_ssa.h:79
int op2_use_chain
Definition zend_ssa.h:90
int op1_use_chain
Definition zend_ssa.h:89
int res_use_chain
Definition zend_ssa.h:91
int result_use
Definition zend_ssa.h:85
int result_def
Definition zend_ssa.h:88
zend_ssa_phi * sym_use_chain
Definition zend_ssa.h:72
int * sources
Definition zend_ssa.h:73
zend_ssa_phi ** use_chains
Definition zend_ssa.h:71
zend_ssa_phi * next
Definition zend_ssa.h:64
bool has_range_constraint
Definition zend_ssa.h:70
zend_ssa_pi_constraint constraint
Definition zend_ssa.h:66
zend_ssa_negative_lat negative
Definition zend_ssa.h:48
zend_ssa_range range
Definition zend_ssa.h:43
zend_long max
Definition zend_ssa.h:27
zend_long min
Definition zend_ssa.h:26
zend_class_entry * ce
Definition zend_ssa.h:53
zend_ssa_range range
Definition zend_ssa.h:131
zend_class_entry * ce
Definition zend_ssa.h:132
unsigned int alias
Definition zend_ssa.h:117
zend_ssa_phi * sym_use_chain
Definition zend_ssa.h:114
unsigned int escape_state
Definition zend_ssa.h:118
zend_ssa_phi * definition_phi
Definition zend_ssa.h:112
zend_ssa_phi * phi_use_chain
Definition zend_ssa.h:113
int vars_count
Definition zend_ssa.h:137
zend_ssa_var * vars
Definition zend_ssa.h:141
zend_cfg cfg
Definition zend_ssa.h:136
zend_ssa_var_info * var_info
Definition zend_ssa.h:142
zend_ssa_op * ops
Definition zend_ssa.h:140
zend_ssa_block * blocks
Definition zend_ssa.h:139
zend_ssa_range_constraint range
Definition zend_ssa.h:57
zend_ssa_type_constraint type
Definition zend_ssa.h:58
struct _zend_arena zend_arena
Definition zend_arena.h:26
struct _zend_cfg zend_cfg
struct _zend_op zend_op
struct _zend_op_array zend_op_array
#define ZEND_API
int32_t zend_long
Definition zend_long.h:42
struct _zend_script zend_script
#define END_EXTERN_C()
#define zend_always_inline
#define ZEND_ASSERT(c)
#define BEGIN_EXTERN_C()
struct _zend_class_entry zend_class_entry
void zend_ssa_remove_uses_of_var(zend_ssa *ssa, int var_num)
Definition zend_ssa.c:1422
void zend_ssa_remove_instr(zend_ssa *ssa, zend_op *opline, zend_ssa_op *ssa_op)
Definition zend_ssa.c:1281
void zend_ssa_unlink_use_chain(zend_ssa *ssa, int op, int var)
Definition zend_ssa.c:1202
void zend_ssa_replace_use_chain(zend_ssa *ssa, int op, int new_op, int var)
Definition zend_ssa.c:1241
void zend_ssa_remove_phi(zend_ssa *ssa, zend_ssa_phi *phi)
Definition zend_ssa.c:1410
ZEND_API int zend_ssa_rename_op(const zend_op_array *op_array, const zend_op *opline, uint32_t k, uint32_t build_flags, int ssa_vars_count, zend_ssa_op *ssa_ops, int *var)
Definition zend_ssa.c:813
void zend_ssa_rename_var_uses(zend_ssa *ssa, int old, int new, bool update_types)
Definition zend_ssa.c:1595
void zend_ssa_remove_predecessor(zend_ssa *ssa, int from, int to)
Definition zend_ssa.c:1455
ZEND_API zend_result zend_build_ssa(zend_arena **arena, const zend_script *script, const zend_op_array *op_array, uint32_t build_flags, zend_ssa *ssa)
Definition zend_ssa.c:936
void zend_ssa_remove_block_from_cfg(zend_ssa *ssa, int i)
Definition zend_ssa.c:1530
ZEND_API void zend_ssa_compute_use_def_chains(zend_arena **arena, const zend_op_array *op_array, zend_ssa *ssa)
Definition zend_ssa.c:1084
void zend_ssa_remove_block(zend_op_array *op_array, zend_ssa *ssa, int i)
Definition zend_ssa.c:1501
void zend_ssa_remove_defs_of_instr(zend_ssa *ssa, zend_ssa_op *ssa_op)
Definition zend_ssa.c:1364
struct _zend_ssa_range zend_ssa_range
struct _zend_ssa_block zend_ssa_block
_zend_ssa_escape_state
Definition zend_ssa.h:100
@ ESCAPE_STATE_UNKNOWN
Definition zend_ssa.h:101
@ ESCAPE_STATE_GLOBAL_ESCAPE
Definition zend_ssa.h:104
@ ESCAPE_STATE_FUNCTION_ESCAPE
Definition zend_ssa.h:103
@ ESCAPE_STATE_NO_ESCAPE
Definition zend_ssa.h:102
void zend_ssa_rename_var_uses(zend_ssa *ssa, int old_var, int new_var, bool update_types)
Definition zend_ssa.c:1595
struct _zend_ssa_type_constraint zend_ssa_type_constraint
union _zend_ssa_pi_constraint zend_ssa_pi_constraint
enum _zend_ssa_alias_kind zend_ssa_alias_kind
struct _zend_ssa zend_ssa
struct _zend_ssa_var zend_ssa_var
struct _zend_ssa_phi zend_ssa_phi
Definition zend_ssa.h:62
enum _zend_ssa_escape_state zend_ssa_escape_state
_zend_ssa_alias_kind
Definition zend_ssa.h:94
@ HTTP_RESPONSE_HEADER_ALIAS
Definition zend_ssa.h:97
@ NO_ALIAS
Definition zend_ssa.h:95
@ SYMTABLE_ALIAS
Definition zend_ssa.h:96
struct _zend_ssa_op zend_ssa_op
struct _zend_ssa_var_info zend_ssa_var_info
struct _zend_ssa_range_constraint zend_ssa_range_constraint
enum _zend_ssa_negative_lat zend_ssa_negative_lat
_zend_ssa_negative_lat
Definition zend_ssa.h:32
@ NEG_USE_LT
Definition zend_ssa.h:36
@ NEG_USE_GT
Definition zend_ssa.h:37
@ NEG_NONE
Definition zend_ssa.h:33
@ NEG_INVARIANT
Definition zend_ssa.h:35
@ NEG_INIT
Definition zend_ssa.h:34
@ NEG_UNKNOWN
Definition zend_ssa.h:38
ZEND_RESULT_CODE zend_result
Definition zend_types.h:64
#define ZEND_FE_FETCH_RW
#define ZEND_FE_FETCH_R
#define ZEND_UNSET_CV
#define ZEND_ADD_ARRAY_ELEMENT
#define ZEND_BIND_GLOBAL
#define ZEND_ADD_ARRAY_UNPACK
#define ZEND_ASSIGN
#define ZEND_BIND_STATIC