2015-11-21 03:06:36 +08:00
|
|
|
/*
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| Zend Engine, CFG - Control Flow Graph |
|
|
|
|
+----------------------------------------------------------------------+
|
2019-02-05 15:33:28 +08:00
|
|
|
| Copyright (c) The PHP Group |
|
2015-11-21 03:06:36 +08:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| This source file is subject to version 3.01 of the PHP license, |
|
|
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
|
|
| available through the world-wide-web at the following url: |
|
|
|
|
| http://www.php.net/license/3_01.txt |
|
|
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
|
|
+----------------------------------------------------------------------+
|
2018-11-02 00:30:28 +08:00
|
|
|
| Authors: Dmitry Stogov <dmitry@php.net> |
|
2015-11-21 03:06:36 +08:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef ZEND_CFG_H
|
|
|
|
#define ZEND_CFG_H
|
|
|
|
|
|
|
|
/* zend_basic_bloc.flags */
|
|
|
|
#define ZEND_BB_START (1<<0) /* fist block */
|
|
|
|
#define ZEND_BB_FOLLOW (1<<1) /* follows the next block */
|
|
|
|
#define ZEND_BB_TARGET (1<<2) /* jump taget */
|
|
|
|
#define ZEND_BB_EXIT (1<<3) /* without successors */
|
|
|
|
#define ZEND_BB_ENTRY (1<<4) /* stackless entry */
|
|
|
|
#define ZEND_BB_TRY (1<<5) /* start of try block */
|
|
|
|
#define ZEND_BB_CATCH (1<<6) /* start of catch block */
|
|
|
|
#define ZEND_BB_FINALLY (1<<7) /* start of finally block */
|
|
|
|
#define ZEND_BB_FINALLY_END (1<<8) /* end of finally block */
|
2016-05-19 00:03:05 +08:00
|
|
|
#define ZEND_BB_UNREACHABLE_FREE (1<<11) /* unreachable loop free */
|
2016-08-30 01:35:17 +08:00
|
|
|
#define ZEND_BB_RECV_ENTRY (1<<12) /* RECV entry */
|
2015-11-21 03:06:36 +08:00
|
|
|
|
|
|
|
#define ZEND_BB_LOOP_HEADER (1<<16)
|
|
|
|
#define ZEND_BB_IRREDUCIBLE_LOOP (1<<17)
|
|
|
|
|
2019-03-06 05:25:21 +08:00
|
|
|
#define ZEND_BB_REACHABLE (1U<<31)
|
2015-11-21 03:06:36 +08:00
|
|
|
|
2019-01-17 23:07:17 +08:00
|
|
|
#define ZEND_BB_PROTECTED (ZEND_BB_ENTRY|ZEND_BB_RECV_ENTRY|ZEND_BB_TRY|ZEND_BB_CATCH|ZEND_BB_FINALLY|ZEND_BB_FINALLY_END|ZEND_BB_UNREACHABLE_FREE)
|
2015-11-21 03:06:36 +08:00
|
|
|
|
|
|
|
typedef struct _zend_basic_block {
|
2017-03-18 04:37:57 +08:00
|
|
|
int *successors; /* successor block indices */
|
2015-11-21 03:06:36 +08:00
|
|
|
uint32_t flags;
|
|
|
|
uint32_t start; /* first opcode number */
|
2016-05-19 00:03:05 +08:00
|
|
|
uint32_t len; /* number of opcodes */
|
2017-03-18 04:37:57 +08:00
|
|
|
int successors_count; /* number of successors */
|
2015-11-21 03:06:36 +08:00
|
|
|
int predecessors_count; /* number of predecessors */
|
|
|
|
int predecessor_offset; /* offset of 1-st predecessor */
|
|
|
|
int idom; /* immediate dominator block */
|
|
|
|
int loop_header; /* closest loop header, or -1 */
|
|
|
|
int level; /* steps away from the entry in the dom. tree */
|
|
|
|
int children; /* list of dominated blocks */
|
|
|
|
int next_child; /* next dominated block */
|
2017-03-18 04:37:57 +08:00
|
|
|
int successors_storage[2]; /* up to 2 successor blocks */
|
2015-11-21 03:06:36 +08:00
|
|
|
} zend_basic_block;
|
|
|
|
|
|
|
|
/*
|
|
|
|
+------------+---+---+---+---+---+
|
|
|
|
| |OP1|OP2|EXT| 0 | 1 |
|
|
|
|
+------------+---+---+---+---+---+
|
|
|
|
|JMP |ADR| | |OP1| - |
|
|
|
|
|JMPZ | |ADR| |OP2|FOL|
|
|
|
|
|JMPNZ | |ADR| |OP2|FOL|
|
|
|
|
|JMPZNZ | |ADR|ADR|OP2|EXT|
|
|
|
|
|JMPZ_EX | |ADR| |OP2|FOL|
|
|
|
|
|JMPNZ_EX | |ADR| |OP2|FOL|
|
|
|
|
|JMP_SET | |ADR| |OP2|FOL|
|
|
|
|
|COALESCE | |ADR| |OP2|FOL|
|
|
|
|
|ASSERT_CHK | |ADR| |OP2|FOL|
|
|
|
|
|NEW | |ADR| |OP2|FOL|
|
|
|
|
|DCL_ANON* |ADR| | |OP1|FOL|
|
|
|
|
|FE_RESET_* | |ADR| |OP2|FOL|
|
|
|
|
|FE_FETCH_* | | |ADR|EXT|FOL|
|
|
|
|
|CATCH | | |ADR|EXT|FOL|
|
|
|
|
|FAST_CALL |ADR| | |OP1|FOL|
|
|
|
|
|FAST_RET | | | | - | - |
|
|
|
|
|RETURN* | | | | - | - |
|
|
|
|
|EXIT | | | | - | - |
|
|
|
|
|THROW | | | | - | - |
|
|
|
|
|* | | | |FOL| - |
|
|
|
|
+------------+---+---+---+---+---+
|
|
|
|
*/
|
|
|
|
|
2015-11-21 06:19:45 +08:00
|
|
|
typedef struct _zend_cfg {
|
2015-11-21 03:06:36 +08:00
|
|
|
int blocks_count; /* number of basic blocks */
|
2017-07-21 04:29:22 +08:00
|
|
|
int edges_count; /* number of edges */
|
2015-11-21 03:06:36 +08:00
|
|
|
zend_basic_block *blocks; /* array of basic blocks */
|
|
|
|
int *predecessors;
|
|
|
|
uint32_t *map;
|
2017-10-18 17:03:07 +08:00
|
|
|
uint32_t flags;
|
2015-11-21 03:06:36 +08:00
|
|
|
} zend_cfg;
|
|
|
|
|
2015-12-16 05:49:44 +08:00
|
|
|
/* Build Flags */
|
2019-03-06 05:25:21 +08:00
|
|
|
#define ZEND_RT_CONSTANTS (1U<<31)
|
2015-12-16 05:49:44 +08:00
|
|
|
#define ZEND_CFG_STACKLESS (1<<30)
|
|
|
|
#define ZEND_SSA_DEBUG_LIVENESS (1<<29)
|
|
|
|
#define ZEND_SSA_DEBUG_PHI_PLACEMENT (1<<28)
|
2015-12-22 21:43:37 +08:00
|
|
|
#define ZEND_SSA_RC_INFERENCE (1<<27)
|
2016-05-19 05:46:20 +08:00
|
|
|
#define ZEND_CFG_NO_ENTRY_PREDECESSORS (1<<25)
|
2016-08-30 01:35:17 +08:00
|
|
|
#define ZEND_CFG_RECV_ENTRY (1<<24)
|
2016-10-25 22:20:27 +08:00
|
|
|
#define ZEND_CALL_TREE (1<<23)
|
2017-03-31 01:14:01 +08:00
|
|
|
#define ZEND_SSA_USE_CV_RESULTS (1<<22)
|
2015-12-16 05:49:44 +08:00
|
|
|
|
2017-10-04 21:53:01 +08:00
|
|
|
#define CRT_CONSTANT_EX(op_array, opline, node, rt_constants) \
|
2015-12-16 05:49:44 +08:00
|
|
|
((rt_constants) ? \
|
2017-10-04 21:53:01 +08:00
|
|
|
RT_CONSTANT(opline, (node)) \
|
2015-12-11 22:24:55 +08:00
|
|
|
: \
|
|
|
|
CT_CONSTANT_EX(op_array, (node).constant) \
|
|
|
|
)
|
|
|
|
|
2015-12-16 05:49:44 +08:00
|
|
|
#define CRT_CONSTANT(node) \
|
2017-10-04 21:53:01 +08:00
|
|
|
CRT_CONSTANT_EX(op_array, opline, node, (build_flags & ZEND_RT_CONSTANTS))
|
2015-12-16 05:49:44 +08:00
|
|
|
|
|
|
|
#define RETURN_VALUE_USED(opline) \
|
2016-02-05 22:23:23 +08:00
|
|
|
((opline)->result_type != IS_UNUSED)
|
2015-12-16 05:49:44 +08:00
|
|
|
|
2015-11-21 03:06:36 +08:00
|
|
|
BEGIN_EXTERN_C()
|
|
|
|
|
2017-10-18 17:03:07 +08:00
|
|
|
int zend_build_cfg(zend_arena **arena, const zend_op_array *op_array, uint32_t build_flags, zend_cfg *cfg);
|
2015-12-16 05:49:44 +08:00
|
|
|
void zend_cfg_remark_reachable_blocks(const zend_op_array *op_array, zend_cfg *cfg);
|
2015-12-11 22:24:55 +08:00
|
|
|
int zend_cfg_build_predecessors(zend_arena **arena, zend_cfg *cfg);
|
2015-12-16 05:49:44 +08:00
|
|
|
int zend_cfg_compute_dominators_tree(const zend_op_array *op_array, zend_cfg *cfg);
|
2017-10-18 17:03:07 +08:00
|
|
|
int zend_cfg_identify_loops(const zend_op_array *op_array, zend_cfg *cfg);
|
2015-11-21 03:06:36 +08:00
|
|
|
|
|
|
|
END_EXTERN_C()
|
|
|
|
|
|
|
|
#endif /* ZEND_CFG_H */
|