Coverage Report

Created: 2026-06-02 06:40

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/php-src/Zend/Optimizer/zend_inference.c
Line
Count
Source
1
/*
2
   +----------------------------------------------------------------------+
3
   | Zend Engine, e-SSA based Type & Range Inference                      |
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_generators.h"
19
#include "zend_inference.h"
20
#include "zend_func_info.h"
21
#include "zend_call_graph.h"
22
#include "zend_closures.h"
23
#include "zend_worklist.h"
24
#include "zend_optimizer_internal.h"
25
26
/* The used range inference algorithm is described in:
27
 *     V. Campos, R. Rodrigues, I. de Assis Costa and F. Pereira.
28
 *     "Speed and Precision in Range Analysis", SBLP'12.
29
 *
30
 * There are a couple degrees of freedom, we use:
31
 *  * Propagation on SCCs.
32
 *  * e-SSA for live range splitting.
33
 *  * Only intra-procedural inference.
34
 *  * Widening with warmup passes, but without jump sets.
35
 */
36
37
/* Whether to handle symbolic range constraints */
38
#define SYM_RANGE
39
40
/* Whether to handle negative range constraints */
41
/* Negative range inference is buggy, so disabled for now */
42
#undef NEG_RANGE
43
44
/* Number of warmup passes to use prior to widening */
45
300k
#define RANGE_WARMUP_PASSES 16
46
47
/* Logging for range inference in general */
48
#if 0
49
#define LOG_SSA_RANGE(...) fprintf(stderr, __VA_ARGS__)
50
#else
51
#define LOG_SSA_RANGE(...)
52
#endif
53
54
/* Logging for negative range constraints */
55
#if 0
56
#define LOG_NEG_RANGE(...) fprintf(stderr, __VA_ARGS__)
57
#else
58
#define LOG_NEG_RANGE(...)
59
#endif
60
61
/* Pop elements in unspecified order from worklist until it is empty */
62
448k
#define WHILE_WORKLIST(worklist, len, i) do { \
63
448k
  bool _done = 0; \
64
1.23M
  while (!_done) { \
65
783k
    _done = 1; \
66
8.73M
    ZEND_BITSET_FOREACH(worklist, len, i) { \
67
1.30M
      zend_bitset_excl(worklist, i); \
68
1.30M
      _done = 0;
69
70
#define WHILE_WORKLIST_END() \
71
1.30M
    } ZEND_BITSET_FOREACH_END(); \
72
783k
  } \
73
448k
} while (0)
74
75
#define CHECK_SCC_VAR(var2) \
76
  do { \
77
    if (!ssa->vars[var2].no_val) { \
78
      if (ssa->vars[var2].scc < 0) { \
79
        zend_ssa_check_scc_var(op_array, ssa, var2, index, stack); \
80
      } \
81
      if (ssa->vars[var2].scc < ssa->vars[var].scc) { \
82
        ssa->vars[var].scc = ssa->vars[var2].scc; \
83
        is_root = 0; \
84
      } \
85
    } \
86
  } while (0)
87
88
#define CHECK_SCC_ENTRY(var2) \
89
1.09M
  do { \
90
1.09M
    if (ssa->vars[var2].scc != ssa->vars[var].scc) { \
91
931k
      ssa->vars[var2].scc_entry = 1; \
92
931k
    } \
93
1.09M
  } while (0)
94
95
#define ADD_SCC_VAR(_var) \
96
169k
  do { \
97
169k
    if (ssa->vars[_var].scc == scc && \
98
169k
        !(ssa->var_info[_var].type & MAY_BE_REF)) { \
99
69.4k
      zend_bitset_incl(worklist, _var); \
100
69.4k
    } \
101
169k
  } while (0)
102
103
#define ADD_SCC_VAR_1(_var) \
104
162k
  do { \
105
162k
    if (ssa->vars[_var].scc == scc && \
106
162k
        !(ssa->var_info[_var].type & MAY_BE_REF) && \
107
162k
        !zend_bitset_in(visited, _var)) { \
108
82.0k
      zend_bitset_incl(worklist, _var); \
109
82.0k
    } \
110
162k
  } while (0)
111
112
#define FOR_EACH_DEFINED_VAR(line, MACRO) \
113
1.26M
  do { \
114
1.26M
    if (ssa->ops[line].op1_def >= 0) { \
115
207k
      MACRO(ssa->ops[line].op1_def); \
116
207k
    } \
117
1.26M
    if (ssa->ops[line].op2_def >= 0) { \
118
12.2k
      MACRO(ssa->ops[line].op2_def); \
119
12.2k
    } \
120
1.26M
    if (ssa->ops[line].result_def >= 0) { \
121
831k
      MACRO(ssa->ops[line].result_def); \
122
831k
    } \
123
1.26M
    if (op_array->opcodes[line].opcode == ZEND_OP_DATA) { \
124
16.3k
      if (ssa->ops[line-1].op1_def >= 0) { \
125
10.4k
        MACRO(ssa->ops[line-1].op1_def); \
126
10.4k
      } \
127
16.3k
      if (ssa->ops[line-1].op2_def >= 0) { \
128
0
        MACRO(ssa->ops[line-1].op2_def); \
129
0
      } \
130
16.3k
      if (ssa->ops[line-1].result_def >= 0) { \
131
4.73k
        MACRO(ssa->ops[line-1].result_def); \
132
4.73k
      } \
133
1.24M
    } else if ((uint32_t)line+1 < op_array->last && \
134
1.24M
               op_array->opcodes[line+1].opcode == ZEND_OP_DATA) { \
135
27.8k
      if (ssa->ops[line+1].op1_def >= 0) { \
136
193
        MACRO(ssa->ops[line+1].op1_def); \
137
193
      } \
138
27.8k
      if (ssa->ops[line+1].op2_def >= 0) { \
139
0
        MACRO(ssa->ops[line+1].op2_def); \
140
0
      } \
141
27.8k
      if (ssa->ops[line+1].result_def >= 0) { \
142
0
        MACRO(ssa->ops[line+1].result_def); \
143
0
      } \
144
27.8k
    } \
145
1.26M
  } while (0)
146
147
148
#define FOR_EACH_VAR_USAGE(_var, MACRO) \
149
1.23M
  do { \
150
1.23M
    zend_ssa_phi *p = ssa->vars[_var].phi_use_chain; \
151
1.23M
    int use = ssa->vars[_var].use_chain; \
152
2.50M
    while (use >= 0) { \
153
1.26M
      FOR_EACH_DEFINED_VAR(use, MACRO); \
154
1.26M
      use = zend_ssa_next_use(ssa->ops, _var, use); \
155
1.26M
    } \
156
1.23M
    p = ssa->vars[_var].phi_use_chain; \
157
1.59M
    while (p) { \
158
355k
      MACRO(p->ssa_var); \
159
355k
      p = zend_ssa_next_use_phi(ssa, _var, p); \
160
355k
    } \
161
1.23M
  } while (0)
162
163
3.54k
static inline bool add_will_overflow(zend_long a, zend_long b) {
164
3.54k
  return (b > 0 && a > ZEND_LONG_MAX - b)
165
3.52k
    || (b < 0 && a < ZEND_LONG_MIN - b);
166
3.54k
}
167
#if 0
168
static inline bool sub_will_overflow(zend_long a, zend_long b) {
169
  return (b > 0 && a < ZEND_LONG_MIN + b)
170
    || (b < 0 && a > ZEND_LONG_MAX + b);
171
}
172
#endif
173
174
#if 0
175
/* Recursive Pearce's SCC algorithm implementation */
176
static void zend_ssa_check_scc_var(const zend_op_array *op_array, zend_ssa *ssa, int var, int *index, zend_worklist_stack *stack) /* {{{ */
177
{
178
  int is_root = 1;
179
#ifdef SYM_RANGE
180
  zend_ssa_phi *p;
181
#endif
182
183
  ssa->vars[var].scc = *index;
184
  (*index)++;
185
186
  FOR_EACH_VAR_USAGE(var, CHECK_SCC_VAR);
187
188
#ifdef SYM_RANGE
189
  /* Process symbolic control-flow constraints */
190
  p = ssa->vars[var].sym_use_chain;
191
  while (p) {
192
    CHECK_SCC_VAR(p->ssa_var);
193
    p = p->sym_use_chain;
194
  }
195
#endif
196
197
  if (is_root) {
198
    ssa->sccs--;
199
    while (stack->len > 0) {
200
      int var2 = zend_worklist_stack_peek(stack);
201
      if (ssa->vars[var2].scc < ssa->vars[var].scc) {
202
        break;
203
      }
204
      zend_worklist_stack_pop(stack);
205
      ssa->vars[var2].scc = ssa->sccs;
206
      (*index)--;
207
    }
208
    ssa->vars[var].scc = ssa->sccs;
209
    ssa->vars[var].scc_entry = 1;
210
    (*index)--;
211
  } else {
212
    zend_worklist_stack_push(stack, var);
213
  }
214
}
215
/* }}} */
216
217
ZEND_API void zend_ssa_find_sccs(const zend_op_array *op_array, zend_ssa *ssa) /* {{{ */
218
{
219
  int index = 0;
220
  zend_worklist_stack stack;
221
  int j;
222
  ALLOCA_FLAG(stack_use_heap)
223
224
  ZEND_WORKLIST_STACK_ALLOCA(&stack, ssa->vars_count, stack_use_heap);
225
226
  /* Find SCCs using Pearce's algorithm. */
227
  ssa->sccs = ssa->vars_count;
228
  for (j = 0; j < ssa->vars_count; j++) {
229
    if (!ssa->vars[j].no_val && ssa->vars[j].scc < 0) {
230
      zend_ssa_check_scc_var(op_array, ssa, j, &index, &stack);
231
    }
232
  }
233
234
  if (ssa->sccs) {
235
    /* Shift SCC indexes. */
236
    for (j = 0; j < ssa->vars_count; j++) {
237
      if (ssa->vars[j].scc >= 0) {
238
        ssa->vars[j].scc -= ssa->sccs;
239
      }
240
    }
241
  }
242
  ssa->sccs = ssa->vars_count - ssa->sccs;
243
244
  for (j = 0; j < ssa->vars_count; j++) {
245
    if (ssa->vars[j].scc >= 0) {
246
      int var = j;
247
      FOR_EACH_VAR_USAGE(var, CHECK_SCC_ENTRY);
248
    }
249
  }
250
251
  ZEND_WORKLIST_STACK_FREE_ALLOCA(&stack, stack_use_heap);
252
}
253
/* }}} */
254
255
#else
256
/* Iterative Pearce's SCC algorithm implementation */
257
258
typedef struct _zend_scc_iterator {
259
  int               state;
260
  int               last;
261
  union {
262
    int           use;
263
    zend_ssa_phi *phi;
264
  };
265
} zend_scc_iterator;
266
267
static int zend_scc_next(const zend_op_array *op_array, const zend_ssa *ssa, int var, zend_scc_iterator *iterator) /* {{{ */
268
2.12M
{
269
2.12M
  zend_ssa_phi *phi;
270
2.12M
  int use, var2;
271
272
2.12M
  switch (iterator->state) {
273
1.04M
    case 0:                       goto state_0;
274
127k
    case 1:  use = iterator->use; goto state_1;
275
8.44k
    case 2:  use = iterator->use; goto state_2;
276
691k
    case 3:  use = iterator->use; goto state_3;
277
6.79k
    case 4:  use = iterator->use; goto state_4;
278
0
    case 5:  use = iterator->use; goto state_5;
279
91
    case 6:  use = iterator->use; goto state_6;
280
0
    case 7:  use = iterator->use; goto state_7;
281
3.82k
    case 8:  use = iterator->use; goto state_8;
282
234k
    case 9:  phi = iterator->phi; goto state_9;
283
0
#ifdef SYM_RANGE
284
4.43k
    case 10: phi = iterator->phi; goto state_10;
285
0
#endif
286
0
    case 11:                      goto state_11;
287
2.12M
  }
288
289
1.04M
state_0:
290
1.04M
  use = ssa->vars[var].use_chain;
291
2.11M
  while (use >= 0) {
292
1.06M
    iterator->use = use;
293
1.06M
    var2 = ssa->ops[use].op1_def;
294
1.06M
    if (var2 >= 0 && !ssa->vars[var2].no_val) {
295
127k
      iterator->state = 1;
296
127k
      return var2;
297
127k
    }
298
1.06M
state_1:
299
1.06M
    var2 = ssa->ops[use].op2_def;
300
1.06M
    if (var2 >= 0 && !ssa->vars[var2].no_val) {
301
8.44k
      iterator->state = 2;
302
8.44k
      return var2;
303
8.44k
    }
304
1.06M
state_2:
305
1.06M
    var2 = ssa->ops[use].result_def;
306
1.06M
    if (var2 >= 0 && !ssa->vars[var2].no_val) {
307
691k
      iterator->state = 3;
308
691k
      return var2;
309
691k
    }
310
1.06M
state_3:
311
1.06M
    if (op_array->opcodes[use].opcode == ZEND_OP_DATA) {
312
13.5k
      var2 = ssa->ops[use-1].op1_def;
313
13.5k
      if (var2 >= 0 && !ssa->vars[var2].no_val) {
314
6.79k
        iterator->state = 4;
315
6.79k
        return var2;
316
6.79k
      }
317
13.5k
state_4:
318
13.5k
      var2 = ssa->ops[use-1].op2_def;
319
13.5k
      if (var2 >= 0 && !ssa->vars[var2].no_val) {
320
0
        iterator->state = 5;
321
0
        return var2;
322
0
      }
323
13.5k
state_5:
324
13.5k
      var2 = ssa->ops[use-1].result_def;
325
13.5k
      if (var2 >= 0 && !ssa->vars[var2].no_val) {
326
3.82k
        iterator->state = 8;
327
3.82k
        return var2;
328
3.82k
      }
329
1.05M
    } else if ((uint32_t)use+1 < op_array->last &&
330
1.04M
               op_array->opcodes[use+1].opcode == ZEND_OP_DATA) {
331
22.6k
      var2 = ssa->ops[use+1].op1_def;
332
22.6k
      if (var2 >= 0 && !ssa->vars[var2].no_val) {
333
91
        iterator->state = 6;
334
91
        return var2;
335
91
      }
336
22.6k
state_6:
337
22.6k
      var2 = ssa->ops[use+1].op2_def;
338
22.6k
      if (var2 >= 0 && !ssa->vars[var2].no_val) {
339
0
        iterator->state = 7;
340
0
        return var2;
341
0
      }
342
22.6k
state_7:
343
22.6k
      var2 = ssa->ops[use+1].result_def;
344
22.6k
      if (var2 >= 0 && !ssa->vars[var2].no_val) {
345
0
        iterator->state = 8;
346
0
        return var2;
347
0
      }
348
22.6k
    }
349
1.06M
state_8:
350
1.06M
    use = zend_ssa_next_use(ssa->ops, var, use);
351
1.06M
  }
352
353
1.04M
  phi = ssa->vars[var].phi_use_chain;
354
1.28M
  while (phi) {
355
236k
    var2 = phi->ssa_var;
356
236k
    if (!ssa->vars[var2].no_val) {
357
234k
      iterator->state = 9;
358
234k
      iterator->phi = phi;
359
234k
      return var2;
360
234k
    }
361
236k
state_9:
362
236k
    phi = zend_ssa_next_use_phi(ssa, var, phi);
363
236k
  }
364
365
1.04M
#ifdef SYM_RANGE
366
  /* Process symbolic control-flow constraints */
367
1.04M
  phi = ssa->vars[var].sym_use_chain;
368
1.04M
  while (phi) {
369
4.63k
    var2 = phi->ssa_var;
370
4.63k
    if (!ssa->vars[var2].no_val) {
371
4.43k
      iterator->state = 10;
372
4.43k
      iterator->phi = phi;
373
4.43k
      return var2;
374
4.43k
    }
375
4.63k
state_10:
376
4.63k
    phi = phi->sym_use_chain;
377
4.63k
  }
378
1.04M
#endif
379
380
1.04M
  iterator->state = 11;
381
1.04M
state_11:
382
1.04M
  return -1;
383
1.04M
}
384
/* }}} */
385
386
static void zend_ssa_check_scc_var(const zend_op_array *op_array, zend_ssa *ssa, int var, int *index, zend_worklist_stack *stack, zend_worklist_stack *vstack, zend_scc_iterator *iterators) /* {{{ */
387
307k
{
388
1.04M
restart:
389
1.04M
  zend_worklist_stack_push(vstack, var);
390
1.04M
  iterators[var].state = 0;
391
1.04M
  iterators[var].last = -1;
392
1.04M
  ssa->vars[var].scc_entry = 1;
393
1.04M
  ssa->vars[var].scc = *index;
394
1.04M
  (*index)++;
395
396
2.08M
  while (vstack->len > 0) {
397
1.78M
    var = zend_worklist_stack_peek(vstack);
398
2.12M
    while (1) {
399
2.12M
      int var2;
400
401
2.12M
      if (iterators[var].last >= 0) {
402
        /* finish edge */
403
1.07M
        var2 = iterators[var].last;
404
1.07M
        if (ssa->vars[var2].scc < ssa->vars[var].scc) {
405
111k
          ssa->vars[var].scc = ssa->vars[var2].scc;
406
111k
          ssa->vars[var].scc_entry = 0;
407
111k
        }
408
1.07M
      }
409
2.12M
      var2 = zend_scc_next(op_array, ssa, var, iterators + var);
410
2.12M
      iterators[var].last = var2;
411
2.12M
      if (var2 < 0) break;
412
      /* begin edge */
413
1.07M
      if (ssa->vars[var2].scc < 0) {
414
737k
        var = var2;
415
737k
        goto restart;
416
737k
      }
417
1.07M
    }
418
419
    /* finish visiting */
420
1.04M
    zend_worklist_stack_pop(vstack);
421
1.04M
    if (ssa->vars[var].scc_entry) {
422
936k
      ssa->sccs--;
423
1.04M
      while (stack->len > 0) {
424
131k
        int var2 = zend_worklist_stack_peek(stack);
425
131k
        if (ssa->vars[var2].scc < ssa->vars[var].scc) {
426
22.2k
          break;
427
22.2k
        }
428
108k
        zend_worklist_stack_pop(stack);
429
108k
        ssa->vars[var2].scc = ssa->sccs;
430
108k
        (*index)--;
431
108k
      }
432
936k
      ssa->vars[var].scc = ssa->sccs;
433
936k
      (*index)--;
434
936k
    } else {
435
108k
      zend_worklist_stack_push(stack, var);
436
108k
    }
437
1.04M
  }
438
1.04M
}
439
/* }}} */
440
441
ZEND_API void zend_ssa_find_sccs(const zend_op_array *op_array, zend_ssa *ssa) /* {{{ */
442
71.2k
{
443
71.2k
  int index = 0;
444
71.2k
  zend_worklist_stack stack, vstack;
445
71.2k
  zend_scc_iterator *iterators;
446
71.2k
  int j;
447
71.2k
  ALLOCA_FLAG(stack_use_heap)
448
71.2k
  ALLOCA_FLAG(vstack_use_heap)
449
71.2k
  ALLOCA_FLAG(iterators_use_heap)
450
451
71.2k
  iterators = do_alloca(sizeof(zend_scc_iterator) * ssa->vars_count, iterators_use_heap);
452
71.2k
  ZEND_WORKLIST_STACK_ALLOCA(&vstack, ssa->vars_count, vstack_use_heap);
453
71.2k
  ZEND_WORKLIST_STACK_ALLOCA(&stack, ssa->vars_count, stack_use_heap);
454
455
  /* Find SCCs using Pearce's algorithm. */
456
71.2k
  ssa->sccs = ssa->vars_count;
457
1.19M
  for (j = 0; j < ssa->vars_count; j++) {
458
1.12M
    if (!ssa->vars[j].no_val && ssa->vars[j].scc < 0) {
459
307k
      zend_ssa_check_scc_var(op_array, ssa, j, &index, &stack, &vstack, iterators);
460
307k
    }
461
1.12M
  }
462
463
71.2k
  if (ssa->sccs) {
464
    /* Shift SCC indexes. */
465
721k
    for (j = 0; j < ssa->vars_count; j++) {
466
691k
      if (ssa->vars[j].scc >= 0) {
467
613k
        ssa->vars[j].scc -= ssa->sccs;
468
613k
      }
469
691k
    }
470
29.2k
  }
471
71.2k
  ssa->sccs = ssa->vars_count - ssa->sccs;
472
473
1.19M
  for (j = 0; j < ssa->vars_count; j++) {
474
1.12M
    if (ssa->vars[j].scc >= 0) {
475
1.04M
      int var = j;
476
1.09M
      FOR_EACH_VAR_USAGE(var, CHECK_SCC_ENTRY);
477
1.04M
    }
478
1.12M
  }
479
480
71.2k
  ZEND_WORKLIST_STACK_FREE_ALLOCA(&stack, stack_use_heap);
481
71.2k
  ZEND_WORKLIST_STACK_FREE_ALLOCA(&vstack, vstack_use_heap);
482
71.2k
  free_alloca(iterators, iterators_use_heap);
483
71.2k
}
484
/* }}} */
485
486
#endif
487
488
ZEND_API void zend_ssa_find_false_dependencies(const zend_op_array *op_array, const zend_ssa *ssa) /* {{{ */
489
71.2k
{
490
71.2k
  zend_ssa_var *ssa_vars = ssa->vars;
491
71.2k
  const zend_ssa_op *ssa_ops = ssa->ops;
492
71.2k
  int ssa_vars_count = ssa->vars_count;
493
71.2k
  zend_bitset worklist;
494
71.2k
  int i, use;
495
71.2k
  const zend_ssa_phi *p;
496
71.2k
  ALLOCA_FLAG(use_heap);
497
498
71.2k
  if (!op_array->function_name || !ssa->vars || !ssa->ops) {
499
30.4k
    return;
500
30.4k
  }
501
502
40.8k
  worklist = do_alloca(sizeof(zend_ulong) * zend_bitset_len(ssa_vars_count), use_heap);
503
40.8k
  memset(worklist, 0, sizeof(zend_ulong) * zend_bitset_len(ssa_vars_count));
504
505
430k
  for (i = 0; i < ssa_vars_count; i++) {
506
389k
    ssa_vars[i].no_val = 1; /* mark as unused */
507
389k
    use = ssa->vars[i].use_chain;
508
419k
    while (use >= 0) {
509
295k
      if (!zend_ssa_is_no_val_use(&op_array->opcodes[use], &ssa->ops[use], i)) {
510
265k
        ssa_vars[i].no_val = 0; /* used directly */
511
265k
        zend_bitset_incl(worklist, i);
512
265k
        break;
513
265k
      }
514
29.8k
      use = zend_ssa_next_use(ssa_ops, i, use);
515
29.8k
    }
516
389k
  }
517
518
1.19M
  WHILE_WORKLIST(worklist, zend_bitset_len(ssa_vars_count), i) {
519
1.19M
    if (ssa_vars[i].definition_phi) {
520
      /* mark all possible sources as used */
521
49.9k
      p = ssa_vars[i].definition_phi;
522
49.9k
      if (p->pi >= 0) {
523
16.2k
        if (ssa_vars[p->sources[0]].no_val) {
524
87
          ssa_vars[p->sources[0]].no_val = 0; /* used indirectly */
525
87
          zend_bitset_incl(worklist, p->sources[0]);
526
87
        }
527
33.6k
      } else {
528
103k
        for (uint32_t j = 0; j < ssa->cfg.blocks[p->block].predecessors_count; j++) {
529
69.7k
          ZEND_ASSERT(p->sources[j] >= 0);
530
69.7k
          if (ssa->vars[p->sources[j]].no_val) {
531
46.1k
            ssa_vars[p->sources[j]].no_val = 0; /* used indirectly */
532
46.1k
            zend_bitset_incl(worklist, p->sources[j]);
533
46.1k
          }
534
69.7k
        }
535
33.6k
      }
536
49.9k
    }
537
1.19M
  } WHILE_WORKLIST_END();
538
539
40.8k
  free_alloca(worklist, use_heap);
540
40.8k
}
541
/* }}} */
542
543
/* From "Hacker's Delight" */
544
static zend_ulong minOR(zend_ulong a, zend_ulong b, zend_ulong c, zend_ulong d)
545
102
{
546
102
  zend_ulong m, temp;
547
548
102
  m = Z_UL(1) << (sizeof(zend_ulong) * 8 - 1);
549
6.11k
  while (m != 0) {
550
6.05k
    if (~a & c & m) {
551
62
      temp = (a | m) & -m;
552
62
      if (temp <= b) {
553
42
        a = temp;
554
42
        break;
555
42
      }
556
5.99k
    } else if (a & ~c & m) {
557
442
      temp = (c | m) & -m;
558
442
      if (temp <= d) {
559
0
        c = temp;
560
0
        break;
561
0
      }
562
442
    }
563
6.01k
    m = m >> 1;
564
6.01k
  }
565
102
  return a | c;
566
102
}
567
568
static zend_ulong maxOR(zend_ulong a, zend_ulong b, zend_ulong c, zend_ulong d)
569
172
{
570
172
  zend_ulong m, temp;
571
572
172
  m = Z_UL(1) << (sizeof(zend_ulong) * 8 - 1);
573
6.23k
  while (m != 0) {
574
6.19k
    if (b & d & m) {
575
130
      temp = (b - m) | (m - 1);
576
130
      if (temp >= a) {
577
130
        b = temp;
578
130
        break;
579
130
      }
580
0
      temp = (d - m) | (m - 1);
581
0
      if (temp >= c) {
582
0
        d = temp;
583
0
        break;
584
0
      }
585
0
    }
586
6.06k
    m = m >> 1;
587
6.06k
  }
588
172
  return b | d;
589
172
}
590
591
static zend_ulong minAND(zend_ulong a, zend_ulong b, zend_ulong c, zend_ulong d)
592
823
{
593
823
  zend_ulong m, temp;
594
595
823
  m = Z_UL(1) << (sizeof(zend_ulong) * 8 - 1);
596
25.0k
  while (m != 0) {
597
24.9k
    if (~a & ~c & m) {
598
17.9k
      temp = (a | m) & -m;
599
17.9k
      if (temp <= b) {
600
533
        a = temp;
601
533
        break;
602
533
      }
603
17.4k
      temp = (c | m) & -m;
604
17.4k
      if (temp <= d) {
605
204
        c = temp;
606
204
        break;
607
204
      }
608
17.4k
    }
609
24.2k
    m = m >> 1;
610
24.2k
  }
611
823
  return a & c;
612
823
}
613
614
static zend_ulong maxAND(zend_ulong a, zend_ulong b, zend_ulong c, zend_ulong d)
615
458
{
616
458
  zend_ulong m, temp;
617
618
458
  m = Z_UL(1) << (sizeof(zend_ulong) * 8 - 1);
619
16.2k
  while (m != 0) {
620
16.1k
    if (b & ~d & m) {
621
18
      temp = (b | ~m) | (m - 1);
622
18
      if (temp >= a) {
623
18
        b = temp;
624
18
        break;
625
18
      }
626
16.1k
    } else if (~b & d & m) {
627
400
      temp = (d | ~m) | (m - 1);
628
400
      if (temp >= c) {
629
400
        d = temp;
630
400
        break;
631
400
      }
632
400
    }
633
15.7k
    m = m >> 1;
634
15.7k
  }
635
458
  return b & d;
636
458
}
637
638
/* Based on "Hacker's Delight" */
639
640
/*
641
0: + + + + 0 0 0 0 => 0 0 + min/max
642
2: + + - + 0 0 1 0 => 1 0 ? min(a,b,c,-1)/max(a,b,0,d)
643
3: + + - - 0 0 1 1 => 1 1 - min/max
644
8: - + + + 1 0 0 0 => 1 0 ? min(a,-1,b,d)/max(0,b,c,d)
645
a: - + - + 1 0 1 0 => 1 0 ? MIN(a,c)/max(0,b,0,d)
646
b: - + - - 1 0 1 1 => 1 1 - c/-1
647
c: - - + + 1 1 0 0 => 1 1 - min/max
648
e: - - - + 1 1 1 0 => 1 1 - a/-1
649
f  - - - - 1 1 1 1 => 1 1 - min/max
650
*/
651
static void zend_ssa_range_or(zend_long a, zend_long b, zend_long c, zend_long d, zend_ssa_range *tmp)
652
181
{
653
181
  int x = ((a < 0) ? 8 : 0) |
654
181
          ((b < 0) ? 4 : 0) |
655
181
          ((c < 0) ? 2 : 0) |
656
181
          ((d < 0) ? 1 : 0);
657
181
  switch (x) {
658
34
    case 0x0:
659
34
    case 0x3:
660
40
    case 0xc:
661
40
    case 0xf:
662
40
      tmp->min = minOR(a, b, c, d);
663
40
      tmp->max = maxOR(a, b, c, d);
664
40
      break;
665
2
    case 0x2:
666
2
      tmp->min = minOR(a, b, c, -1);
667
2
      tmp->max = maxOR(a, b, 0, d);
668
2
      break;
669
60
    case 0x8:
670
60
      tmp->min = minOR(a, -1, c, d);
671
60
      tmp->max = maxOR(0, b, c, d);
672
60
      break;
673
70
    case 0xa:
674
70
      tmp->min = MIN(a, c);
675
70
      tmp->max = maxOR(0, b, 0, d);
676
70
      break;
677
9
    case 0xb:
678
9
      tmp->min = c;
679
9
      tmp->max = -1;
680
9
      break;
681
0
    case 0xe:
682
0
      tmp->min = a;
683
0
      tmp->max = -1;
684
0
      break;
685
181
  }
686
181
}
687
688
/*
689
0: + + + + 0 0 0 0 => 0 0 + min/max
690
2: + + - + 0 0 1 0 => 0 0 + 0/b
691
3: + + - - 0 0 1 1 => 0 0 + min/max
692
8: - + + + 1 0 0 0 => 0 0 + 0/d
693
a: - + - + 1 0 1 0 => 1 0 ? min(a,-1,c,-1)/NAX(b,d)
694
b: - + - - 1 0 1 1 => 1 0 ? min(a,-1,c,d)/max(0,b,c,d)
695
c: - - + + 1 1 0 0 => 1 1 - min/max
696
e: - - - + 1 1 1 0 => 1 0 ? min(a,b,c,-1)/max(a,b,0,d)
697
f  - - - - 1 1 1 1 => 1 1 - min/max
698
*/
699
static void zend_ssa_range_and(zend_long a, zend_long b, zend_long c, zend_long d, zend_ssa_range *tmp)
700
1.21k
{
701
1.21k
  int x = ((a < 0) ? 8 : 0) |
702
1.21k
          ((b < 0) ? 4 : 0) |
703
1.21k
          ((c < 0) ? 2 : 0) |
704
1.21k
          ((d < 0) ? 1 : 0);
705
1.21k
  switch (x) {
706
416
    case 0x0:
707
446
    case 0x3:
708
446
    case 0xc:
709
446
    case 0xf:
710
446
      tmp->min = minAND(a, b, c, d);
711
446
      tmp->max = maxAND(a, b, c, d);
712
446
      break;
713
122
    case 0x2:
714
122
      tmp->min = 0;
715
122
      tmp->max = b;
716
122
      break;
717
272
    case 0x8:
718
272
      tmp->min = 0;
719
272
      tmp->max = d;
720
272
      break;
721
365
    case 0xa:
722
365
      tmp->min = minAND(a, -1, c, -1);
723
365
      tmp->max = MAX(b, d);
724
365
      break;
725
12
    case 0xb:
726
12
      tmp->min = minAND(a, -1, c, d);
727
12
      tmp->max = maxAND(0, b, c, d);
728
12
      break;
729
0
    case 0xe:
730
0
      tmp->min = minAND(a, b, c, -1);
731
0
      tmp->max = maxAND(a, b, 0, d);
732
0
      break;
733
1.21k
  }
734
1.21k
}
735
736
static inline bool zend_abs_range(
737
352
    zend_long min, zend_long max, zend_long *abs_min, zend_long *abs_max) {
738
352
  if (min == ZEND_LONG_MIN) {
739
    /* Cannot take absolute value of LONG_MIN  */
740
29
    return 0;
741
29
  }
742
743
323
  if (min >= 0) {
744
299
    *abs_min = min;
745
299
    *abs_max = max;
746
299
  } else if (max <= 0) {
747
24
    *abs_min = -max;
748
24
    *abs_max = -min;
749
24
  } else {
750
    /* Range crossing zero */
751
0
    *abs_min = 0;
752
0
    *abs_max = MAX(max, -min);
753
0
  }
754
755
323
  return 1;
756
352
}
757
758
718
static inline zend_long safe_shift_left(zend_long n, zend_long s) {
759
718
  return (zend_long) ((zend_ulong) n << (zend_ulong) s);
760
718
}
761
762
303
static inline bool shift_left_overflows(zend_long n, zend_long s) {
763
  /* This considers shifts that shift in the sign bit to be overflowing as well */
764
303
  if (n >= 0) {
765
217
    return s >= SIZEOF_ZEND_LONG * 8 - 1 || safe_shift_left(n, s) < n;
766
217
  } else {
767
86
    return s >= SIZEOF_ZEND_LONG * 8 || safe_shift_left(n, s) > n;
768
86
  }
769
303
}
770
771
/* If b does not divide a exactly, return the two adjacent values between which the real result
772
 * lies. */
