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