Coverage Report

Created: 2025-07-23 06:33

/src/php-src/Zend/Optimizer/zend_cfg.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
   +----------------------------------------------------------------------+
3
   | Zend Engine, CFG - Control Flow Graph                                |
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
#include "zend_compile.h"
20
#include "zend_cfg.h"
21
#include "zend_func_info.h"
22
#include "zend_worklist.h"
23
#include "zend_optimizer.h"
24
#include "zend_optimizer_internal.h"
25
#include "zend_sort.h"
26
27
static void zend_mark_reachable(zend_op *opcodes, zend_cfg *cfg, zend_basic_block *b) /* {{{ */
28
446k
{
29
446k
  zend_basic_block *blocks = cfg->blocks;
30
31
446k
  zend_worklist work;
32
446k
  ALLOCA_FLAG(list_use_heap)
33
446k
  ZEND_WORKLIST_ALLOCA(&work, cfg->blocks_count, list_use_heap);
34
35
446k
  zend_worklist_push(&work, b - cfg->blocks);
36
37
1.75M
  while (zend_worklist_len(&work)) {
38
1.30M
    int i;
39
1.30M
    b = cfg->blocks + zend_worklist_pop(&work);
40
41
1.30M
    b->flags |= ZEND_BB_REACHABLE;
42
1.30M
    if (b->successors_count == 0) {
43
380k
      b->flags |= ZEND_BB_EXIT;
44
380k
      continue;
45
380k
    }
46
47
2.21M
    for (i = 0; i < b->successors_count; i++) {
48
1.28M
      zend_basic_block *succ = blocks + b->successors[i];
49
50
1.28M
      if (b->len != 0) {
51
1.28M
        uint8_t opcode = opcodes[b->start + b->len - 1].opcode;
52
1.28M
        if (opcode == ZEND_MATCH) {
53
2.95k
          succ->flags |= ZEND_BB_TARGET;
54
1.28M
        } else if (opcode == ZEND_SWITCH_LONG || opcode == ZEND_SWITCH_STRING) {
55
1.05k
          if (i == b->successors_count - 1) {
56
191
            succ->flags |= ZEND_BB_FOLLOW | ZEND_BB_TARGET;
57
866
          } else {
58
866
            succ->flags |= ZEND_BB_TARGET;
59
866
          }
60
1.28M
        } else if (b->successors_count == 1) {
61
573k
          if (opcode == ZEND_JMP) {
62
170k
            succ->flags |= ZEND_BB_TARGET;
63
402k
          } else {
64
402k
            succ->flags |= ZEND_BB_FOLLOW;
65
66
402k
            if ((cfg->flags & ZEND_CFG_STACKLESS)) {
67
0
              if (opcode == ZEND_INCLUDE_OR_EVAL ||
68
0
                opcode == ZEND_GENERATOR_CREATE ||
69
0
                opcode == ZEND_YIELD ||
70
0
                opcode == ZEND_YIELD_FROM ||
71
0
                opcode == ZEND_DO_FCALL ||
72
0
                opcode == ZEND_DO_UCALL ||
73
0
                opcode == ZEND_DO_FCALL_BY_NAME) {
74
0
                succ->flags |= ZEND_BB_ENTRY;
75
0
              }
76
0
            }
77
402k
            if ((cfg->flags & ZEND_CFG_RECV_ENTRY)) {
78
0
              if (opcode == ZEND_RECV ||
79
0
                opcode == ZEND_RECV_INIT) {
80
0
                succ->flags |= ZEND_BB_RECV_ENTRY;
81
0
              }
82
0
            }
83
402k
          }
84
707k
        } else {
85
707k
          ZEND_ASSERT(b->successors_count == 2);
86
707k
          if (i == 0) {
87
353k
            succ->flags |= ZEND_BB_TARGET;
88
353k
          } else {
89
353k
            succ->flags |= ZEND_BB_FOLLOW;
90
353k
          }
91
707k
        }
92
1.28M
      } else {
93
846
        succ->flags |= ZEND_BB_FOLLOW;
94
846
      }
95
96
      /* Check reachability of successor */
97
1.28M
      if (!(succ->flags & ZEND_BB_REACHABLE)) {
98
1.07M
        zend_worklist_push(&work, succ - cfg->blocks);
99
1.07M
      }
100
1.28M
    }
101
929k
  }
102
103
446k
  ZEND_WORKLIST_FREE_ALLOCA(&work, list_use_heap);
104
446k
}
105
/* }}} */
106
107
static void zend_mark_reachable_blocks(const zend_op_array *op_array, zend_cfg *cfg, int start) /* {{{ */
108
336k
{
109
336k
  zend_basic_block *blocks = cfg->blocks;
110
111
336k
  blocks[start].flags = ZEND_BB_START;
112
336k
  zend_mark_reachable(op_array->opcodes, cfg, blocks + start);
113
114
336k
  if (op_array->last_try_catch) {
115
90.3k
    zend_basic_block *b;
116
90.3k
    int j, changed;
117
90.3k
    uint32_t *block_map = cfg->map;
118
119
180k
    do {
120
180k
      changed = 0;
121
122
      /* Add exception paths */
123
401k
      for (j = 0; j < op_array->last_try_catch; j++) {
124
125
        /* check for jumps into the middle of try block */
126
220k
        b = blocks + block_map[op_array->try_catch_array[j].try_op];
127
220k
        if (!(b->flags & ZEND_BB_REACHABLE)) {
128
62
          zend_basic_block *end;
129
130
62
          if (op_array->try_catch_array[j].catch_op) {
131
62
            end = blocks + block_map[op_array->try_catch_array[j].catch_op];
132
144
            while (b != end) {
133
94
              if (b->flags & ZEND_BB_REACHABLE) {
134
12
                op_array->try_catch_array[j].try_op = b->start;
135
12
                break;
136
12
              }
137
82
              b++;
138
82
            }
139
62
          }
140
62
          b = blocks + block_map[op_array->try_catch_array[j].try_op];
141
62
          if (!(b->flags & ZEND_BB_REACHABLE)) {
142
50
            if (op_array->try_catch_array[j].finally_op) {
143
12
              end = blocks + block_map[op_array->try_catch_array[j].finally_op];
144
44
              while (b != end) {
145
44
                if (b->flags & ZEND_BB_REACHABLE) {
146
                  /* In case we get here, there is no live try block but there is a live finally block.
147
                   * If we do have catch_op set, we need to set it to the first catch block to satisfy
148
                   * the constraint try_op <= catch_op <= finally_op */
149
12
                  op_array->try_catch_array[j].try_op =
150
12
                    op_array->try_catch_array[j].catch_op ? op_array->try_catch_array[j].catch_op : b->start;
151
12
                  changed = 1;
152
12
                  zend_mark_reachable(op_array->opcodes, cfg, blocks + block_map[op_array->try_catch_array[j].try_op]);
153
12
                  break;
154
12
                }
155
32
                b++;
156
32
              }
157
12
            }
158
50
          }
159
62
        }
160
161
220k
        b = blocks + block_map[op_array->try_catch_array[j].try_op];
162
220k
        if (b->flags & ZEND_BB_REACHABLE) {
163
220k
          b->flags |= ZEND_BB_TRY;
164
220k
          if (op_array->try_catch_array[j].catch_op) {
165
218k
            b = blocks + block_map[op_array->try_catch_array[j].catch_op];
166
218k
            b->flags |= ZEND_BB_CATCH;
167
218k
            if (!(b->flags & ZEND_BB_REACHABLE)) {
168
109k
              changed = 1;
169
109k
              zend_mark_reachable(op_array->opcodes, cfg, b);
170
109k
            }
171
218k
          }
172
220k
          if (op_array->try_catch_array[j].finally_op) {
173
2.90k
            b = blocks + block_map[op_array->try_catch_array[j].finally_op];
174
2.90k
            b->flags |= ZEND_BB_FINALLY;
175
2.90k
            if (!(b->flags & ZEND_BB_REACHABLE)) {
176
350
              changed = 1;
177
350
              zend_mark_reachable(op_array->opcodes, cfg, b);
178
350
            }
179
2.90k
          }
180
220k
          if (op_array->try_catch_array[j].finally_end) {
181
2.90k
            b = blocks + block_map[op_array->try_catch_array[j].finally_end];
182
2.90k
            b->flags |= ZEND_BB_FINALLY_END;
183
2.90k
            if (!(b->flags & ZEND_BB_REACHABLE)) {
184
504
              changed = 1;
185
504
              zend_mark_reachable(op_array->opcodes, cfg, b);
186
504
            }
187
2.90k
          }
188
220k
        } else {
189
38
          if (op_array->try_catch_array[j].catch_op) {
190
38
            ZEND_ASSERT(!(blocks[block_map[op_array->try_catch_array[j].catch_op]].flags & ZEND_BB_REACHABLE));
191
38
          }
192
38
          if (op_array->try_catch_array[j].finally_op) {
193
0
            ZEND_ASSERT(!(blocks[block_map[op_array->try_catch_array[j].finally_op]].flags & ZEND_BB_REACHABLE));
194
0
          }
195
38
          if (op_array->try_catch_array[j].finally_end) {
196
0
            ZEND_ASSERT(!(blocks[block_map[op_array->try_catch_array[j].finally_end]].flags & ZEND_BB_REACHABLE));
197
0
          }
198
38
        }
199
220k
      }
200
180k
    } while (changed);
201
90.3k
  }
202
203
336k
  if (cfg->flags & ZEND_FUNC_FREE_LOOP_VAR) {
204
41.3k
    zend_basic_block *b;
205
41.3k
    int j;
206
41.3k
    uint32_t *block_map = cfg->map;
207
208
    /* Mark blocks that are unreachable, but free a loop var created in a reachable block. */
209
451k
    for (b = blocks; b < blocks + cfg->blocks_count; b++) {
210
409k
      if (b->flags & ZEND_BB_REACHABLE) {
211
395k
        continue;
212
395k
      }
213
214
17.2k
      for (j = b->start; j < b->start + b->len; j++) {
215
2.63k
        zend_op *opline = &op_array->opcodes[j];
216
2.63k
        if (zend_optimizer_is_loop_var_free(opline)) {
217
108
          zend_op *def_opline = zend_optimizer_get_loop_var_def(op_array, opline);
218
108
          if (def_opline) {
219
106
            uint32_t def_block = block_map[def_opline - op_array->opcodes];
220
106
            if (blocks[def_block].flags & ZEND_BB_REACHABLE) {
221
90
              b->flags |= ZEND_BB_UNREACHABLE_FREE;
222
90
              break;
223
90
            }
224
106
          }
225
108
        }
226
2.63k
      }
227
14.6k
    }
228
41.3k
  }
229
336k
}
230
/* }}} */
231
232
void zend_cfg_remark_reachable_blocks(const zend_op_array *op_array, zend_cfg *cfg) /* {{{ */
233
142k
{
234
142k
  zend_basic_block *blocks = cfg->blocks;
235
142k
  int i;
236
142k
  int start = 0;
237
238
142k
  for (i = 0; i < cfg->blocks_count; i++) {
239
142k
    if (blocks[i].flags & ZEND_BB_REACHABLE) {
240
142k
      start = i;
241
142k
      i++;
242
142k
      break;
243
142k
    }
244
142k
  }
245
246
  /* clear all flags */
247
839k
  for (i = 0; i < cfg->blocks_count; i++) {
248
696k
    blocks[i].flags = 0;
249
696k
  }
250
251
142k
  zend_mark_reachable_blocks(op_array, cfg, start);
252
142k
}
253
/* }}} */
254
255
694k
static void initialize_block(zend_basic_block *block) {
256
694k
  block->flags = 0;
257
694k
  block->successors = block->successors_storage;
258
694k
  block->successors_count = 0;
259
694k
  block->predecessors_count = 0;
260
694k
  block->predecessor_offset = -1;
261
694k
  block->idom = -1;
262
694k
  block->loop_header = -1;
263
694k
  block->level = -1;
264
694k
  block->children = -1;
265
694k
  block->next_child = -1;
266
694k
}
267
268
885k
#define BB_START(i) do { \
269
885k
    if (!block_map[i]) { blocks_count++;} \