773
292
static void float_div(zend_long a, zend_long b, zend_long *r1, zend_long *r2) {
774
292
  *r1 = *r2 = a / b;
775
292
  if (a % b != 0) {
776
236
    if (*r2 < 0) {
777
114
      (*r2)--;
778
122
    } else {
779
122
      (*r2)++;
780
122
    }
781
236
  }
782
292
}
783
784
static bool zend_inference_calc_binary_op_range(
785
    const zend_op_array *op_array, const zend_ssa *ssa,
786
84.5k
    const zend_op *opline, const zend_ssa_op *ssa_op, uint8_t opcode, zend_ssa_range *tmp) {
787
84.5k
  zend_long op1_min, op2_min, op1_max, op2_max, t1, t2, t3, t4;
788
789
84.5k
  switch (opcode) {
790
29.6k
    case ZEND_ADD:
791
29.6k
      if (OP1_HAS_RANGE() && OP2_HAS_RANGE()) {
792
18.6k
        op1_min = OP1_MIN_RANGE();
793
18.6k
        op2_min = OP2_MIN_RANGE();
794
18.6k
        op1_max = OP1_MAX_RANGE();
795
18.6k
        op2_max = OP2_MAX_RANGE();
796
18.6k
        if (OP1_RANGE_UNDERFLOW() ||
797
3.62k
          OP2_RANGE_UNDERFLOW() ||
798
17.3k
          zend_add_will_overflow(op1_min, op2_min)) {
799
17.3k
          tmp->underflow = 1;
800
17.3k
          tmp->min = ZEND_LONG_MIN;
801
17.3k
        } else {
802
1.30k
          tmp->min = op1_min + op2_min;
803
1.30k
        }
804
18.6k
        if (OP1_RANGE_OVERFLOW() ||
805
5.04k
          OP2_RANGE_OVERFLOW() ||
806
17.5k
          zend_add_will_overflow(op1_max, op2_max)) {
807
17.5k
          tmp->overflow = 1;
808
17.5k
          tmp->max = ZEND_LONG_MAX;
809
17.5k
        } else {
810
1.11k
          tmp->max = op1_max + op2_max;
811
1.11k
        }
812
18.6k
        return 1;
813
18.6k
      }
814
10.9k
      break;
815
15.2k
    case ZEND_SUB:
816
15.2k
      if (OP1_HAS_RANGE() && OP2_HAS_RANGE()) {
817
12.4k
        op1_min = OP1_MIN_RANGE();
818
12.4k
        op2_min = OP2_MIN_RANGE();
819
12.4k
        op1_max = OP1_MAX_RANGE();
820
12.4k
        op2_max = OP2_MAX_RANGE();
821
12.4k
        if (OP1_RANGE_UNDERFLOW() ||
822
5.70k
          OP2_RANGE_OVERFLOW() ||
823
8.46k
          zend_sub_will_overflow(op1_min, op2_max)) {
824
8.46k
          tmp->underflow = 1;
825
8.46k
          tmp->min = ZEND_LONG_MIN;
826
8.46k
        } else {
827
4.02k
          tmp->min = op1_min - op2_max;
828
4.02k
        }
829
12.4k
        if (OP1_RANGE_OVERFLOW() ||
830
5.56k
          OP2_RANGE_UNDERFLOW() ||
831
8.18k
          zend_sub_will_overflow(op1_max, op2_min)) {
832
8.18k
          tmp->overflow = 1;
833
8.18k
          tmp->max = ZEND_LONG_MAX;
834
8.18k
        } else {
835
4.31k
          tmp->max = op1_max - op2_min;
836
4.31k
        }
837
12.4k
        return 1;
838
12.4k
      }
839
2.75k
      break;
840
11.0k
    case ZEND_MUL:
841
11.0k
      if (OP1_HAS_RANGE() && OP2_HAS_RANGE()) {
842
7.02k
        double dummy;
843
7.02k
        zend_long t1_overflow, t2_overflow, t3_overflow, t4_overflow;
844
7.02k
        op1_min = OP1_MIN_RANGE();
845
7.02k
        op2_min = OP2_MIN_RANGE();
846
7.02k
        op1_max = OP1_MAX_RANGE();
847
7.02k
        op2_max = OP2_MAX_RANGE();
848
        /* Suppress uninit variable warnings, these will only be used if the overflow
849
         * flags are all false. */
850
7.02k
        t1 = t2 = t3 = t4 = 0;
851
7.02k
        ZEND_SIGNED_MULTIPLY_LONG(op1_min, op2_min, t1, dummy, t1_overflow);
852
7.02k
        ZEND_SIGNED_MULTIPLY_LONG(op1_min, op2_max, t2, dummy, t2_overflow);
853
7.02k
        ZEND_SIGNED_MULTIPLY_LONG(op1_max, op2_min, t3, dummy, t3_overflow);
854
7.02k
        ZEND_SIGNED_MULTIPLY_LONG(op1_max, op2_max, t4, dummy, t4_overflow);
855
7.02k
        (void) dummy;
856
857
        // FIXME: more careful overflow checks?
858
7.02k
        if (OP1_RANGE_UNDERFLOW() || OP2_RANGE_UNDERFLOW() ||
859
494
          OP1_RANGE_OVERFLOW() || OP2_RANGE_OVERFLOW()  ||
860
485
          t1_overflow || t2_overflow || t3_overflow || t4_overflow
861
7.02k
        ) {
862
6.74k
          tmp->underflow = 1;
863
6.74k
          tmp->overflow = 1;
864
6.74k
          tmp->min = ZEND_LONG_MIN;
865
6.74k
          tmp->max = ZEND_LONG_MAX;
866
6.74k
        } else {
867
280
          tmp->min = MIN(MIN(t1, t2), MIN(t3, t4));
868
280
          tmp->max = MAX(MAX(t1, t2), MAX(t3, t4));
869
280
        }
870
7.02k
        return 1;
871
7.02k
      }
872
4.02k
      break;
873
4.79k
    case ZEND_DIV:
874
4.79k
      if (OP1_HAS_RANGE() && OP2_HAS_RANGE()) {
875
3.32k
        op1_min = OP1_MIN_RANGE();
876
3.32k
        op2_min = OP2_MIN_RANGE();
877
3.32k
        op1_max = OP1_MAX_RANGE();
878
3.32k
        op2_max = OP2_MAX_RANGE();
879
880
        /* If op2 crosses zero, then floating point values close to zero might be
881
         * possible, which will result in arbitrarily large results (overflow). Also
882
         * avoid dividing LONG_MIN by -1, which is UB. */
883
3.32k
        if (OP1_RANGE_UNDERFLOW() || OP2_RANGE_UNDERFLOW() ||
884
101
          OP1_RANGE_OVERFLOW() || OP2_RANGE_OVERFLOW() ||
885
97
          (op2_min <= 0 && op2_max >= 0) ||
886
73
          (op1_min == ZEND_LONG_MIN && op2_max == -1)
887
3.32k
        ) {
888
3.25k
          tmp->underflow = 1;
889
3.25k
          tmp->overflow = 1;
890
3.25k
          tmp->min = ZEND_LONG_MIN;
891
3.25k
          tmp->max = ZEND_LONG_MAX;
892
3.25k
        } else {
893
73
          zend_long t1_, t2_, t3_, t4_;
894
73
          float_div(op1_min, op2_min, &t1, &t1_);
895
73
          float_div(op1_min, op2_max, &t2, &t2_);
896
73
          float_div(op1_max, op2_min, &t3, &t3_);
897
73
          float_div(op1_max, op2_max, &t4, &t4_);
898
899
73
          tmp->min = MIN(MIN(MIN(t1, t2), MIN(t3, t4)), MIN(MIN(t1_, t2_), MIN(t3_, t4_)));
900
73
          tmp->max = MAX(MAX(MAX(t1, t2), MAX(t3, t4)), MAX(MAX(t1_, t2_), MAX(t3_, t4_)));
901
73
        }
902
3.32k
        return 1;
903
3.32k
      }
904
1.46k
      break;
905
3.18k
    case ZEND_MOD:
906
3.18k
      if (OP1_HAS_RANGE() && OP2_HAS_RANGE()) {
907
1.42k
        if (OP1_RANGE_UNDERFLOW() ||
908
607
          OP2_RANGE_UNDERFLOW() ||
909
361
          OP1_RANGE_OVERFLOW()  ||
910
1.06k
          OP2_RANGE_OVERFLOW()) {
911
1.06k
          tmp->min = ZEND_LONG_MIN;
912
1.06k
          tmp->max = ZEND_LONG_MAX;
913
1.06k
        } else {
914
352
          zend_long op2_abs_min, op2_abs_max;
915
916
352
          op1_min = OP1_MIN_RANGE();
917
352
          op2_min = OP2_MIN_RANGE();
918
352
          op1_max = OP1_MAX_RANGE();
919
352
          op2_max = OP2_MAX_RANGE();
920
352
          if (!zend_abs_range(op2_min, op2_max, &op2_abs_min, &op2_abs_max)) {
921
29
            break;
922
29
          }
923
924
323
          if (op2_abs_max == 0) {
925
            /* Always modulus by zero, nothing we can do */
926
106
            break;
927
106
          }
928
217
          if (op2_abs_min == 0) {
929
            /* Ignore the modulus by zero case, which will throw */
930
0
            op2_abs_min++;
931
0
          }
932
933
217
          if (op1_min >= 0) {
934
12
            tmp->min = op1_max < op2_abs_min ? op1_min : 0;
935
12
            tmp->max = MIN(op1_max, op2_abs_max - 1);
936
205
          } else if (op1_max <= 0) {
937
0
            tmp->min = MAX(op1_min, -op2_abs_max + 1);
938
0
            tmp->max = op1_min > -op2_abs_min ? op1_max : 0;
939
205
          } else {
940
205
            tmp->min = MAX(op1_min, -op2_abs_max + 1);
941
205
            tmp->max = MIN(op1_max, op2_abs_max - 1);
942
205
          }
943
217
        }
944
1.28k
        return 1;
945
1.42k
      }
946
1.76k
      break;
947
1.76k
    case ZEND_SL:
948
1.59k
      if (OP1_HAS_RANGE() && OP2_HAS_RANGE()) {
949
702
        if (OP1_RANGE_UNDERFLOW() ||
950
600
          OP2_RANGE_UNDERFLOW() ||
951
214
          OP1_RANGE_OVERFLOW() ||
952
511
          OP2_RANGE_OVERFLOW()) {
953
511
          tmp->min = ZEND_LONG_MIN;
954
511
          tmp->max = ZEND_LONG_MAX;
955
511
        } else {
956
191
          op1_min = OP1_MIN_RANGE();
957
191
          op2_min = OP2_MIN_RANGE();
958
191
          op1_max = OP1_MAX_RANGE();
959
191
          op2_max = OP2_MAX_RANGE();
960
961
          /* Shifts by negative numbers will throw, ignore them */
962
191
          if (op2_min < 0) {
963
9
            op2_min = 0;
964
9
          }
965
191
          if (op2_max < 0) {
966
4
            op2_max = 0;
967
4
          }
968
969
191
          if (shift_left_overflows(op1_min, op2_max)
970
112
              || shift_left_overflows(op1_max, op2_max)) {
971
79
            tmp->min = ZEND_LONG_MIN;
972
79
            tmp->max = ZEND_LONG_MAX;
973
112
          } else {
974
112
            t1 = safe_shift_left(op1_min, op2_min);
975
112
            t2 = safe_shift_left(op1_min, op2_max);
976
112
            t3 = safe_shift_left(op1_max, op2_min);
977
112
            t4 = safe_shift_left(op1_max, op2_max);
978
112
            tmp->min = MIN(MIN(t1, t2), MIN(t3, t4));
979
112
            tmp->max = MAX(MAX(t1, t2), MAX(t3, t4));
980
112
          }
981
191
        }
982
702
        return 1;
983
702
      }
984
897
      break;
985
1.16k
    case ZEND_SR:
986
1.16k
      if (OP1_HAS_RANGE() && OP2_HAS_RANGE()) {
987
483
        if (OP1_RANGE_UNDERFLOW() ||
988
311
          OP2_RANGE_UNDERFLOW() ||
989
167
          OP1_RANGE_OVERFLOW() ||
990
358
          OP2_RANGE_OVERFLOW()) {
991
358
          tmp->min = ZEND_LONG_MIN;
992
358
          tmp->max = ZEND_LONG_MAX;
993
358
        } else {
994
125
          op1_min = OP1_MIN_RANGE();
995
125
          op2_min = OP2_MIN_RANGE();
996
125
          op1_max = OP1_MAX_RANGE();
997
125
          op2_max = OP2_MAX_RANGE();
998
999
          /* Shifts by negative numbers will throw, ignore them */
1000
125
          if (op2_min < 0) {
1001
74
            op2_min = 0;
1002
74
          }
1003
125
          if (op2_max < 0) {
1004
10
            op2_max = 0;
1005
10
          }
1006
1007
          /* Shifts by more than the integer size will be 0 or -1 */
1008
125
          if (op2_min >= SIZEOF_ZEND_LONG * 8) {
1009
14
            op2_min = SIZEOF_ZEND_LONG * 8 - 1;
1010
14
          }
1011
125
          if (op2_max >= SIZEOF_ZEND_LONG * 8) {
1012
22
            op2_max = SIZEOF_ZEND_LONG * 8 - 1;
1013
22
          }
1014
1015
125
          t1 = op1_min >> op2_min;
1016
125
          t2 = op1_min >> op2_max;
1017
125
          t3 = op1_max >> op2_min;
1018
125
          t4 = op1_max >> op2_max;
1019
125
          tmp->min = MIN(MIN(t1, t2), MIN(t3, t4));
1020
125
          tmp->max = MAX(MAX(t1, t2), MAX(t3, t4));
1021
125
        }
1022
483
        return 1;
1023
483
      }
1024
679
      break;
1025
1.80k
    case ZEND_BW_OR:
1026
1.80k
      if (OP1_HAS_RANGE() && OP2_HAS_RANGE()) {
1027
1.25k
        if (OP1_RANGE_UNDERFLOW() ||
1028
371
          OP2_RANGE_UNDERFLOW() ||
1029
183
          OP1_RANGE_OVERFLOW() ||
1030
1.07k
          OP2_RANGE_OVERFLOW()) {
1031
1.07k
          tmp->min = ZEND_LONG_MIN;
1032
1.07k
          tmp->max = ZEND_LONG_MAX;
1033
1.07k
        } else {
1034
181
          op1_min = OP1_MIN_RANGE();
1035
181
          op2_min = OP2_MIN_RANGE();
1036
181
          op1_max = OP1_MAX_RANGE();
1037
181
          op2_max = OP2_MAX_RANGE();
1038
181
          zend_ssa_range_or(op1_min, op1_max, op2_min, op2_max, tmp);
1039
181
        }
1040
1.25k
        return 1;
1041
1.25k
      }
1042
543
      break;
1043
13.1k
    case ZEND_BW_AND:
1044
13.1k
      if (OP1_HAS_RANGE() && OP2_HAS_RANGE()) {
1045
10.6k
        if (OP1_RANGE_UNDERFLOW() ||
1046
3.78k
          OP2_RANGE_UNDERFLOW() ||
1047
1.22k
          OP1_RANGE_OVERFLOW() ||
1048
9.40k
          OP2_RANGE_OVERFLOW()) {
1049
9.40k
          tmp->min = ZEND_LONG_MIN;
1050
9.40k
          tmp->max = ZEND_LONG_MAX;
1051
9.40k
        } else {
1052
1.21k
          op1_min = OP1_MIN_RANGE();
1053
1.21k
          op2_min = OP2_MIN_RANGE();
1054
1.21k
          op1_max = OP1_MAX_RANGE();
1055
1.21k
          op2_max = OP2_MAX_RANGE();
1056
1.21k
          zend_ssa_range_and(op1_min, op1_max, op2_min, op2_max, tmp);
1057
1.21k
        }
1058
10.6k
        return 1;
1059
10.6k
      }
1060
2.54k
      break;
1061
2.93k
    case ZEND_BW_XOR:
1062
      // TODO
1063
2.93k
      break;
1064
0
    default: ZEND_UNREACHABLE();
1065
84.5k
  }
1066
28.7k
  return 0;
1067
84.5k
}
1068
1069
static bool zend_inference_calc_range(const zend_op_array *op_array, const zend_ssa *ssa, int var, int widening, bool narrowing, zend_ssa_range *tmp)
1070
1.58M
{
1071
1.58M
  uint32_t line;
1072
1.58M
  const zend_op *opline;
1073
1.58M
  const zend_ssa_op *ssa_op;
1074
1075
1.58M
  if (ssa->vars[var].definition_phi) {
1076
312k
    const zend_ssa_phi *p = ssa->vars[var].definition_phi;
1077
1078
312k
    tmp->underflow = 0;
1079
312k
    tmp->min = ZEND_LONG_MAX;
1080
312k
    tmp->max = ZEND_LONG_MIN;
1081
312k
    tmp->overflow = 0;
1082
312k
    if (p->pi >= 0 && p->has_range_constraint) {
1083
38.7k
      const zend_ssa_range_constraint *constraint = &p->constraint.range;
1084
38.7k
      if (constraint->negative) {
1085
3.77k
        int src1 = p->sources[0];
1086
1087
3.77k
        if (ssa->var_info[src1].has_range) {
1088
3.41k
          *tmp = ssa->var_info[src1].range;
1089
3.41k
          if (constraint->range.min == constraint->range.max
1090
3.41k
           && !constraint->range.underflow
1091
3.41k
           && !constraint->range.overflow
1092
3.41k
           && p->constraint.range.min_ssa_var < 0
1093
634
           && p->constraint.range.max_ssa_var < 0
1094
634
           && ssa->vars[src1].definition >= 0) {
1095
            /* Check for constrained induction variable */
1096
131
            line = ssa->vars[src1].definition;
1097
131
            opline = op_array->opcodes + line;
1098
131
            switch (opline->opcode) {
1099
0
              case ZEND_PRE_DEC:
1100
52
              case ZEND_POST_DEC:
1101
52
                if (!tmp->underflow) {
1102
40
                  const zend_ssa_phi *p = ssa->vars[ssa->ops[line].op1_use].definition_phi;
1103
1104
40
                  if (p && p->pi < 0
1105
40
                   && ssa->cfg.blocks[p->block].predecessors_count == 2
1106
40
                   && p->sources[1] == var
1107
40
                   && ssa->var_info[p->sources[0]].has_range
1108
40
                   && ssa->var_info[p->sources[0]].range.min > constraint->range.max) {
1109
40
                    tmp->min = constraint->range.max + 1;
1110
40
                  }
1111
40
                }
1112
52
                break;
1113
0
              case ZEND_PRE_INC:
1114
6
              case ZEND_POST_INC:
1115
6
                if (!tmp->overflow) {
1116
0
                  const zend_ssa_phi *p = ssa->vars[ssa->ops[line].op1_use].definition_phi;
1117
1118
0
                  if (p && p->pi < 0
1119
0
                   && ssa->cfg.blocks[p->block].predecessors_count == 2
1120
0
                   && p->sources[1] == var
1121
0
                   && ssa->var_info[p->sources[0]].has_range
1122
0
                   && ssa->var_info[p->sources[0]].range.max < constraint->range.min) {
1123
0
                    tmp->max = constraint->range.min - 1;
1124
0
                  }
1125
0
                }
1126
6
                break;
1127
131
            }
1128
131
          }
1129
3.41k
        } else if (narrowing) {
1130
325
          tmp->underflow = 1;
1131
325
          tmp->min = ZEND_LONG_MIN;
1132
325
          tmp->max = ZEND_LONG_MAX;
1133
325
          tmp->overflow = 1;
1134
325
        }
1135
1136
#ifdef NEG_RANGE
1137
        if (constraint->min_ssa_var < 0 &&
1138
            constraint->max_ssa_var < 0 &&
1139
            ssa->var_info[p->ssa_var].has_range) {
1140
          LOG_NEG_RANGE("%s() #%d [%ld..%ld] -> [%ld..%ld]?\n",
1141
            ZSTR_VAL(op_array->function_name),
1142
            p->ssa_var,
1143
            ssa->var_info[p->ssa_var].range.min,
1144
            ssa->var_info[p->ssa_var].range.max,
1145
            tmp->min,
1146
            tmp->max);
1147
          if (constraint->negative == NEG_USE_LT &&
1148
              tmp->max >= constraint->range.min) {
1149
            tmp->overflow = 0;
1150
            tmp->max = constraint->range.min - 1;
1151
            LOG_NEG_RANGE("  => [%ld..%ld]\n", tmp->min, tmp->max);
1152
          } else if (constraint->negative == NEG_USE_GT &&
1153
                     tmp->min <= constraint->range.max) {
1154
            tmp->underflow = 0;
1155
            tmp->min = constraint->range.max + 1;
1156
            LOG_NEG_RANGE("  => [%ld..%ld]\n", tmp->min, tmp->max);
1157
          }
1158
        }
1159
#endif
1160
35.0k
      } else if (ssa->var_info[p->sources[0]].has_range) {
1161
        /* intersection */
1162
34.6k
        *tmp = ssa->var_info[p->sources[0]].range;
1163
34.6k
        if (constraint->min_ssa_var < 0) {
1164
28.4k
          tmp->underflow = constraint->range.underflow && tmp->underflow;
1165
28.4k
          tmp->min = MAX(constraint->range.min, tmp->min);
1166
28.4k
#ifdef SYM_RANGE
1167
28.4k
        } else if (narrowing && ssa->var_info[constraint->min_ssa_var].has_range) {
1168
1.72k
          tmp->underflow = ssa->var_info[constraint->min_ssa_var].range.underflow && tmp->underflow;
1169
1.72k
          if (!add_will_overflow(ssa->var_info[constraint->min_ssa_var].range.min, constraint->range.min)) {
1170
1.70k
            tmp->min = MAX(ssa->var_info[constraint->min_ssa_var].range.min + constraint->range.min, tmp->min);
1171
1.70k
          }
1172
1.72k
#endif
1173
1.72k
        }
1174
34.6k
        if (constraint->max_ssa_var < 0) {
1175
26.5k
          tmp->max = MIN(constraint->range.max, tmp->max);
1176
26.5k
          tmp->overflow = constraint->range.overflow && tmp->overflow;
1177
26.5k
#ifdef SYM_RANGE
1178
26.5k
        } else if (narrowing && ssa->var_info[constraint->max_ssa_var].has_range) {
1179
1.82k
          if (!add_will_overflow(ssa->var_info[constraint->max_ssa_var].range.max, constraint->range.max)) {
1180
1.80k
            tmp->max = MIN(ssa->var_info[constraint->max_ssa_var].range.max + constraint->range.max, tmp->max);
1181
1.80k
          }
1182
1.82k
          tmp->overflow = ssa->var_info[constraint->max_ssa_var].range.overflow && tmp->overflow;
1183
1.82k
#endif
1184
1.82k
        }
1185
34.6k
      } else if (narrowing) {
1186
269
        if (constraint->min_ssa_var < 0) {
1187
265
          tmp->underflow = constraint->range.underflow;
1188
265
          tmp->min = constraint->range.min;
1189
265
#ifdef SYM_RANGE
1190
265
        } else if (narrowing && ssa->var_info[constraint->min_ssa_var].has_range) {
1191
0
          if (add_will_overflow(ssa->var_info[constraint->min_ssa_var].range.min, constraint->range.min)) {
1192
0
            tmp->underflow = 1;
1193
0
            tmp->min = ZEND_LONG_MIN;
1194
0
          } else {
1195
0
            tmp->underflow = ssa->var_info[constraint->min_ssa_var].range.underflow;
1196
0
            tmp->min = ssa->var_info[constraint->min_ssa_var].range.min + constraint->range.min;
1197
0
          }
1198
0
#endif
1199
4
        } else {
1200
4
          tmp->underflow = 1;
1201
4
          tmp->min = ZEND_LONG_MIN;
1202
4
        }
1203
269
        if (constraint->max_ssa_var < 0) {
1204
265
          tmp->max = constraint->range.max;
1205
265
          tmp->overflow = constraint->range.overflow;
1206
265
#ifdef SYM_RANGE
1207
265
        } else if (narrowing && ssa->var_info[constraint->max_ssa_var].has_range) {
1208
0
          if (add_will_overflow(ssa->var_info[constraint->max_ssa_var].range.max, constraint->range.max)) {
1209
0
            tmp->overflow = 1;
1210
0
            tmp->max = ZEND_LONG_MAX;
1211
0
          } else {
1212
0
            tmp->max = ssa->var_info[constraint->max_ssa_var].range.max + constraint->range.max;
1213
0
            tmp->overflow = ssa->var_info[constraint->max_ssa_var].range.overflow;
1214
0
          }
1215
0
#endif
1216
4
        } else {
1217
4
          tmp->max = ZEND_LONG_MAX;
1218
4
          tmp->overflow = 1;
1219
4
        }
1220
269
      }
1221
273k
    } else {
1222
826k
      for (uint32_t i = 0; i < ssa->cfg.blocks[p->block].predecessors_count; i++) {
1223
552k
        ZEND_ASSERT(p->sources[i] >= 0);
1224
552k
        if (ssa->var_info[p->sources[i]].has_range) {
1225
          /* union */
1226
470k
          tmp->underflow |= ssa->var_info[p->sources[i]].range.underflow;
1227
470k
          tmp->min = MIN(tmp->min, ssa->var_info[p->sources[i]].range.min);
1228
470k
          tmp->max = MAX(tmp->max, ssa->var_info[p->sources[i]].range.max);
1229
470k
          tmp->overflow |= ssa->var_info[p->sources[i]].range.overflow;
1230
470k
        } else if (narrowing) {
1231
26.3k
          tmp->underflow = 1;
1232
26.3k
          tmp->min = ZEND_LONG_MIN;
1233
26.3k
          tmp->max = ZEND_LONG_MAX;
1234
26.3k
          tmp->overflow = 1;
1235
26.3k
        }
1236
552k
      }
1237
273k
    }
1238
312k
    return (tmp->min <= tmp->max);
1239
1.27M
  } else if (ssa->vars[var].definition < 0) {
1240
11.9k
    return 0;
1241
11.9k
  }
1242
1.26M
  line = ssa->vars[var].definition;
1243
1.26M
  opline = op_array->opcodes + line;
1244
1.26M
  ssa_op = &ssa->ops[line];
1245
1246
1.26M
  return zend_inference_propagate_range(op_array, ssa, opline, ssa_op, var, tmp);
1247
1.58M
}
1248
1249
ZEND_API bool zend_inference_propagate_range(const zend_op_array *op_array, const zend_ssa *ssa, const zend_op *opline, const zend_ssa_op* ssa_op, int var, zend_ssa_range *tmp)
1250
1.26M
{
1251
1.26M
  tmp->underflow = 0;
1252
1.26M
  tmp->overflow = 0;
1253
1.26M
  switch (opline->opcode) {
1254
22.0k
    case ZEND_ADD:
1255
35.2k
    case ZEND_SUB:
1256
44.7k
    case ZEND_MUL:
1257
49.3k
    case ZEND_DIV:
1258
52.1k
    case ZEND_MOD:
1259
53.7k
    case ZEND_SL:
1260
54.9k
    case ZEND_SR:
1261
56.2k
    case ZEND_BW_OR:
1262
59.1k
    case ZEND_BW_AND:
1263
62.0k
    case ZEND_BW_XOR:
1264
62.0k
      if (ssa_op->result_def == var) {
1265
62.0k
        return zend_inference_calc_binary_op_range(
1266
62.0k
          op_array, ssa, opline, ssa_op, opline->opcode, tmp);
1267
62.0k
      }
1268
0
      break;
1269
1270
20.6k
    case ZEND_BW_NOT:
1271
20.6k
      if (ssa_op->result_def == var) {
1272
20.6k
        if (OP1_HAS_RANGE()) {
1273
20.0k
          if (OP1_RANGE_UNDERFLOW() ||
1274
18.9k
              OP1_RANGE_OVERFLOW()) {
1275
1.13k
            tmp->min = ZEND_LONG_MIN;
1276
1.13k
            tmp->max = ZEND_LONG_MAX;
1277
18.8k
          } else {
1278
18.8k
            zend_long op1_min = OP1_MIN_RANGE();
1279
18.8k
            zend_long op1_max = OP1_MAX_RANGE();
1280
18.8k
            tmp->min = ~op1_max;
1281
18.8k
            tmp->max = ~op1_min;
1282
18.8k
          }
1283
20.0k
          return 1;
1284
20.0k
        }
1285
20.6k
      }
1286
625
      break;
1287
2.36k
    case ZEND_CAST:
1288
2.36k
      if (ssa_op->op1_def == var) {
1289
0
        if (ssa_op->op1_def >= 0) {
1290
0
          if (OP1_HAS_RANGE()) {
1291
0
            tmp->underflow = OP1_RANGE_UNDERFLOW();
1292
0
            tmp->min = OP1_MIN_RANGE();
1293
0
            tmp->max = OP1_MAX_RANGE();
1294
0
            tmp->overflow  = OP1_RANGE_OVERFLOW();
1295
0
            return 1;
1296
0
          }
1297
0
        }
1298
2.36k
      } else if (ssa_op->result_def == var) {
1299
2.36k
        if (opline->extended_value == IS_LONG) {
1300
897
          if (OP1_HAS_RANGE()) {
1301
764
            tmp->min = OP1_MIN_RANGE();
1302
764
            tmp->max = OP1_MAX_RANGE();
1303
764
            return 1;
1304
764
          } else {
1305
133
            tmp->min = ZEND_LONG_MIN;
1306
133
            tmp->max = ZEND_LONG_MAX;
1307
133
            return 1;
1308
133
          }
1309
897
        }
1310
2.36k
      }
1311
1.46k
      break;
1312
10.5k
    case ZEND_QM_ASSIGN:
1313
12.2k
    case ZEND_JMP_SET:
1314
15.6k
    case ZEND_COALESCE:
1315
17.3k
    case ZEND_COPY_TMP:
1316
17.3k
      if (ssa_op->op1_def == var) {
1317
0
        if (ssa_op->op1_def >= 0) {
1318
0
          if (OP1_HAS_RANGE()) {
1319
0
            tmp->underflow = OP1_RANGE_UNDERFLOW();
1320
0
            tmp->min = OP1_MIN_RANGE();
1321
0
            tmp->max = OP1_MAX_RANGE();
1322
0
            tmp->overflow  = OP1_RANGE_OVERFLOW();
1323
0
            return 1;
1324
0
          }
1325
0
        }
1326
0
      }
1327
17.3k
      if (ssa_op->result_def == var) {
1328
17.3k
        if (OP1_HAS_RANGE()) {
1329
12.1k
          tmp->min = OP1_MIN_RANGE();
1330
12.1k
          tmp->max = OP1_MAX_RANGE();
1331
12.1k
          tmp->underflow = OP1_RANGE_UNDERFLOW();
1332
12.1k
          tmp->overflow  = OP1_RANGE_OVERFLOW();
1333
12.1k
          return 1;
1334
12.1k
        }
1335
17.3k
      }
1336
5.18k
      break;
1337
5.18k
    case ZEND_SEND_VAR:
1338
0
      if (ssa_op->op1_def == var) {
1339
0
        if (ssa_op->op1_def >= 0) {
1340
0
          if (OP1_HAS_RANGE()) {
1341
0
            tmp->underflow = OP1_RANGE_UNDERFLOW();
1342
0
            tmp->min = OP1_MIN_RANGE();
1343
0
            tmp->max = OP1_MAX_RANGE();
1344
0
            tmp->overflow  = OP1_RANGE_OVERFLOW();
1345
0
            return 1;
1346
0
          }
1347
0
        }
1348
0
      }
1349
0
      break;
1350
24.2k
    case ZEND_PRE_INC:
1351
24.2k
      if (ssa_op->op1_def == var || ssa_op->result_def == var) {
1352
24.2k
        if (OP1_HAS_RANGE()) {
1353
23.9k
          tmp->min = OP1_MIN_RANGE();
1354
23.9k
          tmp->max = OP1_MAX_RANGE();
1355
23.9k
          tmp->underflow = OP1_RANGE_UNDERFLOW();
1356
23.9k
          tmp->overflow = OP1_RANGE_OVERFLOW();
1357
23.9k
          if (tmp->max < ZEND_LONG_MAX) {
1358
20.5k
            tmp->max++;
1359
20.5k
          } else {
1360
3.39k
            tmp->overflow = 1;
1361
3.39k
          }
1362
23.9k
          if (tmp->min < ZEND_LONG_MAX && !tmp->underflow) {
1363
19.7k
            tmp->min++;
1364
19.7k
          }
1365
23.9k
          return 1;
1366
23.9k
        }
1367
24.2k
      }
1368
360
      break;
1369
2.92k
    case ZEND_PRE_DEC:
1370
2.92k
      if (ssa_op->op1_def == var || ssa_op->result_def == var) {
1371
2.92k
        if (OP1_HAS_RANGE()) {
1372
2.66k
          tmp->min = OP1_MIN_RANGE();
1373
2.66k
          tmp->max = OP1_MAX_RANGE();
1374
2.66k
          tmp->underflow = OP1_RANGE_UNDERFLOW();
1375
2.66k
          tmp->overflow = OP1_RANGE_OVERFLOW();
1376
2.66k
          if (tmp->min > ZEND_LONG_MIN) {
1377
626
            tmp->min--;
1378
2.03k
          } else {
1379
2.03k
            tmp->underflow = 1;
1380
2.03k
          }
1381
2.66k
          if (tmp->max > ZEND_LONG_MIN && !tmp->overflow) {
1382
1.24k
            tmp->max--;
1383
1.24k
          }
1384
2.66k
          return 1;
1385
2.66k
        }
1386
2.92k
      }
1387
259
      break;
1388
5.74k
    case ZEND_POST_INC:
1389
5.74k
      if (ssa_op->op1_def == var || ssa_op->result_def == var) {
1390
5.74k
        if (OP1_HAS_RANGE()) {
1391
5.49k
          tmp->min = OP1_MIN_RANGE();
1392
5.49k
          tmp->max = OP1_MAX_RANGE();
1393
5.49k
          tmp->underflow = OP1_RANGE_UNDERFLOW();
1394
5.49k
          tmp->overflow = OP1_RANGE_OVERFLOW();
1395
5.49k
          if (ssa_op->result_def == var) {
1396
883
            return 1;
1397
883
          }
1398
4.61k
          if (tmp->max < ZEND_LONG_MAX) {
1399
3.45k
            tmp->max++;
1400
3.45k
          } else {
1401
1.16k
            tmp->overflow = 1;
1402
1.16k
          }
1403
4.61k
          if (tmp->min < ZEND_LONG_MAX && !tmp->underflow) {
1404
4.10k
            tmp->min++;
1405
4.10k
          }
1406
4.61k
          return 1;
1407
5.49k
        }
1408
5.74k
      }
1409
250
      break;
1410
5.78k
    case ZEND_POST_DEC:
1411
5.78k
      if (ssa_op->op1_def == var || ssa_op->result_def == var) {
1412
5.78k
        if (OP1_HAS_RANGE()) {
1413
4.79k
          tmp->min = OP1_MIN_RANGE();
1414
4.79k
          tmp->max = OP1_MAX_RANGE();
1415
4.79k
          tmp->underflow = OP1_RANGE_UNDERFLOW();
1416
4.79k
          tmp->overflow = OP1_RANGE_OVERFLOW();
1417
4.79k
          if (ssa_op->result_def == var) {
1418
2.17k
            return 1;
1419
2.17k
          }
1420
2.61k
          if (tmp->min > ZEND_LONG_MIN) {
1421
181
            tmp->min--;
1422
2.43k
          } else {
1423
2.43k
            tmp->underflow = 1;
1424
2.43k
          }
1425
2.61k
          if (tmp->max > ZEND_LONG_MIN && !tmp->overflow) {
1426
544
            tmp->max--;
1427
544
          }
1428
2.61k
          return 1;
1429
4.79k
        }
1430
5.78k
      }
1431
994
      break;
1432
2.09k
    case ZEND_UNSET_DIM:
1433
2.14k
    case ZEND_UNSET_OBJ:
1434
2.14k
      if (ssa_op->op1_def == var) {
1435
        /* If op1 is scalar, UNSET_DIM and UNSET_OBJ have no effect, so we can keep
1436
         * the previous ranges. */
1437
2.14k
        if (OP1_HAS_RANGE()) {
1438
1.83k
          tmp->min = OP1_MIN_RANGE();
1439
1.83k
          tmp->max = OP1_MAX_RANGE();
1440
1.83k
          tmp->underflow = OP1_RANGE_UNDERFLOW();
1441
1.83k
          tmp->overflow  = OP1_RANGE_OVERFLOW();
1442
1.83k
          return 1;
1443
1.83k
        }
1444
2.14k
      }
1445
310
      break;
1446
60.8k
    case ZEND_ASSIGN:
1447
60.8k
      if (ssa_op->op1_def == var || ssa_op->op2_def == var || ssa_op->result_def == var) {
1448
60.8k
        if (OP2_HAS_RANGE()) {
1449
52.0k
          tmp->min = OP2_MIN_RANGE();
1450
52.0k
          tmp->max = OP2_MAX_RANGE();
1451
52.0k
          tmp->underflow = OP2_RANGE_UNDERFLOW();
1452
52.0k
          tmp->overflow  = OP2_RANGE_OVERFLOW();
1453
52.0k
          return 1;
1454
52.0k
        }
1455
60.8k
      }
1456
8.72k
      break;
1457
53.7k
    case ZEND_ASSIGN_DIM:
1458
58.7k
    case ZEND_ASSIGN_OBJ:
1459
59.0k
    case ZEND_ASSIGN_STATIC_PROP:
1460
64.0k
    case ZEND_ASSIGN_DIM_OP:
1461
64.6k
    case ZEND_ASSIGN_OBJ_OP:
1462
64.6k
    case ZEND_ASSIGN_STATIC_PROP_OP:
1463
64.6k
      if ((ssa_op+1)->op1_def == var) {
1464
0
        opline++;
1465
0
        ssa_op++;
1466
0
        if (OP1_HAS_RANGE()) {
1467
0
          tmp->min = OP1_MIN_RANGE();
1468
0
          tmp->max = OP1_MAX_RANGE();
1469
0
          tmp->underflow = OP1_RANGE_UNDERFLOW();
1470
0
          tmp->overflow  = OP1_RANGE_OVERFLOW();
1471
0
        }
1472
0
        return 1;
1473
0
      }
1474
64.6k
      break;
1475
64.6k
    case ZEND_ASSIGN_OP:
1476
29.7k
      if (opline->extended_value != ZEND_CONCAT
1477
22.5k
       && opline->extended_value != ZEND_POW) {
1478
22.5k
        if (ssa_op->op1_def == var || ssa_op->result_def == var) {
1479
22.5k
          return zend_inference_calc_binary_op_range(
1480
22.5k
            op_array, ssa, opline, ssa_op,
1481
22.5k
            opline->extended_value, tmp);
1482
22.5k
        }
1483
22.5k
      }
1484
7.20k
      break;
1485
7.20k
    case ZEND_OP_DATA:
1486
6
      if (ssa_op->op1_def == var) {
1487
6
        if ((opline-1)->opcode == ZEND_ASSIGN_DIM ||
1488
6
            (opline-1)->opcode == ZEND_ASSIGN_OBJ ||
1489
6
            (opline-1)->opcode == ZEND_ASSIGN_STATIC_PROP ||
1490
6
            (opline-1)->opcode == ZEND_ASSIGN_DIM_OP ||
1491
6
            (opline-1)->opcode == ZEND_ASSIGN_OBJ_OP ||
1492
6
            (opline-1)->opcode == ZEND_ASSIGN_STATIC_PROP_OP) {
1493
0
          if (OP1_HAS_RANGE()) {
1494
0
            tmp->min = OP1_MIN_RANGE();
1495
0
            tmp->max = OP1_MAX_RANGE();
1496
0
            tmp->underflow = OP1_RANGE_UNDERFLOW();
1497
0
            tmp->overflow  = OP1_RANGE_OVERFLOW();
1498
0
            return 1;
1499
0
          }
1500
0
        }
1501
6
        break;
1502
6
      }
1503
0
      break;
1504
12.8k
    case ZEND_RECV:
1505
14.0k
    case ZEND_RECV_INIT:
1506
14.0k
      if (ssa_op->result_def == var) {
1507
14.0k
        if (op_array->arg_info &&
1508
14.0k
            opline->op1.num <= op_array->num_args) {
1509
14.0k
          zend_type type = op_array->arg_info[opline->op1.num-1].type;
1510
14.0k
          uint32_t mask = ZEND_TYPE_PURE_MASK_WITHOUT_NULL(type);
1511
14.0k
          if (mask == MAY_BE_LONG) {
1512
1.07k
            tmp->underflow = 0;
1513
1.07k
            tmp->min = ZEND_LONG_MIN;
1514
1.07k
            tmp->max = ZEND_LONG_MAX;
1515
1.07k
            tmp->overflow = 0;
1516
1.07k
            return 1;
1517
1.07k
          }
1518
14.0k
        }
1519
14.0k
      }
1520
13.0k
      break;
1521
13.0k
    case ZEND_STRLEN:
1522
1.57k
      if (ssa_op->result_def == var) {
1523
#if SIZEOF_ZEND_LONG == 4
1524
        /* The length of a string is a non-negative integer. However, on 32-bit
1525
         * platforms overflows into negative lengths may occur, so it's better
1526
         * to not assume any particular range. */
1527
        tmp->min = ZEND_LONG_MIN;
1528
#else
1529
1.57k
        tmp->min = 0;
1530
1.57k
#endif
1531
1.57k
        tmp->max = ZEND_LONG_MAX;
1532
1.57k
        return 1;
1533
1.57k
      }
1534
0
      break;
1535
34
    case ZEND_FUNC_NUM_ARGS:
1536
34
      tmp->min = 0;
1537
34
      tmp->max = ZEND_LONG_MAX;
1538
34
      return 1;
1539
980
    case ZEND_COUNT:
1540
      /* count() on Countable objects may return negative numbers */
1541
980
      tmp->min = ZEND_LONG_MIN;
1542
980
      tmp->max = ZEND_LONG_MAX;
1543
980
      return 1;
1544
36.2k
    case ZEND_DO_FCALL:
1545
36.2k
    case ZEND_DO_ICALL:
1546
39.5k
    case ZEND_DO_UCALL:
1547
39.5k
    case ZEND_DO_FCALL_BY_NAME:
1548
39.5k
      if (ssa_op->result_def == var) {
1549
39.5k
        const zend_func_info *func_info = ZEND_FUNC_INFO(op_array);
1550
39.5k
        const zend_call_info *call_info;
1551
39.5k
        if (!func_info || !func_info->call_map) {
1552
5.88k
          break;
1553
5.88k
        }
1554
1555
33.6k
        call_info = func_info->call_map[opline - op_array->opcodes];
1556
33.6k
        if (!call_info || call_info->is_prototype) {
1557
11.5k
          break;
1558
11.5k
        }
1559
22.0k
        if (call_info->callee_func->type == ZEND_USER_FUNCTION) {
1560
6.83k
          func_info = ZEND_FUNC_INFO(&call_info->callee_func->op_array);
1561
6.83k
          if (func_info && func_info->return_info.has_range) {
1562
348
            *tmp = func_info->return_info.range;
1563
348
            return 1;
1564
348
          }
1565
6.83k
        }
1566
//TODO: we can't use type inference for internal functions at this point ???
1567
#if 0
1568
          uint32_t type;
1569
1570
          type = zend_get_func_info(call_info, ssa);
1571
          if (!(type & (MAY_BE_ANY - (MAY_BE_NULL|MAY_BE_FALSE|MAY_BE_TRUE|MAY_BE_LONG)))) {
1572
            tmp->underflow = 0;
1573
            tmp->min = 0;
1574
            tmp->max = 0;
1575
            tmp->overflow = 0;
1576
            if (type & MAY_BE_LONG) {
1577
              tmp->min = ZEND_LONG_MIN;
1578
              tmp->max = ZEND_LONG_MAX;
1579
            } else if (type & MAY_BE_TRUE) {
1580
              if (!(type & (MAY_BE_NULL|MAY_BE_FALSE))) {
1581
                tmp->min = 1;
1582
              }
1583
              tmp->max = 1;
1584
            }
1585
            return 1;
1586
          }
1587
#endif
1588
22.0k
      }
1589
21.7k
      break;
1590
    // FIXME: support for more opcodes
1591
909k
    default:
1592
909k
      break;
1593
1.26M
  }
1594
1.05M
  return 0;
1595
1.26M
}
1596
1597
static void zend_inference_init_range(const zend_op_array *op_array, const zend_ssa *ssa, int var, bool underflow, zend_long min, zend_long max, bool overflow)
1598
825k
{
1599
825k
  if (underflow) {
1600
784k
    min = ZEND_LONG_MIN;
1601
784k
  }
1602
825k
  if (overflow) {
1603
785k
    max = ZEND_LONG_MAX;
1604
785k
  }
1605
825k
  ssa->var_info[var].has_range = 1;
1606
825k
  ssa->var_info[var].range.underflow = underflow;
1607
825k
  ssa->var_info[var].range.min = min;
1608
825k
  ssa->var_info[var].range.max = max;
1609
825k
  ssa->var_info[var].range.overflow = overflow;
1610
825k
  LOG_SSA_RANGE("  change range (init      SCC %2d) %2d [%s%ld..%ld%s]\n", ssa->vars[var].scc, var, (underflow?"-- ":""), min, max, (overflow?" ++":""));
1611
825k
}
1612
1613
static bool zend_inference_widening_meet(zend_ssa_var_info *var_info, zend_ssa_range *r)
1614
44.6k
{
1615
44.6k
  if (!var_info->has_range) {
1616
82
    var_info->has_range = 1;
1617
44.5k
  } else {
1618
44.5k
    if (r->underflow ||
1619
13.9k
        var_info->range.underflow ||
1620
31.8k
        r->min < var_info->range.min) {
1621
31.8k
      r->underflow = 1;
1622
31.8k
      r->min = ZEND_LONG_MIN;
1623
31.8k
    } else {
1624
12.6k
      r->min = var_info->range.min;
1625
12.6k
    }
1626
44.5k
    if (r->overflow ||
1627
12.7k
        var_info->range.overflow ||
1628
34.6k
        r->max > var_info->range.max) {
1629
34.6k
      r->overflow = 1;
1630
34.6k
      r->max = ZEND_LONG_MAX;
1631
34.6k
    } else {
1632
9.94k
      r->max = var_info->range.max;
1633
9.94k
    }
1634
44.5k
    if (var_info->range.min == r->min &&
1635
43.2k
        var_info->range.max == r->max &&
1636
39.5k
        var_info->range.underflow == r->underflow &&
1637
39.4k
        var_info->range.overflow == r->overflow) {
1638
39.4k
      return 0;
1639
39.4k
    }
1640
44.5k
  }
1641
5.17k
  var_info->range = *r;
1642
5.17k
  return 1;
1643
44.6k
}
1644
1645
static bool zend_ssa_range_widening(const zend_op_array *op_array, const zend_ssa *ssa, int var, int scc)
1646
131k
{
1647
131k
  zend_ssa_range tmp;
1648
1649
131k
  if (zend_inference_calc_range(op_array, ssa, var, 1, 0, &tmp)) {
1650
44.6k
    if (zend_inference_widening_meet(&ssa->var_info[var], &tmp)) {
1651
5.17k
      LOG_SSA_RANGE("  change range (widening  SCC %2d) %2d [%s%ld..%ld%s]\n", scc, var, (tmp.underflow?"-- ":""), tmp.min, tmp.max, (tmp.overflow?" ++":""));
1652
5.17k
      return 1;
1653
5.17k
    }
1654
44.6k
  }
1655
126k
  return 0;
1656
131k
}
1657
1658
static bool zend_inference_narrowing_meet(zend_ssa_var_info *var_info, zend_ssa_range *r)
1659
319k
{
1660
319k
  if (!var_info->has_range) {
1661
69.2k
    var_info->has_range = 1;
1662
250k
  } else {
1663
250k
    if (!r->underflow &&
1664
115k
        !var_info->range.underflow &&
1665
113k
        var_info->range.min < r->min) {
1666
142
      r->min = var_info->range.min;
1667
142
    }
1668
250k
    if (!r->overflow &&
1669
112k
        !var_info->range.overflow &&
1670
109k
        var_info->range.max > r->max) {
1671
68
      r->max = var_info->range.max;
1672
68
    }
1673
250k
    if (r->underflow) {
1674
135k
      r->min = ZEND_LONG_MIN;
1675
135k
    }
1676
250k
    if (r->overflow) {
1677
137k
      r->max = ZEND_LONG_MAX;
1678
137k
    }
1679
250k
    if (var_info->range.min == r->min &&
1680
237k
        var_info->range.max == r->max &&
1681
182k
        var_info->range.underflow == r->underflow &&
1682
181k
        var_info->range.overflow == r->overflow) {
1683
181k
      return 0;
1684
181k
    }
1685
250k
  }
1686
138k
  var_info->range = *r;
1687
138k
  return 1;
1688
319k
}
1689
1690
static bool zend_ssa_range_narrowing(const zend_op_array *op_array, const zend_ssa *ssa, int var, int scc)
1691
140k
{
1692
140k
  zend_ssa_range tmp;
1693
1694
140k
  if (zend_inference_calc_range(op_array, ssa, var, 0, 1, &tmp)) {
1695
92.7k
    if (zend_inference_narrowing_meet(&ssa->var_info[var], &tmp)) {
1696
44.2k
      LOG_SSA_RANGE("  change range (narrowing SCC %2d) %2d [%s%ld..%ld%s]\n", scc, var, (tmp.underflow?"-- ":""), tmp.min, tmp.max, (tmp.overflow?" ++":""));
1697
44.2k
      return 1;
1698
44.2k
    }
1699
92.7k
  }
1700
96.1k
  return 0;
1701
140k
}
1702
1703
#ifdef NEG_RANGE
1704
# define CHECK_INNER_CYCLE(var2) \
1705
  do { \
1706
    if (ssa->vars[var2].scc == ssa->vars[var].scc && \
1707
        !ssa->vars[var2].scc_entry && \
1708
        !zend_bitset_in(visited, var2) && \
1709
      zend_check_inner_cycles(op_array, ssa, worklist, visited, var2)) { \
1710
      return 1; \
1711
    } \
1712
  } while (0)
