Coverage Report

Created: 2026-06-13 07:01

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/php-src/Zend/Optimizer/zend_ssa.h
Line
Count
Source
1
/*
2
   +----------------------------------------------------------------------+
3
   | Zend Engine, SSA - Static Single Assignment Form                     |
4
   +----------------------------------------------------------------------+
5
   | Copyright © The PHP Group and Contributors.                          |
6
   +----------------------------------------------------------------------+
7
   | This source file is subject to the Modified BSD License that is      |
8
   | bundled with this package in the file LICENSE, and is available      |
9
   | through the World Wide Web at <https://www.php.net/license/>.        |
10
   |                                                                      |
11
   | SPDX-License-Identifier: BSD-3-Clause                                |
12
   +----------------------------------------------------------------------+
13
   | Authors: Dmitry Stogov <dmitry@php.net>                              |
14
   +----------------------------------------------------------------------+
15
*/
16
17
#ifndef ZEND_SSA_H
18
#define ZEND_SSA_H
19
20
#include "zend_optimizer.h"
21
#include "zend_cfg.h"
22
23
typedef struct _zend_ssa_range {
24
  zend_long              min;
25
  zend_long              max;
26
  bool              underflow;
27
  bool              overflow;
28
} zend_ssa_range;
29
30
typedef enum _zend_ssa_negative_lat {
31
  NEG_NONE      = 0,
32
  NEG_INIT      = 1,
33
  NEG_INVARIANT = 2,
34
  NEG_USE_LT    = 3,
35
  NEG_USE_GT    = 4,
36
  NEG_UNKNOWN   = 5
37
} zend_ssa_negative_lat;
38
39
/* Special kind of SSA Phi function used in eSSA */
40
typedef struct _zend_ssa_range_constraint {
41
  zend_ssa_range         range;       /* simple range constraint */
42
  int                    min_var;
43
  int                    max_var;
44
  int                    min_ssa_var; /* ((min_var>0) ? MIN(ssa_var) : 0) + range.min */
45
  int                    max_ssa_var; /* ((max_var>0) ? MAX(ssa_var) : 0) + range.max */
46
  zend_ssa_negative_lat  negative;
47
} zend_ssa_range_constraint;
48
49
typedef struct _zend_ssa_type_constraint {
50
  uint32_t               type_mask;   /* Type mask to intersect with */
51
  zend_class_entry      *ce;          /* Class entry for instanceof constraints */
52
} zend_ssa_type_constraint;
53
54
typedef union _zend_ssa_pi_constraint {
55
  zend_ssa_range_constraint range;
56
  zend_ssa_type_constraint type;
57
} zend_ssa_pi_constraint;
58
59
/* SSA Phi - ssa_var = Phi(source0, source1, ...sourceN) */
60
typedef struct _zend_ssa_phi zend_ssa_phi;
61
struct _zend_ssa_phi {
62
  zend_ssa_phi          *next;          /* next Phi in the same BB */
63
  int                    pi;            /* if >= 0 this is actually a e-SSA Pi */
64
  zend_ssa_pi_constraint constraint;    /* e-SSA Pi constraint */
65
  int                    var;           /* Original CV, VAR or TMP variable index */
66
  int                    ssa_var;       /* SSA variable index */
67
  uint32_t               block;         /* current BB index */
68
  bool                   has_range_constraint;
69
  zend_ssa_phi         **use_chains;
70
  zend_ssa_phi          *sym_use_chain;
71
  int                   *sources;       /* Array of SSA IDs that produce this var.
72
                           As many as this block has
73
                           predecessors.  */
74
};
75
76
typedef struct _zend_ssa_block {
77
  zend_ssa_phi          *phis;
78
} zend_ssa_block;
79
80
typedef struct _zend_ssa_op {
81
  int                    op1_use;
82
  int                    op2_use;
83
  int                    result_use;
84
  int                    op1_def;
85
  int                    op2_def;
86
  int                    result_def;
87
  int                    op1_use_chain;
88
  int                    op2_use_chain;
89
  int                    res_use_chain;
90
} zend_ssa_op;
91
92
typedef enum _zend_ssa_alias_kind {
93
  NO_ALIAS,
94
  SYMTABLE_ALIAS,
95
  HTTP_RESPONSE_HEADER_ALIAS
96
} zend_ssa_alias_kind;
97
98
typedef enum _zend_ssa_escape_state {
99
  ESCAPE_STATE_UNKNOWN,
100
  ESCAPE_STATE_NO_ESCAPE,
101
  ESCAPE_STATE_FUNCTION_ESCAPE,
102
  ESCAPE_STATE_GLOBAL_ESCAPE
103
} zend_ssa_escape_state;
104
105
typedef struct _zend_ssa_var {
106
  int                    var;            /* original var number; op.var for CVs and following numbers for VARs and TMP_VARs */
107
  int                    scc;            /* strongly connected component */
108
  int                    definition;     /* opcode that defines this value */
109
  int                    use_chain;      /* uses of this value, linked through opN_use_chain */
110
  zend_ssa_phi          *definition_phi; /* phi that defines this value */
111
  zend_ssa_phi          *phi_use_chain;  /* uses of this value in Phi, linked through use_chain */
112
  zend_ssa_phi          *sym_use_chain;  /* uses of this value in Pi constraints */
113
  bool                   no_val : 1;     /* value doesn't matter (used as op1 in ZEND_ASSIGN) */
114
  bool                   scc_entry : 1;
115
  unsigned int           alias : 2;  /* value may be changed indirectly */
116
  unsigned int           escape_state : 2;
117
} zend_ssa_var;
118
119
typedef struct _zend_ssa_var_info {
120
  uint32_t               type; /* inferred type (see zend_inference.h) */
121
  bool                   has_range : 1;
122
  bool                   is_instanceof : 1; /* 0 - class == "ce", 1 - may be child of "ce" */
123
  bool                   recursive : 1;
124
  bool                   use_as_double : 1;
125
  bool                   delayed_fetch_this : 1;
126
  bool                   avoid_refcounting : 1;
127
  bool                   guarded_reference : 1;
128
  bool                   indirect_reference : 1; /* IS_INDIRECT returned by FETCH_DIM_W/FETCH_OBJ_W */
129
  zend_ssa_range         range;
130
  zend_class_entry      *ce;
131
} zend_ssa_var_info;
132
133
typedef struct _zend_ssa {
134
  zend_cfg               cfg;            /* control flow graph             */
135
  int                    vars_count;     /* number of SSA variables        */
136
  int                    sccs;           /* number of SCCs                 */
137
  zend_ssa_block        *blocks;         /* array of SSA blocks            */
138
  zend_ssa_op           *ops;            /* array of SSA instructions      */
139
  zend_ssa_var          *vars;           /* use/def chain of SSA variables */
140
  zend_ssa_var_info     *var_info;
141
} zend_ssa;
142
143
BEGIN_EXTERN_C()
144
145
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);
146
ZEND_API void zend_ssa_compute_use_def_chains(zend_arena **arena, const zend_op_array *op_array, zend_ssa *ssa);
147
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);
148
void zend_ssa_unlink_use_chain(const zend_ssa *ssa, int op, int var);
149
void zend_ssa_replace_use_chain(const zend_ssa *ssa, int op, int new_op, int var);
150
151
void zend_ssa_remove_predecessor(zend_ssa *ssa, int from, int to);
152
void zend_ssa_remove_defs_of_instr(zend_ssa *ssa, zend_ssa_op *ssa_op);
153
void zend_ssa_remove_instr(const zend_ssa *ssa, zend_op *opline, zend_ssa_op *ssa_op);
154
void zend_ssa_remove_phi(const zend_ssa *ssa, zend_ssa_phi *phi);
155
void zend_ssa_remove_uses_of_var(const zend_ssa *ssa, int var_num);
156
void zend_ssa_remove_block(const zend_op_array *op_array, zend_ssa *ssa, uint32_t b);
157
void zend_ssa_rename_var_uses(zend_ssa *ssa, int old_var, int new_var, bool update_types);
158
void zend_ssa_remove_block_from_cfg(zend_ssa *ssa, int b);
159
160
static zend_always_inline void _zend_ssa_remove_def(zend_ssa_var *var)
161
13.1k
{
162
13.1k
  ZEND_ASSERT(var->definition >= 0);
163
13.1k
  ZEND_ASSERT(var->use_chain < 0);
164
13.1k
  ZEND_ASSERT(!var->phi_use_chain);
165
13.1k
  var->definition = -1;
166
13.1k
}
Unexecuted instantiation: zend_persist.c:_zend_ssa_remove_def
Unexecuted instantiation: ZendAccelerator.c:_zend_ssa_remove_def
Unexecuted instantiation: zend_jit_vm_helpers.c:_zend_ssa_remove_def
Unexecuted instantiation: zend_jit.c:_zend_ssa_remove_def
Unexecuted instantiation: block_pass.c:_zend_ssa_remove_def
Unexecuted instantiation: compact_literals.c:_zend_ssa_remove_def
Unexecuted instantiation: compact_vars.c:_zend_ssa_remove_def
Unexecuted instantiation: dce.c:_zend_ssa_remove_def
dfa_pass.c:_zend_ssa_remove_def
Line
Count
Source
161
2.55k
{
162
2.55k
  ZEND_ASSERT(var->definition >= 0);
163
2.55k
  ZEND_ASSERT(var->use_chain < 0);
164
2.55k
  ZEND_ASSERT(!var->phi_use_chain);
165
2.55k
  var->definition = -1;
166
2.55k
}
Unexecuted instantiation: escape_analysis.c:_zend_ssa_remove_def
Unexecuted instantiation: nop_removal.c:_zend_ssa_remove_def
Unexecuted instantiation: optimize_func_calls.c:_zend_ssa_remove_def
Unexecuted instantiation: optimize_temp_vars_5.c:_zend_ssa_remove_def
Unexecuted instantiation: pass1.c:_zend_ssa_remove_def
Unexecuted instantiation: pass3.c:_zend_ssa_remove_def
sccp.c:_zend_ssa_remove_def
Line
Count
Source
161
3.57k
{
162
3.57k
  ZEND_ASSERT(var->definition >= 0);
163
3.57k
  ZEND_ASSERT(var->use_chain < 0);
164
3.57k
  ZEND_ASSERT(!var->phi_use_chain);
165
3.57k
  var->definition = -1;
166
3.57k
}
Unexecuted instantiation: scdf.c:_zend_ssa_remove_def
Unexecuted instantiation: zend_call_graph.c:_zend_ssa_remove_def
Unexecuted instantiation: zend_cfg.c:_zend_ssa_remove_def
Unexecuted instantiation: zend_dump.c:_zend_ssa_remove_def
Unexecuted instantiation: zend_func_info.c:_zend_ssa_remove_def
Unexecuted instantiation: zend_inference.c:_zend_ssa_remove_def
Unexecuted instantiation: zend_optimizer.c:_zend_ssa_remove_def
zend_ssa.c:_zend_ssa_remove_def
Line
Count
Source
161
7.01k
{
162
7.01k
  ZEND_ASSERT(var->definition >= 0);
163
7.01k
  ZEND_ASSERT(var->use_chain < 0);
164
7.01k
  ZEND_ASSERT(!var->phi_use_chain);
165
7.01k
  var->definition = -1;
166
7.01k
}
Unexecuted instantiation: zend_execute.c:_zend_ssa_remove_def
167
168
static zend_always_inline void zend_ssa_remove_result_def(zend_ssa *ssa, zend_ssa_op *ssa_op)
169
12.5k
{
170
12.5k
  zend_ssa_var *var = &ssa->vars[ssa_op->result_def];
171
12.5k
  _zend_ssa_remove_def(var);
172
12.5k
  ssa_op->result_def = -1;
173
12.5k
}
Unexecuted instantiation: zend_persist.c:zend_ssa_remove_result_def
Unexecuted instantiation: ZendAccelerator.c:zend_ssa_remove_result_def
Unexecuted instantiation: zend_jit_vm_helpers.c:zend_ssa_remove_result_def
Unexecuted instantiation: zend_jit.c:zend_ssa_remove_result_def
Unexecuted instantiation: block_pass.c:zend_ssa_remove_result_def
Unexecuted instantiation: compact_literals.c:zend_ssa_remove_result_def
Unexecuted instantiation: compact_vars.c:zend_ssa_remove_result_def
Unexecuted instantiation: dce.c:zend_ssa_remove_result_def
dfa_pass.c:zend_ssa_remove_result_def
Line
Count
Source
169
2.55k
{
170
2.55k
  zend_ssa_var *var = &ssa->vars[ssa_op->result_def];
171
2.55k
  _zend_ssa_remove_def(var);
172
2.55k
  ssa_op->result_def = -1;
173
2.55k
}
Unexecuted instantiation: escape_analysis.c:zend_ssa_remove_result_def
Unexecuted instantiation: nop_removal.c:zend_ssa_remove_result_def
Unexecuted instantiation: optimize_func_calls.c:zend_ssa_remove_result_def
Unexecuted instantiation: optimize_temp_vars_5.c:zend_ssa_remove_result_def
Unexecuted instantiation: pass1.c:zend_ssa_remove_result_def
Unexecuted instantiation: pass3.c:zend_ssa_remove_result_def
sccp.c:zend_ssa_remove_result_def
Line
Count
Source
169
3.53k
{
170
3.53k
  zend_ssa_var *var = &ssa->vars[ssa_op->result_def];
171
3.53k
  _zend_ssa_remove_def(var);
172
3.53k
  ssa_op->result_def = -1;
173
3.53k
}
Unexecuted instantiation: scdf.c:zend_ssa_remove_result_def
Unexecuted instantiation: zend_call_graph.c:zend_ssa_remove_result_def
Unexecuted instantiation: zend_cfg.c:zend_ssa_remove_result_def
Unexecuted instantiation: zend_dump.c:zend_ssa_remove_result_def
Unexecuted instantiation: zend_func_info.c:zend_ssa_remove_result_def
Unexecuted instantiation: zend_inference.c:zend_ssa_remove_result_def
Unexecuted instantiation: zend_optimizer.c:zend_ssa_remove_result_def
zend_ssa.c:zend_ssa_remove_result_def
Line
Count
Source
169
6.41k
{
170
6.41k
  zend_ssa_var *var = &ssa->vars[ssa_op->result_def];
171
6.41k
  _zend_ssa_remove_def(var);
172
6.41k
  ssa_op->result_def = -1;
173
6.41k
}
Unexecuted instantiation: zend_execute.c:zend_ssa_remove_result_def
174
175
static zend_always_inline void zend_ssa_remove_op1_def(zend_ssa *ssa, zend_ssa_op *ssa_op)
176
612
{
177
612
  zend_ssa_var *var = &ssa->vars[ssa_op->op1_def];
178
612
  _zend_ssa_remove_def(var);
179
612
  ssa_op->op1_def = -1;
180
612
}
Unexecuted instantiation: zend_persist.c:zend_ssa_remove_op1_def
Unexecuted instantiation: ZendAccelerator.c:zend_ssa_remove_op1_def
Unexecuted instantiation: zend_jit_vm_helpers.c:zend_ssa_remove_op1_def
Unexecuted instantiation: zend_jit.c:zend_ssa_remove_op1_def
Unexecuted instantiation: block_pass.c:zend_ssa_remove_op1_def
Unexecuted instantiation: compact_literals.c:zend_ssa_remove_op1_def
Unexecuted instantiation: compact_vars.c:zend_ssa_remove_op1_def
Unexecuted instantiation: dce.c:zend_ssa_remove_op1_def
Unexecuted instantiation: dfa_pass.c:zend_ssa_remove_op1_def
Unexecuted instantiation: escape_analysis.c:zend_ssa_remove_op1_def
Unexecuted instantiation: nop_removal.c:zend_ssa_remove_op1_def
Unexecuted instantiation: optimize_func_calls.c:zend_ssa_remove_op1_def
Unexecuted instantiation: optimize_temp_vars_5.c:zend_ssa_remove_op1_def
Unexecuted instantiation: pass1.c:zend_ssa_remove_op1_def
Unexecuted instantiation: pass3.c:zend_ssa_remove_op1_def
sccp.c:zend_ssa_remove_op1_def
Line
Count
Source
176
42
{
177
42
  zend_ssa_var *var = &ssa->vars[ssa_op->op1_def];
178
42
  _zend_ssa_remove_def(var);
179
42
  ssa_op->op1_def = -1;
180
42
}
Unexecuted instantiation: scdf.c:zend_ssa_remove_op1_def
Unexecuted instantiation: zend_call_graph.c:zend_ssa_remove_op1_def
Unexecuted instantiation: zend_cfg.c:zend_ssa_remove_op1_def
Unexecuted instantiation: zend_dump.c:zend_ssa_remove_op1_def
Unexecuted instantiation: zend_func_info.c:zend_ssa_remove_op1_def
Unexecuted instantiation: zend_inference.c:zend_ssa_remove_op1_def
Unexecuted instantiation: zend_optimizer.c:zend_ssa_remove_op1_def
zend_ssa.c:zend_ssa_remove_op1_def
Line
Count
Source
176
570
{
177
570
  zend_ssa_var *var = &ssa->vars[ssa_op->op1_def];
178
570
  _zend_ssa_remove_def(var);
179
570
  ssa_op->op1_def = -1;
180
570
}
Unexecuted instantiation: zend_execute.c:zend_ssa_remove_op1_def
181
182
static zend_always_inline void zend_ssa_remove_op2_def(zend_ssa *ssa, zend_ssa_op *ssa_op)
183
32
{
184
32
  zend_ssa_var *var = &ssa->vars[ssa_op->op2_def];
185
32
  _zend_ssa_remove_def(var);
186
32
  ssa_op->op2_def = -1;
187
32
}
Unexecuted instantiation: zend_persist.c:zend_ssa_remove_op2_def
Unexecuted instantiation: ZendAccelerator.c:zend_ssa_remove_op2_def
Unexecuted instantiation: zend_jit_vm_helpers.c:zend_ssa_remove_op2_def
Unexecuted instantiation: zend_jit.c:zend_ssa_remove_op2_def
Unexecuted instantiation: block_pass.c:zend_ssa_remove_op2_def
Unexecuted instantiation: compact_literals.c:zend_ssa_remove_op2_def
Unexecuted instantiation: compact_vars.c:zend_ssa_remove_op2_def
Unexecuted instantiation: dce.c:zend_ssa_remove_op2_def
Unexecuted instantiation: dfa_pass.c:zend_ssa_remove_op2_def
Unexecuted instantiation: escape_analysis.c:zend_ssa_remove_op2_def
Unexecuted instantiation: nop_removal.c:zend_ssa_remove_op2_def
Unexecuted instantiation: optimize_func_calls.c:zend_ssa_remove_op2_def
Unexecuted instantiation: optimize_temp_vars_5.c:zend_ssa_remove_op2_def
Unexecuted instantiation: pass1.c:zend_ssa_remove_op2_def
Unexecuted instantiation: pass3.c:zend_ssa_remove_op2_def
Unexecuted instantiation: sccp.c:zend_ssa_remove_op2_def
Unexecuted instantiation: scdf.c:zend_ssa_remove_op2_def
Unexecuted instantiation: zend_call_graph.c:zend_ssa_remove_op2_def
Unexecuted instantiation: zend_cfg.c:zend_ssa_remove_op2_def
Unexecuted instantiation: zend_dump.c:zend_ssa_remove_op2_def
Unexecuted instantiation: zend_func_info.c:zend_ssa_remove_op2_def
Unexecuted instantiation: zend_inference.c:zend_ssa_remove_op2_def
Unexecuted instantiation: zend_optimizer.c:zend_ssa_remove_op2_def
zend_ssa.c:zend_ssa_remove_op2_def
Line
Count
Source
183
32
{
184
32
  zend_ssa_var *var = &ssa->vars[ssa_op->op2_def];
185
32
  _zend_ssa_remove_def(var);
186
32
  ssa_op->op2_def = -1;
187
32
}
Unexecuted instantiation: zend_execute.c:zend_ssa_remove_op2_def
188
189
END_EXTERN_C()
190
191
static zend_always_inline int zend_ssa_next_use(const zend_ssa_op *ssa_op, int var, int use)
192
38.4M
{
193
38.4M
  ssa_op += use;
194
38.4M
  if (ssa_op->op1_use == var) {
195
29.7M
    return ssa_op->op1_use_chain;
196
29.7M
  } else if (ssa_op->op2_use == var) {
197
8.26M
    return ssa_op->op2_use_chain;
198
8.26M
  } else {
199
408k
    return ssa_op->res_use_chain;
200
408k
  }
201
38.4M
}
Unexecuted instantiation: zend_persist.c:zend_ssa_next_use
Unexecuted instantiation: ZendAccelerator.c:zend_ssa_next_use
Unexecuted instantiation: zend_jit_vm_helpers.c:zend_ssa_next_use
Unexecuted instantiation: zend_jit.c:zend_ssa_next_use
Unexecuted instantiation: block_pass.c:zend_ssa_next_use
Unexecuted instantiation: compact_literals.c:zend_ssa_next_use
Unexecuted instantiation: compact_vars.c:zend_ssa_next_use
Unexecuted instantiation: dce.c:zend_ssa_next_use
dfa_pass.c:zend_ssa_next_use
Line
Count
Source
192
33.6M
{
193
33.6M
  ssa_op += use;
194
33.6M
  if (ssa_op->op1_use == var) {
195
26.1M
    return ssa_op->op1_use_chain;
196
26.1M
  } else if (ssa_op->op2_use == var) {
197
7.19M
    return ssa_op->op2_use_chain;
198
7.19M
  } else {
199
322k
    return ssa_op->res_use_chain;
200
322k
  }
201
33.6M
}
escape_analysis.c:zend_ssa_next_use
Line
Count
Source
192
33.4k
{
193
33.4k
  ssa_op += use;
194
33.4k
  if (ssa_op->op1_use == var) {
195
21.5k
    return ssa_op->op1_use_chain;
196
21.5k
  } else if (ssa_op->op2_use == var) {
197
5.28k
    return ssa_op->op2_use_chain;
198
6.62k
  } else {
199
6.62k
    return ssa_op->res_use_chain;
200
6.62k
  }
201
33.4k
}
Unexecuted instantiation: nop_removal.c:zend_ssa_next_use
Unexecuted instantiation: optimize_func_calls.c:zend_ssa_next_use
Unexecuted instantiation: optimize_temp_vars_5.c:zend_ssa_next_use
Unexecuted instantiation: pass1.c:zend_ssa_next_use
Unexecuted instantiation: pass3.c:zend_ssa_next_use
sccp.c:zend_ssa_next_use
Line
Count
Source
192
1.05M
{
193
1.05M
  ssa_op += use;
194
1.05M
  if (ssa_op->op1_use == var) {
195
823k
    return ssa_op->op1_use_chain;
196
823k
  } else if (ssa_op->op2_use == var) {
197
204k
    return ssa_op->op2_use_chain;
198
204k
  } else {
199
31.5k
    return ssa_op->res_use_chain;
200
31.5k
  }
201
1.05M
}
Unexecuted instantiation: scdf.c:zend_ssa_next_use
Unexecuted instantiation: zend_call_graph.c:zend_ssa_next_use
Unexecuted instantiation: zend_cfg.c:zend_ssa_next_use
Unexecuted instantiation: zend_dump.c:zend_ssa_next_use
Unexecuted instantiation: zend_func_info.c:zend_ssa_next_use
zend_inference.c:zend_ssa_next_use
Line
Count
Source
192
3.63M
{
193
3.63M
  ssa_op += use;
194
3.63M
  if (ssa_op->op1_use == var) {
195
2.74M
    return ssa_op->op1_use_chain;
196
2.74M
  } else if (ssa_op->op2_use == var) {
197
847k
    return ssa_op->op2_use_chain;
198
847k
  } else {
199
47.9k
    return ssa_op->res_use_chain;
200
47.9k
  }
201
3.63M
}
Unexecuted instantiation: zend_optimizer.c:zend_ssa_next_use
zend_ssa.c:zend_ssa_next_use
Line
Count
Source
192
56.5k
{
193
56.5k
  ssa_op += use;
194
56.5k
  if (ssa_op->op1_use == var) {
195
46.9k
    return ssa_op->op1_use_chain;
196
46.9k
  } else if (ssa_op->op2_use == var) {
197
9.40k
    return ssa_op->op2_use_chain;
198
9.40k
  } else {
199
199
    return ssa_op->res_use_chain;
200
199
  }
201
56.5k
}
Unexecuted instantiation: zend_execute.c:zend_ssa_next_use
202
203
static zend_always_inline zend_ssa_phi* zend_ssa_next_use_phi(const zend_ssa *ssa, int var, const zend_ssa_phi *p)
204
6.30M
{
205
6.30M
  if (p->pi >= 0) {
206
988k
    return p->use_chains[0];
207
5.31M
  } else {
208
8.11M
    for (uint32_t j = 0; j < ssa->cfg.blocks[p->block].predecessors_count; j++) {
209
8.11M
      if (p->sources[j] == var) {
210
5.31M
        return p->use_chains[j];
211
5.31M
      }
212
8.11M
    }
213
5.31M
  }
214
5
  return NULL;
215
6.30M
}
Unexecuted instantiation: zend_persist.c:zend_ssa_next_use_phi
Unexecuted instantiation: ZendAccelerator.c:zend_ssa_next_use_phi
Unexecuted instantiation: zend_jit_vm_helpers.c:zend_ssa_next_use_phi
Unexecuted instantiation: zend_jit.c:zend_ssa_next_use_phi
Unexecuted instantiation: block_pass.c:zend_ssa_next_use_phi
Unexecuted instantiation: compact_literals.c:zend_ssa_next_use_phi
Unexecuted instantiation: compact_vars.c:zend_ssa_next_use_phi
Unexecuted instantiation: dce.c:zend_ssa_next_use_phi
dfa_pass.c:zend_ssa_next_use_phi
Line
Count
Source
204
5.11M
{
205
5.11M
  if (p->pi >= 0) {
206
725k
    return p->use_chains[0];
207
4.39M
  } else {
208
6.66M
    for (uint32_t j = 0; j < ssa->cfg.blocks[p->block].predecessors_count; j++) {
209
6.66M
      if (p->sources[j] == var) {
210
4.39M
        return p->use_chains[j];
211
4.39M
      }
212
6.66M
    }
213
4.39M
  }
214
0
  return NULL;
215
5.11M
}
Unexecuted instantiation: escape_analysis.c:zend_ssa_next_use_phi
Unexecuted instantiation: nop_removal.c:zend_ssa_next_use_phi
Unexecuted instantiation: optimize_func_calls.c:zend_ssa_next_use_phi
Unexecuted instantiation: optimize_temp_vars_5.c:zend_ssa_next_use_phi
Unexecuted instantiation: pass1.c:zend_ssa_next_use_phi
Unexecuted instantiation: pass3.c:zend_ssa_next_use_phi
sccp.c:zend_ssa_next_use_phi
Line
Count
Source
204
230k
{
205
230k
  if (p->pi >= 0) {
206
42.6k
    return p->use_chains[0];
207
187k
  } else {
208
298k
    for (uint32_t j = 0; j < ssa->cfg.blocks[p->block].predecessors_count; j++) {
209
298k
      if (p->sources[j] == var) {
210
187k
        return p->use_chains[j];
211
187k
      }
212
298k
    }
213
187k
  }
214
0
  return NULL;
215
230k
}
Unexecuted instantiation: scdf.c:zend_ssa_next_use_phi
Unexecuted instantiation: zend_call_graph.c:zend_ssa_next_use_phi
Unexecuted instantiation: zend_cfg.c:zend_ssa_next_use_phi
Unexecuted instantiation: zend_dump.c:zend_ssa_next_use_phi
Unexecuted instantiation: zend_func_info.c:zend_ssa_next_use_phi
zend_inference.c:zend_ssa_next_use_phi
Line
Count
Source
204
887k
{
205
887k
  if (p->pi >= 0) {
206
184k
    return p->use_chains[0];
207
703k
  } else {
208
1.10M
    for (uint32_t j = 0; j < ssa->cfg.blocks[p->block].predecessors_count; j++) {
209
1.10M
      if (p->sources[j] == var) {
210
703k
        return p->use_chains[j];
211
703k
      }
212
1.10M
    }
213
703k
  }
214
0
  return NULL;
215
887k
}
Unexecuted instantiation: zend_optimizer.c:zend_ssa_next_use_phi
zend_ssa.c:zend_ssa_next_use_phi
Line
Count
Source
204
62.5k
{
205
62.5k
  if (p->pi >= 0) {
206
35.8k
    return p->use_chains[0];
207
35.8k
  } else {
208
39.8k
    for (uint32_t j = 0; j < ssa->cfg.blocks[p->block].predecessors_count; j++) {
209
39.8k
      if (p->sources[j] == var) {
210
26.6k
        return p->use_chains[j];
211
26.6k
      }
212
39.8k
    }
213
26.7k
  }
214
5
  return NULL;
215
62.5k
}
Unexecuted instantiation: zend_execute.c:zend_ssa_next_use_phi
216
217
static zend_always_inline bool zend_ssa_is_no_val_use(const zend_op *opline, const zend_ssa_op *ssa_op, int var)
218
1.38M
{
219
1.38M
  if (opline->opcode == ZEND_ASSIGN
220
1.24M
       || opline->opcode == ZEND_UNSET_CV
221
1.24M
       || opline->opcode == ZEND_BIND_GLOBAL
222
1.23M
       || opline->opcode == ZEND_BIND_STATIC) {
223
158k
    return ssa_op->op1_use == var && ssa_op->op2_use != var;
224
158k
  }
225
1.22M
  if (opline->opcode == ZEND_FE_FETCH_R || opline->opcode == ZEND_FE_FETCH_RW) {
226
9.90k
    return ssa_op->op2_use == var && ssa_op->op1_use != var;
227
9.90k
  }
228
1.21M
  if (ssa_op->result_use == var
229
773
      && opline->opcode != ZEND_ADD_ARRAY_ELEMENT
230
10
      && opline->opcode != ZEND_ADD_ARRAY_UNPACK) {
231
0
    return ssa_op->op1_use != var && ssa_op->op2_use != var;
232
0
  }
233
1.21M
  return 0;
234
1.21M
}
Unexecuted instantiation: zend_persist.c:zend_ssa_is_no_val_use
Unexecuted instantiation: ZendAccelerator.c:zend_ssa_is_no_val_use
Unexecuted instantiation: zend_jit_vm_helpers.c:zend_ssa_is_no_val_use
Unexecuted instantiation: zend_jit.c:zend_ssa_is_no_val_use
Unexecuted instantiation: block_pass.c:zend_ssa_is_no_val_use
Unexecuted instantiation: compact_literals.c:zend_ssa_is_no_val_use
Unexecuted instantiation: compact_vars.c:zend_ssa_is_no_val_use
dce.c:zend_ssa_is_no_val_use
Line
Count
Source
218
1.06M
{
219
1.06M
  if (opline->opcode == ZEND_ASSIGN
220
968k
       || opline->opcode == ZEND_UNSET_CV
221
966k
       || opline->opcode == ZEND_BIND_GLOBAL
222
965k
       || opline->opcode == ZEND_BIND_STATIC) {
223
109k
    return ssa_op->op1_use == var && ssa_op->op2_use != var;
224
109k
  }
225
957k
  if (opline->opcode == ZEND_FE_FETCH_R || opline->opcode == ZEND_FE_FETCH_RW) {
226
7.48k
    return ssa_op->op2_use == var && ssa_op->op1_use != var;
227
7.48k
  }
228
949k
  if (ssa_op->result_use == var
229
0
      && opline->opcode != ZEND_ADD_ARRAY_ELEMENT
230
0
      && opline->opcode != ZEND_ADD_ARRAY_UNPACK) {
231
0
    return ssa_op->op1_use != var && ssa_op->op2_use != var;
232
0
  }
233
949k
  return 0;
234
949k
}
Unexecuted instantiation: dfa_pass.c:zend_ssa_is_no_val_use
Unexecuted instantiation: escape_analysis.c:zend_ssa_is_no_val_use
Unexecuted instantiation: nop_removal.c:zend_ssa_is_no_val_use
Unexecuted instantiation: optimize_func_calls.c:zend_ssa_is_no_val_use
Unexecuted instantiation: optimize_temp_vars_5.c:zend_ssa_is_no_val_use
Unexecuted instantiation: pass1.c:zend_ssa_is_no_val_use
Unexecuted instantiation: pass3.c:zend_ssa_is_no_val_use
Unexecuted instantiation: sccp.c:zend_ssa_is_no_val_use
Unexecuted instantiation: scdf.c:zend_ssa_is_no_val_use
Unexecuted instantiation: zend_call_graph.c:zend_ssa_is_no_val_use
Unexecuted instantiation: zend_cfg.c:zend_ssa_is_no_val_use
Unexecuted instantiation: zend_dump.c:zend_ssa_is_no_val_use
Unexecuted instantiation: zend_func_info.c:zend_ssa_is_no_val_use
zend_inference.c:zend_ssa_is_no_val_use
Line
Count
Source
218
314k
{
219
314k
  if (opline->opcode == ZEND_ASSIGN
220
276k
       || opline->opcode == ZEND_UNSET_CV
221
275k
       || opline->opcode == ZEND_BIND_GLOBAL
222
274k
       || opline->opcode == ZEND_BIND_STATIC) {
223
48.4k
    return ssa_op->op1_use == var && ssa_op->op2_use != var;
224
48.4k
  }
225
266k
  if (opline->opcode == ZEND_FE_FETCH_R || opline->opcode == ZEND_FE_FETCH_RW) {
226
2.41k
    return ssa_op->op2_use == var && ssa_op->op1_use != var;
227
2.41k
  }
228
264k
  if (ssa_op->result_use == var
229
773
      && opline->opcode != ZEND_ADD_ARRAY_ELEMENT
230
10
      && opline->opcode != ZEND_ADD_ARRAY_UNPACK) {
231
0
    return ssa_op->op1_use != var && ssa_op->op2_use != var;
232
0
  }
233
264k
  return 0;
234
264k
}
Unexecuted instantiation: zend_optimizer.c:zend_ssa_is_no_val_use
Unexecuted instantiation: zend_ssa.c:zend_ssa_is_no_val_use
Unexecuted instantiation: zend_execute.c:zend_ssa_is_no_val_use
235
236
21.5k
static zend_always_inline void zend_ssa_rename_defs_of_instr(zend_ssa *ssa, zend_ssa_op *ssa_op) {
237
  /* Rename def to use if possible. Mark variable as not defined otherwise. */
238
21.5k
  if (ssa_op->op1_def >= 0) {
239
5.83k
    if (ssa_op->op1_use >= 0) {
240
5.83k
      zend_ssa_rename_var_uses(ssa, ssa_op->op1_def, ssa_op->op1_use, true);
241
5.83k
    }
242
5.83k
    ssa->vars[ssa_op->op1_def].definition = -1;
243
5.83k
    ssa_op->op1_def = -1;
244
5.83k
  }
245
21.5k
  if (ssa_op->op2_def >= 0) {
246
0
    if (ssa_op->op2_use >= 0) {
247
0
      zend_ssa_rename_var_uses(ssa, ssa_op->op2_def, ssa_op->op2_use, true);
248
0
    }
249
0
    ssa->vars[ssa_op->op2_def].definition = -1;
250
0
    ssa_op->op2_def = -1;
251
0
  }
252
21.5k
  if (ssa_op->result_def >= 0) {
253
7.39k
    if (ssa_op->result_use >= 0) {
254
115
      zend_ssa_rename_var_uses(ssa, ssa_op->result_def, ssa_op->result_use, true);
255
115
    }
256
7.39k
    ssa->vars[ssa_op->result_def].definition = -1;
257
7.39k
    ssa_op->result_def = -1;
258
7.39k
  }
259
21.5k
}
Unexecuted instantiation: zend_persist.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: ZendAccelerator.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: zend_jit_vm_helpers.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: zend_jit.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: block_pass.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: compact_literals.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: compact_vars.c:zend_ssa_rename_defs_of_instr
dce.c:zend_ssa_rename_defs_of_instr
Line
Count
Source
236
21.5k
static zend_always_inline void zend_ssa_rename_defs_of_instr(zend_ssa *ssa, zend_ssa_op *ssa_op) {
237
  /* Rename def to use if possible. Mark variable as not defined otherwise. */
238
21.5k
  if (ssa_op->op1_def >= 0) {
239
5.83k
    if (ssa_op->op1_use >= 0) {
240
5.83k
      zend_ssa_rename_var_uses(ssa, ssa_op->op1_def, ssa_op->op1_use, true);
241
5.83k
    }
242
5.83k
    ssa->vars[ssa_op->op1_def].definition = -1;
243
5.83k
    ssa_op->op1_def = -1;
244
5.83k
  }
245
21.5k
  if (ssa_op->op2_def >= 0) {
246
0
    if (ssa_op->op2_use >= 0) {
247
0
      zend_ssa_rename_var_uses(ssa, ssa_op->op2_def, ssa_op->op2_use, true);
248
0
    }
249
0
    ssa->vars[ssa_op->op2_def].definition = -1;
250
0
    ssa_op->op2_def = -1;
251
0
  }
252
21.5k
  if (ssa_op->result_def >= 0) {
253
7.39k
    if (ssa_op->result_use >= 0) {
254
      zend_ssa_rename_var_uses(ssa, ssa_op->result_def, ssa_op->result_use, true);
255
115
    }
256
7.39k
    ssa->vars[ssa_op->result_def].definition = -1;
257
7.39k
    ssa_op->result_def = -1;
258
7.39k
  }
259
21.5k
}
Unexecuted instantiation: dfa_pass.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: escape_analysis.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: nop_removal.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: optimize_func_calls.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: optimize_temp_vars_5.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: pass1.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: pass3.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: sccp.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: scdf.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: zend_call_graph.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: zend_cfg.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: zend_dump.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: zend_func_info.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: zend_inference.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: zend_optimizer.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: zend_ssa.c:zend_ssa_rename_defs_of_instr
Unexecuted instantiation: zend_execute.c:zend_ssa_rename_defs_of_instr
260
261
#define NUM_PHI_SOURCES(phi) \
262
2.28M
  ((phi)->pi >= 0 ? 1 : (ssa->cfg.blocks[(phi)->block].predecessors_count))