270
885k
    block_map[i]++; \
271
885k
  } while (0)
272
273
ZEND_API void zend_build_cfg(zend_arena **arena, const zend_op_array *op_array, uint32_t build_flags, zend_cfg *cfg) /* {{{ */
274
193k
{
275
193k
  uint32_t flags = 0;
276
193k
  uint32_t i;
277
193k
  int j;
278
193k
  uint32_t *block_map;
279
193k
  zend_function *fn;
280
193k
  int blocks_count = 0;
281
193k
  zend_basic_block *blocks;
282
193k
  zval *zv;
283
193k
  bool extra_entry_block = 0;
284
285
193k
  cfg->flags = build_flags & (ZEND_CFG_STACKLESS|ZEND_CFG_RECV_ENTRY);
286
287
193k
  cfg->map = block_map = zend_arena_calloc(arena, op_array->last, sizeof(uint32_t));
288
289
  /* Build CFG, Step 1: Find basic blocks starts, calculate number of blocks */
290
193k
  BB_START(0);
291
4.95M
  for (i = 0; i < op_array->last; i++) {
292
4.76M
    zend_op *opline = op_array->opcodes + i;
293
4.76M
    switch (opline->opcode) {
294
42.7k
      case ZEND_RECV:
295
49.1k
      case ZEND_RECV_INIT:
296
49.1k
        if (build_flags & ZEND_CFG_RECV_ENTRY) {
297
0
          BB_START(i + 1);
298
0
        }
299
49.1k
        break;
300
213k
      case ZEND_RETURN:
301
215k
      case ZEND_RETURN_BY_REF:
302
219k
      case ZEND_GENERATOR_RETURN:
303
219k
      case ZEND_VERIFY_NEVER_TYPE:
304
219k
        if (i + 1 < op_array->last) {
305
28.5k
          BB_START(i + 1);
306
28.5k
        }
307
219k
        break;
308
696
      case ZEND_MATCH_ERROR:
309
4.51k
      case ZEND_THROW:
310
        /* Don't treat THROW as terminator if it's used in expression context,
311
         * as we may lose live ranges when eliminating unreachable code. */
312
4.51k
        if (opline->extended_value != ZEND_THROW_IS_EXPR && i + 1 < op_array->last) {
313
2.94k
          BB_START(i + 1);
314
2.94k
        }
315
4.51k
        break;
316
10.8k
      case ZEND_INCLUDE_OR_EVAL:
317
10.8k
        flags |= ZEND_FUNC_INDIRECT_VAR_ACCESS;
318
10.8k
        ZEND_FALLTHROUGH;
319
14.3k
      case ZEND_GENERATOR_CREATE:
320
18.2k
      case ZEND_YIELD:
321
19.2k
      case ZEND_YIELD_FROM:
322
19.2k
        if (build_flags & ZEND_CFG_STACKLESS) {
323
0
          BB_START(i + 1);
324
0
        }
325
19.2k
        break;
326
382k
      case ZEND_DO_FCALL:
327
402k
      case ZEND_DO_UCALL:
328
403k
      case ZEND_DO_FCALL_BY_NAME:
329
403k
        flags |= ZEND_FUNC_HAS_CALLS;
330
403k
        if (build_flags & ZEND_CFG_STACKLESS) {
331
0
          BB_START(i + 1);
332
0
        }
333
403k
        break;
334
0
      case ZEND_DO_ICALL:
335
0
        flags |= ZEND_FUNC_HAS_CALLS;
336
0
        break;
337
195k
      case ZEND_INIT_FCALL:
338
200k
      case ZEND_INIT_NS_FCALL_BY_NAME:
339
200k
        zv = CRT_CONSTANT(opline->op2);
340
200k
        if (opline->opcode == ZEND_INIT_NS_FCALL_BY_NAME) {
341
          /* The third literal is the lowercased unqualified name */
342
5.08k
          zv += 2;
343
5.08k
        }
344
200k
        if ((fn = zend_hash_find_ptr(EG(function_table), Z_STR_P(zv))) != NULL) {
345
160k
          if (fn->type == ZEND_INTERNAL_FUNCTION) {
346
160k
            flags |= zend_optimizer_classify_function(
347
160k
              Z_STR_P(zv), opline->extended_value);
348
160k
          }
349
160k
        }
350
200k
        break;
351
903
      case ZEND_FAST_CALL:
352
903
        BB_START(OP_JMP_ADDR(opline, opline->op1) - op_array->opcodes);
353
903
        BB_START(i + 1);
354
903
        break;
355
696
      case ZEND_FAST_RET:
356
696
        if (i + 1 < op_array->last) {
357
696
          BB_START(i + 1);
358
696
        }
359
696
        break;
360
73.1k
      case ZEND_JMP:
361
73.1k
        BB_START(OP_JMP_ADDR(opline, opline->op1) - op_array->opcodes);
362
73.1k
        if (i + 1 < op_array->last) {
363
72.1k
          BB_START(i + 1);
364
72.1k
        }
365
73.1k
        break;
366
25.1k
      case ZEND_JMPZ:
367
43.1k
      case ZEND_JMPNZ:
368
48.3k
      case ZEND_JMPZ_EX:
369
54.1k
      case ZEND_JMPNZ_EX:
370
56.9k
      case ZEND_JMP_SET:
371
66.2k
      case ZEND_COALESCE:
372
68.7k
      case ZEND_ASSERT_CHECK:
373
148k
      case ZEND_JMP_NULL:
374
148k
      case ZEND_BIND_INIT_STATIC_OR_JMP:
375
148k
      case ZEND_JMP_FRAMELESS:
376
148k
        BB_START(OP_JMP_ADDR(opline, opline->op2) - op_array->opcodes);
377
148k
        BB_START(i + 1);
378
148k
        break;
379
41.4k
      case ZEND_CATCH:
380
41.4k
        if (!(opline->extended_value & ZEND_LAST_CATCH)) {
381
4.88k
          BB_START(OP_JMP_ADDR(opline, opline->op2) - op_array->opcodes);
382
4.88k
        }
383
41.4k
        BB_START(i + 1);
384
41.4k
        break;
385
16.6k
      case ZEND_FE_FETCH_R:
386
18.0k
      case ZEND_FE_FETCH_RW:
387
18.0k
        BB_START(ZEND_OFFSET_TO_OPLINE_NUM(op_array, opline, opline->extended_value));
388
18.0k
        BB_START(i + 1);
389
18.0k
        break;
390
16.6k
      case ZEND_FE_RESET_R:
391
18.0k
      case ZEND_FE_RESET_RW:
392
18.0k
        BB_START(OP_JMP_ADDR(opline, opline->op2) - op_array->opcodes);
393
18.0k
        BB_START(i + 1);
394
18.0k
        break;
395
16
      case ZEND_SWITCH_LONG:
396
116
      case ZEND_SWITCH_STRING:
397
506
      case ZEND_MATCH:
398
506
      {
399
506
        HashTable *jumptable = Z_ARRVAL_P(CRT_CONSTANT(opline->op2));
400
506
        zval *zv;
401
4.70k
        ZEND_HASH_FOREACH_VAL(jumptable, zv) {
402
4.70k
          BB_START(ZEND_OFFSET_TO_OPLINE_NUM(op_array, opline, Z_LVAL_P(zv)));
403
4.70k
        } ZEND_HASH_FOREACH_END();
404
506
        BB_START(ZEND_OFFSET_TO_OPLINE_NUM(op_array, opline, opline->extended_value));
405
506
        BB_START(i + 1);
406
506
        break;
407
116
      }
408
4.43k
      case ZEND_FETCH_R:
409
7.30k
      case ZEND_FETCH_W:
410
8.41k
      case ZEND_FETCH_RW:
411
8.52k
      case ZEND_FETCH_FUNC_ARG:
412
8.61k
      case ZEND_FETCH_IS:
413
8.67k
      case ZEND_FETCH_UNSET:
414
9.01k
      case ZEND_UNSET_VAR:
415
9.39k
      case ZEND_ISSET_ISEMPTY_VAR:
416
9.39k
        if (opline->extended_value & ZEND_FETCH_LOCAL) {
417
7.07k
          flags |= ZEND_FUNC_INDIRECT_VAR_ACCESS;
418
7.07k
        } else if ((opline->extended_value & (ZEND_FETCH_GLOBAL | ZEND_FETCH_GLOBAL_LOCK)) &&
419
2.32k
                   !op_array->function_name) {
420
706
          flags |= ZEND_FUNC_INDIRECT_VAR_ACCESS;
421
706
        }
422
9.39k
        break;
423
302
      case ZEND_FUNC_GET_ARGS:
424
302
        flags |= ZEND_FUNC_VARARG;
425
302
        break;
426
0
      case ZEND_EXT_STMT:
427
0
        flags |= ZEND_FUNC_HAS_EXTENDED_STMT;
428
0
        break;
429
0
      case ZEND_EXT_FCALL_BEGIN:
430
0
      case ZEND_EXT_FCALL_END:
431
0
        flags |= ZEND_FUNC_HAS_EXTENDED_FCALL;
432
0
        break;
433
78.8k
      case ZEND_FREE:
434
97.2k
      case ZEND_FE_FREE:
435
97.2k
        if (zend_optimizer_is_loop_var_free(opline)
436
97.2k
         && ((opline-1)->opcode != ZEND_MATCH_ERROR
437
18.6k
          || (opline-1)->extended_value != ZEND_THROW_IS_EXPR)) {
438
18.5k
          BB_START(i);
439
18.5k
          flags |= ZEND_FUNC_FREE_LOOP_VAR;
440
18.5k
        }
441
97.2k
        break;
442
4.76M
    }
443
4.76M
  }
444
445
  /* If the entry block has predecessors, we may need to split it */
446
193k
  if ((build_flags & ZEND_CFG_NO_ENTRY_PREDECESSORS)
447
193k
      && op_array->last > 0 && block_map[0] > 1) {
448
139
    extra_entry_block = 1;
449
139
  }
450
451
193k
  if (op_array->last_try_catch) {
452
67.3k
    for (j = 0; j < op_array->last_try_catch; j++) {
453
37.1k
      BB_START(op_array->try_catch_array[j].try_op);
454
37.1k
      if (op_array->try_catch_array[j].catch_op) {
455
36.5k
        BB_START(op_array->try_catch_array[j].catch_op);
456
36.5k
      }
457
37.1k
      if (op_array->try_catch_array[j].finally_op) {
458
696
        BB_START(op_array->try_catch_array[j].finally_op);
459
696
      }
460
37.1k
      if (op_array->try_catch_array[j].finally_end) {
461
696
        BB_START(op_array->try_catch_array[j].finally_end);
462
696
      }
463
37.1k
    }
464
30.2k
  }
465
466
193k
  blocks_count += extra_entry_block;
467
193k
  cfg->blocks_count = blocks_count;
468
469
  /* Build CFG, Step 2: Build Array of Basic Blocks */
470
193k
  cfg->blocks = blocks = zend_arena_calloc(arena, sizeof(zend_basic_block), blocks_count);
471
472
193k
  blocks_count = -1;
473
474
193k
  if (extra_entry_block) {
475
139
    initialize_block(&blocks[0]);
476
139
    blocks[0].start = 0;
477
139
    blocks[0].len = 0;
478
139
    blocks_count++;
479
139
  }
480
481
4.95M
  for (i = 0; i < op_array->last; i++) {
482
4.76M
    if (block_map[i]) {
483
694k
      if (blocks_count >= 0) {
484
501k
        blocks[blocks_count].len = i - blocks[blocks_count].start;
485
501k
      }
486
694k
      blocks_count++;
487
694k
      initialize_block(&blocks[blocks_count]);
488
694k
      blocks[blocks_count].start = i;
489
694k
    }
490
4.76M
    block_map[i] = blocks_count;
491
4.76M
  }
492
493
193k
  blocks[blocks_count].len = i - blocks[blocks_count].start;
494
193k
  blocks_count++;
495
496
  /* Build CFG, Step 3: Calculate successors */
497
888k
  for (j = 0; j < blocks_count; j++) {
498
694k
    zend_basic_block *block = &blocks[j];
499
694k
    zend_op *opline;
500
694k
    if (block->len == 0) {
501
139
      block->successors_count = 1;
502
139
      block->successors[0] = j + 1;
503
139
      continue;
504
139
    }
505
506
694k
    opline = op_array->opcodes + block->start + block->len - 1;
507
694k
    switch (opline->opcode) {
508
696
      case ZEND_FAST_RET:
509
213k
      case ZEND_RETURN:
510
216k
      case ZEND_RETURN_BY_REF:
511
220k
      case ZEND_GENERATOR_RETURN:
512
223k
      case ZEND_THROW:
513
224k
      case ZEND_MATCH_ERROR:
514
224k
      case ZEND_VERIFY_NEVER_TYPE:
515
224k
        break;
516
73.1k
      case ZEND_JMP:
517
73.1k
        block->successors_count = 1;
518
73.1k
        block->successors[0] = block_map[OP_JMP_ADDR(opline, opline->op1) - op_array->opcodes];
519
73.1k
        break;
520
25.1k
      case ZEND_JMPZ:
521
43.1k
      case ZEND_JMPNZ:
522
48.3k
      case ZEND_JMPZ_EX:
523
54.1k
      case ZEND_JMPNZ_EX:
524
56.9k
      case ZEND_JMP_SET:
525
66.2k
      case ZEND_COALESCE:
526
68.7k
      case ZEND_ASSERT_CHECK:
527
148k
      case ZEND_JMP_NULL:
528
148k
      case ZEND_BIND_INIT_STATIC_OR_JMP:
529
148k
      case ZEND_JMP_FRAMELESS:
530
148k
        block->successors_count = 2;
531
148k
        block->successors[0] = block_map[OP_JMP_ADDR(opline, opline->op2) - op_array->opcodes];
532
148k
        block->successors[1] = j + 1;
533
148k
        break;
534
41.4k
      case ZEND_CATCH:
535
41.4k
        if (!(opline->extended_value & ZEND_LAST_CATCH)) {
536
4.88k
          block->successors_count = 2;
537
4.88k
          block->successors[0] = block_map[OP_JMP_ADDR(opline, opline->op2) - op_array->opcodes];
538
4.88k
          block->successors[1] = j + 1;
539
36.5k
        } else {
540
36.5k
          block->successors_count = 1;
541
36.5k
          block->successors[0] = j + 1;
542
36.5k
        }
543
41.4k
        break;
544
16.6k
      case ZEND_FE_FETCH_R:
545
18.0k
      case ZEND_FE_FETCH_RW:
546
18.0k
        block->successors_count = 2;
547
18.0k
        block->successors[0] = block_map[ZEND_OFFSET_TO_OPLINE_NUM(op_array, opline, opline->extended_value)];
548
18.0k
        block->successors[1] = j + 1;
549
18.0k
        break;
550
16.6k
      case ZEND_FE_RESET_R:
551
18.0k
      case ZEND_FE_RESET_RW:
552
18.0k
        block->successors_count = 2;
553
18.0k
        block->successors[0] = block_map[OP_JMP_ADDR(opline, opline->op2) - op_array->opcodes];
554
18.0k
        block->successors[1] = j + 1;
555
18.0k
        break;
556
903
      case ZEND_FAST_CALL:
557
903
        block->successors_count = 2;
558
903
        block->successors[0] = block_map[OP_JMP_ADDR(opline, opline->op1) - op_array->opcodes];
559
903
        block->successors[1] = j + 1;
560
903
        break;
561
16
      case ZEND_SWITCH_LONG:
562
116
      case ZEND_SWITCH_STRING:
563
506
      case ZEND_MATCH:
564
506
      {
565
506
        HashTable *jumptable = Z_ARRVAL_P(CRT_CONSTANT(opline->op2));
566
506
        zval *zv;
567
506
        uint32_t s = 0;
568
569
506
        block->successors_count = (opline->opcode == ZEND_MATCH ? 1 : 2) + zend_hash_num_elements(jumptable);
570
506
        block->successors = zend_arena_calloc(arena, block->successors_count, sizeof(int));
571
572
4.70k
        ZEND_HASH_FOREACH_VAL(jumptable, zv) {
573
4.70k
          block->successors[s++] = block_map[ZEND_OFFSET_TO_OPLINE_NUM(op_array, opline, Z_LVAL_P(zv))];
574
4.70k
        } ZEND_HASH_FOREACH_END();
575
576
506
        block->successors[s++] = block_map[ZEND_OFFSET_TO_OPLINE_NUM(op_array, opline, opline->extended_value)];
577
506
        if (opline->opcode != ZEND_MATCH) {
578
116
          block->successors[s++] = j + 1;
579
116
        }
580
506
        break;
581
116
      }
582
169k
      default:
583
169k
        block->successors_count = 1;
584
169k
        block->successors[0] = j + 1;
585
169k
        break;
586
694k
    }
587
694k
  }
588
589
  /* Build CFG, Step 4, Mark Reachable Basic Blocks */
590
193k
  cfg->flags |= flags;
591
193k
  zend_mark_reachable_blocks(op_array, cfg, 0);
592
193k
}
593
/* }}} */
594
595
ZEND_API void zend_cfg_build_predecessors(zend_arena **arena, zend_cfg *cfg) /* {{{ */
596
78.5k
{
597
78.5k
  int j, s, edges;
598
78.5k
  zend_basic_block *b;
599
78.5k
  zend_basic_block *blocks = cfg->blocks;
600
78.5k
  zend_basic_block *end = blocks + cfg->blocks_count;
601
78.5k
  int *predecessors;
602
603
78.5k
  edges = 0;
604
302k
  for (b = blocks; b < end; b++) {
605
223k
    b->predecessors_count = 0;
606
223k
  }
607
302k
  for (b = blocks; b < end; b++) {
608
223k
    if (!(b->flags & ZEND_BB_REACHABLE)) {
609
28
      b->successors_count = 0;
610
28
      b->predecessors_count = 0;
611
223k
    } else {
612
439k
      for (s = 0; s < b->successors_count; s++) {
613
215k
        edges++;
614
215k
        blocks[b->successors[s]].predecessors_count++;
615
215k
      }
616
223k
    }
617
223k
  }
618
619
78.5k
  cfg->edges_count = edges;
620
78.5k
  cfg->predecessors = predecessors = (int*)zend_arena_calloc(arena, sizeof(int), edges);
621
622
78.5k
  edges = 0;
623
302k
  for (b = blocks; b < end; b++) {
624
223k
    if (b->flags & ZEND_BB_REACHABLE) {
625
223k
      b->predecessor_offset = edges;
626
223k
      edges += b->predecessors_count;
627
223k
      b->predecessors_count = 0;
628
223k
    }
629
223k
  }
630
631
302k
  for (j = 0; j < cfg->blocks_count; j++) {
632
223k
    if (blocks[j].flags & ZEND_BB_REACHABLE) {
633
      /* SWITCH_STRING/LONG may have few identical successors */
634
439k
      for (s = 0; s < blocks[j].successors_count; s++) {
635
215k
        int duplicate = 0;
636
215k
        int p;
637
638
291k
        for (p = 0; p < s; p++) {
639
75.3k
          if (blocks[j].successors[p] == blocks[j].successors[s]) {
640
202
            duplicate = 1;
641
202
            break;
642
202
          }
643
75.3k
        }
644
215k
        if (!duplicate) {
645
215k
          zend_basic_block *b = blocks + blocks[j].successors[s];
646
647
215k
          predecessors[b->predecessor_offset + b->predecessors_count] = j;
648
215k
          b->predecessors_count++;
649
215k
        }
650
215k
      }
651
223k
    }
652
223k
  }
653
78.5k
}
654
/* }}} */
655
656
/* Computes a postorder numbering of the CFG */
657
static void compute_postnum_recursive(
658
    int *postnum, int *cur, const zend_cfg *cfg, int block_num) /* {{{ */