1713
1714
static bool zend_check_inner_cycles(const zend_op_array *op_array, zend_ssa *ssa, zend_bitset worklist, zend_bitset visited, int var)
1715
{
1716
  if (zend_bitset_in(worklist, var)) {
1717
    return 1;
1718
  }
1719
  zend_bitset_incl(worklist, var);
1720
  FOR_EACH_VAR_USAGE(var, CHECK_INNER_CYCLE);
1721
  zend_bitset_incl(visited, var);
1722
  return 0;
1723
}
1724
#endif
1725
1726
static void zend_infer_ranges_warmup(const zend_op_array *op_array, zend_ssa *ssa, const int *scc_var, const int *next_scc_var, int scc)
1727
17.6k
{
1728
17.6k
  int worklist_len = zend_bitset_len(ssa->vars_count);
1729
17.6k
  int j, n;
1730
17.6k
  zend_ssa_range tmp;
1731
17.6k
  ALLOCA_FLAG(use_heap)
1732
17.6k
  zend_bitset worklist = do_alloca(sizeof(zend_ulong) * worklist_len * 2, use_heap);
1733
17.6k
  zend_bitset visited = worklist + worklist_len;
1734
#ifdef NEG_RANGE
1735
  bool has_inner_cycles = false;
1736
1737
  memset(worklist, 0, sizeof(zend_ulong) * worklist_len);
1738
  memset(visited, 0, sizeof(zend_ulong) * worklist_len);
1739
  j = scc_var[scc];
1740
  while (j >= 0) {
1741
    if (!zend_bitset_in(visited, j) &&
1742
        zend_check_inner_cycles(op_array, ssa, worklist, visited, j)) {
1743
      has_inner_cycles = true;
1744
      break;
1745
    }
1746
    j = next_scc_var[j];
1747
  }
1748
#endif
1749
1750
17.6k
  memset(worklist, 0, sizeof(zend_ulong) * worklist_len);
1751
1752
300k
  for (n = 0; n < RANGE_WARMUP_PASSES; n++) {
1753
282k
    j= scc_var[scc];
1754
2.30M
    while (j >= 0) {
1755
2.02M
      if (ssa->vars[j].scc_entry
1756
749k
       && !(ssa->var_info[j].type & MAY_BE_REF)) {
1757
469k
        zend_bitset_incl(worklist, j);
1758
469k
      }
1759
2.02M
      j = next_scc_var[j];
1760
2.02M
    }
1761
1762
282k
    memset(visited, 0, sizeof(zend_ulong) * worklist_len);
1763
1764
8.27M
    WHILE_WORKLIST(worklist, worklist_len, j) {
1765
8.27M
      if (zend_inference_calc_range(op_array, ssa, j, 0, 0, &tmp)) {
1766
#ifdef NEG_RANGE
1767
        if (!has_inner_cycles &&
1768
            ssa->var_info[j].has_range &&
1769
            ssa->vars[j].definition_phi &&
1770
            ssa->vars[j].definition_phi->pi >= 0 &&
1771
          ssa->vars[j].definition_phi->has_range_constraint &&
1772
            ssa->vars[j].definition_phi->constraint.range.negative &&
1773
            ssa->vars[j].definition_phi->constraint.range.min_ssa_var < 0 &&
1774
            ssa->vars[j].definition_phi->constraint.range.max_ssa_var < 0) {
1775
          zend_ssa_range_constraint *constraint =
1776
            &ssa->vars[j].definition_phi->constraint.range;
1777
          if (tmp.min == ssa->var_info[j].range.min &&
1778
              tmp.max == ssa->var_info[j].range.max) {
1779
            if (constraint->negative == NEG_INIT) {
1780
              LOG_NEG_RANGE("#%d INVARIANT\n", j);
1781
              constraint->negative = NEG_INVARIANT;
1782
            }
1783
          } else if (tmp.min == ssa->var_info[j].range.min &&
1784
                     tmp.max == ssa->var_info[j].range.max + 1 &&
1785
                     tmp.max < constraint->range.min) {
1786
            if (constraint->negative == NEG_INIT ||
1787
                constraint->negative == NEG_INVARIANT) {
1788
              LOG_NEG_RANGE("#%d LT\n", j);
1789
              constraint->negative = NEG_USE_LT;
1790
//???NEG
1791
            } else if (constraint->negative == NEG_USE_GT) {
1792
              LOG_NEG_RANGE("#%d UNKNOWN\n", j);
1793
              constraint->negative = NEG_UNKNOWN;
1794
            }
1795
          } else if (tmp.max == ssa->var_info[j].range.max &&
1796
                       tmp.min == ssa->var_info[j].range.min - 1 &&
1797
                     tmp.min > constraint->range.max) {
1798
            if (constraint->negative == NEG_INIT ||
1799
                constraint->negative == NEG_INVARIANT) {
1800
              LOG_NEG_RANGE("#%d GT\n", j);
1801
              constraint->negative = NEG_USE_GT;
1802
//???NEG
1803
            } else if (constraint->negative == NEG_USE_LT) {
1804
              LOG_NEG_RANGE("#%d UNKNOWN\n", j);
1805
              constraint->negative = NEG_UNKNOWN;
1806
            }
1807
          } else {
1808
            LOG_NEG_RANGE("#%d UNKNOWN\n", j);
1809
            constraint->negative = NEG_UNKNOWN;
1810
          }
1811
        }
1812
#endif
1813
227k
        if (zend_inference_narrowing_meet(&ssa->var_info[j], &tmp)) {
1814
94.4k
          LOG_SSA_RANGE("  change range (warmup %2d SCC %2d) %2d [%s%ld..%ld%s]\n", n, scc, j, (tmp.underflow?"-- ":""), tmp.min, tmp.max, (tmp.overflow?" ++":""));
1815
94.4k
          zend_bitset_incl(visited, j);
1816
162k
          FOR_EACH_VAR_USAGE(j, ADD_SCC_VAR_1);
1817
94.4k
        }
1818
227k
      }
1819
8.27M
    } WHILE_WORKLIST_END();
1820
282k
  }
1821
17.6k
  free_alloca(worklist, use_heap);
1822
17.6k
}
1823
1824
static void zend_infer_ranges(const zend_op_array *op_array, zend_ssa *ssa) /* {{{ */
1825
71.2k
{
1826
71.2k
  int worklist_len = zend_bitset_len(ssa->vars_count);
1827
71.2k
  zend_bitset worklist;
1828
71.2k
  int *next_scc_var;
1829
71.2k
  int *scc_var;
1830
71.2k
  zend_ssa_phi *p;
1831
71.2k
  zend_ssa_range tmp;
1832
71.2k
  int scc, j;
1833
71.2k
  ALLOCA_FLAG(use_heap);
1834
1835
71.2k
  worklist = do_alloca(
1836
71.2k
    ZEND_MM_ALIGNED_SIZE(sizeof(zend_ulong) * worklist_len) +
1837
71.2k
    ZEND_MM_ALIGNED_SIZE(sizeof(int) * ssa->vars_count) +
1838
71.2k
    sizeof(int) * ssa->sccs, use_heap);
1839
71.2k
  next_scc_var = (int*)((char*)worklist + ZEND_MM_ALIGNED_SIZE(sizeof(zend_ulong) * worklist_len));
1840
71.2k
  scc_var = (int*)((char*)next_scc_var + ZEND_MM_ALIGNED_SIZE(sizeof(int) * ssa->vars_count));
1841
1842
71.2k
  LOG_SSA_RANGE("Range Inference\n");
1843
1844
  /* Create linked lists of SSA variables for each SCC */
1845
71.2k
  memset(scc_var, -1, sizeof(int) * ssa->sccs);
1846
1.19M
  for (j = 0; j < ssa->vars_count; j++) {
1847
1.12M
    if (ssa->vars[j].scc >= 0) {
1848
1.04M
      next_scc_var[j] = scc_var[ssa->vars[j].scc];
1849
1.04M
      scc_var[ssa->vars[j].scc] = j;
1850
1.04M
    }
1851
1.12M
  }
1852
1853
1.00M
  for (scc = 0; scc < ssa->sccs; scc++) {
1854
936k
    j = scc_var[scc];
1855
936k
    if (next_scc_var[j] < 0) {
1856
      /* SCC with a single element */
1857
918k
      if (ssa->var_info[j].type & MAY_BE_REF) {
1858
        /* pass */
1859
774k
      } else if (zend_inference_calc_range(op_array, ssa, j, 0, 1, &tmp)) {
1860
131k
        zend_inference_init_range(op_array, ssa, j, tmp.underflow, tmp.min, tmp.max, tmp.overflow);
1861
643k
      } else {
1862
643k
        zend_inference_init_range(op_array, ssa, j, true, ZEND_LONG_MIN, ZEND_LONG_MAX, true);
1863
643k
      }
1864
918k
    } else {
1865
      /* Find SCC entry points */
1866
17.6k
      memset(worklist, 0, sizeof(zend_ulong) * worklist_len);
1867
126k
      do {
1868
126k
        if (ssa->vars[j].scc_entry
1869
46.8k
         && !(ssa->var_info[j].type & MAY_BE_REF)) {
1870
29.3k
          zend_bitset_incl(worklist, j);
1871
29.3k
        }
1872
126k
        j = next_scc_var[j];
1873
126k
      } while (j >= 0);
1874
1875
17.6k
#if RANGE_WARMUP_PASSES > 0
1876
17.6k
      zend_infer_ranges_warmup(op_array, ssa, scc_var, next_scc_var, scc);
1877
17.6k
      j = scc_var[scc];
1878
126k
      do {
1879
126k
        if (!(ssa->var_info[j].type & MAY_BE_REF)) {
1880
79.6k
          zend_bitset_incl(worklist, j);
1881
79.6k
        }
1882
126k
        j = next_scc_var[j];
1883
126k
      } while (j >= 0);
1884
17.6k
#endif
1885
1886
      /* widening */
1887
638k
      WHILE_WORKLIST(worklist, worklist_len, j) {
1888
638k
        if (zend_ssa_range_widening(op_array, ssa, j, scc)) {
1889
8.63k
          FOR_EACH_VAR_USAGE(j, ADD_SCC_VAR);
1890
4.66k
        }
1891
638k
      } WHILE_WORKLIST_END();
1892
1893
      /* initialize missing ranges */
1894
144k
      for (j = scc_var[scc]; j >= 0; j = next_scc_var[j]) {
1895
126k
        if (!ssa->var_info[j].has_range
1896
97.4k
         && !(ssa->var_info[j].type & MAY_BE_REF)) {
1897
50.4k
          zend_inference_init_range(op_array, ssa, j, true, ZEND_LONG_MIN, ZEND_LONG_MAX,
1898
50.4k
                  true);
1899
61.9k
          FOR_EACH_VAR_USAGE(j, ADD_SCC_VAR);
1900
50.4k
        }
1901
126k
      }
1902
1903
      /* widening (second round) */
1904
327k
      WHILE_WORKLIST(worklist, worklist_len, j) {
1905
327k
        if (zend_ssa_range_widening(op_array, ssa, j, scc)) {
1906
997
          FOR_EACH_VAR_USAGE(j, ADD_SCC_VAR);
1907
515
        }
1908
327k
      } WHILE_WORKLIST_END();
1909
1910
      /* Add all SCC entry variables into worklist for narrowing */
1911
144k
      for (j = scc_var[scc]; j >= 0; j = next_scc_var[j]) {
1912
126k
        if (ssa->vars[j].definition_phi
1913
51.2k
         && ssa->vars[j].definition_phi->pi < 0
1914
34.5k
         && !(ssa->var_info[j].type & MAY_BE_REF)) {
1915
          /* narrowing Phi functions first */
1916
12.1k
          zend_ssa_range_narrowing(op_array, ssa, j, scc);
1917
12.1k
        }
1918
126k
        zend_bitset_incl(worklist, j);
1919
126k
      }
1920
1921
      /* narrowing */
1922
1.02M
      WHILE_WORKLIST(worklist, worklist_len, j) {
1923
1.02M
        if (zend_ssa_range_narrowing(op_array, ssa, j, scc)) {
1924
97.4k
          FOR_EACH_VAR_USAGE(j, ADD_SCC_VAR);
1925
44.2k
#ifdef SYM_RANGE
1926
          /* Process symbolic control-flow constraints */
1927
44.2k
          p = ssa->vars[j].sym_use_chain;
1928
45.1k
          while (p) {
1929
909
            ADD_SCC_VAR(p->ssa_var);
1930
909
            p = p->sym_use_chain;
1931
909
          }
1932
44.2k
#endif
1933
44.2k
        }
1934
1.02M
      } WHILE_WORKLIST_END();
1935
17.6k
    }
1936
936k
  }
1937
1938
71.2k
  free_alloca(worklist, use_heap);
1939
71.2k
}
1940
/* }}} */
1941
1942
0
static uint32_t get_ssa_alias_types(zend_ssa_alias_kind alias) {
1943
0
  if (alias == HTTP_RESPONSE_HEADER_ALIAS) {
1944
0
    return MAY_BE_ARRAY | MAY_BE_ARRAY_KEY_LONG | MAY_BE_ARRAY_OF_STRING | MAY_BE_RC1 | MAY_BE_RCN;
1945
0
  } else {
1946
0
    return MAY_BE_UNDEF | MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_REF | MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
1947
0
  }
1948
0
}
1949
1950
#define UPDATE_SSA_TYPE(_type, _var)                  \
1951
1.10M
  do {                               \
1952
1.10M
    uint32_t __type = (_type) & ~MAY_BE_GUARD;           \
1953
1.10M
    int __var = (_var);                       \
1954
1.10M
    if (__type & MAY_BE_REF) {                   \
1955
171k
      __type |= MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF; \
1956
171k
    }                                \
1957
1.10M
    if (__var >= 0) {                       \
1958
1.10M
      zend_ssa_var *__ssa_var = &ssa_vars[__var];         \
1959
1.10M
      if (__ssa_var->var < op_array->num_args) {         \
1960
33.8k
        if (__type & MAY_BE_RC1) {                              \
1961
29.6k
          /* TODO: may be captured by exception backtreace */ \
1962
29.6k
          __type |= MAY_BE_RCN;                               \
1963
29.6k
        }                                                       \
1964
33.8k
      }                                                           \
1965
1.10M
      if (__ssa_var->var < op_array->last_var) {         \
1966
311k
        if (__type & (MAY_BE_REF|MAY_BE_RCN)) {         \
1967
221k
          __type |= MAY_BE_RC1 | MAY_BE_RCN;         \
1968
221k
        }                            \
1969
311k
        if ((__type & MAY_BE_RC1) && (__type & MAY_BE_STRING)) {\
1970
206k
          /* TODO: support for array keys and ($str . "")*/   \
1971
206k
          __type |= MAY_BE_RCN;                               \
1972
206k
        }                                                       \
1973
311k
        if ((__type & MAY_BE_RC1) && (__type & MAY_BE_OBJECT)) {\
1974
212k
          /* TODO: object may be captured by magic handlers */\
1975
212k
          __type |= MAY_BE_RCN;                               \
1976
212k
        }                                                       \
1977
311k
        if (__ssa_var->alias) {                 \
1978
0
          __type |= get_ssa_alias_types(__ssa_var->alias);  \
1979
0
        }                            \
1980
311k
      }                              \
1981
1.10M
      if (ssa_var_info[__var].type != __type) {           \
1982
915k
        ZEND_ASSERT(ssa_opcodes != NULL ||           \
1983
915k
          __ssa_var->var >= op_array->last_var ||       \
1984
915k
          (ssa_var_info[__var].type & MAY_BE_REF)       \
1985
915k
            == (__type & MAY_BE_REF));            \
1986
915k
        if (ssa_var_info[__var].type & ~__type) {       \
1987
0
          emit_type_narrowing_warning(op_array, ssa, __var);  \
1988
0
          return FAILURE;                   \
1989
0
        }                            \
1990
915k
        ssa_var_info[__var].type = __type;            \
1991
915k
        if (update_worklist) {                   \
1992
915k
          add_usages(op_array, ssa, worklist, __var);     \
1993
915k
        }                            \
1994
915k
      }                              \
1995
1.10M
      /*zend_bitset_excl(worklist, var);*/            \
1996
1.10M
    }                                \
1997
1.10M
  } while (0)
1998
1999
#define UPDATE_SSA_OBJ_TYPE(_ce, _is_instanceof, var)           \
2000
377k
  do {                                                                \
2001
377k
    if (var >= 0) {                         \
2002
377k
      zend_class_entry *__ce = (_ce);               \
2003
377k
      bool __is_instanceof = (_is_instanceof);          \
2004
377k
      if (__ce && (__ce->ce_flags & ZEND_ACC_FINAL)) {     \
2005
9.87k
        __is_instanceof = false;                \
2006
9.87k
      }                              \
2007
377k
      if (ssa_var_info[var].ce != __ce ||             \
2008
377k
          ssa_var_info[var].is_instanceof != __is_instanceof) { \
2009
35.4k
        ssa_var_info[var].ce = __ce;              \
2010
35.4k
        ssa_var_info[var].is_instanceof = __is_instanceof;    \
2011
35.4k
        if (update_worklist) {                   \
2012
35.4k
          add_usages(op_array, ssa, worklist, var);     \
2013
35.4k
        }                            \
2014
35.4k
      }                              \
2015
377k
      /*zend_bitset_excl(worklist, var);*/            \
2016
377k
    }                               \
2017
377k
  } while (0)
2018
2019
156k
#define COPY_SSA_OBJ_TYPE(from_var, to_var) do { \
2020
156k
  if ((from_var) >= 0 && (ssa_var_info[(from_var)].type & MAY_BE_OBJECT) \
2021
156k
      && ssa_var_info[(from_var)].ce && !(ssa_var_info[(to_var)].type & MAY_BE_REF)) { \
2022
5.50k
    UPDATE_SSA_OBJ_TYPE(ssa_var_info[(from_var)].ce, \
2023
5.50k
      ssa_var_info[(from_var)].is_instanceof, (to_var)); \
2024
151k
  } else { \
2025
151k
    UPDATE_SSA_OBJ_TYPE(NULL, 0, (to_var)); \
2026
151k
  } \
