phpdbg_info.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 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. | http://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: Felipe Pena <felipe@php.net> |
  16. | Authors: Joe Watkins <joe.watkins@live.co.uk> |
  17. | Authors: Bob Weinand <bwoebi@php.net> |
  18. +----------------------------------------------------------------------+
  19. */
  20. #include "php.h"
  21. #include "phpdbg.h"
  22. #include "phpdbg_utils.h"
  23. #include "phpdbg_info.h"
  24. #include "phpdbg_bp.h"
  25. #include "phpdbg_prompt.h"
  26. ZEND_EXTERN_MODULE_GLOBALS(phpdbg)
  27. #define PHPDBG_INFO_COMMAND_D(f, h, a, m, l, s, flags) \
  28. PHPDBG_COMMAND_D_EXP(f, h, a, m, l, s, &phpdbg_prompt_commands[13], flags)
  29. const phpdbg_command_t phpdbg_info_commands[] = {
  30. PHPDBG_INFO_COMMAND_D(break, "show breakpoints", 'b', info_break, NULL, 0, PHPDBG_ASYNC_SAFE),
  31. PHPDBG_INFO_COMMAND_D(files, "show included files", 'F', info_files, NULL, 0, PHPDBG_ASYNC_SAFE),
  32. PHPDBG_INFO_COMMAND_D(classes, "show loaded classes", 'c', info_classes, NULL, 0, PHPDBG_ASYNC_SAFE),
  33. PHPDBG_INFO_COMMAND_D(funcs, "show loaded classes", 'f', info_funcs, NULL, 0, PHPDBG_ASYNC_SAFE),
  34. PHPDBG_INFO_COMMAND_D(error, "show last error", 'e', info_error, NULL, 0, PHPDBG_ASYNC_SAFE),
  35. PHPDBG_INFO_COMMAND_D(constants, "show user defined constants", 'd', info_constants, NULL, 0, PHPDBG_ASYNC_SAFE),
  36. PHPDBG_INFO_COMMAND_D(vars, "show active variables", 'v', info_vars, NULL, 0, PHPDBG_ASYNC_SAFE),
  37. PHPDBG_INFO_COMMAND_D(globals, "show superglobals", 'g', info_globals, NULL, 0, PHPDBG_ASYNC_SAFE),
  38. PHPDBG_INFO_COMMAND_D(literal, "show active literal constants", 'l', info_literal, NULL, 0, PHPDBG_ASYNC_SAFE),
  39. PHPDBG_INFO_COMMAND_D(memory, "show memory manager stats", 'm', info_memory, NULL, 0, PHPDBG_ASYNC_SAFE),
  40. PHPDBG_END_COMMAND
  41. };
  42. PHPDBG_INFO(break) /* {{{ */
  43. {
  44. phpdbg_print_breakpoints(PHPDBG_BREAK_FILE);
  45. phpdbg_print_breakpoints(PHPDBG_BREAK_SYM);
  46. phpdbg_print_breakpoints(PHPDBG_BREAK_METHOD);
  47. phpdbg_print_breakpoints(PHPDBG_BREAK_OPLINE);
  48. phpdbg_print_breakpoints(PHPDBG_BREAK_FILE_OPLINE);
  49. phpdbg_print_breakpoints(PHPDBG_BREAK_FUNCTION_OPLINE);
  50. phpdbg_print_breakpoints(PHPDBG_BREAK_METHOD_OPLINE);
  51. phpdbg_print_breakpoints(PHPDBG_BREAK_COND);
  52. phpdbg_print_breakpoints(PHPDBG_BREAK_OPCODE);
  53. return SUCCESS;
  54. } /* }}} */
  55. PHPDBG_INFO(files) /* {{{ */
  56. {
  57. zend_string *fname;
  58. phpdbg_try_access {
  59. phpdbg_notice("includedfilecount", "num=\"%d\"", "Included files: %d", zend_hash_num_elements(&EG(included_files)));
  60. } phpdbg_catch_access {
  61. phpdbg_error("signalsegv", "", "Could not fetch included file count, invalid data source");
  62. return SUCCESS;
  63. } phpdbg_end_try_access();
  64. phpdbg_try_access {
  65. ZEND_HASH_FOREACH_STR_KEY(&EG(included_files), fname) {
  66. phpdbg_writeln("includedfile", "name=\"%s\"", "File: %s", ZSTR_VAL(fname));
  67. } ZEND_HASH_FOREACH_END();
  68. } phpdbg_catch_access {
  69. phpdbg_error("signalsegv", "", "Could not fetch file name, invalid data source, aborting included file listing");
  70. } phpdbg_end_try_access();
  71. return SUCCESS;
  72. } /* }}} */
  73. PHPDBG_INFO(error) /* {{{ */
  74. {
  75. if (PG(last_error_message)) {
  76. phpdbg_try_access {
  77. phpdbg_writeln("lasterror", "error=\"%s\" file=\"%s\" line=\"%d\"", "Last error: %s at %s line %d", PG(last_error_message), PG(last_error_file), PG(last_error_lineno));
  78. } phpdbg_catch_access {
  79. phpdbg_notice("lasterror", "error=\"\"", "No error found!");
  80. } phpdbg_end_try_access();
  81. } else {
  82. phpdbg_notice("lasterror", "error=\"\"", "No error found!");
  83. }
  84. return SUCCESS;
  85. } /* }}} */
  86. PHPDBG_INFO(constants) /* {{{ */
  87. {
  88. HashTable consts;
  89. zend_constant *data;
  90. zend_hash_init(&consts, 8, NULL, NULL, 0);
  91. if (EG(zend_constants)) {
  92. phpdbg_try_access {
  93. ZEND_HASH_FOREACH_PTR(EG(zend_constants), data) {
  94. if (ZEND_CONSTANT_MODULE_NUMBER(data) == PHP_USER_CONSTANT) {
  95. zend_hash_update_ptr(&consts, data->name, data);
  96. }
  97. } ZEND_HASH_FOREACH_END();
  98. } phpdbg_catch_access {
  99. phpdbg_error("signalsegv", "", "Cannot fetch all the constants, invalid data source");
  100. } phpdbg_end_try_access();
  101. }
  102. phpdbg_notice("constantinfo", "num=\"%d\"", "User-defined constants (%d)", zend_hash_num_elements(&consts));
  103. if (zend_hash_num_elements(&consts)) {
  104. phpdbg_out("Address Refs Type Constant\n");
  105. ZEND_HASH_FOREACH_PTR(&consts, data) {
  106. #define VARIABLEINFO(attrs, msg, ...) \
  107. phpdbg_writeln("constant", \
  108. "address=\"%p\" refcount=\"%d\" type=\"%s\" name=\"%.*s\" " attrs, \
  109. "%-18p %-7d %-9s %.*s" msg, &data->value, \
  110. Z_REFCOUNTED(data->value) ? Z_REFCOUNT(data->value) : 1, \
  111. zend_zval_type_name(&data->value), \
  112. (int) ZSTR_LEN(data->name), ZSTR_VAL(data->name), ##__VA_ARGS__)
  113. switch (Z_TYPE(data->value)) {
  114. case IS_STRING:
  115. phpdbg_try_access {
  116. VARIABLEINFO("length=\"%zd\" value=\"%.*s\"", "\nstring (%zd) \"%.*s%s\"", Z_STRLEN(data->value), Z_STRLEN(data->value) < 255 ? (int) Z_STRLEN(data->value) : 255, Z_STRVAL(data->value), Z_STRLEN(data->value) > 255 ? "..." : "");
  117. } phpdbg_catch_access {
  118. VARIABLEINFO("", "");
  119. } phpdbg_end_try_access();
  120. break;
  121. case IS_TRUE:
  122. VARIABLEINFO("value=\"true\"", "\nbool (true)");
  123. break;
  124. case IS_FALSE:
  125. VARIABLEINFO("value=\"false\"", "\nbool (false)");
  126. break;
  127. case IS_LONG:
  128. VARIABLEINFO("value=\"%ld\"", "\nint (%ld)", Z_LVAL(data->value));
  129. break;
  130. case IS_DOUBLE:
  131. VARIABLEINFO("value=\"%lf\"", "\ndouble (%lf)", Z_DVAL(data->value));
  132. break;
  133. default:
  134. VARIABLEINFO("", "");
  135. #undef VARIABLEINFO
  136. }
  137. } ZEND_HASH_FOREACH_END();
  138. }
  139. return SUCCESS;
  140. } /* }}} */
  141. static int phpdbg_arm_auto_global(zval *ptrzv) {
  142. zend_auto_global *auto_global = Z_PTR_P(ptrzv);
  143. if (auto_global->armed) {
  144. if (PHPDBG_G(flags) & PHPDBG_IN_SIGNAL_HANDLER) {
  145. phpdbg_notice("variableinfo", "unreachable=\"%.*s\"", "Cannot show information about superglobal variable %.*s", (int) ZSTR_LEN(auto_global->name), ZSTR_VAL(auto_global->name));
  146. } else {
  147. auto_global->armed = auto_global->auto_global_callback(auto_global->name);
  148. }
  149. }
  150. return 0;
  151. }
  152. static int phpdbg_print_symbols(zend_bool show_globals) {
  153. HashTable vars;
  154. zend_array *symtable;
  155. zend_string *var;
  156. zval *data;
  157. if (!EG(current_execute_data) || !EG(current_execute_data)->func) {
  158. phpdbg_error("inactive", "type=\"op_array\"", "No active op array!");
  159. return SUCCESS;
  160. }
  161. if (show_globals) {
  162. /* that array should only be manipulated during init, so safe for async access during execution */
  163. zend_hash_apply(CG(auto_globals), (apply_func_t) phpdbg_arm_auto_global);
  164. symtable = &EG(symbol_table);
  165. } else if (!(symtable = zend_rebuild_symbol_table())) {
  166. phpdbg_error("inactive", "type=\"symbol_table\"", "No active symbol table!");
  167. return SUCCESS;
  168. }
  169. zend_hash_init(&vars, 8, NULL, NULL, 0);
  170. phpdbg_try_access {
  171. ZEND_HASH_FOREACH_STR_KEY_VAL(symtable, var, data) {
  172. if (zend_is_auto_global(var) ^ !show_globals) {
  173. zend_hash_update(&vars, var, data);
  174. }
  175. } ZEND_HASH_FOREACH_END();
  176. } phpdbg_catch_access {
  177. phpdbg_error("signalsegv", "", "Cannot fetch all data from the symbol table, invalid data source");
  178. } phpdbg_end_try_access();
  179. if (show_globals) {
  180. phpdbg_notice("variableinfo", "num=\"%d\"", "Superglobal variables (%d)", zend_hash_num_elements(&vars));
  181. } else {
  182. zend_op_array *ops = &EG(current_execute_data)->func->op_array;
  183. if (ops->function_name) {
  184. if (ops->scope) {
  185. phpdbg_notice("variableinfo", "method=\"%s::%s\" num=\"%d\"", "Variables in %s::%s() (%d)", ops->scope->name->val, ops->function_name->val, zend_hash_num_elements(&vars));
  186. } else {
  187. phpdbg_notice("variableinfo", "function=\"%s\" num=\"%d\"", "Variables in %s() (%d)", ZSTR_VAL(ops->function_name), zend_hash_num_elements(&vars));
  188. }
  189. } else {
  190. if (ops->filename) {
  191. phpdbg_notice("variableinfo", "file=\"%s\" num=\"%d\"", "Variables in %s (%d)", ZSTR_VAL(ops->filename), zend_hash_num_elements(&vars));
  192. } else {
  193. phpdbg_notice("variableinfo", "opline=\"%p\" num=\"%d\"", "Variables @ %p (%d)", ops, zend_hash_num_elements(&vars));
  194. }
  195. }
  196. }
  197. if (zend_hash_num_elements(&vars)) {
  198. phpdbg_out("Address Refs Type Variable\n");
  199. ZEND_HASH_FOREACH_STR_KEY_VAL(&vars, var, data) {
  200. phpdbg_try_access {
  201. const char *isref = "";
  202. #define VARIABLEINFO(attrs, msg, ...) \
  203. phpdbg_writeln("variable", \
  204. "address=\"%p\" refcount=\"%d\" type=\"%s\" refstatus=\"%s\" name=\"%.*s\" " attrs, \
  205. "%-18p %-7d %-9s %s$%.*s" msg, data, Z_REFCOUNTED_P(data) ? Z_REFCOUNT_P(data) : 1, zend_zval_type_name(data), isref, (int) ZSTR_LEN(var), ZSTR_VAL(var), ##__VA_ARGS__)
  206. retry_switch:
  207. switch (Z_TYPE_P(data)) {
  208. case IS_RESOURCE:
  209. phpdbg_try_access {
  210. const char *type = zend_rsrc_list_get_rsrc_type(Z_RES_P(data));
  211. VARIABLEINFO("type=\"%s\"", "\n|-------(typeof)------> (%s)\n", type ? type : "unknown");
  212. } phpdbg_catch_access {
  213. VARIABLEINFO("type=\"unknown\"", "\n|-------(typeof)------> (unknown)\n");
  214. } phpdbg_end_try_access();
  215. break;
  216. case IS_OBJECT:
  217. phpdbg_try_access {
  218. VARIABLEINFO("instanceof=\"%s\"", "\n|-----(instanceof)----> (%s)\n", ZSTR_VAL(Z_OBJCE_P(data)->name));
  219. } phpdbg_catch_access {
  220. VARIABLEINFO("instanceof=\"%s\"", "\n|-----(instanceof)----> (unknown)\n");
  221. } phpdbg_end_try_access();
  222. break;
  223. case IS_STRING:
  224. phpdbg_try_access {
  225. VARIABLEINFO("length=\"%zd\" value=\"%.*s\"", "\nstring (%zd) \"%.*s%s\"", Z_STRLEN_P(data), Z_STRLEN_P(data) < 255 ? (int) Z_STRLEN_P(data) : 255, Z_STRVAL_P(data), Z_STRLEN_P(data) > 255 ? "..." : "");
  226. } phpdbg_catch_access {
  227. VARIABLEINFO("", "");
  228. } phpdbg_end_try_access();
  229. break;
  230. case IS_TRUE:
  231. VARIABLEINFO("value=\"true\"", "\nbool (true)");
  232. break;
  233. case IS_FALSE:
  234. VARIABLEINFO("value=\"false\"", "\nbool (false)");
  235. break;
  236. case IS_LONG:
  237. VARIABLEINFO("value=\"%ld\"", "\nint (%ld)", Z_LVAL_P(data));
  238. break;
  239. case IS_DOUBLE:
  240. VARIABLEINFO("value=\"%lf\"", "\ndouble (%lf)", Z_DVAL_P(data));
  241. break;
  242. case IS_REFERENCE:
  243. isref = "&";
  244. data = Z_REFVAL_P(data);
  245. goto retry_switch;
  246. case IS_INDIRECT:
  247. data = Z_INDIRECT_P(data);
  248. goto retry_switch;
  249. default:
  250. VARIABLEINFO("", "");
  251. }
  252. #undef VARIABLEINFO
  253. } phpdbg_catch_access {
  254. phpdbg_writeln("variable", "address=\"%p\" name=\"%s\"", "%p\tn/a\tn/a\t$%s", data, ZSTR_VAL(var));
  255. } phpdbg_end_try_access();
  256. } ZEND_HASH_FOREACH_END();
  257. }
  258. zend_hash_destroy(&vars);
  259. return SUCCESS;
  260. } /* }}} */
  261. PHPDBG_INFO(vars) /* {{{ */
  262. {
  263. return phpdbg_print_symbols(0);
  264. }
  265. PHPDBG_INFO(globals) /* {{{ */
  266. {
  267. return phpdbg_print_symbols(1);
  268. }
  269. PHPDBG_INFO(literal) /* {{{ */
  270. {
  271. /* literals are assumed to not be manipulated during executing of their op_array and as such async safe */
  272. zend_bool in_executor = PHPDBG_G(in_execution) && EG(current_execute_data) && EG(current_execute_data)->func;
  273. if (in_executor || PHPDBG_G(ops)) {
  274. zend_op_array *ops = in_executor ? &EG(current_execute_data)->func->op_array : PHPDBG_G(ops);
  275. int literal = 0, count = ops->last_literal - 1;
  276. if (ops->function_name) {
  277. if (ops->scope) {
  278. phpdbg_notice("literalinfo", "method=\"%s::%s\" num=\"%d\"", "Literal Constants in %s::%s() (%d)", ops->scope->name->val, ops->function_name->val, count);
  279. } else {
  280. phpdbg_notice("literalinfo", "function=\"%s\" num=\"%d\"", "Literal Constants in %s() (%d)", ops->function_name->val, count);
  281. }
  282. } else {
  283. if (ops->filename) {
  284. phpdbg_notice("literalinfo", "file=\"%s\" num=\"%d\"", "Literal Constants in %s (%d)", ZSTR_VAL(ops->filename), count);
  285. } else {
  286. phpdbg_notice("literalinfo", "opline=\"%p\" num=\"%d\"", "Literal Constants @ %p (%d)", ops, count);
  287. }
  288. }
  289. while (literal < ops->last_literal) {
  290. if (Z_TYPE(ops->literals[literal]) != IS_NULL) {
  291. phpdbg_write("literal", "id=\"%u\"", "|-------- C%u -------> [", literal);
  292. zend_print_zval(&ops->literals[literal], 0);
  293. phpdbg_out("]\n");
  294. }
  295. literal++;
  296. }
  297. } else {
  298. phpdbg_error("inactive", "type=\"execution\"", "Not executing!");
  299. }
  300. return SUCCESS;
  301. } /* }}} */
  302. PHPDBG_INFO(memory) /* {{{ */
  303. {
  304. size_t used, real, peak_used, peak_real;
  305. zend_mm_heap *heap;
  306. zend_bool is_mm;
  307. if (PHPDBG_G(flags) & PHPDBG_IN_SIGNAL_HANDLER) {
  308. heap = zend_mm_set_heap(phpdbg_original_heap_sigsafe_mem());
  309. }
  310. if ((is_mm = is_zend_mm())) {
  311. used = zend_memory_usage(0);
  312. real = zend_memory_usage(1);
  313. peak_used = zend_memory_peak_usage(0);
  314. peak_real = zend_memory_peak_usage(1);
  315. }
  316. if (PHPDBG_G(flags) & PHPDBG_IN_SIGNAL_HANDLER) {
  317. zend_mm_set_heap(heap);
  318. }
  319. if (is_mm) {
  320. phpdbg_notice("meminfo", "", "Memory Manager Information");
  321. phpdbg_notice("current", "", "Current");
  322. phpdbg_writeln("used", "mem=\"%.3f\"", "|-------> Used:\t%.3f kB", (float) (used / 1024));
  323. phpdbg_writeln("real", "mem=\"%.3f\"", "|-------> Real:\t%.3f kB", (float) (real / 1024));
  324. phpdbg_notice("peak", "", "Peak");
  325. phpdbg_writeln("used", "mem=\"%.3f\"", "|-------> Used:\t%.3f kB", (float) (peak_used / 1024));
  326. phpdbg_writeln("real", "mem=\"%.3f\"", "|-------> Real:\t%.3f kB", (float) (peak_real / 1024));
  327. } else {
  328. phpdbg_error("inactive", "type=\"memory_manager\"", "Memory Manager Disabled!");
  329. }
  330. return SUCCESS;
  331. } /* }}} */
  332. static inline void phpdbg_print_class_name(zend_class_entry *ce) /* {{{ */
  333. {
  334. const char *visibility = ce->type == ZEND_USER_CLASS ? "User" : "Internal";
  335. const char *type = (ce->ce_flags & ZEND_ACC_INTERFACE) ? "Interface" : (ce->ce_flags & ZEND_ACC_ABSTRACT) ? "Abstract Class" : "Class";
  336. phpdbg_writeln("class", "type=\"%s\" flags=\"%s\" name=\"%.*s\" methodcount=\"%d\"", "%s %s %.*s (%d)", visibility, type, (int) ZSTR_LEN(ce->name), ZSTR_VAL(ce->name), zend_hash_num_elements(&ce->function_table));
  337. } /* }}} */
  338. PHPDBG_INFO(classes) /* {{{ */
  339. {
  340. zend_class_entry *ce;
  341. HashTable classes;
  342. zend_hash_init(&classes, 8, NULL, NULL, 0);
  343. phpdbg_try_access {
  344. ZEND_HASH_FOREACH_PTR(EG(class_table), ce) {
  345. if (ce->type == ZEND_USER_CLASS) {
  346. zend_hash_next_index_insert_ptr(&classes, ce);
  347. }
  348. } ZEND_HASH_FOREACH_END();
  349. } phpdbg_catch_access {
  350. phpdbg_notice("signalsegv", "", "Not all classes could be fetched, possibly invalid data source");
  351. } phpdbg_end_try_access();
  352. phpdbg_notice("classinfo", "num=\"%d\"", "User Classes (%d)", zend_hash_num_elements(&classes));
  353. /* once added, assume that classes are stable... until shutdown. */
  354. ZEND_HASH_FOREACH_PTR(&classes, ce) {
  355. phpdbg_print_class_name(ce);
  356. if (ce->parent) {
  357. zend_class_entry *pce;
  358. phpdbg_xml("<parents %r>");
  359. pce = ce->parent;
  360. do {
  361. phpdbg_out("|-------- ");
  362. phpdbg_print_class_name(pce);
  363. } while ((pce = pce->parent));
  364. phpdbg_xml("</parents>");
  365. }
  366. if (ce->info.user.filename) {
  367. phpdbg_writeln("classsource", "file=\"%s\" line=\"%u\"", "|---- in %s on line %u", ZSTR_VAL(ce->info.user.filename), ce->info.user.line_start);
  368. } else {
  369. phpdbg_writeln("classsource", "", "|---- no source code");
  370. }
  371. } ZEND_HASH_FOREACH_END();
  372. zend_hash_destroy(&classes);
  373. return SUCCESS;
  374. } /* }}} */
  375. PHPDBG_INFO(funcs) /* {{{ */
  376. {
  377. zend_function *zf;
  378. HashTable functions;
  379. zend_hash_init(&functions, 8, NULL, NULL, 0);
  380. phpdbg_try_access {
  381. ZEND_HASH_FOREACH_PTR(EG(function_table), zf) {
  382. if (zf->type == ZEND_USER_FUNCTION) {
  383. zend_hash_next_index_insert_ptr(&functions, zf);
  384. }
  385. } ZEND_HASH_FOREACH_END();
  386. } phpdbg_catch_access {
  387. phpdbg_notice("signalsegv", "", "Not all functions could be fetched, possibly invalid data source");
  388. } phpdbg_end_try_access();
  389. phpdbg_notice("functioninfo", "num=\"%d\"", "User Functions (%d)", zend_hash_num_elements(&functions));
  390. ZEND_HASH_FOREACH_PTR(&functions, zf) {
  391. zend_op_array *op_array = &zf->op_array;
  392. phpdbg_write("function", "name=\"%s\"", "|-------- %s", op_array->function_name ? ZSTR_VAL(op_array->function_name) : "{main}");
  393. if (op_array->filename) {
  394. phpdbg_writeln("functionsource", "file=\"%s\" line=\"%d\"", " in %s on line %d", ZSTR_VAL(op_array->filename), op_array->line_start);
  395. } else {
  396. phpdbg_writeln("functionsource", "", " (no source code)");
  397. }
  398. } ZEND_HASH_FOREACH_END();
  399. zend_hash_destroy(&functions);
  400. return SUCCESS;
  401. } /* }}} */