659
228k
{
660
228k
  int s;
661
228k
  zend_basic_block *block = &cfg->blocks[block_num];
662
228k
  if (postnum[block_num] != -1) {
663
70.7k
    return;
664
70.7k
  }
665
666
157k
  postnum[block_num] = -2; /* Marker for "currently visiting" */
667
373k
  for (s = 0; s < block->successors_count; s++) {
668
215k
    compute_postnum_recursive(postnum, cur, cfg, block->successors[s]);
669
215k
  }
670
157k
  postnum[block_num] = (*cur)++;
671
157k
}
672
/* }}} */
673
674
/* Computes dominator tree using algorithm from "A Simple, Fast Dominance Algorithm" by
675
 * Cooper, Harvey and Kennedy. */
676
ZEND_API void zend_cfg_compute_dominators_tree(const zend_op_array *op_array, zend_cfg *cfg) /* {{{ */
677
78.5k
{
678
78.5k
  zend_basic_block *blocks = cfg->blocks;
679
78.5k
  int blocks_count = cfg->blocks_count;
680
78.5k
  int j, k, changed;
681
682
78.5k
  if (cfg->blocks_count == 1) {
683
65.7k
    blocks[0].level = 0;
684
65.7k
    return;
685
65.7k
  }
686
687
12.8k
  ALLOCA_FLAG(use_heap)
688
12.8k
  int *postnum = do_alloca(sizeof(int) * cfg->blocks_count, use_heap);
689
12.8k
  memset(postnum, -1, sizeof(int) * cfg->blocks_count);
690
12.8k
  j = 0;
691
12.8k
  compute_postnum_recursive(postnum, &j, cfg, 0);
692
693
  /* FIXME: move declarations */
694
12.8k
  blocks[0].idom = 0;
695
30.8k
  do {
696
30.8k
    changed = 0;
697
    /* Iterating in RPO here would converge faster */
698
374k
    for (j = 1; j < blocks_count; j++) {
699
344k
      int idom = -1;
700
701
344k
      if ((blocks[j].flags & ZEND_BB_REACHABLE) == 0) {
702
46
        continue;
703
46
      }
704
851k
      for (k = 0; k < blocks[j].predecessors_count; k++) {
705
507k
        int pred = cfg->predecessors[blocks[j].predecessor_offset + k];
706
707
507k
        if (blocks[pred].idom >= 0) {
708
451k
          if (idom < 0) {
709
311k
            idom = pred;
710
311k
          } else {
711
280k
            while (idom != pred) {
712
374k
              while (postnum[pred] < postnum[idom]) pred = blocks[pred].idom;
713
148k
              while (postnum[idom] < postnum[pred]) idom = blocks[idom].idom;
714
140k
            }
715
139k
          }
716
451k
        }
717
507k
      }
718
719
344k
      if (idom >= 0 && blocks[j].idom != idom) {
720
145k
        blocks[j].idom = idom;
721
145k
        changed = 1;
722
145k
      }
723
344k
    }
724
30.8k
  } while (changed);
725
12.8k
  blocks[0].idom = -1;
726
727
157k
  for (j = 1; j < blocks_count; j++) {
728
145k
    if ((blocks[j].flags & ZEND_BB_REACHABLE) == 0) {
729
28
      continue;
730
28
    }
731
145k
    if (blocks[j].idom >= 0) {
732
      /* Sort by block number to traverse children in pre-order */
733
145k
      if (blocks[blocks[j].idom].children < 0 ||
734
145k
          j < blocks[blocks[j].idom].children) {
735
77.6k
        blocks[j].next_child = blocks[blocks[j].idom].children;
736
77.6k
        blocks[blocks[j].idom].children = j;
737
77.6k
      } else {
738
67.5k
        int k = blocks[blocks[j].idom].children;
739
74.5k
        while (blocks[k].next_child >=0 && j > blocks[k].next_child) {
740
6.98k
          k = blocks[k].next_child;
741
6.98k
        }
742
67.5k
        blocks[j].next_child = blocks[k].next_child;
743
67.5k
        blocks[k].next_child = j;
744
67.5k
      }
745
145k
    }
746
145k
  }
747
748
170k
  for (j = 0; j < blocks_count; j++) {
749
157k
    int idom = blocks[j].idom, level = 0;
750
157k
    if ((blocks[j].flags & ZEND_BB_REACHABLE) == 0) {
751
28
      continue;
752
28
    }
753
163k
    while (idom >= 0) {
754
150k
      level++;
755
150k
      if (blocks[idom].level >= 0) {
756
145k
        level += blocks[idom].level;
757
145k
        break;
758
145k
      } else {
759
5.82k
        idom = blocks[idom].idom;
760
5.82k
      }
761
150k
    }
762
157k
    blocks[j].level = level;
763
157k
  }
764
765
12.8k
  free_alloca(postnum, use_heap);
766
12.8k
}
767
/* }}} */
768
769
static bool dominates(zend_basic_block *blocks, int a, int b) /* {{{ */
770
74.0k
{
771
122k
  while (blocks[b].level > blocks[a].level) {
772
48.9k
    b = blocks[b].idom;
773
48.9k
  }
774
74.0k
  return a == b;
775
74.0k
}
776
/* }}} */
777
778
ZEND_API void zend_cfg_identify_loops(const zend_op_array *op_array, zend_cfg *cfg) /* {{{ */
779
78.5k
{
780
78.5k
  int i, j, k, n;
781
78.5k
  int time;
782
78.5k
  zend_basic_block *blocks = cfg->blocks;
783
78.5k
  int *entry_times, *exit_times;
784
78.5k
  zend_worklist work;
785
78.5k
  int flag = ZEND_FUNC_NO_LOOPS;
786
78.5k
  int *sorted_blocks;
787
78.5k
  ALLOCA_FLAG(list_use_heap)
788
78.5k
  ALLOCA_FLAG(tree_use_heap)
789
790
78.5k
  if (cfg->blocks_count == 1) {
791
65.7k
    cfg->flags |= flag;
792
65.7k
    return;
793
65.7k
  }
794
795
12.8k
  ZEND_WORKLIST_ALLOCA(&work, cfg->blocks_count, list_use_heap);
796
797
  /* We don't materialize the DJ spanning tree explicitly, as we are only interested in ancestor
798
   * queries. These are implemented by checking entry/exit times of the DFS search. */
799
12.8k
  entry_times = do_alloca(3 * sizeof(int) * cfg->blocks_count, tree_use_heap);
800
12.8k
  exit_times = entry_times + cfg->blocks_count;
801
12.8k
  sorted_blocks = exit_times + cfg->blocks_count;
802
12.8k
  memset(entry_times, -1, 2 * sizeof(int) * cfg->blocks_count);
803
804
12.8k
  zend_worklist_push(&work, 0);
805
12.8k
  time = 0;
806
170k
  while (zend_worklist_len(&work)) {
807
303k
  next:
808
303k
    i = zend_worklist_peek(&work);
809
303k
    if (entry_times[i] == -1) {
810
157k
      entry_times[i] = time++;
811
157k
    }
812
    /* Visit blocks immediately dominated by i. */
813
464k
    for (j = blocks[i].children; j >= 0; j = blocks[j].next_child) {
814
249k
      if (zend_worklist_push(&work, j)) {
815
88.5k
        goto next;
816
88.5k
      }
817
249k
    }
818
    /* Visit join edges.  */
819
430k
    for (j = 0; j < blocks[i].successors_count; j++) {
820
272k
      int succ = blocks[i].successors[j];
821
272k
      if (blocks[succ].idom == i) {
822
141k
        continue;
823
141k
      } else if (zend_worklist_push(&work, succ)) {
824
56.5k
        goto next;
825
56.5k
      }
826
272k
    }
827
157k
    exit_times[i] = time++;
828
157k
    zend_worklist_pop(&work);
829
157k
  }
830
831
  /* Sort blocks by level, which is the opposite order in which we want to process them */
832
12.8k
  sorted_blocks[0] = 0;
833
12.8k
  j = 0;
834
12.8k
  n = 1;
835
98.4k
  while (j != n) {
836
85.6k
    i = j;
837
85.6k
    j = n;
838
243k
    for (; i < j; i++) {
839
157k
      int child;
840
303k
      for (child = blocks[sorted_blocks[i]].children; child >= 0; child = blocks[child].next_child) {
841
145k
        sorted_blocks[n++] = child;
842
145k
      }
843
157k
    }
844
85.6k
  }
845
846
  /* Identify loops. See Sreedhar et al, "Identifying Loops Using DJ Graphs". */
847
170k
  while (n > 0) {
848
157k
    i = sorted_blocks[--n];
849
850
157k
    if (blocks[i].predecessors_count < 2) {
851
        /* loop header has at least two input edges */
852
91.2k
      continue;
853
91.2k
    }
854
855
203k
    for (j = 0; j < blocks[i].predecessors_count; j++) {
856
137k
      int pred = cfg->predecessors[blocks[i].predecessor_offset + j];
857
858
      /* A join edge is one for which the predecessor does not
859
         immediately dominate the successor. */
860
137k
      if (blocks[i].idom == pred) {
861
63.2k
        continue;
862
63.2k
      }
863
864
      /* In a loop back-edge (back-join edge), the successor dominates
865
         the predecessor.  */
866
74.0k
      if (dominates(blocks, i, pred)) {
867
11.0k
        blocks[i].flags |= ZEND_BB_LOOP_HEADER;
868
11.0k
        flag &= ~ZEND_FUNC_NO_LOOPS;
869
11.0k
        if (!zend_worklist_len(&work)) {
870
10.1k
          zend_bitset_clear(work.visited, zend_bitset_len(cfg->blocks_count));
871
10.1k
        }
872
11.0k
        zend_worklist_push(&work, pred);
873
63.0k
      } else {
874
        /* Otherwise it's a cross-join edge.  See if it's a branch
875
           to an ancestor on the DJ spanning tree.  */
876
63.0k
        if (entry_times[pred] > entry_times[i] && exit_times[pred] < exit_times[i]) {
877
0
          blocks[i].flags |= ZEND_BB_IRREDUCIBLE_LOOP;
878
0
          flag |= ZEND_FUNC_IRREDUCIBLE;
879
0
          flag &= ~ZEND_FUNC_NO_LOOPS;
880
0
        }
881
63.0k
      }
882
74.0k
    }
883
119k
    while (zend_worklist_len(&work)) {
884
52.7k
      j = zend_worklist_pop(&work);
885
55.4k
      while (blocks[j].loop_header >= 0) {
886
2.74k
        j = blocks[j].loop_header;
887
2.74k
      }
888
52.7k
      if (j != i) {
889
41.2k
        if (blocks[j].idom < 0 && j != 0) {
890
          /* Ignore blocks that are unreachable or only abnormally reachable. */
891
0
          continue;
892
0
        }
893
41.2k
        blocks[j].loop_header = i;
894
98.3k
        for (k = 0; k < blocks[j].predecessors_count; k++) {
895
57.0k
          zend_worklist_push(&work, cfg->predecessors[blocks[j].predecessor_offset + k]);
896
57.0k
        }
897
41.2k
      }
898
52.7k
    }
899
66.6k
  }
900
901
12.8k
  free_alloca(entry_times, tree_use_heap);
902
12.8k
  ZEND_WORKLIST_FREE_ALLOCA(&work, list_use_heap);
903
904
12.8k
  cfg->flags |= flag;
905
12.8k
}
906
/* }}} */