2027
156k
} while (0)
2028
2029
static void add_usages(const zend_op_array *op_array, const zend_ssa *ssa, zend_bitset worklist, int var)
2030
951k
{
2031
951k
  if (ssa->vars[var].phi_use_chain) {
2032
153k
    const zend_ssa_phi *p = ssa->vars[var].phi_use_chain;
2033
186k
    do {
2034
186k
      zend_bitset_incl(worklist, p->ssa_var);
2035
186k
      p = zend_ssa_next_use_phi(ssa, var, p);
2036
186k
    } while (p);
2037
153k
  }
2038
951k
  if (ssa->vars[var].use_chain >= 0) {
2039
835k
    int use = ssa->vars[var].use_chain;
2040
835k
    const zend_ssa_op *op;
2041
2042
933k
    do {
2043
933k
      op = ssa->ops + use;
2044
933k
      if (op->result_def >= 0) {
2045
593k
        zend_bitset_incl(worklist, op->result_def);
2046
593k
      }
2047
933k
      if (op->op1_def >= 0) {
2048
113k
        zend_bitset_incl(worklist, op->op1_def);
2049
113k
      }
2050
933k
      if (op->op2_def >= 0) {
2051
9.55k
        zend_bitset_incl(worklist, op->op2_def);
2052
9.55k
      }
2053
933k
      if (op_array->opcodes[use].opcode == ZEND_OP_DATA) {
2054
13.8k
        op--;
2055
13.8k
        if (op->result_def >= 0) {
2056
3.39k
          zend_bitset_incl(worklist, op->result_def);
2057
3.39k
        }
2058
13.8k
        if (op->op1_def >= 0) {
2059
7.53k
          zend_bitset_incl(worklist, op->op1_def);
2060
7.53k
        }
2061
13.8k
        if (op->op2_def >= 0) {
2062
0
          zend_bitset_incl(worklist, op->op2_def);
2063
0
        }
2064
920k
      } else if (use + 1 < op_array->last
2065
907k
       && op_array->opcodes[use + 1].opcode == ZEND_OP_DATA) {
2066
14.3k
        op++;
2067
14.3k
        if (op->result_def >= 0) {
2068
0
          zend_bitset_incl(worklist, op->result_def);
2069
0
        }
2070
14.3k
        if (op->op1_def >= 0) {
2071
133
          zend_bitset_incl(worklist, op->op1_def);
2072
133
        }
2073
14.3k
        if (op->op2_def >= 0) {
2074
0
          zend_bitset_incl(worklist, op->op2_def);
2075
0
        }
2076
14.3k
      }
2077
933k
      use = zend_ssa_next_use(ssa->ops, var, use);
2078
933k
    } while (use >= 0);
2079
835k
  }
2080
951k
}
2081
2082
static void emit_type_narrowing_warning(const zend_op_array *op_array, const zend_ssa *ssa, int var)
2083
0
{
2084
0
  int def_op_num = ssa->vars[var].definition;
2085
0
  const zend_op *def_opline = def_op_num >= 0 ? &op_array->opcodes[def_op_num] : NULL;
2086
0
  const char *def_op_name = def_opline ? zend_get_opcode_name(def_opline->opcode) : "PHI";
2087
0
  uint32_t lineno = def_opline ? def_opline->lineno : 0;
2088
0
  zend_error_at(
2089
0
    E_WARNING, op_array->filename, lineno,
2090
0
    "Narrowing occurred during type inference of %s. Please file a bug report on https://github.com/php/php-src/issues", def_op_name);
2091
0
#if ZEND_DEBUG
2092
0
  ZEND_ASSERT(0 && "Narrowing during type inference");
2093
0
#endif
2094
0
}
2095
2096
ZEND_API uint32_t ZEND_FASTCALL zend_array_type_info(const zval *zv)
2097
15.9k
{
2098
15.9k
  HashTable *ht = Z_ARRVAL_P(zv);
2099
15.9k
  uint32_t tmp = MAY_BE_ARRAY;
2100
15.9k
  zend_string *str;
2101
15.9k
  zval *val;
2102
2103
15.9k
  if (Z_REFCOUNTED_P(zv)) {
2104
10.6k
    tmp |= MAY_BE_RC1 | MAY_BE_RCN;
2105
10.6k
  } else {
2106
5.28k
    tmp |= MAY_BE_RCN;
2107
5.28k
  }
2108
2109
15.9k
  if (zend_hash_num_elements(ht) == 0) {
2110
5.30k
    tmp |=  MAY_BE_ARRAY_EMPTY;
2111
10.6k
  } else if (HT_IS_PACKED(ht)) {
2112
8.13k
    tmp |= MAY_BE_ARRAY_PACKED;
2113
121k
    ZEND_HASH_PACKED_FOREACH_VAL(ht, val) {
2114
121k
      tmp |= 1 << (Z_TYPE_P(val) + MAY_BE_ARRAY_SHIFT);
2115
121k
    } ZEND_HASH_FOREACH_END();
2116
8.13k
  } else {
2117
16.5k
    ZEND_HASH_MAP_FOREACH_STR_KEY_VAL(ht, str, val) {
2118
16.5k
      if (str) {
2119
4.13k
        tmp |= MAY_BE_ARRAY_STRING_HASH;
2120
4.13k
      } else {
2121
1.59k
        tmp |= MAY_BE_ARRAY_NUMERIC_HASH;
2122
1.59k
      }
2123
16.5k
      tmp |= 1 << (Z_TYPE_P(val) + MAY_BE_ARRAY_SHIFT);
2124
16.5k
    } ZEND_HASH_FOREACH_END();
2125
2.52k
  }
2126
15.9k
  return tmp;
2127
15.9k
}
2128
2129
2130
ZEND_API uint32_t zend_array_element_type(uint32_t t1, uint8_t op_type, bool write, bool insert)
2131
23.2k
{
2132
23.2k
  uint32_t tmp = 0;
2133
2134
23.2k
  if (t1 & MAY_BE_OBJECT) {
2135
17.4k
      if (!write) {
2136
      /* can't be REF  because of ZVAL_COPY_DEREF() usage */
2137
8.44k
      tmp |= MAY_BE_ANY | MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
2138
9.04k
    } else {
2139
9.04k
      tmp |= MAY_BE_ANY | MAY_BE_REF | MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
2140
9.04k
      }
2141
17.4k
    if (write) {
2142
9.04k
      tmp |= MAY_BE_INDIRECT;
2143
9.04k
    }
2144
17.4k
  }
2145
23.2k
  if (t1 & MAY_BE_ARRAY) {
2146
21.0k
    if (insert) {
2147
1.56k
      tmp |= MAY_BE_NULL;
2148
19.4k
    } else {
2149
19.4k
      tmp |= MAY_BE_NULL | ((t1 & MAY_BE_ARRAY_OF_ANY) >> MAY_BE_ARRAY_SHIFT);
2150
19.4k
      if (tmp & MAY_BE_ARRAY) {
2151
18.1k
        tmp |= MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
2152
18.1k
      }
2153
19.4k
      if (tmp & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) {
2154
18.5k
        if (!write) {
2155
          /* can't be REF  because of ZVAL_COPY_DEREF() usage */
2156
9.94k
          tmp |= MAY_BE_RCN;
2157
9.94k
          if ((op_type & (IS_VAR|IS_TMP_VAR)) && (t1 & MAY_BE_RC1)) {
2158
2.84k
            tmp |= MAY_BE_RC1;
2159
2.84k
          }
2160
9.94k
        } else if (t1 & MAY_BE_ARRAY_OF_REF) {
2161
7.80k
          tmp |= MAY_BE_REF | MAY_BE_RC1 | MAY_BE_RCN;
2162
7.80k
        } else {
2163
820
          tmp |= MAY_BE_RC1 | MAY_BE_RCN;
2164
820
        }
2165
18.5k
      }
2166
19.4k
    }
2167
21.0k
    if (write) {
2168
10.5k
      tmp |= MAY_BE_INDIRECT;
2169
10.5k
    }
2170
21.0k
  }
2171
23.2k
  if (t1 & MAY_BE_STRING) {
2172
16.4k
    tmp |= MAY_BE_STRING | MAY_BE_RC1;
2173
16.4k
    if (write) {
2174
8.96k
      tmp |= MAY_BE_NULL;
2175
8.96k
    }
2176
16.4k
  }
2177
23.2k
  if (t1 & (MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE)) {
2178
19.8k
    tmp |= MAY_BE_NULL;
2179
19.8k
    if (write) {
2180
10.3k
      tmp |= MAY_BE_INDIRECT;
2181
10.3k
    }
2182
19.8k
  }
2183
23.2k
  if (t1 & (MAY_BE_TRUE|MAY_BE_LONG|MAY_BE_DOUBLE|MAY_BE_RESOURCE)) {
2184
17.5k
    if (!write) {
2185
8.38k
      tmp |= MAY_BE_NULL;
2186
8.38k
    }
2187
17.5k
  }
2188
23.2k
  return tmp;
2189
23.2k
}
2190
2191
static uint32_t assign_dim_array_result_type(
2192
30.8k
    uint32_t arr_type, uint32_t dim_type, uint32_t value_type, uint8_t dim_op_type) {
2193
30.8k
  uint32_t tmp = 0;
2194
  /* Only add key type if we have a value type. We want to maintain the invariant that a
2195
   * key type exists iff a value type exists even in dead code that may use empty types. */
2196
30.8k
  if (value_type & (MAY_BE_ANY|MAY_BE_UNDEF)) {
2197
30.8k
    if (value_type & MAY_BE_UNDEF) {
2198
868
      value_type |= MAY_BE_NULL;
2199
868
    }
2200
30.8k
    if (dim_op_type == IS_UNUSED) {
2201
19.3k
      if (arr_type & (MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE)) {
2202
1.42k
        tmp |= MAY_BE_ARRAY_PACKED;
2203
1.42k
      }
2204
19.3k
      tmp |= MAY_BE_HASH_ONLY(arr_type) ? MAY_BE_ARRAY_NUMERIC_HASH : MAY_BE_ARRAY_KEY_LONG;
2205
19.3k
    } else {
2206
11.5k
      if (dim_type & (MAY_BE_LONG|MAY_BE_FALSE|MAY_BE_TRUE|MAY_BE_RESOURCE|MAY_BE_DOUBLE)) {
2207
4.86k
        if (arr_type & (MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE)) {
2208
3.80k
          tmp |= MAY_BE_ARRAY_PACKED;
2209
3.80k
        }
2210
4.86k
        tmp |= MAY_BE_HASH_ONLY(arr_type) ? MAY_BE_ARRAY_NUMERIC_HASH : MAY_BE_ARRAY_KEY_LONG;
2211
4.86k
      }
2212
11.5k
      if (dim_type & MAY_BE_STRING) {
2213
5.32k
        tmp |= MAY_BE_ARRAY_KEY_STRING;
2214
5.32k
        if (dim_op_type != IS_CONST) {
2215
          // FIXME: numeric string
2216
2.95k
          if (arr_type & (MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE)) {
2217
2.37k
            tmp |= MAY_BE_ARRAY_PACKED;
2218
2.37k
          }
2219
2.95k
          tmp |= MAY_BE_HASH_ONLY(arr_type) ? MAY_BE_ARRAY_NUMERIC_HASH : MAY_BE_ARRAY_KEY_LONG;
2220
2.95k
        }
2221
5.32k
      }
2222
11.5k
      if (dim_type & (MAY_BE_UNDEF|MAY_BE_NULL)) {
2223
4.77k
        tmp |= MAY_BE_ARRAY_KEY_STRING;
2224
4.77k
      }
2225
11.5k
    }
2226
30.8k
  }
2227
  /* Only add value type if we have a key type. It might be that the key type is illegal
2228
   * for arrays. */
2229
30.8k
  if (tmp & MAY_BE_ARRAY_KEY_ANY) {
2230
30.8k
    tmp |= (value_type & MAY_BE_ANY) << MAY_BE_ARRAY_SHIFT;
2231
30.8k
  }
2232
30.8k
  tmp &= ~MAY_BE_ARRAY_EMPTY;
2233
30.8k
  return tmp;
2234
30.8k
}
2235
2236
static uint32_t assign_dim_result_type(
2237
12.2k
    uint32_t arr_type, uint32_t dim_type, uint32_t value_type, uint8_t dim_op_type) {
2238
12.2k
  uint32_t tmp = arr_type & ~(MAY_BE_RC1|MAY_BE_RCN);
2239
2240
12.2k
  if (arr_type & (MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE)) {
2241
7.77k
    tmp &= ~(MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE);
2242
7.77k
    tmp |= MAY_BE_ARRAY|MAY_BE_RC1;
2243
7.77k
  }
2244
12.2k
  if (tmp & (MAY_BE_ARRAY|MAY_BE_STRING)) {
2245
11.8k
    tmp |= MAY_BE_RC1;
2246
11.8k
  }
2247
12.2k
  if (tmp & (MAY_BE_OBJECT|MAY_BE_RESOURCE)) {
2248
6.24k
    tmp |= MAY_BE_RC1 | MAY_BE_RCN;
2249
6.24k
  }
2250
12.2k
  if (tmp & MAY_BE_ARRAY) {
2251
11.8k
    tmp |= assign_dim_array_result_type(arr_type, dim_type, value_type, dim_op_type);
2252
11.8k
  }
2253
12.2k
  return tmp;
2254
12.2k
}
2255
2256
/* For binary ops that have compound assignment operators */
2257
static uint32_t binary_op_result_type(
2258
    const zend_ssa *ssa, uint8_t opcode, uint32_t t1, uint32_t t2, int result_var,
2259
60.1k
    zend_long optimization_level) {
2260
60.1k
  uint32_t tmp = 0;
2261
60.1k
  uint32_t t1_type = (t1 & MAY_BE_ANY) | (t1 & MAY_BE_UNDEF ? MAY_BE_NULL : 0);
2262
60.1k
  uint32_t t2_type = (t2 & MAY_BE_ANY) | (t2 & MAY_BE_UNDEF ? MAY_BE_NULL : 0);
2263
2264
60.1k
  if (!(ZEND_OPTIMIZER_IGNORE_OVERLOADING & optimization_level)) {
2265
    /* Handle potentially overloaded operators.
2266
     * This could be made more precise by checking the class type, if known. */
2267
60.1k
    if ((t1_type & MAY_BE_OBJECT) || (t2_type & MAY_BE_OBJECT)) {
2268
      /* This is somewhat GMP specific. */
2269
25.6k
      tmp |= MAY_BE_OBJECT | MAY_BE_FALSE | MAY_BE_RC1;
2270
25.6k
    }
2271
60.1k
  }
2272
2273
60.1k
  switch (opcode) {
2274
11.9k
    case ZEND_ADD:
2275
11.9k
      if (t1_type == MAY_BE_LONG && t2_type == MAY_BE_LONG) {
2276
547
        if (result_var < 0 ||
2277
539
          !ssa->var_info[result_var].has_range ||
2278
533
            ssa->var_info[result_var].range.underflow ||
2279
372
            ssa->var_info[result_var].range.overflow) {
2280
          /* may overflow */
2281
284
          tmp |= MAY_BE_LONG | MAY_BE_DOUBLE;
2282
284
        } else {
2283
263
          tmp |= MAY_BE_LONG;
2284
263
        }
2285
11.3k
      } else if (t1_type == MAY_BE_DOUBLE || t2_type == MAY_BE_DOUBLE) {
2286
350
        tmp |= MAY_BE_DOUBLE;
2287
11.0k
      } else if (t1_type == MAY_BE_ARRAY && t2_type == MAY_BE_ARRAY) {
2288
62
        tmp |= MAY_BE_ARRAY | MAY_BE_RC1;
2289
62
        tmp |= t1 & (MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF);
2290
62
        tmp |= t2 & (MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF);
2291
10.9k
      } else {
2292
10.9k
        tmp |= MAY_BE_LONG | MAY_BE_DOUBLE;
2293
10.9k
        if ((t1_type & MAY_BE_ARRAY) && (t2_type & MAY_BE_ARRAY)) {
2294
1.57k
          tmp |= MAY_BE_ARRAY | MAY_BE_RC1;
2295
1.57k
          tmp |= t1 & (MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF);
2296
1.57k
          tmp |= t2 & (MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF);
2297
1.57k
        }
2298
10.9k
      }
2299
11.9k
      break;
2300
7.33k
    case ZEND_SUB:
2301
15.6k
    case ZEND_MUL:
2302
15.6k
      if (t1_type == MAY_BE_LONG && t2_type == MAY_BE_LONG) {
2303
1.56k
        if (result_var < 0 ||
2304
1.56k
          !ssa->var_info[result_var].has_range ||
2305
1.55k
            ssa->var_info[result_var].range.underflow ||
2306
1.06k
            ssa->var_info[result_var].range.overflow) {
2307
          /* may overflow */
2308
1.06k
          tmp |= MAY_BE_LONG | MAY_BE_DOUBLE;
2309
1.06k
        } else {
2310
501
          tmp |= MAY_BE_LONG;
2311
501
        }
2312
14.1k
      } else if (t1_type == MAY_BE_DOUBLE || t2_type == MAY_BE_DOUBLE) {
2313
288
        tmp |= MAY_BE_DOUBLE;
2314
13.8k
      } else {
2315
13.8k
        tmp |= MAY_BE_LONG | MAY_BE_DOUBLE;
2316
13.8k
      }
2317
15.6k
      break;
2318
4.47k
    case ZEND_DIV:
2319
4.87k
    case ZEND_POW:
2320
4.87k
      if (t1_type == MAY_BE_DOUBLE || t2_type == MAY_BE_DOUBLE) {
2321
32
        tmp |= MAY_BE_DOUBLE;
2322
4.84k
      } else {
2323
4.84k
        tmp |= MAY_BE_LONG | MAY_BE_DOUBLE;
2324
4.84k
      }
2325
      /* Division by zero results in Inf/-Inf/Nan (double), so it doesn't need any special
2326
       * handling */
2327
4.87k
      break;
2328
2.80k
    case ZEND_MOD:
2329
2.80k
      tmp |= MAY_BE_LONG;
2330
      /* Division by zero results in an exception, so it doesn't need any special handling */
2331
2.80k
      break;
2332
1.37k
    case ZEND_BW_OR:
2333
6.67k
    case ZEND_BW_AND:
2334
9.58k
    case ZEND_BW_XOR:
2335
9.58k
      if ((t1_type & MAY_BE_STRING) && (t2_type & MAY_BE_STRING)) {
2336
1.18k
        tmp |= MAY_BE_STRING | MAY_BE_RC1 | MAY_BE_RCN;
2337
1.18k
      }
2338
9.58k
      if ((t1_type & ~MAY_BE_STRING) || (t2_type & ~MAY_BE_STRING)) {
2339
9.33k
        tmp |= MAY_BE_LONG;
2340
9.33k
      }
2341
9.58k
      break;
2342
1.84k
    case ZEND_SL:
2343
3.26k
    case ZEND_SR:
2344
3.26k
      tmp |= MAY_BE_LONG;
2345
3.26k
      break;
2346
12.0k
    case ZEND_CONCAT:
2347
12.0k
    case ZEND_FAST_CONCAT:
2348
      /* TODO: +MAY_BE_OBJECT ??? */
2349
12.0k
      tmp = MAY_BE_STRING | MAY_BE_RC1 | MAY_BE_RCN;
2350
12.0k
      break;
2351
0
    default: ZEND_UNREACHABLE();
2352
60.1k
  }
2353
60.1k
  return tmp;
2354
60.1k
}
2355
2356
32.6k
static uint32_t zend_convert_type_declaration_mask(uint32_t type_mask) {
2357
32.6k
  uint32_t result_mask = type_mask & MAY_BE_ANY;
2358
32.6k
  if (type_mask & MAY_BE_VOID) {
2359
2.30k
    result_mask |= MAY_BE_NULL;
2360
2.30k
  }
2361
32.6k
  if (type_mask & MAY_BE_CALLABLE) {
2362
192
    result_mask |= MAY_BE_STRING|MAY_BE_OBJECT|MAY_BE_ARRAY|MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF;
2363
192
  }
2364
32.6k
  if (type_mask & MAY_BE_STATIC) {
2365
158
    result_mask |= MAY_BE_OBJECT;
2366
158
  }
2367
32.6k
  if (type_mask & MAY_BE_ARRAY) {
2368
7.40k
    result_mask |= MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF;
2369
7.40k
  }
2370
32.6k
  return result_mask;
2371
32.6k
}
2372
2373
static uint32_t zend_convert_type(const zend_script *script, zend_type type, zend_class_entry **pce)
2374
55.5k
{
2375
55.5k
  if (pce) {
2376
55.5k
    *pce = NULL;
2377
55.5k
  }
2378
2379
55.5k
  if (!ZEND_TYPE_IS_SET(type)) {
2380
22.8k
    return MAY_BE_ANY|MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF|MAY_BE_RC1|MAY_BE_RCN;
2381
22.8k
  }
2382
2383
32.6k
  uint32_t tmp = zend_convert_type_declaration_mask(ZEND_TYPE_PURE_MASK(type));
2384
32.6k
  if (ZEND_TYPE_IS_COMPLEX(type)) {
2385
5.03k
    tmp |= MAY_BE_OBJECT;
2386
5.03k
    if (pce) {
2387
      /* As we only have space to store one CE,
2388
       * we use a plain object type for class unions. */
2389
5.03k
      if (ZEND_TYPE_HAS_NAME(type)) {
2390
4.15k
        zend_string *lcname = zend_string_tolower(ZEND_TYPE_NAME(type));
2391
        // TODO: Pass through op_array.
2392
4.15k
        *pce = zend_optimizer_get_class_entry(script, NULL, lcname);
2393
4.15k
        zend_string_release_ex(lcname, 0);
2394
4.15k
      }
2395
5.03k
    }
2396
5.03k
  }
2397
32.6k
  if (tmp & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) {
2398
22.2k
    tmp |= MAY_BE_RC1 | MAY_BE_RCN;
2399
22.2k
  }
2400
32.6k
  return tmp;
2401
55.5k
}
2402
2403
ZEND_API uint32_t zend_fetch_arg_info_type(const zend_script *script, const zend_arg_info *arg_info, zend_class_entry **pce)
2404
49.2k
{
2405
49.2k
  return zend_convert_type(script, arg_info->type, pce);
2406
49.2k
}
2407
2408
7.61k
static const zend_property_info *lookup_prop_info(const zend_class_entry *ce, zend_string *name, zend_class_entry *scope) {
2409
7.61k
  const zend_property_info *prop_info;
2410
2411
  /* If the class is linked, reuse the precise runtime logic. */
2412
7.61k
  if ((ce->ce_flags & ZEND_ACC_LINKED)
2413
6.75k
   && (!scope || (scope->ce_flags & ZEND_ACC_LINKED))) {
2414
6.72k
    const zend_class_entry *prev_scope = EG(fake_scope);
2415
6.72k
    EG(fake_scope) = scope;
2416
6.72k
    prop_info = zend_get_property_info(ce, name, 1);
2417
6.72k
    EG(fake_scope) = prev_scope;
2418
6.72k
    if (prop_info && prop_info != ZEND_WRONG_PROPERTY_INFO) {
2419
5.61k
      return prop_info;
2420
5.61k
    }
2421
1.10k
    return NULL;
2422
6.72k
  }
2423
2424
  /* Otherwise, handle only some safe cases */
2425
891
  prop_info = zend_hash_find_ptr(&ce->properties_info, name);
2426
891
  if (prop_info &&
2427
688
    ((prop_info->ce == scope) ||
2428
32
     (!scope && (prop_info->flags & ZEND_ACC_PUBLIC)))
2429
891
  ) {
2430
656
    return prop_info;
2431
656
  }
2432
235
  return NULL;
2433
891
}
2434
2435
static const zend_property_info *zend_fetch_prop_info(const zend_op_array *op_array, const zend_ssa *ssa, const zend_op *opline, const zend_ssa_op *ssa_op)
2436
53.4k
{
2437
53.4k
  const zend_property_info *prop_info = NULL;
2438
53.4k
  if (opline->op2_type == IS_CONST) {
2439
52.0k
    const zend_class_entry *ce = NULL;
2440
2441
52.0k
    if (opline->op1_type == IS_UNUSED && !(op_array->fn_flags & ZEND_ACC_TRAIT_CLONE)) {
2442
4.77k
      ce = op_array->scope;
2443
47.2k
    } else if (ssa_op->op1_use >= 0) {
2444
47.2k
      ce = ssa->var_info[ssa_op->op1_use].ce;
2445
47.2k
    }
2446
52.0k
    if (ce) {
2447
5.46k
      prop_info = lookup_prop_info(ce,
2448
5.46k
        Z_STR_P(CRT_CONSTANT(opline->op2)),
2449
5.46k
        op_array->scope);
2450
5.46k
      if (prop_info && (prop_info->flags & ZEND_ACC_STATIC)) {
2451
28
        prop_info = NULL;
2452
28
      }
2453
5.46k
    }
2454
52.0k
  }
2455
53.4k
  return prop_info;
2456
53.4k
}
2457
2458
static const zend_property_info *zend_fetch_static_prop_info(const zend_script *script, const zend_op_array *op_array, const zend_ssa *ssa, const zend_op *opline)
2459
2.54k
{
2460
2.54k
  const zend_property_info *prop_info = NULL;
2461
2.54k
  if (opline->op1_type == IS_CONST) {
2462
2.34k
    const zend_class_entry *ce = NULL;
2463
2.34k
    if (opline->op2_type == IS_UNUSED) {
2464
878
      uint32_t fetch_type = opline->op2.num & ZEND_FETCH_CLASS_MASK;
2465
878
      switch (fetch_type) {
2466
776
        case ZEND_FETCH_CLASS_SELF:
2467
858
        case ZEND_FETCH_CLASS_STATIC:
2468
          /* We enforce that static property types cannot change during inheritance, so
2469
           * handling static the same way as self here is legal. */
2470
858
          ce = op_array->scope;
2471
858
          break;
2472
20
        case ZEND_FETCH_CLASS_PARENT:
2473
20
          if (op_array->scope && (op_array->scope->ce_flags & ZEND_ACC_LINKED)) {
2474
16
            ce = op_array->scope->parent;
2475
16
          }
2476
20
          break;
2477
878
      }
2478
1.46k
    } else if (opline->op2_type == IS_CONST) {
2479
1.35k
      const zval *zv = CRT_CONSTANT(opline->op2);
2480
1.35k
      ce = zend_optimizer_get_class_entry(script, op_array, Z_STR_P(zv + 1));
2481
1.35k
    }
2482
2483
2.34k
    if (ce) {
2484
2.14k
      const zval *zv = CRT_CONSTANT(opline->op1);
2485
2.14k
      prop_info = lookup_prop_info(ce, Z_STR_P(zv), op_array->scope);
2486
2.14k
      if (prop_info && !(prop_info->flags & ZEND_ACC_STATIC)) {
2487
56
        prop_info = NULL;
2488
56
      }
2489
2.14k
    }
2490
2.34k
  }
2491
2.54k
  return prop_info;
2492
2.54k
}
2493
2494
static uint32_t zend_fetch_prop_type(const zend_script *script, const zend_property_info *prop_info, zend_class_entry **pce)
2495
56.0k
{
2496
56.0k
  if (!prop_info) {
2497
49.8k
    if (pce) {
2498
49.2k
      *pce = NULL;
2499
49.2k
    }
2500
49.8k
    return MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF | MAY_BE_RC1 | MAY_BE_RCN;
2501
49.8k
  }
2502
2503
6.21k
  return zend_convert_type(script, prop_info->type, pce);
2504
56.0k
}
2505
2506
static bool result_may_be_separated(const zend_ssa *ssa, const zend_ssa_op *ssa_op)
2507
4.91k
{
2508
4.91k
  int tmp_var = ssa_op->result_def;
2509
2510
4.91k
  if (ssa->vars[tmp_var].use_chain >= 0
2511
4.91k
   && !ssa->vars[tmp_var].phi_use_chain) {
2512
4.91k
    const zend_ssa_op *use_op = &ssa->ops[ssa->vars[tmp_var].use_chain];
2513
2514
    /* TODO: analyze instructions between ssa_op and use_op */
2515
4.91k
    if (use_op == ssa_op + 1) {
2516
4.58k
      if ((use_op->op1_use == tmp_var && use_op->op1_use_chain < 0)
2517
4.57k
       || (use_op->op2_use == tmp_var && use_op->op2_use_chain < 0)) {
2518
4.57k
        return 0;
2519
4.57k
      }
2520
4.58k
    }
2521
4.91k
  }
2522
339
  return 1;
2523
4.91k
}
2524
2525
static zend_always_inline zend_result _zend_update_type_info(
2526
      const zend_op_array *op_array,
2527
      zend_ssa            *ssa,
2528
      const zend_script   *script,
2529
      zend_bitset          worklist,
2530
      const zend_op       *opline,
2531
      zend_ssa_op         *ssa_op,
2532
      const zend_op      **ssa_opcodes,
2533
      zend_long            optimization_level,
2534
      bool            update_worklist)