263
264
/* FOREACH_USE and FOREACH_PHI_USE explicitly support "continue"
265
 * and changing the use chain of the current element */
266
13.3M
#define FOREACH_USE(var, use) do { \
267
13.3M
  int _var_num = (var) - ssa->vars, next; \
268
31.3M
  for (use = (var)->use_chain; use >= 0; use = next) { \
269
23.8M
    next = zend_ssa_next_use(ssa->ops, _var_num, use);
270
#define FOREACH_USE_END() \
271
23.8M
  } \
272
13.3M
} while (0)
273
274
8.49M
#define FOREACH_PHI_USE(var, phi) do { \
275
8.49M
  int _var_num = (var) - ssa->vars; \
276
8.49M
  zend_ssa_phi *next_phi; \
277
10.2M
  for (phi = (var)->phi_use_chain; phi; phi = next_phi) { \
278
3.00M
    next_phi = zend_ssa_next_use_phi(ssa, _var_num, phi);
279
#define FOREACH_PHI_USE_END() \
280
3.00M
  } \
281
8.49M
} while (0)
282
283
1.54M
#define FOREACH_PHI_SOURCE(phi, source) do { \
284
1.54M
  zend_ssa_phi *_phi = (phi); \
285
1.54M
  uint32_t _i, _end = NUM_PHI_SOURCES(phi); \
