/src/php-src/Zend/Optimizer/zend_call_graph.c
Line | Count | Source |
1 | | /* |
2 | | +----------------------------------------------------------------------+ |
3 | | | Zend Engine, Call Graph | |
4 | | +----------------------------------------------------------------------+ |
5 | | | Copyright (c) The PHP Group | |
6 | | +----------------------------------------------------------------------+ |
7 | | | This source file is subject to version 3.01 of the PHP license, | |
8 | | | that is bundled with this package in the file LICENSE, and is | |
9 | | | available through the world-wide-web at the following url: | |
10 | | | https://www.php.net/license/3_01.txt | |
11 | | | If you did not receive a copy of the PHP license and are unable to | |
12 | | | obtain it through the world-wide-web, please send a note to | |
13 | | | license@php.net so we can mail you a copy immediately. | |
14 | | +----------------------------------------------------------------------+ |
15 | | | Authors: Dmitry Stogov <dmitry@php.net> | |
16 | | +----------------------------------------------------------------------+ |
17 | | */ |
18 | | |
19 | | #include "zend_compile.h" |
20 | | #include "zend_extensions.h" |
21 | | #include "Optimizer/zend_optimizer.h" |
22 | | #include "zend_optimizer_internal.h" |
23 | | #include "zend_inference.h" |
24 | | #include "zend_call_graph.h" |
25 | | #include "zend_func_info.h" |
26 | | |
27 | | static void zend_op_array_calc(zend_op_array *op_array, void *context) |
28 | 83.6k | { |
29 | 83.6k | zend_call_graph *call_graph = context; |
30 | 83.6k | call_graph->op_arrays_count++; |
31 | 83.6k | } |
32 | | |
33 | | static void zend_op_array_collect(zend_op_array *op_array, void *context) |
34 | 83.6k | { |
35 | 83.6k | zend_call_graph *call_graph = context; |
36 | 83.6k | zend_func_info *func_info = call_graph->func_infos + call_graph->op_arrays_count; |
37 | | |
38 | 83.6k | ZEND_SET_FUNC_INFO(op_array, func_info); |
39 | 83.6k | call_graph->op_arrays[call_graph->op_arrays_count] = op_array; |
40 | 83.6k | func_info->num = call_graph->op_arrays_count; |
41 | 83.6k | call_graph->op_arrays_count++; |
42 | 83.6k | } |
43 | | |
44 | | ZEND_API void zend_analyze_calls(zend_arena **arena, zend_script *script, uint32_t build_flags, zend_op_array *op_array, zend_func_info *func_info) |
45 | 83.6k | { |
46 | 83.6k | zend_op *opline = op_array->opcodes; |
47 | 83.6k | zend_op *end = opline + op_array->last; |
48 | 83.6k | zend_function *func; |
49 | 83.6k | zend_call_info *call_info; |
50 | 83.6k | int call = 0; |
51 | 83.6k | zend_call_info **call_stack; |
52 | 83.6k | ALLOCA_FLAG(use_heap); |
53 | 83.6k | bool is_prototype; |
54 | | |
55 | 83.6k | call_stack = do_alloca((op_array->last / 2) * sizeof(zend_call_info*), use_heap); |
56 | 83.6k | call_info = NULL; |
57 | 2.17M | while (opline != end) { |
58 | 2.08M | switch (opline->opcode) { |
59 | 93.9k | case ZEND_INIT_FCALL: |
60 | 128k | case ZEND_INIT_METHOD_CALL: |
61 | 134k | case ZEND_INIT_STATIC_METHOD_CALL: |
62 | 134k | case ZEND_INIT_PARENT_PROPERTY_HOOK_CALL: |
63 | 134k | call_stack[call] = call_info; |
64 | 134k | func = zend_optimizer_get_called_func( |
65 | 134k | script, op_array, opline, &is_prototype); |
66 | 134k | if (func) { |
67 | 97.4k | call_info = zend_arena_calloc(arena, 1, sizeof(zend_call_info) + (sizeof(zend_send_arg_info) * ((int)opline->extended_value - 1))); |
68 | 97.4k | call_info->caller_op_array = op_array; |
69 | 97.4k | call_info->caller_init_opline = opline; |
70 | 97.4k | call_info->caller_call_opline = NULL; |
71 | 97.4k | call_info->callee_func = func; |
72 | 97.4k | call_info->num_args = opline->extended_value; |
73 | 97.4k | call_info->next_callee = func_info->callee_info; |
74 | 97.4k | call_info->is_prototype = is_prototype; |
75 | 97.4k | call_info->is_frameless = false; |
76 | 97.4k | func_info->callee_info = call_info; |
77 | | |
78 | 97.4k | if (build_flags & ZEND_CALL_TREE) { |
79 | 0 | call_info->next_caller = NULL; |
80 | 97.4k | } else if (func->type == ZEND_INTERNAL_FUNCTION |
81 | 77.7k | || func->op_array.filename != script->filename) { |
82 | 77.7k | call_info->next_caller = NULL; |
83 | 77.7k | } else { |
84 | 19.7k | zend_func_info *callee_func_info = ZEND_FUNC_INFO(&func->op_array); |
85 | 19.7k | if (callee_func_info) { |
86 | 19.7k | call_info->next_caller = callee_func_info->caller_info; |
87 | 19.7k | callee_func_info->caller_info = call_info; |
88 | 19.7k | } else { |
89 | 16 | call_info->next_caller = NULL; |
90 | 16 | } |
91 | 19.7k | } |
92 | 97.4k | } else { |
93 | 37.1k | call_info = NULL; |
94 | 37.1k | } |
95 | 134k | call++; |
96 | 134k | break; |
97 | 4.17k | case ZEND_INIT_FCALL_BY_NAME: |
98 | 6.19k | case ZEND_INIT_NS_FCALL_BY_NAME: |
99 | 10.3k | case ZEND_INIT_DYNAMIC_CALL: |
100 | 58.2k | case ZEND_NEW: |
101 | 58.9k | case ZEND_INIT_USER_CALL: |
102 | 58.9k | call_stack[call] = call_info; |
103 | 58.9k | call_info = NULL; |
104 | 58.9k | call++; |
105 | 58.9k | break; |
106 | 0 | case ZEND_FRAMELESS_ICALL_0: |
107 | 0 | case ZEND_FRAMELESS_ICALL_1: |
108 | 0 | case ZEND_FRAMELESS_ICALL_2: |
109 | 0 | case ZEND_FRAMELESS_ICALL_3: { |
110 | 0 | func = ZEND_FLF_FUNC(opline); |
111 | 0 | zend_call_info *call_info = zend_arena_calloc(arena, 1, sizeof(zend_call_info)); |
112 | 0 | call_info->caller_op_array = op_array; |
113 | 0 | call_info->caller_init_opline = opline; |
114 | 0 | call_info->caller_call_opline = NULL; |
115 | 0 | call_info->callee_func = func; |
116 | 0 | call_info->num_args = ZEND_FLF_NUM_ARGS(opline->opcode); |
117 | 0 | call_info->next_callee = func_info->callee_info; |
118 | 0 | call_info->is_prototype = false; |
119 | 0 | call_info->is_frameless = true; |
120 | 0 | call_info->next_caller = NULL; |
121 | 0 | func_info->callee_info = call_info; |
122 | 0 | break; |
123 | 0 | } |
124 | 184k | case ZEND_DO_FCALL: |
125 | 184k | case ZEND_DO_ICALL: |
126 | 192k | case ZEND_DO_UCALL: |
127 | 193k | case ZEND_DO_FCALL_BY_NAME: |
128 | 193k | case ZEND_CALLABLE_CONVERT: |
129 | 193k | func_info->flags |= ZEND_FUNC_HAS_CALLS; |
130 | 193k | if (call_info) { |
131 | 97.4k | call_info->caller_call_opline = opline; |
132 | 97.4k | } |
133 | 193k | call--; |
134 | 193k | call_info = call_stack[call]; |
135 | 193k | break; |
136 | 93.9k | case ZEND_SEND_VAL: |
137 | 151k | case ZEND_SEND_VAR: |
138 | 164k | case ZEND_SEND_VAL_EX: |
139 | 174k | case ZEND_SEND_VAR_EX: |
140 | 174k | case ZEND_SEND_FUNC_ARG: |
141 | 176k | case ZEND_SEND_REF: |
142 | 177k | case ZEND_SEND_VAR_NO_REF: |
143 | 178k | case ZEND_SEND_VAR_NO_REF_EX: |
144 | 181k | case ZEND_SEND_USER: |
145 | 181k | if (call_info) { |
146 | 104k | if (opline->op2_type == IS_CONST) { |
147 | 1.12k | call_info->named_args = true; |
148 | 1.12k | break; |
149 | 1.12k | } |
150 | | |
151 | 103k | uint32_t num = opline->op2.num; |
152 | 103k | if (num > 0) { |
153 | 103k | num--; |
154 | 103k | } |
155 | 103k | call_info->arg_info[num].opline = opline; |
156 | 103k | } |
157 | 180k | break; |
158 | 180k | case ZEND_SEND_ARRAY: |
159 | 1.11k | case ZEND_SEND_UNPACK: |
160 | 1.11k | if (call_info) { |
161 | 708 | call_info->send_unpack = true; |
162 | 708 | } |
163 | 1.11k | break; |
164 | 2.08M | } |
165 | 2.08M | opline++; |
166 | 2.08M | } |
167 | 83.6k | free_alloca(call_stack, use_heap); |
168 | 83.6k | } |
169 | | |
170 | | static bool zend_is_indirectly_recursive(const zend_op_array *root, const zend_op_array *op_array, zend_bitset visited) |
171 | 21.6k | { |
172 | 21.6k | const zend_func_info *func_info; |
173 | 21.6k | zend_call_info *call_info; |
174 | 21.6k | bool ret = false; |
175 | | |
176 | 21.6k | if (op_array == root) { |
177 | 0 | return true; |
178 | 0 | } |
179 | | |
180 | 21.6k | func_info = ZEND_FUNC_INFO(op_array); |
181 | 21.6k | if (zend_bitset_in(visited, func_info->num)) { |
182 | 1.15k | return false; |
183 | 1.15k | } |
184 | 20.4k | zend_bitset_incl(visited, func_info->num); |
185 | 20.4k | call_info = func_info->caller_info; |
186 | 22.8k | while (call_info) { |
187 | 2.38k | if (zend_is_indirectly_recursive(root, call_info->caller_op_array, visited)) { |
188 | 0 | call_info->recursive = true; |
189 | 0 | ret = true; |
190 | 0 | } |
191 | 2.38k | call_info = call_info->next_caller; |
192 | 2.38k | } |
193 | 20.4k | return ret; |
194 | 21.6k | } |
195 | | |
196 | | static void zend_analyze_recursion(zend_call_graph *call_graph) |
197 | 45.5k | { |
198 | 45.5k | const zend_op_array *op_array; |
199 | 45.5k | zend_func_info *func_info; |
200 | 45.5k | zend_call_info *call_info; |
201 | 45.5k | uint32_t set_len = zend_bitset_len(call_graph->op_arrays_count); |
202 | 45.5k | zend_bitset visited; |
203 | 45.5k | ALLOCA_FLAG(use_heap); |
204 | | |
205 | 45.5k | visited = ZEND_BITSET_ALLOCA(set_len, use_heap); |
206 | 129k | for (uint32_t i = 0; i < call_graph->op_arrays_count; i++) { |
207 | 83.6k | op_array = call_graph->op_arrays[i]; |
208 | 83.6k | func_info = call_graph->func_infos + i; |
209 | 83.6k | call_info = func_info->caller_info; |
210 | 103k | for (; call_info; call_info = call_info->next_caller) { |
211 | 19.7k | if (call_info->is_prototype) { |
212 | | /* Might be calling an overridden child method and not actually recursive. */ |
213 | 337 | continue; |
214 | 337 | } |
215 | 19.3k | if (call_info->caller_op_array == op_array) { |
216 | 160 | call_info->recursive = true; |
217 | 160 | func_info->flags |= ZEND_FUNC_RECURSIVE | ZEND_FUNC_RECURSIVE_DIRECTLY; |
218 | 19.2k | } else { |
219 | 19.2k | memset(visited, 0, sizeof(zend_ulong) * set_len); |
220 | 19.2k | if (zend_is_indirectly_recursive(op_array, call_info->caller_op_array, visited)) { |
221 | 0 | call_info->recursive = true; |
222 | 0 | func_info->flags |= ZEND_FUNC_RECURSIVE | ZEND_FUNC_RECURSIVE_INDIRECTLY; |
223 | 0 | } |
224 | 19.2k | } |
225 | 19.3k | } |
226 | 83.6k | } |
227 | | |
228 | 45.5k | free_alloca(visited, use_heap); |
229 | 45.5k | } |
230 | | |
231 | | static void zend_sort_op_arrays(zend_call_graph *call_graph) |
232 | 45.5k | { |
233 | 45.5k | (void) call_graph; |
234 | | |
235 | | // TODO: perform topological sort of cyclic call graph |
236 | 45.5k | } |
237 | | |
238 | | ZEND_API void zend_build_call_graph(zend_arena **arena, zend_script *script, zend_call_graph *call_graph) /* {{{ */ |
239 | 45.5k | { |
240 | 45.5k | call_graph->op_arrays_count = 0; |
241 | 45.5k | zend_foreach_op_array(script, zend_op_array_calc, call_graph); |
242 | | |
243 | 45.5k | call_graph->op_arrays = (zend_op_array**)zend_arena_calloc(arena, call_graph->op_arrays_count, sizeof(zend_op_array*)); |
244 | 45.5k | call_graph->func_infos = (zend_func_info*)zend_arena_calloc(arena, call_graph->op_arrays_count, sizeof(zend_func_info)); |
245 | 45.5k | call_graph->op_arrays_count = 0; |
246 | 45.5k | zend_foreach_op_array(script, zend_op_array_collect, call_graph); |
247 | 45.5k | } |
248 | | /* }}} */ |
249 | | |
250 | | ZEND_API void zend_analyze_call_graph(zend_arena **arena, zend_script *script, zend_call_graph *call_graph) /* {{{ */ |
251 | 45.5k | { |
252 | 129k | for (uint32_t i = 0; i < call_graph->op_arrays_count; i++) { |
253 | 83.6k | zend_analyze_calls(arena, script, 0, call_graph->op_arrays[i], call_graph->func_infos + i); |
254 | 83.6k | } |
255 | 45.5k | zend_analyze_recursion(call_graph); |
256 | 45.5k | zend_sort_op_arrays(call_graph); |
257 | 45.5k | } |
258 | | /* }}} */ |
259 | | |
260 | | ZEND_API zend_call_info **zend_build_call_map(zend_arena **arena, const zend_func_info *info, const zend_op_array *op_array) /* {{{ */ |
261 | 83.6k | { |
262 | 83.6k | zend_call_info **map, *call; |
263 | 83.6k | if (!info->callee_info) { |
264 | | /* Don't build call map if function contains no calls */ |
265 | 46.4k | return NULL; |
266 | 46.4k | } |
267 | | |
268 | 37.2k | map = zend_arena_calloc(arena, sizeof(zend_call_info *), op_array->last); |
269 | 134k | for (call = info->callee_info; call; call = call->next_callee) { |
270 | 97.4k | map[call->caller_init_opline - op_array->opcodes] = call; |
271 | 97.4k | if (call->caller_call_opline) { |
272 | 97.4k | map[call->caller_call_opline - op_array->opcodes] = call; |
273 | 97.4k | } |
274 | 97.4k | if (!call->is_frameless) { |
275 | 200k | for (uint32_t i = 0; i < call->num_args; i++) { |
276 | 103k | if (call->arg_info[i].opline) { |
277 | 103k | map[call->arg_info[i].opline - op_array->opcodes] = call; |
278 | 103k | } |
279 | 103k | } |
280 | 97.4k | } |
281 | 97.4k | } |
282 | 37.2k | return map; |
283 | 83.6k | } |
284 | | /* }}} */ |