2535
884k
{
2536
884k
  uint32_t t1, t2;
2537
884k
  uint32_t tmp, orig;
2538
884k
  zend_ssa_var *ssa_vars = ssa->vars;
2539
884k
  zend_ssa_var_info *ssa_var_info = ssa->var_info;
2540
884k
  zend_class_entry *ce;
2541
884k
  int j;
2542
2543
884k
  if (opline->opcode == ZEND_OP_DATA) {
2544
335
    opline--;
2545
335
    ssa_op--;
2546
335
  }
2547
2548
884k
  t1 = OP1_INFO();
2549
884k
  t2 = OP2_INFO();
2550
2551
  /* If one of the operands cannot have any type, this means the operand derives from
2552
   * unreachable code. Propagate the empty result early, so that that the following
2553
   * code may assume that operands have at least one type. */
2554
884k
  if (!(t1 & (MAY_BE_ANY|MAY_BE_UNDEF|MAY_BE_CLASS))
2555
881k
   || !(t2 & (MAY_BE_ANY|MAY_BE_UNDEF|MAY_BE_CLASS))
2556
881k
   || (ssa_op->result_use >= 0 && !(RES_USE_INFO() & (MAY_BE_ANY|MAY_BE_UNDEF|MAY_BE_CLASS)))
2557
881k
   || ((opline->opcode == ZEND_ASSIGN_DIM_OP
2558
879k
     || opline->opcode == ZEND_ASSIGN_OBJ_OP
2559
878k
     || opline->opcode == ZEND_ASSIGN_STATIC_PROP_OP
2560
878k
     || opline->opcode == ZEND_ASSIGN_DIM
2561
867k
     || opline->opcode == ZEND_ASSIGN_OBJ)
2562
19.5k
      && !(OP1_DATA_INFO() & (MAY_BE_ANY|MAY_BE_UNDEF|MAY_BE_CLASS)) /*&& 0*/)) {
2563
3.12k
    tmp = 0;
2564
3.12k
    if (ssa_op->result_def >= 0 && !(ssa_var_info[ssa_op->result_def].type & MAY_BE_REF)) {
2565
2.47k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
2566
2.47k
    }
2567
3.12k
    if (ssa_op->op1_def >= 0 && !(ssa_var_info[ssa_op->op1_def].type & MAY_BE_REF)) {
2568
1.01k
      UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
2569
1.01k
    }
2570
3.12k
    if (ssa_op->op2_def >= 0 && !(ssa_var_info[ssa_op->op2_def].type & MAY_BE_REF)) {
2571
4
      UPDATE_SSA_TYPE(tmp, ssa_op->op2_def);
2572
4
    }
2573
3.12k
    if (opline->opcode == ZEND_ASSIGN_DIM_OP
2574
3.12k
     || opline->opcode == ZEND_ASSIGN_OBJ_OP
2575
3.11k
     || opline->opcode == ZEND_ASSIGN_STATIC_PROP_OP
2576
3.11k
     || opline->opcode == ZEND_ASSIGN_DIM
2577
2.79k
     || opline->opcode == ZEND_ASSIGN_OBJ) {
2578
491
      if ((ssa_op+1)->op1_def >= 0 && !(ssa_var_info[(ssa_op+1)->op1_def].type & MAY_BE_REF)) {
2579
0
        UPDATE_SSA_TYPE(tmp, (ssa_op+1)->op1_def);
2580
0
      }
2581
491
    }
2582
3.12k
    return SUCCESS;
2583
3.12k
  }
2584
2585
880k
  switch (opline->opcode) {
2586
8.88k
    case ZEND_ADD:
2587
15.6k
    case ZEND_SUB:
2588
22.1k
    case ZEND_MUL:
2589
26.4k
    case ZEND_DIV:
2590
26.8k
    case ZEND_POW:
2591
29.4k
    case ZEND_MOD:
2592
30.6k
    case ZEND_BW_OR:
2593
33.4k
    case ZEND_BW_AND:
2594
36.3k
    case ZEND_BW_XOR:
2595
38.1k
    case ZEND_SL:
2596
39.1k
    case ZEND_SR:
2597
47.9k
    case ZEND_CONCAT:
2598
47.9k
      tmp = binary_op_result_type(ssa, opline->opcode, t1, t2, ssa_op->result_def, optimization_level);
2599
47.9k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
2600
47.9k
      break;
2601
47.9k
    case ZEND_BW_NOT:
2602
19.6k
      tmp = 0;
2603
19.6k
      if (t1 & MAY_BE_STRING) {
2604
18.1k
        tmp |= MAY_BE_STRING | MAY_BE_RC1 | MAY_BE_RCN;
2605
18.1k
      }
2606
19.6k
      if (t1 & (MAY_BE_ANY-MAY_BE_STRING)) {
2607
19.3k
        tmp |= MAY_BE_LONG;
2608
19.3k
      }
2609
19.6k
      if (!(ZEND_OPTIMIZER_IGNORE_OVERLOADING & optimization_level)) {
2610
19.6k
        if (t1 & MAY_BE_OBJECT) {
2611
          /* Potentially overloaded operator. */
2612
17.7k
          tmp |= MAY_BE_OBJECT | MAY_BE_RC1;
2613
17.7k
        }
2614
19.6k
      }
2615
19.6k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
2616
19.6k
      break;
2617
77.5k
    case ZEND_BEGIN_SILENCE:
2618
77.5k
      UPDATE_SSA_TYPE(MAY_BE_LONG, ssa_op->result_def);
2619
77.5k
      break;
2620
77.5k
    case ZEND_BOOL_NOT:
2621
3.18k
    case ZEND_BOOL_XOR:
2622
5.38k
    case ZEND_IS_IDENTICAL:
2623
6.30k
    case ZEND_IS_NOT_IDENTICAL:
2624
9.73k
    case ZEND_IS_EQUAL:
2625
13.2k
    case ZEND_IS_NOT_EQUAL:
2626
22.6k
    case ZEND_IS_SMALLER:
2627
25.8k
    case ZEND_IS_SMALLER_OR_EQUAL:
2628
26.1k
    case ZEND_INSTANCEOF:
2629
27.0k
    case ZEND_JMPZ_EX:
2630
28.0k
    case ZEND_JMPNZ_EX:
2631
28.1k
    case ZEND_CASE:
2632
28.3k
    case ZEND_CASE_STRICT:
2633
29.5k
    case ZEND_BOOL:
2634
29.6k
    case ZEND_ISSET_ISEMPTY_CV:
2635
29.7k
    case ZEND_ISSET_ISEMPTY_VAR:
2636
31.4k
    case ZEND_ISSET_ISEMPTY_DIM_OBJ:
2637
32.2k
    case ZEND_ISSET_ISEMPTY_PROP_OBJ:
2638
32.3k
    case ZEND_ISSET_ISEMPTY_STATIC_PROP:
2639
32.4k
    case ZEND_ASSERT_CHECK:
2640
32.4k
    case ZEND_IN_ARRAY:
2641
32.6k
    case ZEND_ARRAY_KEY_EXISTS:
2642
32.6k
      UPDATE_SSA_TYPE(MAY_BE_FALSE|MAY_BE_TRUE, ssa_op->result_def);
2643
32.6k
      break;
2644
32.6k
    case ZEND_CAST:
2645
2.06k
      if (ssa_op->op1_def >= 0) {
2646
0
        tmp = t1;
2647
0
        if ((t1 & (MAY_BE_ARRAY|MAY_BE_OBJECT)) &&
2648
0
            (opline->extended_value == IS_ARRAY ||
2649
0
             opline->extended_value == IS_OBJECT)) {
2650
0
          tmp |= MAY_BE_RCN;
2651
0
        } else if ((t1 & MAY_BE_STRING) &&
2652
0
            opline->extended_value == IS_STRING) {
2653
0
          tmp |= MAY_BE_RCN;
2654
0
        }
2655
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
2656
0
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
2657
0
      }
2658
2.06k
      tmp = 1 << opline->extended_value;
2659
2.06k
      if (tmp & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) {
2660
1.41k
        if ((tmp & MAY_BE_ANY) == (t1 & MAY_BE_ANY)) {
2661
57
          tmp |= (t1 & MAY_BE_RC1) | MAY_BE_RCN;
2662
1.36k
        } else if ((opline->extended_value == IS_ARRAY ||
2663
1.13k
              opline->extended_value == IS_OBJECT) &&
2664
577
               (t1 & (MAY_BE_ARRAY|MAY_BE_OBJECT))) {
2665
511
            tmp |= MAY_BE_RC1 | MAY_BE_RCN;
2666
849
        } else if (opline->extended_value == IS_STRING &&
2667
783
               (t1 & (MAY_BE_STRING|MAY_BE_OBJECT))) {
2668
550
          tmp |= MAY_BE_RC1 | MAY_BE_RCN;
2669
550
        } else {
2670
299
          tmp |= MAY_BE_RC1;
2671
299
          if (opline->extended_value == IS_ARRAY
2672
19
           && (t1 & (MAY_BE_UNDEF|MAY_BE_NULL))) {
2673
9
            tmp |= MAY_BE_RCN;
2674
9
          }
2675
299
        }
2676
1.41k
      }
2677
2.06k
      if (opline->extended_value == IS_ARRAY) {
2678
224
        if (t1 & (MAY_BE_UNDEF|MAY_BE_NULL)) {
2679
140
          tmp |= MAY_BE_ARRAY_EMPTY;
2680
140
        }
2681
224
        if (t1 & MAY_BE_ARRAY) {
2682
133
          tmp |= t1 & (MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF);
2683
133
        }
2684
224
        if (t1 & MAY_BE_OBJECT) {
2685
203
          tmp |= MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
2686
203
        } else if (t1 & (MAY_BE_ANY - MAY_BE_NULL)) {
2687
18
          tmp |= ((t1 & (MAY_BE_ANY - MAY_BE_NULL)) << MAY_BE_ARRAY_SHIFT) | ((t1 & MAY_BE_NULL) ? MAY_BE_ARRAY_KEY_LONG : MAY_BE_ARRAY_PACKED);
2688
18
        }
2689
224
      }
2690
2.06k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
2691
2.06k
      break;
2692
10.0k
    case ZEND_QM_ASSIGN:
2693
11.4k
    case ZEND_JMP_SET:
2694
14.9k
    case ZEND_COALESCE:
2695
16.6k
    case ZEND_COPY_TMP:
2696
16.6k
      if (ssa_op->op1_def >= 0) {
2697
0
        tmp = t1;
2698
0
        if (t1 & (MAY_BE_RC1|MAY_BE_REF)) {
2699
0
          tmp |= MAY_BE_RCN;
2700
0
        }
2701
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
2702
0
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
2703
0
      }
2704
16.6k
      tmp = t1 & ~MAY_BE_UNDEF;
2705
16.6k
      if (opline->opcode != ZEND_COPY_TMP || opline->op1_type != IS_VAR) {
2706
16.6k
        tmp &= ~MAY_BE_REF;
2707
16.6k
      }
2708
16.6k
      if (t1 & MAY_BE_UNDEF) {
2709
2.30k
        tmp |= MAY_BE_NULL;
2710
2.30k
      }
2711
16.6k
      if (t1 & (MAY_BE_RC1|MAY_BE_RCN)) {
2712
8.49k
        tmp |= (t1 & (MAY_BE_RC1|MAY_BE_RCN));
2713
8.49k
        if (opline->opcode == ZEND_COPY_TMP || opline->op1_type == IS_CV) {
2714
2.39k
          tmp |= MAY_BE_RCN;
2715
2.39k
        }
2716
8.49k
      }
2717
16.6k
      if (opline->opcode == ZEND_COALESCE || opline->opcode == ZEND_JMP_SET) {
2718
        /* COALESCE and JMP_SET result can't be null */
2719
4.88k
        tmp &= ~MAY_BE_NULL;
2720
4.88k
        if (opline->opcode == ZEND_JMP_SET) {
2721
          /* JMP_SET result can't be false either */
2722
1.39k
          tmp &= ~MAY_BE_FALSE;
2723
1.39k
        }
2724
4.88k
      }
2725
16.6k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
2726
16.6k
      COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->result_def);
2727
16.6k
      break;
2728
29.6k
    case ZEND_JMP_NULL:
2729
29.6k
    {
2730
29.6k
      uint32_t short_circuiting_type = opline->extended_value & ZEND_SHORT_CIRCUITING_CHAIN_MASK;
2731
29.6k
      if (short_circuiting_type == ZEND_SHORT_CIRCUITING_CHAIN_EXPR) {
2732
29.4k
        tmp = MAY_BE_NULL;
2733
29.4k
      } else if (short_circuiting_type == ZEND_SHORT_CIRCUITING_CHAIN_ISSET) {
2734
66
        tmp = MAY_BE_FALSE;
2735
66
      } else {
2736
66
        ZEND_ASSERT(short_circuiting_type == ZEND_SHORT_CIRCUITING_CHAIN_EMPTY);
2737
66
        tmp = MAY_BE_TRUE;
2738
66
      }
2739
29.6k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
2740
29.6k
      break;
2741
29.6k
    }
2742
29.6k
    case ZEND_ASSIGN_OP:
2743
11.5k
    case ZEND_ASSIGN_DIM_OP:
2744
12.2k
    case ZEND_ASSIGN_OBJ_OP:
2745
12.2k
    case ZEND_ASSIGN_STATIC_PROP_OP:
2746
12.2k
    {
2747
12.2k
      const zend_property_info *prop_info = NULL;
2748
12.2k
      orig = 0;
2749
12.2k
      tmp = 0;
2750
12.2k
      if (opline->opcode == ZEND_ASSIGN_OBJ_OP) {
2751
616
        prop_info = zend_fetch_prop_info(op_array, ssa, opline, ssa_op);
2752
616
        orig = t1;
2753
616
        t1 = zend_fetch_prop_type(script, prop_info, NULL);
2754
616
        t2 = OP1_DATA_INFO();
2755
11.5k
      } else if (opline->opcode == ZEND_ASSIGN_DIM_OP) {
2756
2.25k
        if (t1 & MAY_BE_ARRAY_OF_REF) {
2757
1.61k
              tmp |= MAY_BE_REF;
2758
1.61k
        }
2759
2.25k
        orig = t1;
2760
2.25k
        t1 = zend_array_element_type(t1, opline->op1_type, 1, 0);
2761
2.25k
        t2 = OP1_DATA_INFO();
2762
9.34k
      } else if (opline->opcode == ZEND_ASSIGN_STATIC_PROP_OP) {
2763
12
        prop_info = zend_fetch_static_prop_info(script, op_array, ssa, opline);
2764
12
        t1 = zend_fetch_prop_type(script, prop_info, NULL);
2765
12
        t2 = OP1_DATA_INFO();
2766
9.33k
      } else {
2767
9.33k
        if (t1 & MAY_BE_REF) {
2768
2.48k
              tmp |= MAY_BE_REF;
2769
2.48k
        }
2770
9.33k
      }
2771
2772
12.2k
      tmp |= binary_op_result_type(
2773
12.2k
        ssa, opline->extended_value, t1, t2,
2774
12.2k
        opline->opcode == ZEND_ASSIGN_OP ? ssa_op->op1_def : -1, optimization_level);
2775
12.2k
      if (tmp & (MAY_BE_STRING|MAY_BE_ARRAY)) {
2776
4.37k
        tmp |= MAY_BE_RC1 | MAY_BE_RCN;
2777
4.37k
      }
2778
12.2k
      if (tmp & (MAY_BE_OBJECT|MAY_BE_RESOURCE)) {
2779
4.31k
        tmp |= MAY_BE_RC1 | MAY_BE_RCN;
2780
4.31k
      }
2781
2782
12.2k
      if (opline->opcode == ZEND_ASSIGN_DIM_OP) {
2783
2.25k
        if (opline->op1_type == IS_CV) {
2784
2.10k
          orig = assign_dim_result_type(orig, OP2_INFO(), tmp, opline->op2_type);
2785
2.10k
          UPDATE_SSA_TYPE(orig, ssa_op->op1_def);
2786
2.10k
          COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
2787
2.10k
        }
2788
9.95k
      } else if (opline->opcode == ZEND_ASSIGN_OBJ_OP) {
2789
616
        if (opline->op1_type == IS_CV) {
2790
584
          orig = (orig & (MAY_BE_REF|MAY_BE_OBJECT))|MAY_BE_RC1|MAY_BE_RCN;
2791
584
          UPDATE_SSA_TYPE(orig, ssa_op->op1_def);
2792
584
          COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
2793
584
        }
2794
9.34k
      } else if (opline->opcode == ZEND_ASSIGN_STATIC_PROP_OP) {
2795
        /* Nothing to do */
2796
9.33k
      } else {
2797
9.33k
        if (opline->opcode == ZEND_ASSIGN_OP && ssa_op->result_def >= 0 && (tmp & MAY_BE_RC1)) {
2798
3.18k
          tmp |= MAY_BE_RCN;
2799
3.18k
        }
2800
9.33k
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
2801
9.33k
      }
2802
12.2k
      if (ssa_op->result_def >= 0) {
2803
6.68k
        ce = NULL;
2804
6.68k
        if (opline->opcode == ZEND_ASSIGN_DIM_OP) {
2805
1.02k
          if (opline->op2_type == IS_UNUSED) {
2806
            /* When appending to an array and the LONG_MAX key is already used
2807
             * null will be returned. */
2808
30
            tmp |= MAY_BE_NULL;
2809
30
          }
2810
1.02k
          if (t2 & (MAY_BE_ARRAY | MAY_BE_OBJECT)) {
2811
            /* Arrays and objects cannot be used as keys. */
2812
296
            tmp |= MAY_BE_NULL;
2813
296
          }
2814
1.02k
          if (t1 & (MAY_BE_ANY - (MAY_BE_NULL | MAY_BE_FALSE | MAY_BE_STRING | MAY_BE_ARRAY))) {
2815
            /* null and false are implicitly converted to array, anything else
2816
             * results in a null return value. */
2817
921
            tmp |= MAY_BE_NULL;
2818
921
          }
2819
1.02k
          if (tmp & MAY_BE_REF) {
2820
            /* Typed reference may cause auto conversion */
2821
913
            tmp |= MAY_BE_ANY;
2822
913
          }
2823
5.66k
        } else if (opline->opcode == ZEND_ASSIGN_OBJ_OP) {
2824
          /* The return value must also satisfy the property type */
2825
248
          if (prop_info) {
2826
16
            t1 = zend_fetch_prop_type(script, prop_info, &ce);
2827
16
            if ((t1 & (MAY_BE_LONG|MAY_BE_DOUBLE)) == MAY_BE_LONG
2828
12
             && (tmp & (MAY_BE_LONG|MAY_BE_DOUBLE)) == MAY_BE_DOUBLE) {
2829
              /* DOUBLE may be auto-converted to LONG */
2830
0
              tmp |= MAY_BE_LONG;
2831
0
              tmp &= ~MAY_BE_DOUBLE;
2832
16
            } else if ((t1 & (MAY_BE_LONG|MAY_BE_DOUBLE|MAY_BE_STRING)) == MAY_BE_STRING
2833
0
             && (tmp & (MAY_BE_LONG|MAY_BE_DOUBLE))) {
2834
              /* LONG/DOUBLE may be auto-converted to STRING */
2835
0
              tmp |= MAY_BE_STRING;
2836
0
              tmp &= ~(MAY_BE_LONG|MAY_BE_DOUBLE);
2837
0
            }
2838
16
            tmp &= t1;
2839
232
          } else {
2840
232
            tmp |= MAY_BE_LONG | MAY_BE_STRING;
2841
232
          }
2842
5.41k
        } else if (opline->opcode == ZEND_ASSIGN_STATIC_PROP_OP) {
2843
          /* The return value must also satisfy the property type */
2844
12
          if (prop_info) {
2845
10
            t1 = zend_fetch_prop_type(script, prop_info, &ce);
2846
10
            if ((t1 & (MAY_BE_LONG|MAY_BE_DOUBLE)) == MAY_BE_LONG
2847
0
             && (tmp & (MAY_BE_LONG|MAY_BE_DOUBLE)) == MAY_BE_DOUBLE) {
2848
              /* DOUBLE may be auto-converted to LONG */
2849
0
              tmp |= MAY_BE_LONG;
2850
0
              tmp &= ~MAY_BE_DOUBLE;
2851
10
            } else if ((t1 & (MAY_BE_LONG|MAY_BE_DOUBLE|MAY_BE_STRING)) == MAY_BE_STRING
2852
10
             && (tmp & (MAY_BE_LONG|MAY_BE_DOUBLE))) {
2853
              /* LONG/DOUBLE may be auto-converted to STRING */
2854
10
              tmp |= MAY_BE_STRING;
2855
10
              tmp &= ~(MAY_BE_LONG|MAY_BE_DOUBLE);
2856
10
            }
2857
10
            tmp &= t1;
2858
10
          } else {
2859
2
            tmp |= MAY_BE_LONG | MAY_BE_STRING;
2860
2
          }
2861
5.40k
        } else {
2862
5.40k
          if (tmp & MAY_BE_REF) {
2863
            /* Typed reference may cause auto conversion */
2864
1.42k
            tmp |= MAY_BE_ANY;
2865
1.42k
          }
2866
5.40k
        }
2867
6.68k
        tmp &= ~MAY_BE_REF;
2868
6.68k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
2869
6.68k
        if (ce) {
2870
0
          UPDATE_SSA_OBJ_TYPE(ce, 1, ssa_op->result_def);
2871
0
        }
2872
6.68k
      }
2873
12.2k
      break;
2874
12.2k
    }
2875
12.2k
    case ZEND_PRE_INC:
2876
8.45k
    case ZEND_PRE_DEC:
2877
8.45k
      tmp = 0;
2878
8.45k
      if (t1 & (MAY_BE_RC1|MAY_BE_RCN)) {
2879
2.73k
        tmp |= MAY_BE_RC1;
2880
2.73k
        if (ssa_op->result_def >= 0) {
2881
600
          tmp |= MAY_BE_RCN;
2882
600
        }
2883
2.73k
      }
2884
8.45k
      if ((t1 & (MAY_BE_ANY|MAY_BE_UNDEF)) == MAY_BE_LONG) {
2885
1.88k
        if (!ssa_var_info[ssa_op->op1_use].has_range ||
2886
1.88k
            (opline->opcode == ZEND_PRE_DEC &&
2887
232
             (ssa_var_info[ssa_op->op1_use].range.underflow ||
2888
207
              ssa_var_info[ssa_op->op1_use].range.min == ZEND_LONG_MIN)) ||
2889
1.69k
             (opline->opcode == ZEND_PRE_INC &&
2890
1.64k
              (ssa_var_info[ssa_op->op1_use].range.overflow ||
2891
1.17k
               ssa_var_info[ssa_op->op1_use].range.max == ZEND_LONG_MAX))) {
2892
          /* may overflow */
2893
653
          tmp |= MAY_BE_LONG | MAY_BE_DOUBLE;
2894
1.22k
        } else {
2895
1.22k
          tmp |= MAY_BE_LONG;
2896
1.22k
        }
2897
6.57k
      } else {
2898
6.57k
        if (t1 & (MAY_BE_UNDEF | MAY_BE_NULL)) {
2899
5.49k
          if (opline->opcode == ZEND_PRE_INC) {
2900
4.53k
            tmp |= MAY_BE_LONG;
2901
4.53k
          } else {
2902
962
            tmp |= MAY_BE_NULL;
2903
962
          }
2904
5.49k
        }
2905
6.57k
        if (t1 & MAY_BE_LONG) {
2906
5.13k
          tmp |= MAY_BE_LONG | MAY_BE_DOUBLE;
2907
5.13k
        }
2908
6.57k
        if (t1 & MAY_BE_DOUBLE) {
2909
4.38k
          tmp |= MAY_BE_DOUBLE;
2910
4.38k
        }
2911
6.57k
        if (t1 & MAY_BE_STRING) {
2912
2.62k
          tmp |= MAY_BE_STRING | MAY_BE_LONG | MAY_BE_DOUBLE;
2913
2.62k
        }
2914
6.57k
        tmp |= t1 & (MAY_BE_FALSE | MAY_BE_TRUE | MAY_BE_OBJECT);
2915
6.57k
      }
2916
8.45k
      if (ssa_op->result_def >= 0) {
2917
2.27k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
2918
2.27k
      }
2919
8.45k
      if (ssa_op->op1_def >= 0) {
2920
8.42k
        if (t1 & MAY_BE_REF) {
2921
2.37k
          tmp |= MAY_BE_REF;
2922
2.37k
        }
2923
8.42k
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
2924
8.42k
      }
2925
8.45k
      break;
2926
8.45k
    case ZEND_POST_INC:
2927
5.59k
    case ZEND_POST_DEC:
2928
5.59k
      if (ssa_op->result_def >= 0) {
2929
5.59k
        tmp = 0;
2930
5.59k
        if (t1 & (MAY_BE_RC1|MAY_BE_RCN)) {
2931
1.51k
          tmp |= MAY_BE_RC1|MAY_BE_RCN;
2932
1.51k
        }
2933
5.59k
        tmp |= t1 & ~(MAY_BE_UNDEF|MAY_BE_REF|MAY_BE_RCN);
2934
5.59k
        if (t1 & MAY_BE_UNDEF) {
2935
1.55k
          tmp |= MAY_BE_NULL;
2936
1.55k
        }
2937
5.59k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
2938
5.59k
      }
2939
5.59k
      tmp = 0;
2940
5.59k
      if (t1 & MAY_BE_REF) {
2941
1.04k
        tmp |= MAY_BE_REF;
2942
1.04k
      }
2943
5.59k
      if (t1 & (MAY_BE_RC1|MAY_BE_RCN)) {
2944
1.51k
        tmp |= MAY_BE_RC1;
2945
1.51k
      }
2946
5.59k
      if ((t1 & (MAY_BE_ANY|MAY_BE_UNDEF)) == MAY_BE_LONG) {
2947
852
        if (!ssa_var_info[ssa_op->op1_use].has_range ||
2948
852
             (opline->opcode == ZEND_POST_DEC &&
2949
308
              (ssa_var_info[ssa_op->op1_use].range.underflow ||
2950
288
               ssa_var_info[ssa_op->op1_use].range.min == ZEND_LONG_MIN)) ||
2951
588
              (opline->opcode == ZEND_POST_INC &&
2952
544
               (ssa_var_info[ssa_op->op1_use].range.overflow ||
2953
526
                ssa_var_info[ssa_op->op1_use].range.max == ZEND_LONG_MAX))) {
2954
          /* may overflow */
2955
526
          tmp |= MAY_BE_LONG | MAY_BE_DOUBLE;
2956
526
        } else {
2957
326
          tmp |= MAY_BE_LONG;
2958
326
        }
2959
4.73k
      } else {
2960
4.73k
        if (t1 & (MAY_BE_UNDEF | MAY_BE_NULL)) {
2961
2.92k
          if (opline->opcode == ZEND_POST_INC) {
2962
1.25k
            tmp |= MAY_BE_LONG;
2963
1.67k
          } else {
2964
1.67k
            tmp |= MAY_BE_NULL;
2965
1.67k
          }
2966
2.92k
        }
2967
4.73k
        if (t1 & MAY_BE_LONG) {
2968
3.48k
          tmp |= MAY_BE_LONG | MAY_BE_DOUBLE;
2969
3.48k
        }
2970
4.73k
        if (t1 & MAY_BE_DOUBLE) {
2971
3.23k
          tmp |= MAY_BE_DOUBLE;
2972
3.23k
        }
2973
4.73k
        if (t1 & MAY_BE_STRING) {
2974
1.39k
          tmp |= MAY_BE_STRING | MAY_BE_LONG | MAY_BE_DOUBLE;
2975
1.39k
        }
2976
4.73k
        tmp |= t1 & (MAY_BE_FALSE | MAY_BE_TRUE | MAY_BE_RESOURCE | MAY_BE_ARRAY | MAY_BE_OBJECT | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF | MAY_BE_ARRAY_KEY_ANY);
2977
4.73k
      }
2978
5.59k
      if (ssa_op->op1_def >= 0) {
2979
5.52k
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
2980
5.52k
      }
2981
5.59k
      break;
2982
10.8k
    case ZEND_ASSIGN_DIM:
2983
10.8k
      if (opline->op1_type == IS_CV) {
2984
10.1k
        tmp = assign_dim_result_type(t1, t2, OP1_DATA_INFO(), opline->op2_type);
2985
10.1k
        tmp |= ssa->var_info[ssa_op->op1_def].type & (MAY_BE_ARRAY_PACKED|MAY_BE_ARRAY_NUMERIC_HASH|MAY_BE_ARRAY_STRING_HASH);
2986
10.1k
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
2987
10.1k
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
2988
10.1k
      }
2989
10.8k
      if (ssa_op->result_def >= 0) {
2990
6.00k
        tmp = 0;
2991
6.00k
        if (t1 & MAY_BE_STRING) {
2992
2.06k
          tmp |= MAY_BE_STRING | MAY_BE_NULL;
2993
2.06k
        }
2994
6.00k
        if (t1 & MAY_BE_OBJECT) {
2995
2.05k
          tmp |= (MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF);
2996
2.05k
        }
2997
6.00k
        if (t1 & (MAY_BE_ARRAY|MAY_BE_FALSE|MAY_BE_NULL|MAY_BE_UNDEF)) {
2998
5.94k
          tmp |= (OP1_DATA_INFO() & (MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF));
2999
3000
5.94k
          if (OP1_DATA_INFO() & MAY_BE_UNDEF) {
3001
396
            tmp |= MAY_BE_NULL;
3002
396
          }
3003
5.94k
          if (t1 & MAY_BE_ARRAY_OF_REF) {
3004
            /* A scalar type conversion may occur when assigning to a typed reference. */
3005
2.14k
            tmp |= MAY_BE_NULL|MAY_BE_FALSE|MAY_BE_TRUE|MAY_BE_LONG|MAY_BE_DOUBLE|MAY_BE_STRING;
3006
2.14k
          }
3007
5.94k
        }
3008
6.00k
        if (t1 & (MAY_BE_TRUE|MAY_BE_LONG|MAY_BE_DOUBLE|MAY_BE_RESOURCE)) {
3009
2.07k
          tmp |= MAY_BE_NULL;
3010
2.07k
        }
3011
6.00k
        tmp |= MAY_BE_RC1 | MAY_BE_RCN;
3012
6.00k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3013
6.00k
      }
3014
10.8k
      if ((ssa_op+1)->op1_def >= 0) {
3015
0
        opline++;
3016
0
        ssa_op++;
3017
0
        tmp = OP1_INFO();
3018
0
        if (tmp & (MAY_BE_ANY | MAY_BE_REF)) {
3019
0
          if (tmp & MAY_BE_RC1) {
3020
0
            tmp |= MAY_BE_RCN;
3021
0
          }
3022
0
        }
3023
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3024
0
      }
3025
10.8k
      break;
3026
10.8k
    case ZEND_ASSIGN_OBJ:
3027
5.51k
      if (opline->op1_type == IS_CV) {
3028
5.30k
        const zend_class_entry *ce = ssa_var_info[ssa_op->op1_use].ce;
3029
5.30k
        bool add_rc = (t1 & (MAY_BE_OBJECT|MAY_BE_REF)) && (!ce
3030
931
          || ce->__set
3031
          /* Non-default write_property may be set within create_object. */
3032
931
          || ce->create_object
3033
927
          || ce->default_object_handlers->write_property != zend_std_write_property
3034
927
          || ssa_var_info[ssa_op->op1_use].is_instanceof);
3035
5.30k
        tmp = (t1 & (MAY_BE_REF|MAY_BE_OBJECT|MAY_BE_RC1|MAY_BE_RCN))|(add_rc ? (MAY_BE_RC1|MAY_BE_RCN) : 0);
3036
5.30k
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3037
5.30k
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
3038
5.30k
      }
3039
5.51k
      if (ssa_op->result_def >= 0) {
3040
        // TODO: If there is no __set we might do better
3041
1.46k
        tmp = zend_fetch_prop_type(script,
3042
1.46k
          zend_fetch_prop_info(op_array, ssa, opline, ssa_op), &ce);
3043
1.46k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3044
1.46k
        if (ce) {
3045
2
          UPDATE_SSA_OBJ_TYPE(ce, 1, ssa_op->result_def);
3046
2
        }
3047
1.46k
      }
3048
5.51k
      if ((ssa_op+1)->op1_def >= 0) {
3049
0
        opline++;
3050
0
        ssa_op++;
3051
0
        tmp = OP1_INFO();
3052
0
        if (tmp & MAY_BE_RC1) {
3053
0
          tmp |= MAY_BE_RCN;
3054
0
        }
3055
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3056
0
      }
3057
5.51k
      break;
3058
5.51k
    case ZEND_ASSIGN_STATIC_PROP:
3059
298
      if (ssa_op->result_def >= 0) {
3060
298
        tmp = MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF | MAY_BE_RC1 | MAY_BE_RCN;
3061
298
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3062
298
      }
3063
298
      if ((ssa_op+1)->op1_def >= 0) {
3064
0
        opline++;
3065
0
        ssa_op++;
3066
0
        tmp = OP1_INFO();
3067
0
        if (tmp & MAY_BE_RC1) {
3068
0
          tmp |= MAY_BE_RCN;
3069
0
        }
3070
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3071
0
      }
3072
298
      break;
3073
394
    case ZEND_PRE_INC_OBJ:
3074
594
    case ZEND_PRE_DEC_OBJ:
3075
784
    case ZEND_POST_INC_OBJ:
3076
852
    case ZEND_POST_DEC_OBJ:
3077
852
      if (opline->op1_type == IS_CV) {
3078
758
        tmp = (t1 & (MAY_BE_REF|MAY_BE_OBJECT))|MAY_BE_RC1|MAY_BE_RCN;
3079
758
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3080
758
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
3081
758
      }
3082
852
      if (ssa_op->result_def >= 0) {
3083
        // TODO: ???
3084
546
        tmp = MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3085
546
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3086
546
      }
3087
852
      break;
3088
79.7k
    case ZEND_ASSIGN:
3089
79.7k
      if (ssa_op->op2_def >= 0) {
3090
0
        tmp = t2;
3091
0
        if (tmp & MAY_BE_RC1) {
3092
0
          tmp |= MAY_BE_RCN;
3093
0
        }
3094
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op2_def);
3095
0
      }
3096
79.7k
      tmp = t2 & ~(MAY_BE_UNDEF|MAY_BE_REF|MAY_BE_RC1|MAY_BE_RCN);
3097
79.7k
      if (t2 & MAY_BE_UNDEF) {
3098
2.60k
        tmp |= MAY_BE_NULL;
3099
2.60k
      }
3100
79.7k
      if (t1 & MAY_BE_REF) {
3101
42.5k
        tmp |= MAY_BE_REF;
3102
42.5k
      }
3103
79.7k
      if (t2 & MAY_BE_REF) {
3104
2.15k
        tmp |= MAY_BE_RC1 | MAY_BE_RCN;
3105
77.5k
      } else if (opline->op2_type & (IS_TMP_VAR|IS_VAR)) {
3106
50.8k
        tmp |= t2 & (MAY_BE_RC1|MAY_BE_RCN);
3107
50.8k
      } else if (t2 & (MAY_BE_RC1|MAY_BE_RCN)) {
3108
8.82k
        tmp |= MAY_BE_RCN;
3109
8.82k
      }
3110
79.7k
      if (RETURN_VALUE_USED(opline) && (tmp & MAY_BE_RC1)) {
3111
12.3k
        tmp |= MAY_BE_RCN;
3112
12.3k
      }
3113
79.7k
      if (ssa_op->op1_def >= 0) {
3114
79.7k
        if (ssa_var_info[ssa_op->op1_def].use_as_double) {
3115
4
          tmp &= ~MAY_BE_LONG;
3116
4
          tmp |= MAY_BE_DOUBLE;
3117
4
        }
3118
79.7k
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3119
79.7k
        COPY_SSA_OBJ_TYPE(ssa_op->op2_use, ssa_op->op1_def);
3120
79.7k
      }
3121
79.7k
      if (ssa_op->result_def >= 0) {
3122
26.4k
        if (tmp & MAY_BE_REF) {
3123
          /* A scalar type conversion may occur when assigning to a typed reference. */
3124
9.10k
          tmp &= ~MAY_BE_REF;
3125
9.10k
          tmp |= MAY_BE_NULL|MAY_BE_FALSE|MAY_BE_TRUE|MAY_BE_LONG|MAY_BE_DOUBLE|MAY_BE_STRING|MAY_BE_RC1|MAY_BE_RCN;
3126
9.10k
        }
3127
26.4k
        if ((tmp & (MAY_BE_RC1|MAY_BE_RCN)) == MAY_BE_RCN) {
3128
          /* refcount may be indirectly decremented. Make an exception if the result is used in the next instruction */
3129
884
          if (!ssa_opcodes) {
3130
884
            if (ssa->vars[ssa_op->result_def].use_chain < 0
3131
884
             || opline + 1 != op_array->opcodes + ssa->vars[ssa_op->result_def].use_chain) {
3132
64
              tmp |= MAY_BE_RC1;
3133
64
              }
3134
884
          } else {
3135
0
            if (ssa->vars[ssa_op->result_def].use_chain < 0
3136
0
             || opline + 1 != ssa_opcodes[ssa->vars[ssa_op->result_def].use_chain]) {
3137
0
              tmp |= MAY_BE_RC1;
3138
0
              }
3139
0
          }
3140
884
        }
3141
26.4k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3142
26.4k
        COPY_SSA_OBJ_TYPE(ssa_op->op2_use, ssa_op->result_def);
3143
26.4k
      }
3144
79.7k
      break;
3145
79.7k
    case ZEND_ASSIGN_REF:
3146
// TODO: ???
3147
3.80k
      if (opline->op2_type == IS_CV) {
3148
2.57k
        tmp = (MAY_BE_REF | t2) & ~(MAY_BE_UNDEF|MAY_BE_RC1|MAY_BE_RCN);
3149
2.57k
        if (t2 & MAY_BE_UNDEF) {
3150
619
          tmp |= MAY_BE_NULL;
3151
619
        }
3152
2.57k
        UPDATE_SSA_TYPE(tmp, ssa_op->op2_def);
3153
2.57k
      }
3154
3.80k
      if (opline->op2_type == IS_VAR && opline->extended_value == ZEND_RETURNS_FUNCTION) {
3155
176
        tmp = (MAY_BE_REF | MAY_BE_RCN | MAY_BE_RC1 | t2) & ~MAY_BE_UNDEF;
3156
3.63k
      } else {
3157
3.63k
        tmp = (MAY_BE_REF | t2) & ~(MAY_BE_UNDEF|MAY_BE_RC1|MAY_BE_RCN);
3158
3.63k
      }
3159
3.80k
      if (t2 & MAY_BE_UNDEF) {
3160
619
        tmp |= MAY_BE_NULL;
3161
619
      }
3162
3.80k
      UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3163
3.80k
      if (ssa_op->result_def >= 0) {
3164
1.47k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3165
1.47k
      }
3166
3.80k
      break;
3167
3.80k
    case ZEND_ASSIGN_OBJ_REF:
3168
351
      if (opline->op1_type == IS_CV && ssa_op->op1_def >= 0) {
3169
0
        tmp = t1;
3170
0
        if (tmp & MAY_BE_OBJECT) {
3171
0
          tmp |= MAY_BE_RC1 | MAY_BE_RCN;
3172
0
        }
3173
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3174
0
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
3175
0
      }
3176
3177
351
      t2 = OP1_DATA_INFO();
3178
351
      if ((opline+1)->op1_type == IS_VAR && (opline->extended_value & ZEND_RETURNS_FUNCTION)) {
3179
2
        tmp = (MAY_BE_REF | MAY_BE_RCN | MAY_BE_RC1 | t2) & ~MAY_BE_UNDEF;
3180
349
      } else {
3181
349
        tmp = (MAY_BE_REF | t2) & ~(MAY_BE_UNDEF|MAY_BE_RC1|MAY_BE_RCN);
3182
349
      }
3183
351
      if (t2 & MAY_BE_UNDEF) {
3184
74
        tmp |= MAY_BE_NULL;
3185
74
      }
3186
351
      if (ssa_op->result_def >= 0) {
3187
124
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3188
124
      }
3189
351
      if ((opline+1)->op1_type == IS_CV) {
3190
343
        opline++;
3191
343
        ssa_op++;
3192
343
        tmp = (MAY_BE_REF | t2) & ~(MAY_BE_UNDEF|MAY_BE_RC1|MAY_BE_RCN);
3193
343
        if (t2 & MAY_BE_UNDEF) {
3194
74
          tmp |= MAY_BE_NULL;
3195
74
        }
3196
343
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3197
343
      }
3198
351
      break;
3199
351
    case ZEND_ASSIGN_STATIC_PROP_REF:
3200
90
      if (ssa_op->result_def >= 0) {
3201
72
        UPDATE_SSA_TYPE(MAY_BE_REF, ssa_op->result_def);
3202
72
      }
3203
90
      if ((opline+1)->op1_type == IS_CV) {
3204
82
        opline++;
3205
82
        ssa_op++;
3206
82
        UPDATE_SSA_TYPE(MAY_BE_REF, ssa_op->op1_def);
3207
82
      }
3208
90
      break;
3209
801
    case ZEND_BIND_GLOBAL:
3210
801
      tmp = MAY_BE_REF | MAY_BE_ANY
3211
801
        | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3212
801
      UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3213
801
      break;
3214
7.63k
    case ZEND_BIND_STATIC:
3215
7.63k
      tmp = MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF
3216
7.63k
        | ((opline->extended_value & ZEND_BIND_REF) ? MAY_BE_REF : (MAY_BE_RC1 | MAY_BE_RCN));
3217
7.63k
      if (opline->extended_value & ZEND_BIND_IMPLICIT) {
3218
5.90k
        tmp |= MAY_BE_UNDEF;
3219
5.90k
      }
3220
7.63k
      UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3221
7.63k
      break;
3222
7.63k
    case ZEND_BIND_INIT_STATIC_OR_JMP:
3223
161
      tmp = MAY_BE_UNDEF | MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF | MAY_BE_REF;
3224
161
      UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3225
161
      break;
3226
161
    case ZEND_SEND_VAR:
3227
0
      if (ssa_op->op1_def >= 0) {
3228
0
        tmp = t1;
3229
0
        if (t1 & (MAY_BE_RC1|MAY_BE_REF)) {
3230
0
          tmp |= MAY_BE_RCN;
3231
0
        }
3232
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3233
0
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
3234
0
      }
3235
0
      break;
3236
482
    case ZEND_BIND_LEXICAL:
3237
482
      if (ssa_op->op2_def >= 0) {
3238
482
        if (opline->extended_value & ZEND_BIND_REF) {
3239
482
          tmp = t2 | MAY_BE_REF;
3240
482
        } else {
3241
0
          tmp = t2 & ~(MAY_BE_RC1|MAY_BE_RCN);
3242
0
          if (t2 & (MAY_BE_RC1|MAY_BE_RCN)) {
3243
0
            tmp |= MAY_BE_RCN;
3244
0
          }
3245
0
        }
3246
482
        UPDATE_SSA_TYPE(tmp, ssa_op->op2_def);
3247
482
        COPY_SSA_OBJ_TYPE(ssa_op->op2_use, ssa_op->op2_def);
3248
482
      }
3249
482
      break;
3250
486
    case ZEND_YIELD:
3251
486
      if (ssa_op->op1_def >= 0) {
3252
66
        if (op_array->fn_flags & ZEND_ACC_RETURN_REFERENCE) {
3253
66
          tmp = t1 | MAY_BE_REF;
3254
66
        } else {
3255
0
          tmp = t1 & ~(MAY_BE_RC1|MAY_BE_RCN);
3256
0
          if (t1 & (MAY_BE_RC1|MAY_BE_RCN)) {
3257
0
            tmp |= MAY_BE_RCN;
3258
0
          }
3259
0
        }
3260
66
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3261
66
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
3262
66
      }
3263
486
      if (ssa_op->result_def >= 0) {
3264
432
        tmp = MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF
3265
432
          | MAY_BE_RC1 | MAY_BE_RCN;
3266
432
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3267
432
      }
3268
486
      break;
3269
5.46k
    case ZEND_SEND_VAR_EX:
3270
5.46k
    case ZEND_SEND_FUNC_ARG:
3271
5.46k
      if (ssa_op->op1_def >= 0) {
3272
5.46k
        tmp = (t1 & MAY_BE_UNDEF)|MAY_BE_REF|MAY_BE_RC1|MAY_BE_RCN|MAY_BE_ANY|MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF;
3273
5.46k
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3274
5.46k
      }
3275
5.46k
      break;
3276
5.46k
    case ZEND_SEND_REF:
3277
1.91k
      if (ssa_op->op1_def >= 0) {
3278
1.91k
        tmp = MAY_BE_REF|MAY_BE_RC1|MAY_BE_RCN|MAY_BE_ANY|MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF;
3279
1.91k
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3280
1.91k
      }
3281
1.91k
      break;
3282
1.91k
    case ZEND_SEND_UNPACK:
3283
300
      if (ssa_op->op1_def >= 0) {
3284
300
        tmp = t1;
3285
300
        if (t1 & MAY_BE_ARRAY) {
3286
296
          tmp |= MAY_BE_RC1 | MAY_BE_RCN;
3287
296
          if (t1 & MAY_BE_ARRAY_OF_ANY) {
3288
            /* SEND_UNPACK may acquire references into the array */
3289
296
            tmp |= MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3290
296
          }
3291
296
        }
3292
300
        if (t1 & MAY_BE_OBJECT) {
3293
218
          tmp |= MAY_BE_RC1 | MAY_BE_RCN;
3294
218
        }
3295
300
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3296
300
      }
3297
300
      break;
3298
3.91k
    case ZEND_FAST_CONCAT:
3299
17.8k
    case ZEND_ROPE_INIT:
3300
260k
    case ZEND_ROPE_ADD:
3301
274k
    case ZEND_ROPE_END:
3302
274k
      UPDATE_SSA_TYPE(MAY_BE_STRING|MAY_BE_RC1|MAY_BE_RCN, ssa_op->result_def);
3303
274k
      break;
3304
274k
    case ZEND_RECV:
3305
24.6k
    case ZEND_RECV_INIT:
3306
24.9k
    case ZEND_RECV_VARIADIC:
3307
24.9k
    {
3308
      /* Typehinting */
3309
24.9k
      zend_arg_info *arg_info = &op_array->arg_info[opline->op1.num-1];
3310
3311
24.9k
      ce = NULL;
3312
24.9k
      tmp = zend_fetch_arg_info_type(script, arg_info, &ce);
3313
24.9k
      if (ZEND_ARG_SEND_MODE(arg_info)) {
3314
1.82k
        tmp |= MAY_BE_REF;
3315
1.82k
        ce = NULL;
3316
1.82k
      }
3317
3318
24.9k
      if (opline->opcode == ZEND_RECV_VARIADIC) {
3319
370
        uint32_t elem_type = tmp & MAY_BE_REF
3320
370
          ? MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF
3321
370
          : (tmp & MAY_BE_ANY) << MAY_BE_ARRAY_SHIFT;
3322
370
        tmp = MAY_BE_RC1|MAY_BE_RCN|MAY_BE_ARRAY|MAY_BE_ARRAY_KEY_ANY|elem_type;
3323
370
        ce = NULL;
3324
370
      }
3325
3326
24.9k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3327
24.9k
      if (ce) {
3328
846
        UPDATE_SSA_OBJ_TYPE(ce, 1, ssa_op->result_def);
3329
24.1k
      } else {
3330
24.1k
        UPDATE_SSA_OBJ_TYPE(NULL, 0, ssa_op->result_def);
3331
24.1k
      }
3332
24.9k
      break;
3333
24.9k
    }
3334
537
    case ZEND_DECLARE_ANON_CLASS:
3335
537
      UPDATE_SSA_TYPE(MAY_BE_CLASS, ssa_op->result_def);
3336
537
      if (script && (ce = zend_hash_find_ptr(&script->class_table, Z_STR_P(CRT_CONSTANT(opline->op1)))) != NULL) {
3337
537
        UPDATE_SSA_OBJ_TYPE(ce, 0, ssa_op->result_def);
3338
537
      }
3339
537
      break;
3340
876
    case ZEND_FETCH_CLASS:
3341
876
      UPDATE_SSA_TYPE(MAY_BE_CLASS, ssa_op->result_def);
3342
876
      if (opline->op2_type == IS_UNUSED) {
3343
0
        switch (opline->op1.num & ZEND_FETCH_CLASS_MASK) {
3344
0
          case ZEND_FETCH_CLASS_SELF:
3345
0
            if (op_array->scope) {
3346
0
              UPDATE_SSA_OBJ_TYPE(op_array->scope, 0, ssa_op->result_def);
3347
0
            } else {
3348
0
              UPDATE_SSA_OBJ_TYPE(NULL, 0, ssa_op->result_def);
3349
0
            }
3350
0
            break;
3351
0
          case ZEND_FETCH_CLASS_PARENT:
3352
0
            if (op_array->scope && op_array->scope->parent && (op_array->scope->ce_flags & ZEND_ACC_LINKED)) {
3353
0
              UPDATE_SSA_OBJ_TYPE(op_array->scope->parent, 0, ssa_op->result_def);
3354
0
            } else {
3355
0
              UPDATE_SSA_OBJ_TYPE(NULL, 0, ssa_op->result_def);
3356
0
            }
3357
0
            break;
3358
0
          case ZEND_FETCH_CLASS_STATIC:
3359
0
            if (op_array->scope && (op_array->scope->ce_flags & ZEND_ACC_FINAL)) {
3360
0
              UPDATE_SSA_OBJ_TYPE(op_array->scope, 0, ssa_op->result_def);
3361
0
            } else {
3362
0
              UPDATE_SSA_OBJ_TYPE(NULL, 0, ssa_op->result_def);
3363
0
            }
3364
0
            break;
3365
0
          default:
3366
0
            UPDATE_SSA_OBJ_TYPE(NULL, 0, ssa_op->result_def);
3367
0
            break;
3368
0
        }
3369
876
      } else if (opline->op2_type == IS_CONST) {
3370
34
        zval *zv = CRT_CONSTANT(opline->op2);
3371
34
        if (Z_TYPE_P(zv) == IS_STRING) {
3372
34
          ce = zend_optimizer_get_class_entry(script, op_array, Z_STR_P(zv+1));
3373
34
          UPDATE_SSA_OBJ_TYPE(ce, 0, ssa_op->result_def);
3374
34
        } else {
3375
0
          UPDATE_SSA_OBJ_TYPE(NULL, 0, ssa_op->result_def);
3376
0
        }
3377
842
      } else {
3378
842
        COPY_SSA_OBJ_TYPE(ssa_op->op2_use, ssa_op->result_def);
3379
842
      }
3380
876
      break;
3381
21.4k
    case ZEND_NEW:
3382
21.4k
      tmp = MAY_BE_RC1|MAY_BE_RCN|MAY_BE_OBJECT;
3383
21.4k
      ce = zend_optimizer_get_class_entry_from_op1(script, op_array, opline);
3384
21.4k
      if (ce) {
3385
18.1k
        UPDATE_SSA_OBJ_TYPE(ce, 0, ssa_op->result_def);
3386
18.1k
      } else if ((t1 & MAY_BE_CLASS) && ssa_op->op1_use >= 0 && ssa_var_info[ssa_op->op1_use].ce) {
3387
543
        UPDATE_SSA_OBJ_TYPE(ssa_var_info[ssa_op->op1_use].ce, ssa_var_info[ssa_op->op1_use].is_instanceof, ssa_op->result_def);
3388
543
        if (!ssa_var_info[ssa_op->result_def].is_instanceof) {
3389
541
          ce = ssa_var_info[ssa_op->op1_use].ce;
3390
541
        }
3391
2.75k
      } else {
3392
2.75k
        UPDATE_SSA_OBJ_TYPE(NULL, 0, ssa_op->result_def);
3393
2.75k
      }
3394
      /* New objects without constructors cannot escape. */
3395
21.4k
      if (ce
3396
18.6k
       && !ce->constructor
3397
10.3k
       && !ce->create_object
3398
9.88k
       && ce->default_object_handlers->get_constructor == zend_std_get_constructor) {
3399
9.88k
        tmp &= ~MAY_BE_RCN;
3400
9.88k
      }
3401
21.4k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3402
21.4k
      break;
3403
21.4k
    case ZEND_CLONE:
3404
423
      UPDATE_SSA_TYPE(MAY_BE_RC1|MAY_BE_RCN|MAY_BE_OBJECT, ssa_op->result_def);
3405
423
      COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->result_def);
3406
423
      break;
3407
3.80k
    case ZEND_INIT_ARRAY:
3408
19.1k
    case ZEND_ADD_ARRAY_ELEMENT:
3409
19.1k
      if (ssa_op->op1_def >= 0) {
3410
494
        if (opline->extended_value & ZEND_ARRAY_ELEMENT_REF) {
3411
494
          tmp = (MAY_BE_REF | t1) & ~(MAY_BE_UNDEF|MAY_BE_RC1|MAY_BE_RCN);
3412
494
          if (t1 & MAY_BE_UNDEF) {
3413
178
            tmp |= MAY_BE_NULL;
3414
178
          }
3415
494
        } else if ((t1 & (MAY_BE_REF|MAY_BE_RC1|MAY_BE_RCN)) == MAY_BE_REF) {
3416
0
          tmp = (MAY_BE_REF | t1) & ~(MAY_BE_UNDEF|MAY_BE_RC1|MAY_BE_RCN);
3417
0
          if (t1 & MAY_BE_UNDEF) {
3418
0
            tmp |= MAY_BE_NULL;
3419
0
          }
3420
0
        } else if (t1 & MAY_BE_REF) {
3421
0
          tmp = (MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_REF | t1);
3422
0
        } else {
3423
0
          tmp = t1;
3424
0
          if (t1 & MAY_BE_RC1) {
3425
0
            tmp |= MAY_BE_RCN;
3426
0
          }
3427
0
        }
3428
494
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3429
494
      }