286
2.50M
  for (_i = 0; _i < _end; _i++) { \
287
2.33M
    ZEND_ASSERT(_phi->sources[_i] >= 0); \
288
2.33M
    source = _phi->sources[_i];
289
#define FOREACH_PHI_SOURCE_END() \
290
2.33M
  } \
291
1.54M
} while (0)
292
293
515k
#define FOREACH_PHI(phi) do { \
294
1.99M
  for (uint32_t _i = 0; _i < ssa->cfg.blocks_count; _i++) { \
295
1.47M
    phi = ssa->blocks[_i].phis; \
296
2.49M
    for (; phi; phi = phi->next) {
297
#define FOREACH_PHI_END() \
298
1.01M
    } \
299
1.47M
  } \
300
515k
} while (0)
301
302
370k
#define FOREACH_BLOCK(block) do { \
303
1.44M
  for (uint32_t _i = 0; _i < ssa->cfg.blocks_count; _i++) { \
304
1.07M
    (block) = &ssa->cfg.blocks[_i]; \
305
1.07M
    if (!((block)->flags & ZEND_BB_REACHABLE)) { \
306
29.5k
      continue; \
307
29.5k
    }
308
#define FOREACH_BLOCK_END() \
309
1.04M
  } \
310
370k
} while (0)
311
312
/* Does not support "break" */
313
370k
#define FOREACH_INSTR_NUM(i) do { \
314
370k
  zend_basic_block *_block; \
315
1.07M
  FOREACH_BLOCK(_block) { \
316
1.04M
    uint32_t _end = _block->start + _block->len; \
317
8.84M
    for ((i) = _block->start; (i) < _end; (i)++) {
318
#define FOREACH_INSTR_NUM_END() \
319
7.80M
    } \
320
1.04M
  } FOREACH_BLOCK_END(); \
321
370k
} while (0)
322
323
#endif /* ZEND_SSA_H */