Coverage Report

Created: 2026-06-02 06:39

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/php-src/Zend/Optimizer/zend_dfg.c
Line
Count
Source
1
/*
2
   +----------------------------------------------------------------------+
3
   | Zend Engine, DFG - Data Flow Graph                                   |
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
#include "zend_compile.h"
18
#include "zend_dfg.h"
19
20
static zend_always_inline void zend_dfg_add_use_def_op_impl(const zend_op_array *op_array, const zend_op *opline, uint32_t build_flags, zend_bitset use, zend_bitset def) /* {{{ */
21
714k
{
22
714k
  uint32_t var_num;
23
714k
  const zend_op *next;
24
25
714k
  if (opline->op1_type & (IS_CV|IS_VAR|IS_TMP_VAR)) {
26
415k
    var_num = EX_VAR_TO_NUM(opline->op1.var);
27
415k
    if (!zend_bitset_in(def, var_num)) {
28
113k
      zend_bitset_incl(use, var_num);
29
113k
    }
30
415k
  }
31
714k
  if (((opline->op2_type & (IS_VAR|IS_TMP_VAR)) != 0
32
59.4k
    && opline->opcode != ZEND_FE_FETCH_R
33
59.3k
    && opline->opcode != ZEND_FE_FETCH_RW)
34
654k
   || (opline->op2_type == IS_CV)) {
35
121k
    var_num = EX_VAR_TO_NUM(opline->op2.var);
36
121k
    if (!zend_bitset_in(def, var_num)) {
37
73.0k
      zend_bitset_incl(use, var_num);
38
73.0k
    }
39
121k
  }
40
714k
  if ((build_flags & ZEND_SSA_USE_CV_RESULTS)
41
0
   && opline->result_type == IS_CV
42
0
   && opline->opcode != ZEND_RECV) {
43
0
    var_num = EX_VAR_TO_NUM(opline->result.var);
44
0
    if (!zend_bitset_in(def, var_num)) {
45
0
      zend_bitset_incl(use, var_num);
46
0
    }
47
0
  }
48
49
714k
  switch (opline->opcode) {
50
30.6k
    case ZEND_ASSIGN:
51
30.6k
      if ((build_flags & ZEND_SSA_RC_INFERENCE) && opline->op2_type == IS_CV) {
52
0
        zend_bitset_incl(def, EX_VAR_TO_NUM(opline->op2.var));
53
0
      }
54
30.6k
      if (opline->op1_type == IS_CV) {
55
56.6k
add_op1_def:
56
56.6k
        zend_bitset_incl(def, EX_VAR_TO_NUM(opline->op1.var));
57
56.6k
      }
58
56.7k
      break;
59
56.7k
    case ZEND_ASSIGN_REF:
60
1.42k
      if (opline->op2_type == IS_CV) {
61
630
        zend_bitset_incl(def, EX_VAR_TO_NUM(opline->op2.var));
62
630
      }
63
1.42k
      if (opline->op1_type == IS_CV) {
64
1.04k
        goto add_op1_def;
65
1.04k
      }
66
382
      break;
67
4.29k
    case ZEND_ASSIGN_DIM:
68
8.93k
    case ZEND_ASSIGN_OBJ:
69
8.93k
      next = opline + 1;
70
8.93k
      if (next->op1_type & (IS_CV|IS_VAR|IS_TMP_VAR)) {
71
5.02k
        var_num = EX_VAR_TO_NUM(next->op1.var);
72
5.02k
        if (!zend_bitset_in(def, var_num)) {
73
732
          zend_bitset_incl(use, var_num);
74
732
        }
75
5.02k
        if (build_flags & ZEND_SSA_RC_INFERENCE && next->op1_type == IS_CV) {
76
0
          zend_bitset_incl(def, var_num);
77
0
        }
78
5.02k
      }
79
8.93k
      if (opline->op1_type == IS_CV) {
80
5.37k
        goto add_op1_def;
81
5.37k
      }
82
3.56k
      break;
83
3.56k
    case ZEND_ASSIGN_OBJ_REF:
84
199
      next = opline + 1;
85
199
      if (next->op1_type & (IS_CV|IS_VAR|IS_TMP_VAR)) {
86
199
        var_num = EX_VAR_TO_NUM(next->op1.var);
87
199
        if (!zend_bitset_in(def, var_num)) {
88
38
          zend_bitset_incl(use, var_num);
89
38
        }
90
199
        if (next->op1_type == IS_CV) {
91
145
          zend_bitset_incl(def, var_num);
92
145
        }
93
199
      }
94
199
      if (opline->op1_type == IS_CV) {
95
106
        goto add_op1_def;
96
106
      }
97
93
      break;
98
461
    case ZEND_ASSIGN_STATIC_PROP:
99
461
      next = opline + 1;
100
461
      if (next->op1_type & (IS_CV|IS_VAR|IS_TMP_VAR)) {
101
262
        var_num = EX_VAR_TO_NUM(next->op1.var);
102
262
        if (!zend_bitset_in(def, var_num)) {
103
76
          zend_bitset_incl(use, var_num);
104
76
        }
105
262
        if ((build_flags & ZEND_SSA_RC_INFERENCE) && next->op1_type == IS_CV) {
106
0
          zend_bitset_incl(def, var_num);
107
0
        }
108
262
      }
109
461
      break;
110
102
    case ZEND_ASSIGN_STATIC_PROP_REF:
111
102
      next = opline + 1;
112
102
      if (next->op1_type & (IS_CV|IS_VAR|IS_TMP_VAR)) {
113
102
        var_num = EX_VAR_TO_NUM(next->op1.var);
114
102
        if (!zend_bitset_in(def, var_num)) {
115
6
          zend_bitset_incl(use, var_num);
116
6
        }
117
102
        if (next->op1_type == IS_CV) {
118
24
          zend_bitset_incl(def, var_num);
119
24
        }
120
102
      }
121
102
      break;
122
18
    case ZEND_ASSIGN_STATIC_PROP_OP:
123
18
    case ZEND_FRAMELESS_ICALL_3:
124
18
      next = opline + 1;
125
18
      if (next->op1_type & (IS_CV|IS_VAR|IS_TMP_VAR)) {
126
2
        var_num = EX_VAR_TO_NUM(next->op1.var);
127
2
        if (!zend_bitset_in(def, var_num)) {
128
0
          zend_bitset_incl(use, var_num);
129
0
        }
130
2
      }
131
18
      break;
132
943
    case ZEND_ASSIGN_DIM_OP:
133
1.32k
    case ZEND_ASSIGN_OBJ_OP:
134
1.32k
      next = opline + 1;
135
1.32k
      if (next->op1_type & (IS_CV|IS_VAR|IS_TMP_VAR)) {
136
728
        var_num = EX_VAR_TO_NUM(next->op1.var);
137
728
        if (!zend_bitset_in(def, var_num)) {
138
130
          zend_bitset_incl(use, var_num);
139
130
        }
140
728
      }
141
1.32k
      if (opline->op1_type == IS_CV) {
142
962
        goto add_op1_def;
143
962
      }
144
358
      break;
145
2.96k
    case ZEND_ASSIGN_OP:
146
5.28k
    case ZEND_PRE_INC:
147
5.64k
    case ZEND_PRE_DEC:
148
6.16k
    case ZEND_POST_INC:
149
6.54k
    case ZEND_POST_DEC:
150
6.95k
    case ZEND_BIND_GLOBAL:
151
9.08k
    case ZEND_BIND_STATIC:
152
9.17k
    case ZEND_BIND_INIT_STATIC_OR_JMP:
153
9.29k
    case ZEND_SEND_VAR_NO_REF:
154
9.86k
    case ZEND_SEND_VAR_NO_REF_EX:
155
13.0k
    case ZEND_SEND_VAR_EX:
156
13.5k
    case ZEND_SEND_FUNC_ARG:
157
14.6k
    case ZEND_SEND_REF:
158
15.0k
    case ZEND_SEND_UNPACK:
159
15.4k
    case ZEND_FE_RESET_RW:
160
16.0k
    case ZEND_MAKE_REF:
161
16.2k
    case ZEND_PRE_INC_OBJ:
162
16.3k
    case ZEND_PRE_DEC_OBJ:
163
16.3k
    case ZEND_POST_INC_OBJ:
164
16.3k
    case ZEND_POST_DEC_OBJ:
165
16.9k
    case ZEND_UNSET_DIM:
166
17.3k
    case ZEND_UNSET_OBJ:
167
18.9k
    case ZEND_FETCH_DIM_W:
168
19.5k
    case ZEND_FETCH_DIM_RW:
169
19.7k
    case ZEND_FETCH_DIM_FUNC_ARG:
170
19.9k
    case ZEND_FETCH_DIM_UNSET:
171
20.1k
    case ZEND_FETCH_LIST_W:
172
20.1k
      if (opline->op1_type == IS_CV) {
173
16.3k
        goto add_op1_def;
174
16.3k
      }
175
3.71k
      break;
176
9.67k
    case ZEND_SEND_VAR:
177
10.6k
    case ZEND_CAST:
178
15.7k
    case ZEND_QM_ASSIGN:
179
16.5k
    case ZEND_JMP_SET:
180
18.0k
    case ZEND_COALESCE:
181
19.4k
    case ZEND_FE_RESET_R:
182
19.4k
      if ((build_flags & ZEND_SSA_RC_INFERENCE) && opline->op1_type == IS_CV) {
183
0
        goto add_op1_def;
184
0
      }
185
19.4k
      break;
186
19.4k
    case ZEND_ADD_ARRAY_UNPACK:
187
79
      var_num = EX_VAR_TO_NUM(opline->result.var);
188
79
      if (!zend_bitset_in(def, var_num)) {
189
0
        zend_bitset_incl(use, var_num);
190
0
      }
191
79
      break;
192
9.55k
    case ZEND_ADD_ARRAY_ELEMENT:
193
9.55k
      var_num = EX_VAR_TO_NUM(opline->result.var);
194
9.55k
      if (!zend_bitset_in(def, var_num)) {
195
310
        zend_bitset_incl(use, var_num);
196
310
      }
197
9.55k
      ZEND_FALLTHROUGH;
198
11.2k
    case ZEND_INIT_ARRAY:
199
11.2k
      if (((build_flags & ZEND_SSA_RC_INFERENCE)
200
11.2k
            || (opline->extended_value & ZEND_ARRAY_ELEMENT_REF))
201
142
          && opline->op1_type == IS_CV) {
202
126
        goto add_op1_def;
203
126
      }
204
11.1k
      break;
205
11.1k
    case ZEND_YIELD:
206
963
      if (opline->op1_type == IS_CV
207
170
          && ((op_array->fn_flags & ZEND_ACC_RETURN_REFERENCE)
208
142
            || (build_flags & ZEND_SSA_RC_INFERENCE))) {
209
28
        goto add_op1_def;
210
28
      }
211
935
      break;
212
1.23k
    case ZEND_UNSET_CV:
213
1.23k
      goto add_op1_def;
214
1.87k
    case ZEND_VERIFY_RETURN_TYPE:
215
1.87k
      if (opline->op1_type & (IS_TMP_VAR|IS_VAR|IS_CV)) {
216
884
        goto add_op1_def;
217
884
      }
218
993
      break;
219
1.38k
    case ZEND_FE_FETCH_R:
220
1.77k
    case ZEND_FE_FETCH_RW:
221
#if 0
222
      /* This special case was handled above the switch */
223
      if (opline->op2_type != IS_CV) {
224
        op2_use = -1; /* not used */
225
      }
226
#endif
227
1.77k
      zend_bitset_incl(def, EX_VAR_TO_NUM(opline->op2.var));
228
1.77k
      break;
229
1.86k
    case ZEND_BIND_LEXICAL:
230
1.86k
      if ((opline->extended_value & ZEND_BIND_REF) || (build_flags & ZEND_SSA_RC_INFERENCE)) {
231
238
        zend_bitset_incl(def, EX_VAR_TO_NUM(opline->op2.var));
232
238
      }
233
1.86k
      break;
234
612k
    default:
235
612k
      break;
236
714k
  }
237
238
714k
  if (opline->result_type & (IS_CV|IS_VAR|IS_TMP_VAR)) {
239
379k
    zend_bitset_incl(def, EX_VAR_TO_NUM(opline->result.var));
240
379k
  }
241
714k
}
242
/* }}} */
243
244
ZEND_API void zend_dfg_add_use_def_op(const zend_op_array *op_array, const zend_op *opline, uint32_t build_flags, zend_bitset use, zend_bitset def) /* {{{ */
245
0
{
246
0
  zend_dfg_add_use_def_op_impl(op_array, opline, build_flags, use, def);
247
0
}
248
/* }}} */
249
250
void zend_build_dfg(const zend_op_array *op_array, const zend_cfg *cfg, const zend_dfg *dfg, uint32_t build_flags) /* {{{ */
251
37.1k
{
252
37.1k
  uint32_t set_size = dfg->size;
253
37.1k
  zend_basic_block *blocks = cfg->blocks;
254
37.1k
  uint32_t blocks_count = cfg->blocks_count;
255
37.1k
  zend_bitset tmp, def, use, in, out;
256
37.1k
  uint32_t j;
257
258
37.1k
  tmp = dfg->tmp;
259
37.1k
  def = dfg->def;
260
37.1k
  use = dfg->use;
261
37.1k
  in  = dfg->in;
262
37.1k
  out = dfg->out;
263
264
  /* Collect "def" and "use" sets */
265
143k
  for (j = 0; j < blocks_count; j++) {
266
106k
    const zend_op *opline, *end;
267
106k
    zend_bitset b_use, b_def;
268
269
106k
    if ((blocks[j].flags & ZEND_BB_REACHABLE) == 0) {
270
14
      continue;
271
14
    }
272
273
106k
    opline = op_array->opcodes + blocks[j].start;
274
106k
    end = opline + blocks[j].len;
275
106k
    b_use = DFG_BITSET(use, set_size, j);
276
106k
    b_def = DFG_BITSET(def, set_size, j);
277
831k
    for (; opline < end; opline++) {
278
725k
      if (opline->opcode != ZEND_OP_DATA) {
279
714k
        zend_dfg_add_use_def_op_impl(op_array, opline, build_flags, b_use, b_def);
280
714k
      }
281
725k
    }
282
106k
  }
283
284
  /* Calculate "in" and "out" sets */
285
37.1k
  {
286
37.1k
    uint32_t worklist_len = zend_bitset_len(blocks_count);
287
37.1k
    zend_bitset worklist;
288
37.1k
    ALLOCA_FLAG(use_heap);
289
37.1k
    worklist = ZEND_BITSET_ALLOCA(worklist_len, use_heap);
290
37.1k
    memset(worklist, 0, worklist_len * ZEND_BITSET_ELM_SIZE);
291
143k
    for (j = 0; j < blocks_count; j++) {
292
106k
      zend_bitset_incl(worklist, j);
293
106k
    }
294
163k
    while (!zend_bitset_empty(worklist, worklist_len)) {
295
      /* We use the last block on the worklist, because predecessors tend to be located
296
       * before the succeeding block, so this converges faster. */
297
126k
      j = zend_bitset_last(worklist, worklist_len);
298
126k
      zend_bitset_excl(worklist, j);
299
300
126k
      if ((blocks[j].flags & ZEND_BB_REACHABLE) == 0) {
301
14
        continue;
302
14
      }
303
126k
      if (blocks[j].successors_count != 0) {
304
88.6k
        zend_bitset_copy(DFG_BITSET(out, set_size, j), DFG_BITSET(in, set_size, blocks[j].successors[0]), set_size);
305
133k
        for (uint32_t k = 1; k < blocks[j].successors_count; k++) {
306
44.8k
          zend_bitset_union(DFG_BITSET(out, set_size, j), DFG_BITSET(in, set_size, blocks[j].successors[k]), set_size);
307
44.8k
        }
308
88.6k
      } else {
309
38.0k
        zend_bitset_clear(DFG_BITSET(out, set_size, j), set_size);
310
38.0k
      }
311
126k
      zend_bitset_union_with_difference(tmp, DFG_BITSET(use, set_size, j), DFG_BITSET(out, set_size, j), DFG_BITSET(def, set_size, j), set_size);
312
126k
      if (!zend_bitset_equal(DFG_BITSET(in, set_size, j), tmp, set_size)) {
313
91.4k
        zend_bitset_copy(DFG_BITSET(in, set_size, j), tmp, set_size);
314
315
        /* Add predecessors of changed block to worklist */
316
91.4k
        {
317
91.4k
          const int *predecessors = &cfg->predecessors[blocks[j].predecessor_offset];
318
206k
          for (uint32_t k = 0; k < blocks[j].predecessors_count; k++) {
319
115k
            zend_bitset_incl(worklist, predecessors[k]);
320
115k
          }
321
91.4k
        }
322
91.4k
      }
323
126k
    }
324
325
    free_alloca(worklist, use_heap);
326
37.1k
  }
327
37.1k
}
328
/* }}} */