3430
19.1k
      if (ssa_op->result_def >= 0) {
3431
19.1k
        uint32_t arr_type;
3432
19.1k
        if (opline->opcode == ZEND_INIT_ARRAY) {
3433
3.80k
          arr_type = 0;
3434
15.3k
        } else {
3435
15.3k
          arr_type = RES_USE_INFO();
3436
15.3k
        }
3437
19.1k
        tmp = MAY_BE_RC1|MAY_BE_ARRAY|arr_type;
3438
19.1k
        if (opline->opcode == ZEND_INIT_ARRAY && opline->op1_type == IS_UNUSED) {
3439
114
          tmp |= MAY_BE_ARRAY_EMPTY;
3440
114
        }
3441
19.1k
        if (opline->op1_type != IS_UNUSED
3442
19.0k
         && (opline->op2_type == IS_UNUSED
3443
19.0k
          || (t2 & (MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE|MAY_BE_TRUE|MAY_BE_LONG|MAY_BE_DOUBLE|MAY_BE_RESOURCE|MAY_BE_STRING)))) {
3444
19.0k
          tmp |= assign_dim_array_result_type(arr_type, t2, t1, opline->op2_type);
3445
19.0k
          if (opline->extended_value & ZEND_ARRAY_ELEMENT_REF) {
3446
513
            tmp |= MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF;
3447
513
          }
3448
19.0k
        }
3449
19.1k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3450
19.1k
      }
3451
19.1k
      break;
3452
19.1k
    case ZEND_ADD_ARRAY_UNPACK:
3453
150
      tmp = ssa_var_info[ssa_op->result_use].type;
3454
150
      ZEND_ASSERT(tmp & MAY_BE_ARRAY);
3455
150
      tmp |= t1 & (MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF);
3456
150
      if (t1 & MAY_BE_OBJECT) {
3457
114
        tmp |= MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY;
3458
114
      }
3459
150
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3460
150
      break;
3461
1.91k
    case ZEND_UNSET_CV:
3462
1.91k
      tmp = MAY_BE_UNDEF;
3463
1.91k
      if (!op_array->function_name) {
3464
        /* In global scope, we know nothing */
3465
1.25k
        tmp |= MAY_BE_REF;
3466
1.25k
      }
3467
1.91k
      UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3468
1.91k
      break;
3469
1.91k
    case ZEND_UNSET_DIM:
3470
1.17k
    case ZEND_UNSET_OBJ:
3471
1.17k
      if (ssa_op->op1_def >= 0) {
3472
1.17k
        UPDATE_SSA_TYPE(t1, ssa_op->op1_def);
3473
1.17k
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
3474
1.17k
      }
3475
1.17k
      break;
3476
3.03k
    case ZEND_FE_RESET_R:
3477
4.37k
    case ZEND_FE_RESET_RW:
3478
4.37k
      if (ssa_op->op1_def >= 0) {
3479
1.24k
        tmp = t1;
3480
1.24k
        if (opline->opcode == ZEND_FE_RESET_RW) {
3481
1.24k
          tmp |= MAY_BE_REF;
3482
1.24k
        } else if (t1 & MAY_BE_RC1) {
3483
0
          tmp |= MAY_BE_RCN;
3484
0
        }
3485
1.24k
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3486
1.24k
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
3487
1.24k
      }
3488
4.37k
      if (opline->opcode == ZEND_FE_RESET_RW) {
3489
//???
3490
1.33k
        tmp = MAY_BE_REF | (t1 & (MAY_BE_ARRAY | MAY_BE_OBJECT));
3491
3.03k
      } else {
3492
3.03k
        tmp = MAY_BE_RC1 | MAY_BE_RCN | (t1 & (MAY_BE_ARRAY | MAY_BE_OBJECT | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF));
3493
3.03k
      }
3494
      /* The result is set to UNDEF for invalid foreach inputs. */
3495
4.37k
      if ((t1 & (MAY_BE_ANY | MAY_BE_UNDEF)) & ~(MAY_BE_ARRAY | MAY_BE_OBJECT)) {
3496
3.43k
        tmp |= MAY_BE_UNDEF;
3497
3.43k
      }
3498
4.37k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3499
4.37k
      COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->result_def);
3500
4.37k
      break;
3501
4.45k
    case ZEND_FE_FETCH_R:
3502
5.42k
    case ZEND_FE_FETCH_RW:
3503
5.42k
      tmp = 0;
3504
5.42k
      if (opline->op2_type == IS_CV) {
3505
5.04k
        tmp = t2 & MAY_BE_REF;
3506
5.04k
      }
3507
5.42k
      if (t1 & MAY_BE_OBJECT) {
3508
3.97k
        if (opline->opcode == ZEND_FE_FETCH_RW) {
3509
976
          tmp |= MAY_BE_REF | MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3510
2.99k
        } else {
3511
2.99k
          tmp |= MAY_BE_RCN | MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3512
2.99k
          if (opline->op2_type != IS_CV) {
3513
194
            tmp |= MAY_BE_REF;
3514
194
          }
3515
2.99k
        }
3516
3.97k
      }
3517
5.42k
      if (t1 & MAY_BE_ARRAY) {
3518
4.86k
        if (opline->opcode == ZEND_FE_FETCH_RW) {
3519
976
          tmp |= MAY_BE_REF | MAY_BE_RCN | MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3520
3.88k
        } else {
3521
3.88k
          tmp |= ((t1 & MAY_BE_ARRAY_OF_ANY) >> MAY_BE_ARRAY_SHIFT);
3522
3.88k
          if (tmp & MAY_BE_ARRAY) {
3523
3.23k
            tmp |= MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3524
3.23k
          }
3525
3.88k
          if (t1 & MAY_BE_ARRAY_OF_REF) {
3526
2.97k
            tmp |= MAY_BE_RC1 | MAY_BE_RCN;
3527
2.97k
            if (opline->op2_type != IS_CV) {
3528
210
              tmp |= MAY_BE_REF;
3529
210
            }
3530
2.97k
          } else if (tmp & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) {
3531
476
            tmp |= MAY_BE_RC1 | MAY_BE_RCN;
3532
476
          }
3533
3.88k
        }
3534
4.86k
      }
3535
5.42k
      UPDATE_SSA_TYPE(tmp, ssa_op->op2_def);
3536
5.42k
      if (ssa_op->result_def >= 0) {
3537
2.50k
        tmp = (ssa_op->result_use >= 0) ? RES_USE_INFO() : 0;
3538
2.50k
        if (t1 & MAY_BE_OBJECT) {
3539
1.97k
          tmp |= MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3540
1.97k
        }
3541
2.50k
        if (t1 & MAY_BE_ARRAY) {
3542
2.27k
          if (t1 & MAY_BE_ARRAY_KEY_LONG) {
3543
2.18k
            tmp |= MAY_BE_LONG;
3544
2.18k
          }
3545
2.27k
          if (t1 & MAY_BE_ARRAY_KEY_STRING) {
3546
2.09k
            tmp |= MAY_BE_STRING | MAY_BE_RCN;
3547
2.09k
          }
3548
2.27k
        }
3549
2.50k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3550
2.50k
      }
3551
5.42k
      break;
3552
8.72k
    case ZEND_FETCH_DIM_R:
3553
10.0k
    case ZEND_FETCH_DIM_IS:
3554
10.9k
    case ZEND_FETCH_DIM_RW:
3555
18.1k
    case ZEND_FETCH_DIM_W:
3556
18.4k
    case ZEND_FETCH_DIM_UNSET:
3557
19.3k
    case ZEND_FETCH_DIM_FUNC_ARG:
3558
20.6k
    case ZEND_FETCH_LIST_R:
3559
21.0k
    case ZEND_FETCH_LIST_W:
3560
21.0k
      if (ssa_op->op1_def >= 0) {
3561
6.43k
        uint32_t key_type = 0;
3562
6.43k
        tmp = t1 & ~(MAY_BE_RC1|MAY_BE_RCN);
3563
6.43k
        if (opline->opcode == ZEND_FETCH_DIM_W ||
3564
1.62k
            opline->opcode == ZEND_FETCH_DIM_RW ||
3565
976
            opline->opcode == ZEND_FETCH_DIM_FUNC_ARG ||
3566
6.20k
            opline->opcode == ZEND_FETCH_LIST_W) {
3567
6.20k
          if (t1 & (MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE)) {
3568
5.42k
            if (opline->opcode != ZEND_FETCH_DIM_FUNC_ARG) {
3569
4.75k
              tmp &= ~(MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE);
3570
4.75k
            }
3571
5.42k
            tmp |= MAY_BE_ARRAY | MAY_BE_RC1;
3572
5.42k
          }
3573
6.20k
          if (t1 & (MAY_BE_STRING|MAY_BE_ARRAY)) {
3574
5.22k
            tmp |= MAY_BE_RC1;
3575
5.22k
            if (opline->opcode == ZEND_FETCH_DIM_FUNC_ARG) {
3576
686
              tmp |= t1 & MAY_BE_RCN;
3577
686
            }
3578
5.22k
          }
3579
6.20k
          if (t1 & (MAY_BE_OBJECT|MAY_BE_RESOURCE)) {
3580
4.13k
            tmp |= t1 & (MAY_BE_RC1|MAY_BE_RCN);
3581
4.13k
          }
3582
6.20k
          if (opline->op2_type == IS_UNUSED) {
3583
582
            if (t1 & (MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE)) {
3584
476
              key_type |= MAY_BE_ARRAY_PACKED;
3585
476
            }
3586
582
            if (t1 & MAY_BE_ARRAY) {
3587
471
              key_type |= MAY_BE_HASH_ONLY(t1) ?
3588
463
                MAY_BE_ARRAY_NUMERIC_HASH : MAY_BE_ARRAY_KEY_LONG;
3589
471
            }
3590
5.62k
          } else {
3591
5.62k
            if (t2 & (MAY_BE_LONG|MAY_BE_FALSE|MAY_BE_TRUE|MAY_BE_RESOURCE|MAY_BE_DOUBLE)) {
3592
3.60k
              if (t1 & (MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE)) {
3593
3.16k
                key_type |= MAY_BE_ARRAY_PACKED;
3594
3.16k
              }
3595
3.60k
              if (t1 & MAY_BE_ARRAY) {
3596
2.93k
                key_type |= MAY_BE_HASH_ONLY(t1) ?
3597
2.89k
                  MAY_BE_ARRAY_NUMERIC_HASH : MAY_BE_ARRAY_KEY_LONG;
3598
2.93k
                }
3599
3.60k
            }
3600
5.62k
            if (t2 & MAY_BE_STRING) {
3601
2.37k
              key_type |= MAY_BE_ARRAY_KEY_STRING;
3602
2.37k
              if (opline->op2_type != IS_CONST) {
3603
                // FIXME: numeric string
3604
1.09k
                if (t1 & (MAY_BE_UNDEF|MAY_BE_NULL|MAY_BE_FALSE)) {
3605
1.07k
                  key_type |= MAY_BE_ARRAY_PACKED;
3606
1.07k
                }
3607
1.09k
                if (t1 & MAY_BE_ARRAY) {
3608
1.04k
                  key_type |= MAY_BE_HASH_ONLY(t1) ?
3609
1.02k
                    MAY_BE_ARRAY_NUMERIC_HASH : MAY_BE_ARRAY_KEY_LONG;
3610
1.04k
                  }
3611
1.09k
              }
3612
2.37k
            }
3613
5.62k
            if (t2 & (MAY_BE_UNDEF | MAY_BE_NULL)) {
3614
970
              key_type |= MAY_BE_ARRAY_KEY_STRING;
3615
970
            }
3616
5.62k
          }
3617
6.20k
        } else if (opline->opcode == ZEND_FETCH_DIM_UNSET) {
3618
232
          if (t1 & MAY_BE_ARRAY) {
3619
216
            tmp |= MAY_BE_RC1;
3620
216
          }
3621
232
          if (t1 & (MAY_BE_OBJECT|MAY_BE_RESOURCE)) {
3622
184
            tmp |= t1 & (MAY_BE_RC1|MAY_BE_RCN);
3623
184
          }
3624
232
        }
3625
6.43k
        if ((key_type & (MAY_BE_ARRAY_KEY_LONG|MAY_BE_ARRAY_KEY_STRING))
3626
5.97k
            && (opline->opcode == ZEND_FETCH_DIM_RW
3627
5.33k
            || opline->opcode == ZEND_FETCH_DIM_W
3628
732
            || opline->opcode == ZEND_FETCH_DIM_FUNC_ARG
3629
5.97k
            || opline->opcode == ZEND_FETCH_LIST_W)) {
3630
5.97k
          j = ssa_vars[ssa_op->result_def].use_chain;
3631
5.97k
          if (j < 0) {
3632
            /* no uses */
3633
0
            tmp |= key_type | MAY_BE_ARRAY | MAY_BE_ARRAY_OF_NULL;
3634
0
          }
3635
11.9k
          while (j >= 0) {
3636
5.97k
            uint8_t opcode;
3637
3638
5.97k
            if (!ssa_opcodes) {
3639
5.97k
              if (j != (opline - op_array->opcodes) + 1) {
3640
                /* Use must be in next opline */
3641
0
                tmp |= key_type | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3642
0
                break;
3643
0
              }
3644
5.97k
              opcode = op_array->opcodes[j].opcode;
3645
5.97k
            } else {
3646
0
              if (ssa_opcodes[j] != opline + 1) {
3647
                /* Use must be in next opline */
3648
0
                tmp |= key_type | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3649
0
                break;
3650
0
              }
3651
0
              opcode = ssa_opcodes[j]->opcode;
3652
0
            }
3653
5.97k
            switch (opcode) {
3654
292
              case ZEND_FETCH_DIM_W:
3655
490
              case ZEND_FETCH_DIM_RW:
3656
530
              case ZEND_FETCH_DIM_FUNC_ARG:
3657
530
              case ZEND_FETCH_LIST_W:
3658
2.69k
              case ZEND_ASSIGN_DIM:
3659
3.06k
              case ZEND_ASSIGN_DIM_OP:
3660
3.06k
                tmp |= key_type | MAY_BE_ARRAY | MAY_BE_ARRAY_OF_ARRAY;
3661
3.06k
                break;
3662
0
              case ZEND_SEND_VAR_EX:
3663
658
              case ZEND_SEND_FUNC_ARG:
3664
658
              case ZEND_SEND_VAR_NO_REF:
3665
658
              case ZEND_SEND_VAR_NO_REF_EX:
3666
922
              case ZEND_SEND_REF:
3667
2.34k
              case ZEND_ASSIGN_REF:
3668
2.35k
              case ZEND_YIELD:
3669
2.36k
              case ZEND_INIT_ARRAY:
3670
2.36k
              case ZEND_ADD_ARRAY_ELEMENT:
3671
2.37k
              case ZEND_RETURN_BY_REF:
3672
2.39k
              case ZEND_VERIFY_RETURN_TYPE:
3673
2.60k
              case ZEND_MAKE_REF:
3674
2.60k
              case ZEND_FE_RESET_RW:
3675
2.60k
                tmp |= key_type | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3676
2.60k
                break;
3677
56
              case ZEND_PRE_INC:
3678
56
              case ZEND_PRE_DEC:
3679
72
              case ZEND_POST_INC:
3680
72
              case ZEND_POST_DEC:
3681
72
                if (tmp & MAY_BE_ARRAY_OF_LONG) {
3682
                  /* may overflow */
3683
60
                  tmp |= key_type | MAY_BE_ARRAY_OF_DOUBLE;
3684
60
                } else if (!(tmp & (MAY_BE_ARRAY_OF_LONG|MAY_BE_ARRAY_OF_DOUBLE))) {
3685
12
                  tmp |= key_type | MAY_BE_ARRAY_OF_LONG | MAY_BE_ARRAY_OF_DOUBLE;
3686
12
                }
3687
72
                break;
3688
32
              case ZEND_FETCH_OBJ_W:
3689
32
              case ZEND_FETCH_OBJ_RW:
3690
50
              case ZEND_FETCH_OBJ_FUNC_ARG:
3691
196
              case ZEND_ASSIGN_OBJ:
3692
196
              case ZEND_ASSIGN_OBJ_OP:
3693
218
              case ZEND_ASSIGN_OBJ_REF:
3694
222
              case ZEND_PRE_INC_OBJ:
3695
222
              case ZEND_PRE_DEC_OBJ:
3696
222
              case ZEND_POST_INC_OBJ:
3697
222
              case ZEND_POST_DEC_OBJ:
3698
                /* These will result in an error exception, unless the element
3699
                 * is already an object. */
3700
222
                break;
3701
12
              case ZEND_SEND_VAR:
3702
16
              case ZEND_FETCH_DIM_R:
3703
                /* This can occur if a DIM_FETCH_FUNC_ARG with UNUSED op2 is left
3704
                 * behind, because it can't be converted to DIM_FETCH_R. */
3705
16
                break;
3706
0
              case ZEND_FREE:
3707
                /* This may happen if the using opcode is DCEd.  */
3708
0
                break;
3709
0
              default: ZEND_UNREACHABLE();
3710
5.97k
            }
3711
5.97k
            j = zend_ssa_next_use(ssa->ops, ssa_op->result_def, j);
3712
5.97k
            if (j >= 0) {
3713
0
              tmp |= key_type | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
3714
0
              break;
3715
0
            }
3716
5.97k
          }
3717
5.97k
          if (opline->opcode != ZEND_FETCH_DIM_FUNC_ARG) {
3718
5.24k
            tmp &= ~MAY_BE_ARRAY_EMPTY;
3719
5.24k
          }
3720
5.97k
        }
3721
6.43k
        if (!(tmp & MAY_BE_ARRAY)
3722
6.23k
         || (tmp & MAY_BE_ARRAY_KEY_ANY)
3723
36
         || opline->opcode == ZEND_FETCH_DIM_FUNC_ARG
3724
22
         || opline->opcode == ZEND_FETCH_DIM_R
3725
22
         || opline->opcode == ZEND_FETCH_DIM_IS
3726
22
         || opline->opcode == ZEND_FETCH_DIM_UNSET
3727
6.41k
         || opline->opcode == ZEND_FETCH_LIST_R) {
3728
6.41k
          UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3729
6.41k
        } else {
3730
          /* invalid key type */
3731
22
          return SUCCESS;
3732
22
        }
3733
6.41k
        COPY_SSA_OBJ_TYPE(ssa_op->op1_use, ssa_op->op1_def);
3734
6.41k
      }
3735
      /* FETCH_LIST on a string behaves like FETCH_R on null */
3736
21.0k
      tmp = zend_array_element_type(
3737
21.0k
        opline->opcode != ZEND_FETCH_LIST_R ? t1 : ((t1 & ~MAY_BE_STRING) | MAY_BE_NULL),
3738
21.0k
        opline->op1_type,
3739
21.0k
        opline->opcode != ZEND_FETCH_DIM_R && opline->opcode != ZEND_FETCH_DIM_IS
3740
10.9k
          && opline->opcode != ZEND_FETCH_LIST_R,
3741
21.0k
        opline->op2_type == IS_UNUSED);
3742
21.0k
      if (opline->opcode == ZEND_FETCH_DIM_FUNC_ARG && (t1 & (MAY_BE_TRUE|MAY_BE_LONG|MAY_BE_DOUBLE|MAY_BE_RESOURCE))) {
3743
637
        tmp |= MAY_BE_NULL;
3744
637
      }
3745
21.0k
      if (opline->opcode == ZEND_FETCH_DIM_IS && (t1 & MAY_BE_STRING)) {
3746
955
        tmp |= MAY_BE_NULL;
3747
955
      }
3748
21.0k
      if ((tmp & (MAY_BE_RC1|MAY_BE_RCN)) == MAY_BE_RCN && opline->result_type == IS_TMP_VAR) {
3749
        /* refcount may be indirectly decremented. Make an exception if the result is used in the next instruction */
3750
1.51k
        if (!ssa_opcodes) {
3751
1.51k
          if (ssa->vars[ssa_op->result_def].use_chain < 0
3752
1.51k
           || opline + 1 != op_array->opcodes + ssa->vars[ssa_op->result_def].use_chain) {
3753
327
            tmp |= MAY_BE_RC1;
3754
327
            }
3755
1.51k
        } else {
3756
0
          if (ssa->vars[ssa_op->result_def].use_chain < 0
3757
0
           || opline + 1 != ssa_opcodes[ssa->vars[ssa_op->result_def].use_chain]) {
3758
0
            tmp |= MAY_BE_RC1;
3759
0
            }
3760
0
        }
3761
1.51k
      }
3762
21.0k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3763
21.0k
      break;
3764
21.0k
    case ZEND_FETCH_THIS:
3765
2.39k
      if (!(op_array->fn_flags & ZEND_ACC_TRAIT_CLONE)) {
3766
2.39k
        UPDATE_SSA_OBJ_TYPE(op_array->scope, 1, ssa_op->result_def);
3767
2.39k
      }
3768
2.39k
      UPDATE_SSA_TYPE(MAY_BE_RCN|MAY_BE_OBJECT, ssa_op->result_def);
3769
2.39k
      break;
3770
47.1k
    case ZEND_FETCH_OBJ_R:
3771
48.3k
    case ZEND_FETCH_OBJ_IS:
3772
48.4k
    case ZEND_FETCH_OBJ_RW:
3773
51.6k
    case ZEND_FETCH_OBJ_W:
3774
51.9k
    case ZEND_FETCH_OBJ_UNSET:
3775
52.1k
    case ZEND_FETCH_OBJ_FUNC_ARG:
3776
52.1k
      if (ssa_op->result_def >= 0) {
3777
52.1k
        uint32_t tmp = 0;
3778
52.1k
        ce = NULL;
3779
52.1k
        if (opline->op1_type != IS_UNUSED
3780
47.3k
            && (t1 & (MAY_BE_ANY | MAY_BE_UNDEF) & ~MAY_BE_OBJECT)) {
3781
46.2k
          tmp |= MAY_BE_NULL;
3782
46.2k
        }
3783
52.1k
        if (opline->op1_type == IS_UNUSED || (t1 & MAY_BE_OBJECT)) {
3784
51.3k
          const zend_property_info *prop_info = zend_fetch_prop_info(op_array, ssa, opline, ssa_op);
3785
51.3k
          tmp |= zend_fetch_prop_type(script, prop_info, &ce);
3786
51.3k
          if (opline->opcode != ZEND_FETCH_OBJ_R && opline->opcode != ZEND_FETCH_OBJ_IS) {
3787
3.72k
            tmp |= MAY_BE_REF | MAY_BE_INDIRECT;
3788
3.72k
            if ((opline->extended_value & ZEND_FETCH_OBJ_FLAGS) == ZEND_FETCH_DIM_WRITE) {
3789
1.63k
              tmp |= MAY_BE_UNDEF;
3790
1.63k
            }
3791
3.72k
            ce = NULL;
3792
47.6k
          } else if (!(opline->op1_type & (IS_VAR|IS_TMP_VAR)) || !(t1 & MAY_BE_RC1)) {
3793
43.0k
            const zend_class_entry *ce = NULL;
3794
3795
43.0k
            if (opline->op1_type == IS_UNUSED) {
3796
3.44k
              ce = op_array->scope;
3797
39.5k
            } else if (ssa_op->op1_use >= 0 && !ssa->var_info[ssa_op->op1_use].is_instanceof) {
3798
39.2k
              ce = ssa->var_info[ssa_op->op1_use].ce;
3799
39.2k
            }
3800
            /* Unset properties will resort back to __get/__set */
3801
43.0k
            if (ce
3802
3.71k
             && !ce->create_object
3803
3.68k
             && ce->default_object_handlers->read_property == zend_std_read_property
3804
3.68k
             && !ce->__get
3805
3.13k
             && !result_may_be_separated(ssa, ssa_op)) {
3806
2.82k
              tmp &= ~MAY_BE_RC1;
3807
2.82k
            }
3808
43.0k
            if (opline->opcode == ZEND_FETCH_OBJ_IS) {
3809
              /* IS check may return null for uninitialized typed property. */
3810
962
              tmp |= MAY_BE_NULL;
3811
962
            }
3812
43.0k
          }
3813
51.3k
        }
3814
52.1k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3815
52.1k
        if (ce) {
3816
76
          UPDATE_SSA_OBJ_TYPE(ce, 1, ssa_op->result_def);
3817
76
        }
3818
52.1k
      }
3819
52.1k
      break;
3820
52.1k
    case ZEND_FETCH_STATIC_PROP_R:
3821
1.77k
    case ZEND_FETCH_STATIC_PROP_IS:
3822
1.77k
    case ZEND_FETCH_STATIC_PROP_RW:
3823
2.03k
    case ZEND_FETCH_STATIC_PROP_W:
3824
2.11k
    case ZEND_FETCH_STATIC_PROP_UNSET:
3825
2.44k
    case ZEND_FETCH_STATIC_PROP_FUNC_ARG:
3826
2.44k
      tmp = zend_fetch_prop_type(script,
3827
2.44k
        zend_fetch_static_prop_info(script, op_array, ssa, opline), &ce);
3828
2.44k
      if (opline->opcode != ZEND_FETCH_STATIC_PROP_R
3829
944
          && opline->opcode != ZEND_FETCH_STATIC_PROP_IS) {
3830
668
        tmp |= MAY_BE_REF | MAY_BE_INDIRECT;
3831
668
        if ((opline->extended_value & ZEND_FETCH_OBJ_FLAGS) == ZEND_FETCH_DIM_WRITE) {
3832
196
          tmp |= MAY_BE_UNDEF;
3833
196
        }
3834
668
        ce = NULL;
3835
1.77k
      } else {
3836
1.77k
        if (!result_may_be_separated(ssa, ssa_op)) {
3837
1.75k
          tmp &= ~MAY_BE_RC1;
3838
1.75k
        }
3839
1.77k
        if (opline->opcode == ZEND_FETCH_STATIC_PROP_IS) {
3840
276
          tmp |= MAY_BE_NULL;
3841
276
        }
3842
1.77k
      }
3843
2.44k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3844
2.44k
      if (ce) {
3845
26
        UPDATE_SSA_OBJ_TYPE(ce, 1, ssa_op->result_def);
3846
26
      }
3847
2.44k
      break;
3848
0
    case ZEND_FRAMELESS_ICALL_1:
3849
0
    case ZEND_FRAMELESS_ICALL_2:
3850
0
    case ZEND_FRAMELESS_ICALL_3:
3851
0
      if (ssa_op->op1_def >= 0) {
3852
0
        ZEND_ASSERT(ssa_op->op1_use >= 0);
3853
0
        tmp = ssa->var_info[ssa_op->op1_use].type;
3854
0
        if (tmp & MAY_BE_RC1) {
3855
0
          tmp |= MAY_BE_RCN;
3856
0
        }
3857
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3858
0
      }
3859
0
      if (ssa_op->op2_def >= 0) {
3860
0
        ZEND_ASSERT(ssa_op->op2_use >= 0);
3861
0
        tmp = ssa->var_info[ssa_op->op2_use].type;
3862
0
        if (tmp & MAY_BE_RC1) {
3863
0
          tmp |= MAY_BE_RCN;
3864
0
        }
3865
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op2_def);
3866
0
      }
3867
0
      if (opline->opcode == ZEND_FRAMELESS_ICALL_3) {
3868
0
        zend_ssa_op *next_ssa_op = ssa_op + 1;
3869
0
        if (next_ssa_op->op1_def >= 0) {
3870
0
          ZEND_ASSERT(next_ssa_op->op1_use >= 0);
3871
0
          tmp = ssa->var_info[next_ssa_op->op1_use].type;
3872
0
          if (tmp & MAY_BE_RC1) {
3873
0
            tmp |= MAY_BE_RCN;
3874
0
          }
3875
0
          UPDATE_SSA_TYPE(tmp, next_ssa_op->op1_def);
3876
0
        }
3877
0
      }
3878
0
      ZEND_FALLTHROUGH;
3879
0
    case ZEND_FRAMELESS_ICALL_0:
3880
36.2k
    case ZEND_DO_FCALL:
3881
36.2k
    case ZEND_DO_ICALL:
3882
39.5k
    case ZEND_DO_UCALL:
3883
39.5k
    case ZEND_DO_FCALL_BY_NAME:
3884
39.5k
      if (ssa_op->result_def >= 0) {
3885
39.5k
        zend_func_info *func_info = ZEND_FUNC_INFO(op_array);
3886
39.5k
        zend_call_info *call_info;
3887
3888
39.5k
        if (!func_info || !func_info->call_map) {
3889
5.88k
          goto unknown_opcode;
3890
5.88k
        }
3891
33.6k
        call_info = func_info->call_map[opline - op_array->opcodes];
3892
33.6k
        if (!call_info) {
3893
11.4k
          goto unknown_opcode;
3894
11.4k
        }
3895
3896
22.1k
        zend_class_entry *ce;
3897
22.1k
        bool ce_is_instanceof;
3898
22.1k
        tmp = zend_get_func_info(call_info, ssa, &ce, &ce_is_instanceof);
3899
22.1k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3900
22.1k
        if (ce) {
3901
1.10k
          UPDATE_SSA_OBJ_TYPE(ce, ce_is_instanceof, ssa_op->result_def);
3902
1.10k
        }
3903
22.1k
      }
3904
22.1k
      break;
3905
22.1k
    case ZEND_CALLABLE_CONVERT:
3906
448
      UPDATE_SSA_TYPE(MAY_BE_OBJECT | MAY_BE_RC1 | MAY_BE_RCN, ssa_op->result_def);
3907
448
      UPDATE_SSA_OBJ_TYPE(zend_ce_closure, /* is_instanceof */ false, ssa_op->result_def);
3908
448
      break;
3909
19.1k
    case ZEND_FETCH_CONSTANT:
3910
19.1k
      UPDATE_SSA_TYPE(MAY_BE_RC1|MAY_BE_RCN|MAY_BE_ANY|MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY, ssa_op->result_def);
3911
19.1k
      break;
3912
19.1k
    case ZEND_FETCH_CLASS_CONSTANT: {
3913
2.55k
      bool is_prototype;
3914
2.55k
      const zend_class_constant *cc = zend_fetch_class_const_info(script, op_array, opline, &is_prototype);
3915
2.55k
      if (!cc || !ZEND_TYPE_IS_SET(cc->type)) {
3916
2.43k
        UPDATE_SSA_TYPE(MAY_BE_RC1|MAY_BE_RCN|MAY_BE_ANY|MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY, ssa_op->result_def);
3917
2.43k
        break;
3918
2.43k
      }
3919
120
      UPDATE_SSA_TYPE(zend_convert_type(script, cc->type, &ce), ssa_op->result_def);
3920
120
      if (ce) {
3921
0
        UPDATE_SSA_OBJ_TYPE(ce, /* is_instanceof */ true, ssa_op->result_def);
3922
0
      }
3923
120
      break;
3924
120
    }
3925
1.57k
    case ZEND_STRLEN:
3926
2.39k
    case ZEND_COUNT:
3927
2.43k
    case ZEND_FUNC_NUM_ARGS:
3928
2.43k
      UPDATE_SSA_TYPE(MAY_BE_LONG, ssa_op->result_def);
3929
2.43k
      break;
3930
2.43k
    case ZEND_FUNC_GET_ARGS:
3931
80
      UPDATE_SSA_TYPE(MAY_BE_RC1|MAY_BE_RCN|MAY_BE_ARRAY|MAY_BE_ARRAY_EMPTY|MAY_BE_ARRAY_PACKED|MAY_BE_ARRAY_OF_ANY, ssa_op->result_def);
3932
80
      break;
3933
438
    case ZEND_GET_CLASS:
3934
508
    case ZEND_GET_CALLED_CLASS:
3935
508
      UPDATE_SSA_TYPE(MAY_BE_STRING|MAY_BE_RCN, ssa_op->result_def);
3936
508
      break;
3937
508
    case ZEND_GET_TYPE:
3938
68
      UPDATE_SSA_TYPE(MAY_BE_STRING|MAY_BE_RC1|MAY_BE_RCN, ssa_op->result_def);
3939
68
      break;
3940
1.16k
    case ZEND_TYPE_CHECK: {
3941
1.16k
      uint32_t expected_type_mask = opline->extended_value;
3942
1.16k
      if (t1 & MAY_BE_UNDEF) {
3943
160
        t1 |= MAY_BE_NULL;
3944
160
      }
3945
1.16k
      tmp = 0;
3946
1.16k
      if (t1 & expected_type_mask) {
3947
1.05k
        tmp |= MAY_BE_TRUE;
3948
1.05k
        if ((t1 & expected_type_mask) & MAY_BE_RESOURCE) {
3949
151
          tmp |= MAY_BE_FALSE;
3950
151
        }
3951
1.05k
      }
3952
1.16k
      if (t1 & (MAY_BE_ANY - expected_type_mask)) {
3953
1.07k
        tmp |= MAY_BE_FALSE;
3954
1.07k
      }
3955
1.16k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3956
1.16k
      break;
3957
1.16k
    }
3958
1.16k
    case ZEND_DEFINED:
3959
93
      UPDATE_SSA_TYPE(MAY_BE_FALSE|MAY_BE_TRUE, ssa_op->result_def);
3960
93
      break;
3961
1.92k
    case ZEND_VERIFY_RETURN_TYPE:
3962
1.92k
      if (t1 & MAY_BE_REF) {
3963
90
        tmp = t1;
3964
90
        ce = NULL;
3965
1.83k
      } else {
3966
1.83k
        zend_arg_info *ret_info = op_array->arg_info - 1;
3967
1.83k
        tmp = zend_fetch_arg_info_type(script, ret_info, &ce);
3968
1.83k
        if ((tmp & MAY_BE_NULL) && opline->op1_type == IS_CV) {
3969
86
          tmp |= MAY_BE_UNDEF;
3970
86
        }
3971
1.83k
        tmp |= (t1 & MAY_BE_INDIRECT);
3972
3973
        // TODO: We could model more precisely how illegal types are converted.
3974
1.83k
        uint32_t extra_types = t1 & ~tmp;
3975
1.83k
        if (!extra_types) {
3976
1.16k
          tmp &= t1;
3977
1.16k
        }
3978
1.83k
      }
3979
1.92k
      if (opline->op1_type & (IS_TMP_VAR|IS_VAR|IS_CV)) {
3980
1.77k
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
3981
1.77k
        if (ce) {
3982
209
          UPDATE_SSA_OBJ_TYPE(ce, 1, ssa_op->op1_def);
3983
1.56k
        } else {
3984
1.56k
          UPDATE_SSA_OBJ_TYPE(NULL, 0, ssa_op->op1_def);
3985
1.56k
        }
3986
1.77k
      } else {
3987
144
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3988
144
        if (ce) {
3989
18
          UPDATE_SSA_OBJ_TYPE(ce, 1, ssa_op->result_def);
3990
126
        } else {
3991
126
          UPDATE_SSA_OBJ_TYPE(NULL, 0, ssa_op->result_def);
3992
126
        }
3993
144
      }
3994
1.92k
      break;
3995
1.92k
    case ZEND_MAKE_REF:
3996
1.09k
      tmp = MAY_BE_REF|MAY_BE_RC1|MAY_BE_RCN|MAY_BE_ANY|MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF;
3997
1.09k
      UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
3998
1.09k
      if (ssa_op->op1_def >= 0) {
3999
252
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
4000
252
      }
4001
1.09k
      break;
4002
1.09k
    case ZEND_CATCH:
4003
      /* Forbidden opcodes */
4004
0
      ZEND_UNREACHABLE();
4005
0
      break;
4006
426
    case ZEND_FETCH_CLASS_NAME:
4007
426
      UPDATE_SSA_TYPE(MAY_BE_STRING|MAY_BE_RCN, ssa_op->result_def);
4008
426
      break;
4009
426
    case ZEND_ISSET_ISEMPTY_THIS:
4010
80
      UPDATE_SSA_TYPE(MAY_BE_BOOL, ssa_op->result_def);
4011
80
      break;
4012
6.11k
    case ZEND_DECLARE_LAMBDA_FUNCTION:
4013
6.11k
      UPDATE_SSA_TYPE(MAY_BE_OBJECT|MAY_BE_RC1|MAY_BE_RCN, ssa_op->result_def);
4014
6.11k
      UPDATE_SSA_OBJ_TYPE(zend_ce_closure, /* is_instanceof */ false, ssa_op->result_def);
4015
6.11k
      break;
4016
0
    case ZEND_PRE_DEC_STATIC_PROP:
4017
10
    case ZEND_PRE_INC_STATIC_PROP:
4018
28
    case ZEND_POST_DEC_STATIC_PROP:
4019
82
    case ZEND_POST_INC_STATIC_PROP: {
4020
82
      if (ssa_op->result_def >= 0) {
4021
82
        const zend_property_info *prop_info = zend_fetch_static_prop_info(script, op_array, ssa, opline);
4022
82
        zend_class_entry *prop_ce;
4023
82
        tmp = zend_fetch_prop_type(script, prop_info, &prop_ce);
4024
        /* Internal objects may result in essentially anything. */
4025
82
        if (tmp & MAY_BE_OBJECT) {
4026
76
          goto unknown_opcode;
4027
76
        }
4028
6
        tmp &= MAY_BE_LONG|MAY_BE_DOUBLE|MAY_BE_STRING|MAY_BE_BOOL|MAY_BE_NULL;
4029
6
        if (tmp & MAY_BE_STRING) {
4030
0
          tmp |= MAY_BE_RC1 | MAY_BE_RCN;
4031
0
        }
4032
6
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
4033
6
      }
4034
6
      break;
4035
82
    }
4036
55
    case ZEND_SPACESHIP:
4037
55
      UPDATE_SSA_TYPE(MAY_BE_LONG, ssa_op->result_def);
4038
55
      break;
4039
485
    case ZEND_FETCH_GLOBALS:
4040
485
      UPDATE_SSA_TYPE(MAY_BE_ARRAY|MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_ARRAY_OF_REF|MAY_BE_RC1|MAY_BE_RCN, ssa_op->result_def);
4041
485
      break;
4042
1.11k
    default:
4043
#ifdef ZEND_DEBUG_TYPE_INFERENCE
4044
      if (ssa_op->result_def >= 0) {
4045
        switch (opline->opcode) {
4046
          case ZEND_FETCH_R:
4047
          case ZEND_FETCH_W:
4048
          case ZEND_FETCH_RW:
4049
          case ZEND_FETCH_IS:
4050
          case ZEND_FETCH_UNSET:
4051
          case ZEND_YIELD_FROM:
4052
          /* Currently unimplemented due to some assumptions in JIT. See:
4053
           * https://github.com/php/php-src/pull/13304#issuecomment-1926668141 */
4054
          case ZEND_SEPARATE:
4055
            break;
4056
          default:
4057
            fprintf(stderr, "Missing result type inference for opcode %s, line %d\n", zend_get_opcode_name(opline->opcode), opline->lineno);
4058
            break;
4059
        }
4060
      }
4061
      if (ssa_op->op1_def >= 0) {
4062
        fprintf(stderr, "Missing op1 type inference for opcode %s, line %d\n", zend_get_opcode_name(opline->opcode), opline->lineno);
4063
      }
4064
      if (ssa_op->op2_def >= 0) {
4065
        fprintf(stderr, "Missing op2 type inference for opcode %s, line %d\n", zend_get_opcode_name(opline->opcode), opline->lineno);
4066
      }
4067
#endif
4068
1.11k
      if (ssa_op->op1_def >= 0) {
4069
0
        tmp = MAY_BE_ANY | MAY_BE_REF | MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
4070
0
        UPDATE_SSA_TYPE(tmp, ssa_op->op1_def);
4071
0
      }
4072
18.5k
unknown_opcode:
4073
18.5k
      if (ssa_op->result_def >= 0) {
4074
18.5k
        tmp = MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
4075
18.5k
        if (opline->result_type == IS_TMP_VAR) {
4076
16.7k
          if (opline->opcode == ZEND_FETCH_R || opline->opcode == ZEND_FETCH_IS) {
4077
            /* Variable reference counter may be decremented before use */
4078
            /* See: ext/opcache/tests/jit/fetch_r_001.phpt */
4079
230
            tmp |= MAY_BE_RC1 | MAY_BE_RCN;
4080
16.5k
          } else {
4081
16.5k
            tmp |= MAY_BE_RC1 | MAY_BE_RCN;
4082
16.5k
          }
4083
16.7k
        } else if (opline->result_type == IS_CV) {
4084
0
          tmp |= MAY_BE_RC1 | MAY_BE_RCN;
4085
1.78k
        } else {
4086
1.78k
          tmp |= MAY_BE_REF | MAY_BE_RC1 | MAY_BE_RCN;
4087
1.78k
          switch (opline->opcode) {
4088
450
            case ZEND_FETCH_W:
4089
496
            case ZEND_FETCH_RW:
4090
507
            case ZEND_FETCH_FUNC_ARG:
4091
515
            case ZEND_FETCH_UNSET:
4092
515
            case ZEND_FETCH_DIM_W:
4093
515
            case ZEND_FETCH_DIM_RW:
4094
515
            case ZEND_FETCH_DIM_FUNC_ARG:
4095
515
            case ZEND_FETCH_DIM_UNSET:
4096
515
            case ZEND_FETCH_OBJ_W:
4097
515
            case ZEND_FETCH_OBJ_RW:
4098
515
            case ZEND_FETCH_OBJ_FUNC_ARG:
4099
515
            case ZEND_FETCH_OBJ_UNSET:
4100
515
            case ZEND_FETCH_STATIC_PROP_W:
4101
515
            case ZEND_FETCH_STATIC_PROP_RW:
4102
515
            case ZEND_FETCH_STATIC_PROP_FUNC_ARG:
4103
515
            case ZEND_FETCH_STATIC_PROP_UNSET:
4104
515
              tmp |= MAY_BE_INDIRECT;
4105
515
              break;
4106
1.78k
          }
4107
1.78k
        }
4108
18.5k
        UPDATE_SSA_TYPE(tmp, ssa_op->result_def);
4109
18.5k
      }
4110
18.5k
      break;
4111
880k
  }
4112
4113
880k
  return SUCCESS;
4114
880k
}
4115
4116
ZEND_API zend_result zend_update_type_info(
4117
      const zend_op_array *op_array,
4118
      zend_ssa            *ssa,
4119
      const zend_script   *script,
4120
      const zend_op       *opline,
4121
      zend_ssa_op         *ssa_op,
4122
      const zend_op      **ssa_opcodes,
4123
      zend_long            optimization_level)
4124
0
{
4125
0
  return _zend_update_type_info(op_array, ssa, script, NULL, opline, ssa_op, ssa_opcodes, optimization_level,
4126
0
              false);
4127
0
}
4128
4129
32
static uint32_t get_class_entry_rank(const zend_class_entry *ce) {
4130
32
  uint32_t rank = 0;
4131
32
  if (ce->ce_flags & ZEND_ACC_LINKED) {
4132
32
    while (ce->parent) {
4133
0
      rank++;
4134
0
      ce = ce->parent;
4135
0
    }
4136
32
  }
4137
32
  return rank;
4138
32
}
4139
4140
/* Compute least common ancestor on class inheritance tree only */
4141
static zend_class_entry *join_class_entries(
4142
51.6k
    zend_class_entry *ce1, zend_class_entry *ce2, bool *is_instanceof) {
4143
51.6k
  uint32_t rank1, rank2;
4144
51.6k
  if (ce1 == ce2) {
4145
51.5k
    return ce1;
4146
51.5k
  }
4147
118
  if (!ce1 || !ce2) {
4148
102
    return NULL;
4149
102
  }
4150
4151
16
  rank1 = get_class_entry_rank(ce1);
4152
16
  rank2 = get_class_entry_rank(ce2);
4153
4154
16
  while (rank1 != rank2) {
4155
0
    if (rank1 > rank2) {
4156
0
      ce1 = !(ce1->ce_flags & ZEND_ACC_LINKED) ? NULL : ce1->parent;
4157
0
      rank1--;
4158
0
    } else {
4159
0
      ce2 = !(ce2->ce_flags & ZEND_ACC_LINKED) ? NULL : ce2->parent;
4160
0
      rank2--;
4161
0
    }
4162
0
  }
4163
4164
32
  while (ce1 != ce2) {
4165
16
    ce1 = !(ce1->ce_flags & ZEND_ACC_LINKED) ? NULL : ce1->parent;
4166
16
    ce2 = !(ce2->ce_flags & ZEND_ACC_LINKED) ? NULL : ce2->parent;
4167
16
  }
4168
4169
16
  if (ce1) {
4170
0
    *is_instanceof = true;
4171
0
  }
4172
16
  return ce1;
4173
118
}
4174
4175
0
static bool safe_instanceof(const zend_class_entry *ce1, const zend_class_entry *ce2) {
4176
0
  if (ce1 == ce2) {
4177
0
    return 1;
4178
0
  }
4179
0
  if (!(ce1->ce_flags & ZEND_ACC_LINKED)) {
4180
    /* This case could be generalized, similarly to unlinked_instanceof */
4181
0
    return 0;
4182
0
  }
4183
0
  return instanceof_function(ce1, ce2);
4184
0
}
4185
4186
static zend_result zend_infer_types_ex(const zend_op_array *op_array, const zend_script *script, zend_ssa *ssa, zend_bitset worklist, zend_long optimization_level)
4187
71.2k
{
4188
71.2k
  const zend_basic_block *blocks = ssa->cfg.blocks;
4189
71.2k
  zend_ssa_var *ssa_vars = ssa->vars;
4190
71.2k
  zend_ssa_var_info *ssa_var_info = ssa->var_info;
4191
71.2k
  int ssa_vars_count = ssa->vars_count;
4192
71.2k
  int j;
4193
71.2k
  uint32_t tmp, worklist_len = zend_bitset_len(ssa_vars_count);
4194
71.2k
  bool update_worklist = 1;
4195
71.2k
  const zend_op **ssa_opcodes = NULL;
4196
4197
1.11M
  while (!zend_bitset_empty(worklist, worklist_len)) {
4198
1.04M
    j = zend_bitset_first(worklist, worklist_len);
4199
1.04M
    zend_bitset_excl(worklist, j);
4200
1.04M
    if (ssa_vars[j].definition_phi) {
4201
162k
      zend_ssa_phi *p = ssa_vars[j].definition_phi;
4202
162k
      if (p->pi >= 0) {
4203
42.9k
        zend_class_entry *ce = ssa_var_info[p->sources[0]].ce;
4204
42.9k
        bool is_instanceof = ssa_var_info[p->sources[0]].is_instanceof;
4205
42.9k
        tmp = get_ssa_var_info(ssa, p->sources[0]);
4206
4207
42.9k
        if (!p->has_range_constraint) {
4208
30.2k
          const zend_ssa_type_constraint *constraint = &p->constraint.type;
4209
30.2k
          tmp &= constraint->type_mask;
4210
30.2k
          if (!(tmp & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE))) {
4211
1.72k
            tmp &= ~(MAY_BE_RC1|MAY_BE_RCN);
4212
1.72k
          }
4213
30.2k
          if ((tmp & MAY_BE_OBJECT) && constraint->ce && ce != constraint->ce) {
4214
22
            if (!ce) {
4215
22
              ce = constraint->ce;
4216
22
              is_instanceof = true;
4217
22
            } else if (is_instanceof && safe_instanceof(constraint->ce, ce)) {
4218
0
              ce = constraint->ce;
4219
0
            } else {
4220
              /* Ignore the constraint (either ce instanceof constraint->ce or
4221
               * they are unrelated, as far as we can statically determine) */
4222
0
            }
4223
22
          }
4224
30.2k
        }
4225
4226
42.9k
        UPDATE_SSA_TYPE(tmp, j);
4227
42.9k
        if (tmp & MAY_BE_REF) {
4228
24.1k
          UPDATE_SSA_OBJ_TYPE(NULL, 0, j);
4229
24.1k
        } else {
4230
18.8k
          UPDATE_SSA_OBJ_TYPE(ce, is_instanceof, j);
4231
18.8k
        }
4232
119k
      } else {
4233
119k
        bool first = true;
4234
119k
        bool is_instanceof = false;
4235
119k
        zend_class_entry *ce = NULL;
4236
119k
        uint32_t i;
4237
4238
119k
        tmp = 0;
4239
366k
        for (i = 0; i < blocks[p->block].predecessors_count; i++) {
4240
247k
          tmp |= get_ssa_var_info(ssa, p->sources[i]);
4241
247k
        }
4242
119k
        UPDATE_SSA_TYPE(tmp, j);
4243
366k
        for (i = 0; i < blocks[p->block].predecessors_count; i++) {
4244
247k
          zend_ssa_var_info *info;
4245
4246
247k
          ZEND_ASSERT(p->sources[i] >= 0);
4247
247k
          info = &ssa_var_info[p->sources[i]];
4248
247k
          if (info->type & MAY_BE_OBJECT) {
4249
132k
            if (first) {
4250
80.9k
              ce = info->ce;
4251
80.9k
              is_instanceof = info->is_instanceof;
4252
80.9k
              first = false;
4253
80.9k
            } else {
4254
51.6k
              is_instanceof |= info->is_instanceof;
4255
51.6k
              ce = join_class_entries(ce, info->ce, &is_instanceof);
4256
51.6k
            }
4257
132k
          }
4258
247k
        }
4259
119k
        UPDATE_SSA_OBJ_TYPE(ce, ce ? is_instanceof : 0, j);
4260
119k
      }
4261
884k
    } else if (ssa_vars[j].definition >= 0) {
4262
884k
      int i = ssa_vars[j].definition;
4263
884k
      if (_zend_update_type_info(op_array, ssa, script, worklist, op_array->opcodes + i, ssa->ops + i, NULL, optimization_level, true) == FAILURE) {
4264
0
        return FAILURE;
4265
0
      }
4266
884k
    }
4267
1.04M
  }
4268
71.2k
  return SUCCESS;
4269
71.2k
}
4270
4271
3.24k
static bool is_narrowable_instr(const zend_op *opline)  {
4272
3.24k
  return opline->opcode == ZEND_ADD || opline->opcode == ZEND_SUB
4273
3.04k
    || opline->opcode == ZEND_MUL || opline->opcode == ZEND_DIV;
4274
3.24k
}
4275
4276
46
static bool is_effective_op1_double_cast(const zend_op *opline, const zval *op2) {
4277
46
  return (opline->opcode == ZEND_ADD && Z_LVAL_P(op2) == 0)
4278
34
    || (opline->opcode == ZEND_SUB && Z_LVAL_P(op2) == 0)
4279
34
    || (opline->opcode == ZEND_MUL && Z_LVAL_P(op2) == 1)
4280
34
    || (opline->opcode == ZEND_DIV && Z_LVAL_P(op2) == 1);
4281
46
}
4282
13
static bool is_effective_op2_double_cast(const zend_op *opline, const zval *op1) {
4283
  /* In PHP it holds that (double)(0-$int) is bitwise identical to 0.0-(double)$int,
4284
   * so allowing SUB here is fine. */
4285
13
  return (opline->opcode == ZEND_ADD && Z_LVAL_P(op1) == 0)
4286
4
    || (opline->opcode == ZEND_SUB && Z_LVAL_P(op1) == 0)
4287
4
    || (opline->opcode == ZEND_MUL && Z_LVAL_P(op1) == 1);
4288
13
}
4289
4290
/* This function recursively checks whether it's possible to convert an integer variable
4291
 * initialization to a double initialization. The basic idea is that if the value is used
4292
 * only in add/sub/mul/div ("narrowable" instructions) with a double result value, then it
4293
 * will be cast to double at that point anyway, so we may as well do it earlier already.
4294
 *
4295
 * The tricky case are chains of operations, where it's not necessarily a given that converting
4296
 * an integer to double before the chain of operations is the same as converting it after the
4297
 * chain. What this function does is detect two cases where it is safe:
4298
 *  * If the operations only involve constants, then we can simply verify that performing the
4299
 *    calculation on integers and doubles yields the same value.
4300
 *  * Even if one operand is not known, we may be able to determine that the operations with the
4301
 *    integer replaced by a double only acts as an effective double cast on the unknown operand.
4302
 *    E.g. 0+$i and 0.0+$i only differ by that cast. If then the consuming instruction of this
4303
 *    result will perform a double cast anyway, the conversion is safe.
4304
 *
4305
 * The checks happens recursively, while keeping track of which variables are already visited to
4306
 * avoid infinite loops. An iterative, worklist driven approach would be possible, but the state
4307
 * management more cumbersome to implement, so we don't bother for now.
4308
 */
4309
static bool can_convert_to_double(
4310
    const zend_op_array *op_array, zend_ssa *ssa, int var_num,
4311
6.25k
    zval *value, zend_bitset visited) {
4312
6.25k
  zend_ssa_var *var = &ssa->vars[var_num];
4313
6.25k
  zend_ssa_phi *phi;
4314
6.25k
  int use;
4315
6.25k
  uint32_t type;
4316
4317
6.25k
  if (zend_bitset_in(visited, var_num)) {
4318
6
    return 1;
4319
6
  }
4320
6.24k
  zend_bitset_incl(visited, var_num);
4321
4322
6.47k
  for (use = var->use_chain; use >= 0; use = zend_ssa_next_use(ssa->ops, var_num, use)) {
4323
3.45k
    zend_op *opline = &op_array->opcodes[use];
4324
3.45k
    zend_ssa_op *ssa_op = &ssa->ops[use];
4325
4326
3.45k
    if (zend_ssa_is_no_val_use(opline, ssa_op, var_num)) {
4327
215
      continue;
4328
215
    }
4329
4330
3.24k
    if (!is_narrowable_instr(opline)) {
4331
3.02k
      return 0;
4332
3.02k
    }
4333
4334
    /* Instruction always returns double, the conversion is certainly fine */
4335
219
    type = ssa->var_info[ssa_op->result_def].type;
4336
219
    if ((type & MAY_BE_ANY) == MAY_BE_DOUBLE) {
4337
8
      continue;
4338
8
    }
4339
4340
    /* UNDEF signals that the previous result is an effective double cast, this is only allowed
4341
     * if this instruction would have done the cast anyway (previous check). */
4342
211
    if (Z_ISUNDEF_P(value)) {
4343
4
      return 0;
4344
4
    }
4345
4346
    /* Check that narrowing can actually be useful */
4347
207
    if ((type & MAY_BE_ANY) & ~(MAY_BE_LONG|MAY_BE_DOUBLE)) {
4348
104
      return 0;
4349
104
    }
4350
4351
103
    {
4352
      /* For calculation on original values */
4353
103
      zval orig_op1, orig_op2, orig_result;
4354
      /* For calculation with var_num cast to double */
4355
103
      zval dval_op1, dval_op2, dval_result;
4356
4357
103
      ZVAL_UNDEF(&orig_op1);
4358
103
      ZVAL_UNDEF(&dval_op1);
4359
103
      if (ssa_op->op1_use == var_num) {
4360
54
        ZVAL_COPY_VALUE(&orig_op1, value);
4361
54
        ZVAL_DOUBLE(&dval_op1, (double) Z_LVAL_P(value));
4362
54
      } else if (opline->op1_type == IS_CONST) {
4363
3
        zval *zv = CRT_CONSTANT(opline->op1);
4364
3
        if (Z_TYPE_P(zv) == IS_LONG || Z_TYPE_P(zv) == IS_DOUBLE) {
4365
3
          ZVAL_COPY_VALUE(&orig_op1, zv);
4366
3
          ZVAL_COPY_VALUE(&dval_op1, zv);
4367
3
        }
4368
3
      }
4369
4370
103
      ZVAL_UNDEF(&orig_op2);
4371
103
      ZVAL_UNDEF(&dval_op2);
4372
103
      if (ssa_op->op2_use == var_num) {
4373
49
        ZVAL_COPY_VALUE(&orig_op2, value);
4374
49
        ZVAL_DOUBLE(&dval_op2, (double) Z_LVAL_P(value));
4375
54
      } else if (opline->op2_type == IS_CONST) {
4376
41
        zval *zv = CRT_CONSTANT(opline->op2);
4377
41
        if (Z_TYPE_P(zv) == IS_LONG || Z_TYPE_P(zv) == IS_DOUBLE) {
4378
41
          ZVAL_COPY_VALUE(&orig_op2, zv);
4379
41
          ZVAL_COPY_VALUE(&dval_op2, zv);
4380
41
        }
4381
41
      }
4382
4383
103
      ZEND_ASSERT(!Z_ISUNDEF(orig_op1) || !Z_ISUNDEF(orig_op2));
4384
103
      if (Z_ISUNDEF(orig_op1)) {
4385
46
        if (opline->opcode == ZEND_MUL && Z_LVAL(orig_op2) == 0) {
4386
0
          ZVAL_LONG(&orig_result, 0);
4387
46
        } else if (is_effective_op1_double_cast(opline, &orig_op2)) {
4388
12
          ZVAL_UNDEF(&orig_result);
4389
34
        } else {
4390
34
          return 0;
4391
34
        }
4392
57
      } else if (Z_ISUNDEF(orig_op2)) {
4393
13
        if (opline->opcode == ZEND_MUL && Z_LVAL(orig_op1) == 0) {
4394
0
          ZVAL_LONG(&orig_result, 0);
4395
13
        } else if (is_effective_op2_double_cast(opline, &orig_op1)) {
4396
9
          ZVAL_UNDEF(&orig_result);
4397
9
        } else {
4398
4
          return 0;
4399
4
        }
4400
44
      } else {
4401
44
        uint8_t opcode = opline->opcode;
4402
4403
44
        if (opcode == ZEND_ASSIGN_OP) {
4404
0
          opcode = opline->extended_value;
4405
0
        }
4406
4407
        /* Avoid division by zero */
4408
44
        if (opcode == ZEND_DIV && zval_get_double(&orig_op2) == 0.0) {
4409
0
          return 0;
4410
0
        }
4411
4412
44
        get_binary_op(opcode)(&orig_result, &orig_op1, &orig_op2);
4413
44
        get_binary_op(opcode)(&dval_result, &dval_op1, &dval_op2);
4414
44
        ZEND_ASSERT(Z_TYPE(dval_result) == IS_DOUBLE);
4415
44
        if (zval_get_double(&orig_result) != Z_DVAL(dval_result)) {
4416
4
          return 0;
4417
4
        }
4418
44
      }
4419
4420
61
      if (!can_convert_to_double(op_array, ssa, ssa_op->result_def, &orig_result, visited)) {
4421
55
        return 0;
4422
55
      }
4423
61
    }
4424
61
  }
4425
4426
3.03k
  for (phi = var->phi_use_chain; phi; phi = zend_ssa_next_use_phi(ssa, var_num, phi)) {
4427
    /* Check that narrowing can actually be useful */
4428
3.01k
    type = ssa->var_info[phi->ssa_var].type;
4429
3.01k
    if ((type & MAY_BE_ANY) & ~(MAY_BE_LONG|MAY_BE_DOUBLE)) {
4430
263
      return 0;
4431
263
    }
4432
4433
2.75k
    if (!can_convert_to_double(op_array, ssa, phi->ssa_var, value, visited)) {
4434
2.73k
      return 0;
4435
2.73k
    }
4436
2.75k
  }
4437
4438
18
  return 1;
4439
3.01k
}
4440
4441
static zend_result zend_type_narrowing(const zend_op_array *op_array, const zend_script *script, zend_ssa *ssa, zend_long optimization_level)
4442
71.2k
{
4443
71.2k
  uint32_t bitset_len = zend_bitset_len(ssa->vars_count);
4444
71.2k
  zend_bitset visited, worklist;
4445
71.2k
  int i, v;
4446
71.2k
  zend_op *opline;
4447
71.2k
  bool narrowed = 0;
4448
71.2k
  ALLOCA_FLAG(use_heap)
4449
4450
71.2k
  visited = ZEND_BITSET_ALLOCA(2 * bitset_len, use_heap);
4451
71.2k
  worklist = visited + bitset_len;
4452
4453
71.2k
  zend_bitset_clear(worklist, bitset_len);
4454
4455
1.07M
  for (v = op_array->last_var; v < ssa->vars_count; v++) {
4456
1.00M
    if ((ssa->var_info[v].type & (MAY_BE_REF | MAY_BE_ANY | MAY_BE_UNDEF)) != MAY_BE_LONG) continue;
4457
115k
    if (ssa->vars[v].definition < 0) continue;
4458
108k
    if (ssa->vars[v].no_val) continue;
4459
106k
    opline = op_array->opcodes + ssa->vars[v].definition;
4460
    /* Go through assignments of literal integers and check if they can be converted to
4461
     * doubles instead, in the hope that we'll narrow long|double to double. */
4462
106k
    if (opline->opcode == ZEND_ASSIGN && opline->result_type == IS_UNUSED &&
4463
4.54k
        opline->op1_type == IS_CV && opline->op2_type == IS_CONST) {
4464
3.44k
      zval *value = CRT_CONSTANT(opline->op2);
4465
4466
3.44k
      zend_bitset_clear(visited, bitset_len);
4467
3.44k
      if (can_convert_to_double(op_array, ssa, v, value, visited)) {
4468
4
        narrowed = 1;
4469
4
        ssa->var_info[v].use_as_double = 1;
4470
        /* The "visited" vars are exactly those which may change their type due to
4471
         * narrowing. Reset their types and add them to the type inference worklist */
4472
86
        ZEND_BITSET_FOREACH(visited, bitset_len, i) {
4473
10
          ssa->var_info[i].type &= ~MAY_BE_ANY;
4474
10
        } ZEND_BITSET_FOREACH_END();
4475
4
        zend_bitset_union(worklist, visited, bitset_len);
4476
4
      }
4477
3.44k
    }
4478
106k
  }
4479
4480
71.2k
  if (!narrowed) {
4481
71.2k
    free_alloca(visited, use_heap);
4482
71.2k
    return SUCCESS;
4483
71.2k
  }
4484
4485
4
  if (zend_infer_types_ex(op_array, script, ssa, worklist, optimization_level) == FAILURE) {
4486
0
    free_alloca(visited, use_heap);
4487
0
    return FAILURE;
4488
0
  }
4489
4490
4
  free_alloca(visited, use_heap);
4491
4
  return SUCCESS;
4492
4
}
4493
4494
static bool is_recursive_tail_call(const zend_op_array *op_array, const zend_op *opline)
4495
69.5k
{
4496
69.5k
  const zend_func_info *info = ZEND_FUNC_INFO(op_array);
4497
4498
69.5k
  if (info->ssa.ops && info->ssa.vars && info->call_map &&
4499
32.5k
      info->ssa.ops[opline - op_array->opcodes].op1_use >= 0 &&
4500
2.43k
      info->ssa.vars[info->ssa.ops[opline - op_array->opcodes].op1_use].definition >= 0) {
4501
4502
2.38k
    const zend_op *op = op_array->opcodes + info->ssa.vars[info->ssa.ops[opline - op_array->opcodes].op1_use].definition;
4503
4504
2.38k
    if (op->opcode == ZEND_DO_UCALL) {
4505
144
      const zend_call_info *call_info = info->call_map[op - op_array->opcodes];
4506
144
      if (call_info && op_array == &call_info->callee_func->op_array) {
4507
8
        return 1;
4508
8
      }
4509
144
    }
4510
2.38k
  }
4511
69.5k
  return 0;
4512
69.5k
}
4513
4514
uint32_t zend_get_return_info_from_signature_only(
4515
    const zend_function *func, const zend_script *script,
4516
28.1k
    zend_class_entry **ce, bool *ce_is_instanceof, bool use_tentative_return_info) {
4517
28.1k
  uint32_t type;
4518
28.1k
  if (func->common.fn_flags & ZEND_ACC_HAS_RETURN_TYPE &&
4519
22.4k
    (use_tentative_return_info || !ZEND_ARG_TYPE_IS_TENTATIVE(func->common.arg_info - 1))
4520
28.1k
  ) {
4521
22.4k
    const zend_arg_info *ret_info = func->common.arg_info - 1;
4522
22.4k
    type = zend_fetch_arg_info_type(script, ret_info, ce);
4523
22.4k
    *ce_is_instanceof = ce != NULL;
4524
22.4k
  } else {
4525
5.74k
    type = MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF
4526
5.74k
      | MAY_BE_RC1 | MAY_BE_RCN;
4527
5.74k
    *ce = NULL;
4528
5.74k
    *ce_is_instanceof = false;
4529
5.74k
  }
4530
4531
  /* For generators RETURN_REFERENCE refers to the yielded values. */
4532
28.1k
  if ((func->common.fn_flags & ZEND_ACC_RETURN_REFERENCE)
4533
990
      && !(func->common.fn_flags & ZEND_ACC_GENERATOR)) {
4534
918
    type |= MAY_BE_REF;
4535
918
    *ce = NULL;
4536
918
    *ce_is_instanceof = false;
4537
918
  }
4538
28.1k
  return type;
4539
28.1k
}
4540
4541
ZEND_API void zend_init_func_return_info(
4542
  const zend_op_array *op_array, const zend_script *script, zend_ssa_var_info *ret)
4543
7.16k
{
4544
7.16k
  ZEND_ASSERT((op_array->fn_flags & ZEND_ACC_HAS_RETURN_TYPE));
4545
4546
7.16k
  zend_ssa_range tmp_range = {0, 0, 0, 0};
4547
7.16k
  bool is_instanceof = false;
4548
7.16k
  ret->type = zend_get_return_info_from_signature_only(
4549
7.16k
    (zend_function *) op_array, script, &ret->ce, &is_instanceof, /* use_tentative_return_info */ true);
4550
7.16k
  ret->is_instanceof = is_instanceof;
4551
7.16k
  ret->range = tmp_range;
4552
7.16k
  ret->has_range = 0;
4553
7.16k
}
4554
4555
static void zend_func_return_info(const zend_op_array   *op_array,
4556
                                  const zend_script     *script,
4557
                                  bool                   recursive,
4558
                                  bool                   widening,
4559
                                  zend_ssa_var_info     *ret)
4560
71.2k
{
4561
71.2k
  const zend_func_info *info = ZEND_FUNC_INFO(op_array);
4562
71.2k
  const zend_ssa *ssa = &info->ssa;
4563
71.2k
  uint32_t blocks_count = info->ssa.cfg.blocks_count;
4564
71.2k
  const zend_basic_block *blocks = info->ssa.cfg.blocks;
4565
71.2k
  uint32_t t1;
4566
71.2k
  uint32_t tmp = 0;
4567
71.2k
  zend_class_entry *tmp_ce = NULL;
4568
71.2k
  int tmp_is_instanceof = -1;
4569
71.2k
  zend_class_entry *arg_ce;
4570
71.2k
  bool arg_is_instanceof;
4571
71.2k
  zend_ssa_range tmp_range = {0, 0, 0, 0};
4572
71.2k
  int tmp_has_range = -1;
4573
4574
71.2k
  if (op_array->fn_flags & ZEND_ACC_GENERATOR) {
4575
1.70k
    ret->type = MAY_BE_OBJECT | MAY_BE_RC1 | MAY_BE_RCN;
4576
1.70k
    ret->ce = zend_ce_generator;
4577
1.70k
    ret->is_instanceof = 0;
4578
1.70k
    ret->range = tmp_range;
4579
1.70k
    ret->has_range = 0;
4580
1.70k
    return;
4581
1.70k
  }
4582
4583
69.5k
  if (!ret->type) {
4584
    /* We will intersect the type later. */
4585
62.6k
    ret->type = MAY_BE_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF | MAY_BE_ARRAY_KEY_ANY
4586
62.6k
      | MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_REF;
4587
62.6k
  }
4588
4589
266k
  for (uint32_t j = 0; j < blocks_count; j++) {
4590
196k
    if ((blocks[j].flags & ZEND_BB_REACHABLE) && blocks[j].len != 0) {
4591
196k
      zend_op *opline = op_array->opcodes + blocks[j].start + blocks[j].len - 1;
4592
4593
196k
      if (opline->opcode == ZEND_RETURN || opline->opcode == ZEND_RETURN_BY_REF) {
4594
69.5k
        const zend_ssa_op *ssa_op = ssa->ops ? &ssa->ops[opline - op_array->opcodes] : NULL;
4595
69.5k
        if (!recursive && ssa_op && info->ssa.var_info &&
4596
0
            ssa_op->op1_use >= 0 &&
4597
0
            info->ssa.var_info[ssa_op->op1_use].recursive) {
4598
0
          continue;
4599
0
        }
4600
69.5k
        if (is_recursive_tail_call(op_array, opline)) {
4601
8
          continue;
4602
8
        }
4603
69.5k
        t1 = OP1_INFO();
4604
69.5k
        if (t1 & MAY_BE_UNDEF) {
4605
205
          t1 |= MAY_BE_NULL;
4606
205
        }
4607
69.5k
        if (opline->opcode == ZEND_RETURN) {
4608
68.4k
          if (t1 & MAY_BE_RC1) {
4609
6.37k
            t1 |= MAY_BE_RCN;
4610
6.37k
          }
4611
68.4k
          t1 &= ~(MAY_BE_UNDEF | MAY_BE_REF);
4612
68.4k
        } else {
4613
1.13k
          t1 |= MAY_BE_REF;
4614
1.13k
          t1 &= ~(MAY_BE_UNDEF | MAY_BE_RC1 | MAY_BE_RCN);
4615
1.13k
        }
4616
69.5k
        tmp |= t1;
4617
4618
69.5k
        if (ssa_op && info->ssa.var_info &&
4619
69.5k
            ssa_op->op1_use >= 0 && !(t1 & MAY_BE_REF) &&
4620
8.89k
            info->ssa.var_info[ssa_op->op1_use].ce) {
4621
1.70k
          arg_ce = info->ssa.var_info[ssa_op->op1_use].ce;
4622
1.70k
          arg_is_instanceof = info->ssa.var_info[ssa_op->op1_use].is_instanceof;
4623
67.8k
        } else {
4624
67.8k
          arg_ce = NULL;
4625
67.8k
          arg_is_instanceof = false;
4626
67.8k
        }
4627
4628
69.5k
        if (tmp_is_instanceof < 0) {
4629
67.5k
          tmp_ce = arg_ce;
4630
67.5k
          tmp_is_instanceof = arg_is_instanceof;
4631
67.5k
        } else if (arg_ce && arg_ce == tmp_ce) {
4632
30
          if (tmp_is_instanceof != arg_is_instanceof) {
4633
0
            tmp_is_instanceof = 1;
4634
0
          }
4635
1.99k
        } else {
4636
1.99k
          tmp_ce = NULL;
4637
1.99k
          tmp_is_instanceof = 0;
4638
1.99k
        }
4639
4640
69.5k
        if (opline->op1_type == IS_CONST) {
4641
59.6k
          const zval *zv = CRT_CONSTANT(opline->op1);
4642
4643
59.6k
          if (Z_TYPE_P(zv) == IS_LONG) {
4644
30.7k
            if (tmp_has_range < 0) {
4645
30.3k
              tmp_has_range = 1;
4646
30.3k
              tmp_range.underflow = 0;
4647
30.3k
              tmp_range.min = Z_LVAL_P(zv);
4648
30.3k
              tmp_range.max = Z_LVAL_P(zv);
4649
30.3k
              tmp_range.overflow = 0;
4650
30.3k
            } else if (tmp_has_range) {
4651
207
              if (!tmp_range.underflow) {
4652
102
                tmp_range.min = MIN(tmp_range.min, Z_LVAL_P(zv));
4653
102
              }
4654
207
              if (!tmp_range.overflow) {
4655
102
                tmp_range.max = MAX(tmp_range.max, Z_LVAL_P(zv));
4656
102
              }
4657
207
            }
4658
30.7k
          } else {
4659
28.9k
            tmp_has_range = 0;
4660
28.9k
          }
4661
59.6k
        } else if (ssa_op && info->ssa.var_info && ssa_op->op1_use >= 0) {
4662
9.91k
          if (info->ssa.var_info[ssa_op->op1_use].has_range) {
4663
9.36k
            if (tmp_has_range < 0) {
4664
8.69k
              tmp_has_range = 1;
4665
8.69k
              tmp_range = info->ssa.var_info[ssa_op->op1_use].range;
4666
8.69k
            } else if (tmp_has_range) {
4667
              /* union */
4668
553
              if (info->ssa.var_info[ssa_op->op1_use].range.underflow) {
4669
551
                tmp_range.underflow = 1;
4670
551
                tmp_range.min = ZEND_LONG_MIN;
4671
551
              } else {
4672
2
                tmp_range.min = MIN(tmp_range.min, info->ssa.var_info[ssa_op->op1_use].range.min);
4673
2
              }
4674
553
              if (info->ssa.var_info[ssa_op->op1_use].range.overflow) {
4675
551
                tmp_range.overflow = 1;
4676
551
                tmp_range.max = ZEND_LONG_MAX;
4677
551
              } else {
4678
2
                tmp_range.max = MAX(tmp_range.max, info->ssa.var_info[ssa_op->op1_use].range.max);
4679
2
              }
4680
553
            }
4681
9.36k
          } else if (!widening) {
4682
559
            tmp_has_range = 1;
4683
559
            tmp_range.underflow = 1;
4684
559
            tmp_range.min = ZEND_LONG_MIN;
4685
559
            tmp_range.max = ZEND_LONG_MAX;
4686
559
            tmp_range.overflow = 1;
4687
559
          }
4688
9.91k
        } else {
4689
0
          tmp_has_range = 0;
4690
0
        }
4691
69.5k
      }
4692
196k
    }
4693
196k
  }
4694
4695
69.5k
  if (!(op_array->fn_flags & ZEND_ACC_HAS_RETURN_TYPE)) {
4696
62.6k
    if (tmp_is_instanceof < 0) {
4697
1.85k
      tmp_is_instanceof = 0;
4698
1.85k
      tmp_ce = NULL;
4699
1.85k
    }
4700
62.6k
    if (tmp_has_range < 0) {
4701
1.85k
      tmp_has_range = 0;
4702
1.85k
    }
4703
62.6k
    ret->ce = tmp_ce;
4704
62.6k
    ret->is_instanceof = tmp_is_instanceof;
4705
62.6k
  }
4706
69.5k
  ret->type &= tmp;
4707
69.5k
  ret->range = tmp_range;
4708
69.5k
  ret->has_range = tmp_has_range;
4709
69.5k
}
4710
4711
static zend_result zend_infer_types(const zend_op_array *op_array, const zend_script *script, zend_ssa *ssa, zend_long optimization_level)
4712
71.2k
{
4713
71.2k
  int ssa_vars_count = ssa->vars_count;
4714
71.2k
  int j;
4715
71.2k
  zend_bitset worklist;
4716
71.2k
  ALLOCA_FLAG(use_heap);
4717
4718
71.2k
  worklist = do_alloca(sizeof(zend_ulong) * zend_bitset_len(ssa_vars_count), use_heap);
4719
71.2k
  memset(worklist, 0, sizeof(zend_ulong) * zend_bitset_len(ssa_vars_count));
4720
4721
  /* Type Inference */
4722
1.07M
  for (j = op_array->last_var; j < ssa_vars_count; j++) {
4723
1.00M
    zend_bitset_incl(worklist, j);
4724
1.00M
  }
4725
4726
71.2k
  if (zend_infer_types_ex(op_array, script, ssa, worklist, optimization_level) == FAILURE) {
4727
0
    free_alloca(worklist,  use_heap);
4728
0
    return FAILURE;
4729
0
  }
4730
4731
71.2k
  if (optimization_level & ZEND_OPTIMIZER_NARROW_TO_DOUBLE) {
4732
    /* Narrowing integer initialization to doubles */
4733
71.2k
    zend_type_narrowing(op_array, script, ssa, optimization_level);
4734
71.2k
  }
4735
4736
71.2k
  if (ZEND_FUNC_INFO(op_array)) {
4737
71.2k
    zend_func_return_info(op_array, script, 1, 0, &ZEND_FUNC_INFO(op_array)->return_info);
4738
71.2k
  }
4739
4740
71.2k
  free_alloca(worklist,  use_heap);
4741
71.2k
  return SUCCESS;
4742
71.2k
}
4743
4744
static void zend_mark_cv_references(const zend_op_array *op_array, const zend_script *script, const zend_ssa *ssa)
4745
71.2k
{
4746
71.2k
  int var, def;
4747
71.2k
  const zend_op *opline;
4748
71.2k
  zend_arg_info *arg_info;
4749
71.2k
  uint32_t worklist_len = zend_bitset_len(ssa->vars_count);
4750
71.2k
  zend_bitset worklist;
4751
71.2k
  ALLOCA_FLAG(use_heap);
4752
4753
71.2k
  worklist = do_alloca(sizeof(zend_ulong) * worklist_len, use_heap);
4754
71.2k
  memset(worklist, 0, sizeof(zend_ulong) * worklist_len);
4755
4756
  /* Collect SSA variables which definitions creates PHP reference */
4757
1.19M
  for (var = 0; var < ssa->vars_count; var++) {
4758
1.12M
    def = ssa->vars[var].definition;
4759
1.12M
    if (def >= 0 && ssa->vars[var].var < op_array->last_var) {
4760
144k
      opline = op_array->opcodes + def;
4761
144k
      if (ssa->ops[def].result_def == var) {
4762
24.9k
        switch (opline->opcode) {
4763
21.6k
          case ZEND_RECV:
4764
24.6k
          case ZEND_RECV_INIT:
4765
24.6k
            arg_info = &op_array->arg_info[opline->op1.num-1];
4766
24.6k
            if (!ZEND_ARG_SEND_MODE(arg_info)) {
4767
22.8k
              continue;
4768
22.8k
            }
4769
1.77k
            break;
4770
1.77k
          default:
4771
370
            continue;
4772
24.9k
        }
4773
119k
      } else if (ssa->ops[def].op1_def == var) {
4774
114k
        switch (opline->opcode) {
4775
1.93k
          case ZEND_ASSIGN_REF:
4776
2.06k
          case ZEND_MAKE_REF:
4777
2.69k
          case ZEND_FE_RESET_RW:
4778
3.50k
          case ZEND_BIND_GLOBAL:
4779
5.41k
          case ZEND_SEND_REF:
4780
10.8k
          case ZEND_SEND_VAR_EX:
4781
10.8k
          case ZEND_SEND_FUNC_ARG:
4782
11.0k
          case ZEND_BIND_INIT_STATIC_OR_JMP:
4783
11.0k
            break;
4784
193
          case ZEND_INIT_ARRAY:
4785
247
          case ZEND_ADD_ARRAY_ELEMENT:
4786
247
            if (!(opline->extended_value & ZEND_ARRAY_ELEMENT_REF)) {
4787
0
              continue;
4788
0
            }
4789
247
            break;
4790
7.63k
          case ZEND_BIND_STATIC:
4791
7.63k
            if (!(opline->extended_value & ZEND_BIND_REF)) {
4792
6.53k
              continue;
4793
6.53k
            }
4794
1.09k
            break;
4795
1.09k
          case ZEND_YIELD:
4796
60
            if (!(op_array->fn_flags & ZEND_ACC_RETURN_REFERENCE)) {
4797
0
              continue;
4798
0
            }
4799
60
            break;
4800
335
          case ZEND_OP_DATA:
4801
335
            switch ((opline-1)->opcode) {
4802
285
              case ZEND_ASSIGN_OBJ_REF:
4803
335
              case ZEND_ASSIGN_STATIC_PROP_REF:
4804
335
                break;
4805
0
              default:
4806
0
                continue;
4807
335
            }
4808
335
            break;
4809
95.3k
          default:
4810
95.3k
            continue;
4811
114k
        }
4812
114k
      } else if (ssa->ops[def].op2_def == var) {
4813
5.21k
        switch (opline->opcode) {
4814
1.26k
          case ZEND_ASSIGN_REF:
4815
1.98k
          case ZEND_FE_FETCH_RW:
4816
1.98k
            break;
4817
482
          case ZEND_BIND_LEXICAL:
4818
482
            if (!(opline->extended_value & ZEND_BIND_REF)) {
4819
0
              continue;
4820
0
            }
4821
482
            break;
4822
2.75k
          default:
4823
2.75k
            continue;
4824
5.21k
        }
4825
5.21k
      } else {
4826
0
        ZEND_UNREACHABLE();
4827
0
      }
4828
17.0k
      zend_bitset_incl(worklist, var);
4829
978k
    } else if (ssa->var_info[var].type & MAY_BE_REF) {
4830
61.1k
      zend_bitset_incl(worklist, var);
4831
917k
    } else if (ssa->vars[var].alias == SYMTABLE_ALIAS) {
4832
0
      zend_bitset_incl(worklist, var);
4833
0
    }
4834
1.12M
  }
4835
4836
  /* Set and propagate MAY_BE_REF */
4837
2.68M
  WHILE_WORKLIST(worklist, worklist_len, var) {
4838
4839
2.68M
    ssa->var_info[var].type |= MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_REF | MAY_BE_ANY | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
4840
4841
2.68M
    if (ssa->vars[var].phi_use_chain) {
4842
74.3k
      zend_ssa_phi *p = ssa->vars[var].phi_use_chain;
4843
98.4k
      do {
4844
98.4k
        if (!(ssa->var_info[p->ssa_var].type & MAY_BE_REF)) {
4845
84.7k
          zend_bitset_incl(worklist, p->ssa_var);
4846
84.7k
        }
4847
98.4k
        p = zend_ssa_next_use_phi(ssa, var, p);
4848
98.4k
      } while (p);
4849
74.3k
    }
4850
4851
2.68M
    if (ssa->vars[var].use_chain >= 0) {
4852
157k
      int use = ssa->vars[var].use_chain;
4853
411k
      FOREACH_USE(&ssa->vars[var], use) {
4854
411k
        zend_ssa_op *op = ssa->ops + use;
4855
411k
        if (op->op1_use == var && op->op1_def >= 0) {
4856
64.2k
          if (!(ssa->var_info[op->op1_def].type & MAY_BE_REF)) {
4857
            /* Unset breaks references (outside global scope). */
4858
60.0k
            if (op_array->opcodes[use].opcode == ZEND_UNSET_CV
4859
1.30k
                && op_array->function_name) {
4860
60
              continue;
4861
60
            }
4862
59.9k
            zend_bitset_incl(worklist, op->op1_def);
4863
59.9k
          }
4864
64.2k
        }
4865
254k
        if (op->op2_use == var && op->op2_def >= 0) {
4866
3.95k
          if (!(ssa->var_info[op->op2_def].type & MAY_BE_REF)) {
4867
2.27k
            zend_bitset_incl(worklist, op->op2_def);
4868
2.27k
          }
4869
3.95k
        }
4870
254k
        if (op->result_use == var && op->result_def >= 0) {
4871
0
          if (!(ssa->var_info[op->result_def].type & MAY_BE_REF)) {
4872
0
            zend_bitset_incl(worklist, op->result_def);
4873
0
          }
4874
0
        }
4875
254k
      } FOREACH_USE_END();
4876
157k
    }
4877
2.68M
  } WHILE_WORKLIST_END();
4878
4879
71.2k
  free_alloca(worklist,  use_heap);
4880
71.2k
}
4881
4882
ZEND_API zend_result zend_ssa_inference(zend_arena **arena, const zend_op_array *op_array, const zend_script *script, zend_ssa *ssa, zend_long optimization_level) /* {{{ */
4883
71.2k
{
4884
71.2k
  zend_ssa_var_info *ssa_var_info;
4885
71.2k
  int i;
4886
4887
71.2k
  if (!ssa->var_info) {
4888
71.2k
    ssa->var_info = zend_arena_calloc(arena, ssa->vars_count, sizeof(zend_ssa_var_info));
4889
71.2k
  }
4890
71.2k
  ssa_var_info = ssa->var_info;
4891
4892
71.2k
  if (!op_array->function_name) {
4893
91.5k
    for (i = 0; i < op_array->last_var; i++) {
4894
61.1k
      ssa_var_info[i].type = MAY_BE_UNDEF | MAY_BE_RC1 | MAY_BE_RCN | MAY_BE_REF | MAY_BE_ANY  | MAY_BE_ARRAY_KEY_ANY | MAY_BE_ARRAY_OF_ANY | MAY_BE_ARRAY_OF_REF;
4895
61.1k
      ssa_var_info[i].has_range = 0;
4896
61.1k
    }
4897
40.8k
  } else {
4898
99.4k
    for (i = 0; i < op_array->last_var; i++) {
4899
58.5k
      ssa_var_info[i].type = MAY_BE_UNDEF;
4900
58.5k
      ssa_var_info[i].has_range = 0;
4901
58.5k
      if (ssa->vars[i].alias) {
4902
0
        ssa_var_info[i].type |= get_ssa_alias_types(ssa->vars[i].alias);
4903
0
      }
4904
58.5k
    }
4905
40.8k
  }
4906
1.07M
  for (i = op_array->last_var; i < ssa->vars_count; i++) {
4907
1.00M
    ssa_var_info[i].type = 0;
4908
1.00M
    ssa_var_info[i].has_range = 0;
4909
1.00M
  }
4910
4911
71.2k
  zend_mark_cv_references(op_array, script, ssa);
4912
4913
71.2k
  zend_infer_ranges(op_array, ssa);
4914
4915
71.2k
  if (zend_infer_types(op_array, script, ssa, optimization_level) == FAILURE) {
4916
0
    return FAILURE;
4917
0
  }
4918
4919
71.2k
  return SUCCESS;
4920
71.2k
}
4921
/* }}} */
4922
4923
ZEND_API bool zend_may_throw_ex(const zend_op *opline, const zend_ssa_op *ssa_op, const zend_op_array *op_array, const zend_ssa *ssa, uint32_t t1, uint32_t t2)
4924
61.0k
{
4925
61.0k
  if (opline->op1_type == IS_CV) {
4926
20.5k
    if (t1 & MAY_BE_UNDEF) {
4927
12.5k
      switch (opline->opcode) {
4928
0
        case ZEND_UNSET_VAR:
4929
0
        case ZEND_ISSET_ISEMPTY_VAR:
4930
0
          return 1;
4931
8
        case ZEND_ISSET_ISEMPTY_DIM_OBJ:
4932
8
        case ZEND_ISSET_ISEMPTY_PROP_OBJ:
4933
7.42k
        case ZEND_ASSIGN:
4934
7.63k
        case ZEND_ASSIGN_DIM:
4935
7.63k
        case ZEND_ASSIGN_REF:
4936
7.63k
        case ZEND_BIND_GLOBAL:
4937
7.74k
        case ZEND_BIND_STATIC:
4938
7.74k
        case ZEND_BIND_INIT_STATIC_OR_JMP:
4939
7.83k
        case ZEND_FETCH_DIM_IS:
4940
7.83k
        case ZEND_FETCH_OBJ_IS:
4941
7.83k
        case ZEND_SEND_REF:
4942
8.36k
        case ZEND_UNSET_CV:
4943
8.36k
        case ZEND_ISSET_ISEMPTY_CV:
4944
8.36k
        case ZEND_MAKE_REF:
4945
8.36k
        case ZEND_FETCH_DIM_W:
4946
8.36k
          break;
4947
4.22k
        default:
4948
          /* undefined variable warning */
4949
4.22k
          return 1;
4950
12.5k
      }
4951
12.5k
    }
4952
40.4k
  } else if (opline->op1_type & (IS_TMP_VAR|IS_VAR)) {
4953
33.1k
    if ((t1 & MAY_BE_RC1)
4954
22.0k
     && (t1 & (MAY_BE_OBJECT|MAY_BE_RESOURCE|MAY_BE_ARRAY_OF_OBJECT|MAY_BE_ARRAY_OF_RESOURCE|MAY_BE_ARRAY_OF_ARRAY))) {
4955
15.8k
      switch (opline->opcode) {
4956
10
        case ZEND_CASE:
4957
10
        case ZEND_CASE_STRICT:
4958
10
        case ZEND_FE_FETCH_R:
4959
10
        case ZEND_FE_FETCH_RW:
4960
10
        case ZEND_FETCH_LIST_R:
4961
10
        case ZEND_QM_ASSIGN:
4962
10
        case ZEND_SEND_VAL:
4963
10
        case ZEND_SEND_VAL_EX:
4964
10
        case ZEND_SEND_VAR:
4965
10
        case ZEND_SEND_VAR_EX:
4966
10
        case ZEND_SEND_FUNC_ARG:
4967
10
        case ZEND_SEND_VAR_NO_REF:
4968
10
        case ZEND_SEND_VAR_NO_REF_EX:
4969
10
        case ZEND_SEND_REF:
4970
10
        case ZEND_SEPARATE:
4971
10
        case ZEND_END_SILENCE:
4972
10
        case ZEND_MAKE_REF:
4973
10
          break;
4974
15.7k
        default:
4975
          /* destructor may be called */
4976
15.7k
          return 1;
4977
15.8k
      }
4978
15.8k
    }
4979
33.1k
  }
4980
4981
41.0k
  if (opline->op2_type == IS_CV) {
4982
1.91k
    if (t2 & MAY_BE_UNDEF) {
4983
1.08k
      switch (opline->opcode) {
4984
0
        case ZEND_ASSIGN_REF:
4985
0
        case ZEND_FE_FETCH_R:
4986
0
        case ZEND_FE_FETCH_RW:
4987
0
          break;
4988
1.08k
        default:
4989
          /* undefined variable warning */
4990
1.08k
          return 1;
4991
1.08k
      }
4992
1.08k
    }
4993
39.0k
  } else if (opline->op2_type & (IS_TMP_VAR|IS_VAR)) {
4994
6.25k
    if ((t2 & MAY_BE_RC1)
4995
2.06k
     && (t2 & (MAY_BE_OBJECT|MAY_BE_RESOURCE|MAY_BE_ARRAY_OF_OBJECT|MAY_BE_ARRAY_OF_RESOURCE|MAY_BE_ARRAY_OF_ARRAY))) {
4996
1.35k
      switch (opline->opcode) {
4997
204
        case ZEND_ASSIGN:
4998
204
        case ZEND_FE_FETCH_R:
4999
204
        case ZEND_FE_FETCH_RW:
5000
204
          break;
5001
1.14k
        default:
5002
          /* destructor may be called */
5003
1.14k
          return 1;
5004
1.35k
      }
5005
1.35k
    }
5006
6.25k
  }
5007
5008
38.7k
  switch (opline->opcode) {
5009
3.18k
    case ZEND_NOP:
5010
5.31k
    case ZEND_QM_ASSIGN:
5011
5.31k
    case ZEND_JMP:
5012
5.35k
    case ZEND_CHECK_VAR:
5013
5.35k
    case ZEND_MAKE_REF:
5014
5.35k
    case ZEND_BEGIN_SILENCE:
5015
5.35k
    case ZEND_END_SILENCE:
5016
14.9k
    case ZEND_FREE:
5017
15.6k
    case ZEND_FE_FREE:
5018
15.6k
    case ZEND_SEPARATE:
5019
15.8k
    case ZEND_TYPE_CHECK:
5020
15.8k
    case ZEND_DEFINED:
5021
15.8k
    case ZEND_ISSET_ISEMPTY_THIS:
5022
15.8k
    case ZEND_COALESCE:
5023
15.8k
    case ZEND_SWITCH_LONG:
5024
15.8k
    case ZEND_SWITCH_STRING:
5025
15.8k
    case ZEND_MATCH:
5026
15.8k
    case ZEND_ISSET_ISEMPTY_VAR:
5027
15.8k
    case ZEND_ISSET_ISEMPTY_CV:
5028
15.8k
    case ZEND_FUNC_NUM_ARGS:
5029
15.8k
    case ZEND_FUNC_GET_ARGS:
5030
16.9k
    case ZEND_COPY_TMP:
5031
16.9k
    case ZEND_JMP_NULL:
5032
16.9k
    case ZEND_JMP_FRAMELESS:
5033
16.9k
      return 0;
5034
99
    case ZEND_IS_IDENTICAL:
5035
126
    case ZEND_IS_NOT_IDENTICAL:
5036
126
    case ZEND_CASE_STRICT:
5037
      /* Array to array comparison may lead to recursion. */
5038
126
      return (t1 & t2) & MAY_BE_ARRAY_OF_ARRAY;
5039
0
    case ZEND_SEND_VAR:
5040
0
    case ZEND_SEND_VAL:
5041
0
    case ZEND_SEND_REF:
5042
0
    case ZEND_SEND_VAR_EX:
5043
0
    case ZEND_SEND_FUNC_ARG:
5044
0
    case ZEND_CHECK_FUNC_ARG:
5045
      /* May throw for named params. */
5046
0
      return opline->op2_type == IS_CONST;
5047
0
    case ZEND_INIT_FCALL:
5048
      /* can't throw, because call is resolved at compile time */
5049
0
      return 0;
5050
0
    case ZEND_BIND_GLOBAL:
5051
0
      if ((opline+1)->opcode == ZEND_BIND_GLOBAL) {
5052
0
        return zend_may_throw(opline + 1, ssa_op ? ssa_op + 1 : NULL, op_array, ssa);
5053
0
      }
5054
0
      return 0;
5055
942
    case ZEND_ADD:
5056
942
      if ((t1 & MAY_BE_ANY) == MAY_BE_ARRAY
5057
2
       && (t2 & MAY_BE_ANY) == MAY_BE_ARRAY) {
5058
2
        return 0;
5059
2
      }
5060
940
      return (t1 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5061
874
        (t2 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5062
57
    case ZEND_DIV:
5063
57
      if (!OP2_HAS_RANGE() ||
5064
51
        (OP2_MIN_RANGE() <= 0 && OP2_MAX_RANGE() >= 0)) {
5065
        /* Division by zero */
5066
51
        return 1;
5067
51
      }
5068
6
      ZEND_FALLTHROUGH;
5069
526
    case ZEND_SUB:
5070
1.06k
    case ZEND_MUL:
5071
1.09k
    case ZEND_POW:
5072
1.09k
      return (t1 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5073
829
        (t2 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5074
    /* Ops may throw if not an integer */
5075
171
    case ZEND_MOD:
5076
171
      if (!OP2_HAS_RANGE() ||
5077
127
        (OP2_MIN_RANGE() <= 0 && OP2_MAX_RANGE() >= 0)) {
5078
        /* Division by zero */
5079
88
        return 1;
5080
88
      }
5081
83
      ZEND_FALLTHROUGH;
5082
287
    case ZEND_SL:
5083
434
    case ZEND_SR:
5084
434
      return (t1 & (MAY_BE_STRING|MAY_BE_DOUBLE|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5085
205
        (t2 & (MAY_BE_STRING|MAY_BE_DOUBLE|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5086
123
        !OP2_HAS_RANGE() ||
5087
123
        OP2_MIN_RANGE() < 0;
5088
515
    case ZEND_CONCAT:
5089
711
    case ZEND_FAST_CONCAT:
5090
711
      return (t1 & (MAY_BE_ARRAY|MAY_BE_OBJECT)) ||
5091
709
        (t2 & (MAY_BE_ARRAY|MAY_BE_OBJECT));
5092
106
    case ZEND_BW_OR:
5093
240
    case ZEND_BW_AND:
5094
536
    case ZEND_BW_XOR:
5095
536
      if ((t1 & MAY_BE_ANY) == MAY_BE_STRING
5096
15
       && (t2 & MAY_BE_ANY) == MAY_BE_STRING) {
5097
4
        return 0;
5098
4
      }
5099
532
      return (t1 & (MAY_BE_STRING|MAY_BE_DOUBLE|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5100
466
        (t2 & (MAY_BE_STRING|MAY_BE_DOUBLE|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5101
756
    case ZEND_BW_NOT:
5102
756
      return (t1 & (MAY_BE_NULL|MAY_BE_FALSE|MAY_BE_TRUE|MAY_BE_DOUBLE|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5103
1.53k
    case ZEND_PRE_INC:
5104
1.78k
    case ZEND_POST_INC:
5105
1.78k
      return (t1 & (MAY_BE_FALSE|MAY_BE_TRUE|MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5106
    /* null emits a warning as it has no effect compared to ++ which converts the value to 1 */
5107
141
    case ZEND_PRE_DEC:
5108
245
    case ZEND_POST_DEC:
5109
245
      return (t1 & (MAY_BE_NULL|MAY_BE_FALSE|MAY_BE_TRUE|MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5110
0
    case ZEND_JMPZ:
5111
0
    case ZEND_JMPNZ:
5112
0
    case ZEND_JMPZ_EX:
5113
0
    case ZEND_JMPNZ_EX:
5114
0
    case ZEND_JMP_SET:
5115
0
      return (t1 & MAY_BE_OBJECT);
5116
332
    case ZEND_BOOL:
5117
563
    case ZEND_BOOL_NOT:
5118
      /* NAN Cast to bool will warn, but if we have a range it is fine */
5119
563
      return (t1 & MAY_BE_OBJECT) || ((t1 & MAY_BE_DOUBLE) && !OP1_HAS_RANGE());
5120
272
    case ZEND_BOOL_XOR:
5121
272
      return (t1 & MAY_BE_OBJECT) || (t2 & MAY_BE_OBJECT);
5122
164
    case ZEND_IS_EQUAL:
5123
370
    case ZEND_IS_NOT_EQUAL:
5124
635
    case ZEND_IS_SMALLER:
5125
688
    case ZEND_IS_SMALLER_OR_EQUAL:
5126
698
    case ZEND_CASE:
5127
698
    case ZEND_SPACESHIP:
5128
698
      if ((t1 & MAY_BE_ANY) == MAY_BE_NULL
5129
646
       || (t2 & MAY_BE_ANY) == MAY_BE_NULL) {
5130
91
        return 0;
5131
91
      }
5132
607
      return (t1 & (MAY_BE_OBJECT|MAY_BE_ARRAY_OF_ARRAY|MAY_BE_ARRAY_OF_OBJECT)) || (t2 & (MAY_BE_OBJECT|MAY_BE_ARRAY_OF_ARRAY|MAY_BE_ARRAY_OF_OBJECT));
5133
10
    case ZEND_ASSIGN_OP:
5134
10
      if (opline->extended_value == ZEND_ADD) {
5135
10
        if ((t1 & MAY_BE_ANY) == MAY_BE_ARRAY
5136
0
         && (t2 & MAY_BE_ANY) == MAY_BE_ARRAY) {
5137
0
          return 0;
5138
0
        }
5139
10
        return (t1 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5140
6
          (t2 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5141
10
      } else if (opline->extended_value == ZEND_DIV ||
5142
0
        opline->extended_value == ZEND_MOD) {
5143
0
        if (!OP2_HAS_RANGE() ||
5144
0
          (OP2_MIN_RANGE() <= 0 && OP2_MAX_RANGE() >= 0)) {
5145
          /* Division by zero */
5146
0
          return 1;
5147
0
        }
5148
0
        return (t1 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5149
0
          (t2 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5150
0
      } else if (opline->extended_value == ZEND_SUB ||
5151
0
        opline->extended_value == ZEND_MUL ||
5152
0
        opline->extended_value == ZEND_POW) {
5153
0
        return (t1 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5154
0
          (t2 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5155
0
      } else if (opline->extended_value == ZEND_SL ||
5156
0
        opline->extended_value == ZEND_SR) {
5157
0
        return (t1 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5158
0
          (t2 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5159
0
          !OP2_HAS_RANGE() ||
5160
0
          OP2_MIN_RANGE() < 0;
5161
0
      } else if (opline->extended_value == ZEND_CONCAT) {
5162
0
        return (t1 & (MAY_BE_ARRAY|MAY_BE_OBJECT)) ||
5163
0
          (t2 & (MAY_BE_ARRAY|MAY_BE_OBJECT));
5164
0
      } else if (opline->extended_value == ZEND_BW_OR ||
5165
0
        opline->extended_value == ZEND_BW_AND ||
5166
0
        opline->extended_value == ZEND_BW_XOR) {
5167
0
        if ((t1 & MAY_BE_ANY) == MAY_BE_STRING
5168
0
         && (t2 & MAY_BE_ANY) == MAY_BE_STRING) {
5169
0
          return 0;
5170
0
        }
5171
0
        return (t1 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE)) ||
5172
0
          (t2 & (MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5173
0
      }
5174
0
      return 1;
5175
10.7k
    case ZEND_ASSIGN:
5176
10.7k
      if (t1 & MAY_BE_REF) {
5177
0
        return 1;
5178
0
      }
5179
10.7k
      ZEND_FALLTHROUGH;
5180
10.7k
    case ZEND_UNSET_VAR:
5181
10.7k
      return (t1 & (MAY_BE_OBJECT|MAY_BE_RESOURCE|MAY_BE_ARRAY_OF_OBJECT|MAY_BE_ARRAY_OF_RESOURCE|MAY_BE_ARRAY_OF_ARRAY));
5182
108
    case ZEND_BIND_STATIC:
5183
108
    case ZEND_BIND_INIT_STATIC_OR_JMP:
5184
108
      if (t1 & (MAY_BE_OBJECT|MAY_BE_RESOURCE|MAY_BE_ARRAY_OF_OBJECT|MAY_BE_ARRAY_OF_RESOURCE|MAY_BE_ARRAY_OF_ARRAY)) {
5185
        /* Destructor may throw. */
5186
48
        return 1;
5187
48
      }
5188
60
      return 0;
5189
221
    case ZEND_ASSIGN_DIM:
5190
221
      if ((opline+1)->op1_type == IS_CV) {
5191
83
        if (OP1_DATA_INFO() & MAY_BE_UNDEF) {
5192
77
          return 1;
5193
77
        }
5194
83
      }
5195
144
      if (t1 & (MAY_BE_ARRAY_OF_OBJECT|MAY_BE_ARRAY_OF_RESOURCE|MAY_BE_ARRAY_OF_ARRAY|MAY_BE_ARRAY_OF_REF)) {
5196
14
        return 1;
5197
14
      }
5198
130
      return (t1 & (MAY_BE_OBJECT|MAY_BE_RESOURCE|MAY_BE_TRUE|MAY_BE_FALSE|MAY_BE_STRING|MAY_BE_LONG|MAY_BE_DOUBLE)) || opline->op2_type == IS_UNUSED ||
5199
57
        (t2 & (MAY_BE_UNDEF|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5200
71
    case ZEND_ASSIGN_OBJ:
5201
71
      if (t1 & (MAY_BE_ANY-MAY_BE_OBJECT)) {
5202
0
        return 1;
5203
0
      }
5204
71
      if ((opline+1)->op1_type == IS_CV) {
5205
25
        if (OP1_DATA_INFO() & MAY_BE_UNDEF) {
5206
17
          return 1;
5207
17
        }
5208
25
      }
5209
54
      if (ssa_op->op1_use) {
5210
54
        const zend_ssa_var_info *var_info = ssa->var_info + ssa_op->op1_use;
5211
54
        const zend_class_entry *ce = var_info->ce;
5212
5213
54
        if (var_info->is_instanceof
5214
54
         || !ce
5215
40
         || ce->create_object
5216
40
         || ce->default_object_handlers->write_property != zend_std_write_property
5217
40
         || ce->default_object_handlers->get_property_ptr_ptr != zend_std_get_property_ptr_ptr
5218
40
         || ce->__get
5219
40
         || ce->__set
5220
40
         || ce->parent) {
5221
14
          return 1;
5222
14
        }
5223
5224
40
        if (opline->op2_type != IS_CONST) {
5225
0
          return 1;
5226
0
        }
5227
5228
40
        zend_string *prop_name = Z_STR_P(CRT_CONSTANT(opline->op2));
5229
40
        if (ZSTR_LEN(prop_name) > 0 && ZSTR_VAL(prop_name)[0] == '\0') {
5230
0
          return 1;
5231
0
        }
5232
5233
40
        zend_property_info *prop_info =
5234
40
          zend_hash_find_ptr(&ce->properties_info, prop_name);
5235
40
        if (prop_info) {
5236
11
          if (ZEND_TYPE_IS_SET(prop_info->type)) {
5237
4
            return 1;
5238
4
          }
5239
7
          return !(prop_info->flags & ZEND_ACC_PUBLIC)
5240
0
            && prop_info->ce != op_array->scope;
5241
29
        } else {
5242
29
          return !(ce->ce_flags & ZEND_ACC_ALLOW_DYNAMIC_PROPERTIES);
5243
29
        }
5244
40
      }
5245
0
      return 1;
5246
0
    case ZEND_ROPE_INIT:
5247
0
    case ZEND_ROPE_ADD:
5248
0
    case ZEND_ROPE_END:
5249
0
      return t2 & (MAY_BE_ARRAY|MAY_BE_OBJECT);
5250
122
    case ZEND_INIT_ARRAY:
5251
122
      return (opline->op2_type != IS_UNUSED) && (t2 & (MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5252
105
    case ZEND_ADD_ARRAY_ELEMENT:
5253
105
      return (opline->op2_type == IS_UNUSED) || (t2 & (MAY_BE_NULL|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5254
1.03k
    case ZEND_STRLEN:
5255
1.03k
      return (t1 & MAY_BE_ANY) != MAY_BE_STRING;
5256
157
    case ZEND_COUNT:
5257
157
      return (t1 & MAY_BE_ANY) != MAY_BE_ARRAY;
5258
0
    case ZEND_RECV_INIT:
5259
0
      if (Z_TYPE_P(CRT_CONSTANT(opline->op2)) == IS_CONSTANT_AST) {
5260
0
        return 1;
5261
0
      }
5262
0
      if (op_array->fn_flags & ZEND_ACC_HAS_TYPE_HINTS) {
5263
0
        uint32_t arg_num = opline->op1.num;
5264
0
        const zend_arg_info *cur_arg_info;
5265
5266
0
        if (EXPECTED(arg_num <= op_array->num_args)) {
5267
0
          cur_arg_info = &op_array->arg_info[arg_num-1];
5268
0
        } else if (UNEXPECTED(op_array->fn_flags & ZEND_ACC_VARIADIC)) {
5269
0
          cur_arg_info = &op_array->arg_info[op_array->num_args];
5270
0
        } else {
5271
0
          return 0;
5272
0
        }
5273
0
        return ZEND_TYPE_IS_SET(cur_arg_info->type);
5274
0
      } else {
5275
0
        return 0;
5276
0
      }
5277
0
    case ZEND_FETCH_IS:
5278
0
      return (t2 & (MAY_BE_ARRAY|MAY_BE_OBJECT));
5279
130
    case ZEND_ISSET_ISEMPTY_DIM_OBJ:
5280
130
      return (t1 & MAY_BE_OBJECT) || (t2 & (MAY_BE_DOUBLE|MAY_BE_ARRAY|MAY_BE_OBJECT));
5281
95
    case ZEND_FETCH_DIM_IS:
5282
95
      return (t1 & MAY_BE_OBJECT) || (t2 & (MAY_BE_DOUBLE|MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE));
5283
131
    case ZEND_CAST:
5284
131
      switch (opline->extended_value) {
5285
55
        case IS_LONG:
5286
55
          return (t1 & (MAY_BE_DOUBLE|MAY_BE_STRING|MAY_BE_OBJECT));
5287
18
        case IS_DOUBLE:
5288
18
          return (t1 & MAY_BE_OBJECT);
5289
58
        case IS_STRING:
5290
58
          return (t1 & (MAY_BE_ARRAY|MAY_BE_OBJECT));
5291
0
        case IS_ARRAY:
5292
0
          return (t1 & MAY_BE_OBJECT);
5293
0
        case IS_OBJECT:
5294
0
          return 0;
5295
0
        default: ZEND_UNREACHABLE();
5296
131
      }
5297
      /* GCC is getting confused here for the Wimplicit-fallthrough warning with
5298
       * default: ZEND_UNREACHABLE(); macro */
5299
0
      return 0;
5300
2
    case ZEND_ARRAY_KEY_EXISTS:
5301
2
      if ((t2 & MAY_BE_ANY) != MAY_BE_ARRAY) {
5302
0
        return 1;
5303
0
      }
5304
2
      if ((t1 & (MAY_BE_ARRAY|MAY_BE_OBJECT|MAY_BE_RESOURCE))) {
5305
0
        return 1;
5306
0
      }
5307
2
      return 0;
5308
4
    case ZEND_FE_RESET_R:
5309
4
    case ZEND_FE_RESET_RW:
5310
4
      if ((t1 & (MAY_BE_ANY|MAY_BE_REF)) != MAY_BE_ARRAY) {
5311
0
        return 1;
5312
0
      }
5313
4
      return 0;
5314
0
    case ZEND_FE_FETCH_R:
5315
0
      if ((t1 & (MAY_BE_ANY|MAY_BE_REF)) != MAY_BE_ARRAY) {
5316
0
        return 1;
5317
0
      }
5318
0
      if (opline->op2_type == IS_CV
5319
0
       && (t2 & MAY_BE_RC1)
5320
0
       && (t2 & (MAY_BE_OBJECT|MAY_BE_RESOURCE|MAY_BE_ARRAY_OF_OBJECT|MAY_BE_ARRAY_OF_RESOURCE|MAY_BE_ARRAY_OF_ARRAY))) {
5321
0
        return 1;
5322
0
      }
5323
0
      return 0;
5324
0
    case ZEND_FETCH_DIM_W:
5325
0
    case ZEND_FETCH_LIST_W:
5326
0
      if (t1 & (MAY_BE_FALSE|MAY_BE_TRUE|MAY_BE_LONG|MAY_BE_DOUBLE|MAY_BE_STRING|MAY_BE_OBJECT|MAY_BE_RESOURCE|MAY_BE_REF)) {
5327
0
        return 1;
5328
0
      }
5329
0
      if (t2 & (MAY_BE_RESOURCE|MAY_BE_ARRAY|MAY_BE_OBJECT)) {
5330
0
        return 1;
5331
0
      }
5332
0
      if (opline->op2_type == IS_UNUSED) {
5333
0
        return 1;
5334
0
      }
5335
0
      return 0;
5336
14
    case ZEND_YIELD_FROM: {
5337
14
      uint32_t t1 = OP1_INFO();
5338
14
      if ((t1 & (MAY_BE_ANY|MAY_BE_UNDEF)) == MAY_BE_ARRAY && MAY_BE_EMPTY_ONLY(t1)) {
5339
14
        return false;
5340
14
      }
5341
0
      return true;
5342
14
    }
5343
596
    default:
5344
596
      return 1;
5345
38.7k
  }
5346
38.7k
}
5347
5348
ZEND_API bool zend_may_throw(const zend_op *opline, const zend_ssa_op *ssa_op, const zend_op_array *op_array, const zend_ssa *ssa)
5349
61.0k
{
5350
61.0k
  return zend_may_throw_ex(opline, ssa_op, op_array, ssa, OP1_INFO(), OP2_INFO());
5351
61.0k
}