phpdbg_utils.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868
  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 "zend.h"
  21. #include "php.h"
  22. #include "phpdbg.h"
  23. #include "phpdbg_opcode.h"
  24. #include "phpdbg_utils.h"
  25. #include "ext/standard/php_string.h"
  26. /* FASYNC under Solaris */
  27. #ifdef HAVE_SYS_FILE_H
  28. # include <sys/file.h>
  29. #endif
  30. #ifdef HAVE_SYS_IOCTL_H
  31. # include "sys/ioctl.h"
  32. # ifndef GWINSZ_IN_SYS_IOCTL
  33. # include <termios.h>
  34. # endif
  35. #endif
  36. ZEND_EXTERN_MODULE_GLOBALS(phpdbg)
  37. /* {{{ color structures */
  38. const static phpdbg_color_t colors[] = {
  39. PHPDBG_COLOR_D("none", "0;0"),
  40. PHPDBG_COLOR_D("white", "0;64"),
  41. PHPDBG_COLOR_D("white-bold", "1;64"),
  42. PHPDBG_COLOR_D("white-underline", "4;64"),
  43. PHPDBG_COLOR_D("red", "0;31"),
  44. PHPDBG_COLOR_D("red-bold", "1;31"),
  45. PHPDBG_COLOR_D("red-underline", "4;31"),
  46. PHPDBG_COLOR_D("green", "0;32"),
  47. PHPDBG_COLOR_D("green-bold", "1;32"),
  48. PHPDBG_COLOR_D("green-underline", "4;32"),
  49. PHPDBG_COLOR_D("yellow", "0;33"),
  50. PHPDBG_COLOR_D("yellow-bold", "1;33"),
  51. PHPDBG_COLOR_D("yellow-underline", "4;33"),
  52. PHPDBG_COLOR_D("blue", "0;34"),
  53. PHPDBG_COLOR_D("blue-bold", "1;34"),
  54. PHPDBG_COLOR_D("blue-underline", "4;34"),
  55. PHPDBG_COLOR_D("purple", "0;35"),
  56. PHPDBG_COLOR_D("purple-bold", "1;35"),
  57. PHPDBG_COLOR_D("purple-underline", "4;35"),
  58. PHPDBG_COLOR_D("cyan", "0;36"),
  59. PHPDBG_COLOR_D("cyan-bold", "1;36"),
  60. PHPDBG_COLOR_D("cyan-underline", "4;36"),
  61. PHPDBG_COLOR_D("black", "0;30"),
  62. PHPDBG_COLOR_D("black-bold", "1;30"),
  63. PHPDBG_COLOR_D("black-underline", "4;30"),
  64. PHPDBG_COLOR_END
  65. }; /* }}} */
  66. /* {{{ */
  67. const static phpdbg_element_t elements[] = {
  68. PHPDBG_ELEMENT_D("prompt", PHPDBG_COLOR_PROMPT),
  69. PHPDBG_ELEMENT_D("error", PHPDBG_COLOR_ERROR),
  70. PHPDBG_ELEMENT_D("notice", PHPDBG_COLOR_NOTICE),
  71. PHPDBG_ELEMENT_END
  72. }; /* }}} */
  73. PHPDBG_API int phpdbg_is_numeric(const char *str) /* {{{ */
  74. {
  75. if (!str)
  76. return 0;
  77. for (; *str; str++) {
  78. if (isspace(*str) || *str == '-') {
  79. continue;
  80. }
  81. return isdigit(*str);
  82. }
  83. return 0;
  84. } /* }}} */
  85. PHPDBG_API int phpdbg_is_empty(const char *str) /* {{{ */
  86. {
  87. if (!str)
  88. return 1;
  89. for (; *str; str++) {
  90. if (isspace(*str)) {
  91. continue;
  92. }
  93. return 0;
  94. }
  95. return 1;
  96. } /* }}} */
  97. PHPDBG_API int phpdbg_is_addr(const char *str) /* {{{ */
  98. {
  99. return str[0] && str[1] && memcmp(str, "0x", 2) == 0;
  100. } /* }}} */
  101. PHPDBG_API int phpdbg_is_class_method(const char *str, size_t len, char **class, char **method) /* {{{ */
  102. {
  103. char *sep = NULL;
  104. if (strstr(str, "#") != NULL)
  105. return 0;
  106. if (strstr(str, " ") != NULL)
  107. return 0;
  108. sep = strstr(str, "::");
  109. if (!sep || sep == str || sep+2 == str+len-1) {
  110. return 0;
  111. }
  112. if (class != NULL) {
  113. if (str[0] == '\\') {
  114. str++;
  115. len--;
  116. }
  117. *class = estrndup(str, sep - str);
  118. (*class)[sep - str] = 0;
  119. }
  120. if (method != NULL) {
  121. *method = estrndup(sep+2, str + len - (sep + 2));
  122. }
  123. return 1;
  124. } /* }}} */
  125. PHPDBG_API char *phpdbg_resolve_path(const char *path) /* {{{ */
  126. {
  127. char resolved_name[MAXPATHLEN];
  128. if (expand_filepath(path, resolved_name) == NULL) {
  129. return NULL;
  130. }
  131. return estrdup(resolved_name);
  132. } /* }}} */
  133. PHPDBG_API const char *phpdbg_current_file(void) /* {{{ */
  134. {
  135. const char *file = zend_get_executed_filename();
  136. if (memcmp(file, "[no active file]", sizeof("[no active file]")) == 0) {
  137. return PHPDBG_G(exec);
  138. }
  139. return file;
  140. } /* }}} */
  141. PHPDBG_API const zend_function *phpdbg_get_function(const char *fname, const char *cname) /* {{{ */
  142. {
  143. zend_function *func = NULL;
  144. zend_string *lfname = zend_string_init(fname, strlen(fname), 0);
  145. zend_string *tmp = zend_string_tolower(lfname);
  146. zend_string_release(lfname);
  147. lfname = tmp;
  148. if (cname) {
  149. zend_class_entry *ce;
  150. zend_string *lcname = zend_string_init(cname, strlen(cname), 0);
  151. tmp = zend_string_tolower(lcname);
  152. zend_string_release(lcname);
  153. lcname = tmp;
  154. ce = zend_lookup_class(lcname);
  155. zend_string_release(lcname);
  156. if (ce) {
  157. func = zend_hash_find_ptr(&ce->function_table, lfname);
  158. }
  159. } else {
  160. func = zend_hash_find_ptr(EG(function_table), lfname);
  161. }
  162. zend_string_release(lfname);
  163. return func;
  164. } /* }}} */
  165. PHPDBG_API char *phpdbg_trim(const char *str, size_t len, size_t *new_len) /* {{{ */
  166. {
  167. const char *p = str;
  168. char *new = NULL;
  169. while (p && isspace(*p)) {
  170. ++p;
  171. --len;
  172. }
  173. while (*p && isspace(*(p + len -1))) {
  174. --len;
  175. }
  176. if (len == 0) {
  177. new = estrndup("", sizeof(""));
  178. *new_len = 0;
  179. } else {
  180. new = estrndup(p, len);
  181. *(new + len) = '\0';
  182. if (new_len) {
  183. *new_len = len;
  184. }
  185. }
  186. return new;
  187. } /* }}} */
  188. PHPDBG_API const phpdbg_color_t *phpdbg_get_color(const char *name, size_t name_length) /* {{{ */
  189. {
  190. const phpdbg_color_t *color = colors;
  191. while (color && color->name) {
  192. if (name_length == color->name_length &&
  193. memcmp(name, color->name, name_length) == SUCCESS) {
  194. phpdbg_debug("phpdbg_get_color(%s, %lu): %s", name, name_length, color->code);
  195. return color;
  196. }
  197. ++color;
  198. }
  199. phpdbg_debug("phpdbg_get_color(%s, %lu): failed", name, name_length);
  200. return NULL;
  201. } /* }}} */
  202. PHPDBG_API void phpdbg_set_color(int element, const phpdbg_color_t *color) /* {{{ */
  203. {
  204. PHPDBG_G(colors)[element] = color;
  205. } /* }}} */
  206. PHPDBG_API void phpdbg_set_color_ex(int element, const char *name, size_t name_length) /* {{{ */
  207. {
  208. const phpdbg_color_t *color = phpdbg_get_color(name, name_length);
  209. if (color) {
  210. phpdbg_set_color(element, color);
  211. } else PHPDBG_G(colors)[element] = colors;
  212. } /* }}} */
  213. PHPDBG_API const phpdbg_color_t* phpdbg_get_colors(void) /* {{{ */
  214. {
  215. return colors;
  216. } /* }}} */
  217. PHPDBG_API int phpdbg_get_element(const char *name, size_t len) {
  218. const phpdbg_element_t *element = elements;
  219. while (element && element->name) {
  220. if (len == element->name_length) {
  221. if (strncasecmp(name, element->name, len) == SUCCESS) {
  222. return element->id;
  223. }
  224. }
  225. element++;
  226. }
  227. return PHPDBG_COLOR_INVALID;
  228. }
  229. PHPDBG_API void phpdbg_set_prompt(const char *prompt) /* {{{ */
  230. {
  231. /* free formatted prompt */
  232. if (PHPDBG_G(prompt)[1]) {
  233. free(PHPDBG_G(prompt)[1]);
  234. PHPDBG_G(prompt)[1] = NULL;
  235. }
  236. /* free old prompt */
  237. if (PHPDBG_G(prompt)[0]) {
  238. free(PHPDBG_G(prompt)[0]);
  239. PHPDBG_G(prompt)[0] = NULL;
  240. }
  241. /* copy new prompt */
  242. PHPDBG_G(prompt)[0] = strdup(prompt);
  243. } /* }}} */
  244. PHPDBG_API const char *phpdbg_get_prompt(void) /* {{{ */
  245. {
  246. /* find cached prompt */
  247. if (PHPDBG_G(prompt)[1]) {
  248. return PHPDBG_G(prompt)[1];
  249. }
  250. /* create cached prompt */
  251. #ifndef HAVE_LIBEDIT
  252. /* TODO: libedit doesn't seems to support coloured prompt */
  253. if ((PHPDBG_G(flags) & PHPDBG_IS_COLOURED)) {
  254. ZEND_IGNORE_VALUE(asprintf(&PHPDBG_G(prompt)[1], "\033[%sm%s\033[0m ",
  255. PHPDBG_G(colors)[PHPDBG_COLOR_PROMPT]->code,
  256. PHPDBG_G(prompt)[0]));
  257. } else
  258. #endif
  259. {
  260. ZEND_IGNORE_VALUE(asprintf(&PHPDBG_G(prompt)[1], "%s ", PHPDBG_G(prompt)[0]));
  261. }
  262. return PHPDBG_G(prompt)[1];
  263. } /* }}} */
  264. int phpdbg_rebuild_symtable(void) {
  265. if (!EG(current_execute_data) || !EG(current_execute_data)->func) {
  266. phpdbg_error("inactive", "type=\"op_array\"", "No active op array!");
  267. return FAILURE;
  268. }
  269. if (!zend_rebuild_symbol_table()) {
  270. phpdbg_error("inactive", "type=\"symbol_table\"", "No active symbol table!");
  271. return FAILURE;
  272. }
  273. return SUCCESS;
  274. }
  275. PHPDBG_API int phpdbg_get_terminal_width(void) /* {{{ */
  276. {
  277. int columns;
  278. #ifdef _WIN32
  279. CONSOLE_SCREEN_BUFFER_INFO csbi;
  280. GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &csbi);
  281. columns = csbi.srWindow.Right - csbi.srWindow.Left + 1;
  282. #elif defined(HAVE_SYS_IOCTL_H) && defined(TIOCGWINSZ)
  283. struct winsize w;
  284. columns = ioctl(fileno(stdout), TIOCGWINSZ, &w) == 0 ? w.ws_col : 80;
  285. #else
  286. columns = 80;
  287. #endif
  288. return columns;
  289. } /* }}} */
  290. PHPDBG_API int phpdbg_get_terminal_height(void) /* {{{ */
  291. {
  292. int lines;
  293. #ifdef _WIN32
  294. CONSOLE_SCREEN_BUFFER_INFO csbi;
  295. GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &csbi);
  296. lines = csbi.srWindow.Bottom - csbi.srWindow.Top + 1;
  297. #elif defined(HAVE_SYS_IOCTL_H) && defined(TIOCGWINSZ)
  298. struct winsize w;
  299. lines = ioctl(fileno(stdout), TIOCGWINSZ, &w) == 0 ? w.ws_row : 40;
  300. #else
  301. lines = 40;
  302. #endif
  303. return lines;
  304. } /* }}} */
  305. PHPDBG_API void phpdbg_set_async_io(int fd) {
  306. #if !defined(_WIN32) && defined(FASYNC)
  307. int flags;
  308. fcntl(STDIN_FILENO, F_SETOWN, getpid());
  309. flags = fcntl(STDIN_FILENO, F_GETFL);
  310. fcntl(STDIN_FILENO, F_SETFL, flags | FASYNC);
  311. #endif
  312. }
  313. int phpdbg_safe_class_lookup(const char *name, int name_length, zend_class_entry **ce) {
  314. if (PHPDBG_G(flags) & PHPDBG_IN_SIGNAL_HANDLER) {
  315. char *lc_name, *lc_free;
  316. int lc_length;
  317. if (name == NULL || !name_length) {
  318. return FAILURE;
  319. }
  320. lc_free = lc_name = emalloc(name_length + 1);
  321. zend_str_tolower_copy(lc_name, name, name_length);
  322. lc_length = name_length + 1;
  323. if (lc_name[0] == '\\') {
  324. lc_name += 1;
  325. lc_length -= 1;
  326. }
  327. phpdbg_try_access {
  328. *ce = zend_hash_str_find_ptr(EG(class_table), lc_name, lc_length);
  329. } phpdbg_catch_access {
  330. phpdbg_error("signalsegv", "class=\"%.*s\"", "Could not fetch class %.*s, invalid data source", name_length, name);
  331. } phpdbg_end_try_access();
  332. efree(lc_free);
  333. } else {
  334. zend_string *str_name = zend_string_init(name, name_length, 0);
  335. *ce = zend_lookup_class(str_name);
  336. efree(str_name);
  337. }
  338. return *ce ? SUCCESS : FAILURE;
  339. }
  340. char *phpdbg_get_property_key(char *key) {
  341. if (*key != 0) {
  342. return key;
  343. }
  344. return strchr(key + 1, 0) + 1;
  345. }
  346. static int phpdbg_parse_variable_arg_wrapper(char *name, size_t len, char *keyname, size_t keylen, HashTable *parent, zval *zv, phpdbg_parse_var_func callback) {
  347. return callback(name, len, keyname, keylen, parent, zv);
  348. }
  349. PHPDBG_API int phpdbg_parse_variable(char *input, size_t len, HashTable *parent, size_t i, phpdbg_parse_var_func callback, zend_bool silent) {
  350. return phpdbg_parse_variable_with_arg(input, len, parent, i, (phpdbg_parse_var_with_arg_func) phpdbg_parse_variable_arg_wrapper, NULL, silent, callback);
  351. }
  352. PHPDBG_API int phpdbg_parse_variable_with_arg(char *input, size_t len, HashTable *parent, size_t i, phpdbg_parse_var_with_arg_func callback, phpdbg_parse_var_with_arg_func step_cb, zend_bool silent, void *arg) {
  353. int ret = FAILURE;
  354. zend_bool new_index = 1;
  355. char *last_index;
  356. size_t index_len = 0;
  357. zval *zv;
  358. if (len < 2 || *input != '$') {
  359. goto error;
  360. }
  361. while (i++ < len) {
  362. if (i == len) {
  363. new_index = 1;
  364. } else {
  365. switch (input[i]) {
  366. case '[':
  367. new_index = 1;
  368. break;
  369. case ']':
  370. break;
  371. case '>':
  372. if (last_index[index_len - 1] == '-') {
  373. new_index = 1;
  374. index_len--;
  375. }
  376. break;
  377. default:
  378. if (new_index) {
  379. last_index = input + i;
  380. new_index = 0;
  381. }
  382. if (input[i - 1] == ']') {
  383. goto error;
  384. }
  385. index_len++;
  386. }
  387. }
  388. if (new_index && index_len == 0) {
  389. zend_ulong numkey;
  390. zend_string *strkey;
  391. ZEND_HASH_FOREACH_KEY_PTR(parent, numkey, strkey, zv) {
  392. while (Z_TYPE_P(zv) == IS_INDIRECT) {
  393. zv = Z_INDIRECT_P(zv);
  394. }
  395. if (i == len || (i == len - 1 && input[len - 1] == ']')) {
  396. char *key, *propkey;
  397. size_t namelen, keylen;
  398. char *name;
  399. char *keyname = estrndup(last_index, index_len);
  400. if (strkey) {
  401. key = ZSTR_VAL(strkey);
  402. keylen = ZSTR_LEN(strkey);
  403. } else {
  404. keylen = spprintf(&key, 0, ZEND_ULONG_FMT, numkey);
  405. }
  406. propkey = phpdbg_get_property_key(key);
  407. name = emalloc(i + keylen + 2);
  408. namelen = sprintf(name, "%.*s%.*s%s", (int) i, input, (int) (keylen - (propkey - key)), propkey, input[len - 1] == ']'?"]":"");
  409. if (!strkey) {
  410. efree(key);
  411. }
  412. ret = callback(name, namelen, keyname, index_len, parent, zv, arg) == SUCCESS || ret == SUCCESS?SUCCESS:FAILURE;
  413. } else retry_ref: if (Z_TYPE_P(zv) == IS_OBJECT) {
  414. if (step_cb) {
  415. char *name = estrndup(input, i);
  416. char *keyname = estrndup(last_index, index_len);
  417. ret = step_cb(name, i, keyname, index_len, parent, zv, arg) == SUCCESS || ret == SUCCESS?SUCCESS:FAILURE;
  418. }
  419. phpdbg_parse_variable_with_arg(input, len, Z_OBJPROP_P(zv), i, callback, step_cb, silent, arg);
  420. } else if (Z_TYPE_P(zv) == IS_ARRAY) {
  421. if (step_cb) {
  422. char *name = estrndup(input, i);
  423. char *keyname = estrndup(last_index, index_len);
  424. ret = step_cb(name, i, keyname, index_len, parent, zv, arg) == SUCCESS || ret == SUCCESS?SUCCESS:FAILURE;
  425. }
  426. phpdbg_parse_variable_with_arg(input, len, Z_ARRVAL_P(zv), i, callback, step_cb, silent, arg);
  427. } else if (Z_ISREF_P(zv)) {
  428. if (step_cb) {
  429. char *name = estrndup(input, i);
  430. char *keyname = estrndup(last_index, index_len);
  431. ret = step_cb(name, i, keyname, index_len, parent, zv, arg) == SUCCESS || ret == SUCCESS?SUCCESS:FAILURE;
  432. }
  433. ZVAL_DEREF(zv);
  434. goto retry_ref;
  435. } else {
  436. /* Ignore silently */
  437. }
  438. } ZEND_HASH_FOREACH_END();
  439. return ret;
  440. } else if (new_index) {
  441. char last_chr = last_index[index_len];
  442. last_index[index_len] = 0;
  443. if (!(zv = zend_symtable_str_find(parent, last_index, index_len))) {
  444. if (!silent) {
  445. phpdbg_error("variable", "type=\"undefined\" variable=\"%.*s\"", "%.*s is undefined", (int) input[i] == ']' ? i + 1 : i, input);
  446. }
  447. return FAILURE;
  448. }
  449. while (Z_TYPE_P(zv) == IS_INDIRECT) {
  450. zv = Z_INDIRECT_P(zv);
  451. }
  452. last_index[index_len] = last_chr;
  453. if (i == len) {
  454. char *name = estrndup(input, i);
  455. char *keyname = estrndup(last_index, index_len);
  456. ret = callback(name, i, keyname, index_len, parent, zv, arg) == SUCCESS || ret == SUCCESS?SUCCESS:FAILURE;
  457. } else retry_ref_end: if (Z_TYPE_P(zv) == IS_OBJECT) {
  458. if (step_cb) {
  459. char *name = estrndup(input, i);
  460. char *keyname = estrndup(last_index, index_len);
  461. ret = step_cb(name, i, keyname, index_len, parent, zv, arg) == SUCCESS || ret == SUCCESS?SUCCESS:FAILURE;
  462. }
  463. parent = Z_OBJPROP_P(zv);
  464. } else if (Z_TYPE_P(zv) == IS_ARRAY) {
  465. if (step_cb) {
  466. char *name = estrndup(input, i);
  467. char *keyname = estrndup(last_index, index_len);
  468. ret = step_cb(name, i, keyname, index_len, parent, zv, arg) == SUCCESS || ret == SUCCESS?SUCCESS:FAILURE;
  469. }
  470. parent = Z_ARRVAL_P(zv);
  471. } else if (Z_ISREF_P(zv)) {
  472. if (step_cb) {
  473. char *name = estrndup(input, i);
  474. char *keyname = estrndup(last_index, index_len);
  475. ret = step_cb(name, i, keyname, index_len, parent, zv, arg) == SUCCESS || ret == SUCCESS?SUCCESS:FAILURE;
  476. }
  477. ZVAL_DEREF(zv);
  478. goto retry_ref_end;
  479. } else {
  480. phpdbg_error("variable", "type=\"notiterable\" variable=\"%.*s\"", "%.*s is nor an array nor an object", (int) (input[i] == '>' ? i - 1 : i), input);
  481. return FAILURE;
  482. }
  483. index_len = 0;
  484. }
  485. }
  486. return ret;
  487. error:
  488. phpdbg_error("variable", "type=\"invalidinput\"", "Malformed input");
  489. return FAILURE;
  490. }
  491. int phpdbg_is_auto_global(char *name, int len) {
  492. return zend_is_auto_global_str(name, len);
  493. }
  494. static int phpdbg_xml_array_element_dump(zval *zv, zend_string *key, zend_ulong num) {
  495. phpdbg_xml("<element");
  496. phpdbg_try_access {
  497. if (key) { /* string key */
  498. phpdbg_xml(" name=\"%.*s\"", (int) ZSTR_LEN(key), ZSTR_VAL(key));
  499. } else { /* numeric key */
  500. phpdbg_xml(" name=\"%ld\"", num);
  501. }
  502. } phpdbg_catch_access {
  503. phpdbg_xml(" severity=\"error\" ></element>");
  504. return 0;
  505. } phpdbg_end_try_access();
  506. phpdbg_xml(">");
  507. phpdbg_xml_var_dump(zv);
  508. phpdbg_xml("</element>");
  509. return 0;
  510. }
  511. static int phpdbg_xml_object_property_dump(zval *zv, zend_string *key, zend_ulong num) {
  512. phpdbg_xml("<property");
  513. phpdbg_try_access {
  514. if (key) { /* string key */
  515. const char *prop_name, *class_name;
  516. int unmangle = zend_unmangle_property_name(key, &class_name, &prop_name);
  517. if (class_name && unmangle == SUCCESS) {
  518. phpdbg_xml(" name=\"%s\"", prop_name);
  519. if (class_name[0] == '*') {
  520. phpdbg_xml(" protection=\"protected\"");
  521. } else {
  522. phpdbg_xml(" class=\"%s\" protection=\"private\"", class_name);
  523. }
  524. } else {
  525. phpdbg_xml(" name=\"%.*s\" protection=\"public\"", (int) ZSTR_LEN(key), ZSTR_VAL(key));
  526. }
  527. } else { /* numeric key */
  528. phpdbg_xml(" name=\"%ld\" protection=\"public\"", num);
  529. }
  530. } phpdbg_catch_access {
  531. phpdbg_xml(" severity=\"error\" ></property>");
  532. return 0;
  533. } phpdbg_end_try_access();
  534. phpdbg_xml(">");
  535. phpdbg_xml_var_dump(zv);
  536. phpdbg_xml("</property>");
  537. return 0;
  538. }
  539. #define COMMON (is_ref ? "&" : "")
  540. PHPDBG_API void phpdbg_xml_var_dump(zval *zv) {
  541. HashTable *myht;
  542. zend_string *class_name, *key;
  543. zend_ulong num;
  544. zval *val;
  545. int (*element_dump_func)(zval *zv, zend_string *key, zend_ulong num);
  546. zend_bool is_ref = 0;
  547. int is_temp;
  548. phpdbg_try_access {
  549. is_ref = Z_ISREF_P(zv) && GC_REFCOUNT(Z_COUNTED_P(zv)) > 1;
  550. ZVAL_DEREF(zv);
  551. switch (Z_TYPE_P(zv)) {
  552. case IS_TRUE:
  553. phpdbg_xml("<bool refstatus=\"%s\" value=\"true\" />", COMMON);
  554. break;
  555. case IS_FALSE:
  556. phpdbg_xml("<bool refstatus=\"%s\" value=\"false\" />", COMMON);
  557. break;
  558. case IS_NULL:
  559. phpdbg_xml("<null refstatus=\"%s\" />", COMMON);
  560. break;
  561. case IS_LONG:
  562. phpdbg_xml("<int refstatus=\"%s\" value=\"" ZEND_LONG_FMT "\" />", COMMON, Z_LVAL_P(zv));
  563. break;
  564. case IS_DOUBLE:
  565. phpdbg_xml("<float refstatus=\"%s\" value=\"%.*G\" />", COMMON, (int) EG(precision), Z_DVAL_P(zv));
  566. break;
  567. case IS_STRING:
  568. phpdbg_xml("<string refstatus=\"%s\" length=\"%zd\" value=\"%.*s\" />", COMMON, Z_STRLEN_P(zv), (int) Z_STRLEN_P(zv), Z_STRVAL_P(zv));
  569. break;
  570. case IS_ARRAY:
  571. myht = Z_ARRVAL_P(zv);
  572. if (!(GC_FLAGS(myht) & GC_IMMUTABLE)) {
  573. if (GC_IS_RECURSIVE(myht)) {
  574. phpdbg_xml("<recursion />");
  575. break;
  576. }
  577. GC_PROTECT_RECURSION(myht);
  578. }
  579. phpdbg_xml("<array refstatus=\"%s\" num=\"%d\">", COMMON, zend_hash_num_elements(myht));
  580. element_dump_func = phpdbg_xml_array_element_dump;
  581. is_temp = 0;
  582. goto head_done;
  583. case IS_OBJECT:
  584. myht = Z_OBJDEBUG_P(zv, is_temp);
  585. if (myht && GC_IS_RECURSIVE(myht)) {
  586. phpdbg_xml("<recursion />");
  587. break;
  588. }
  589. class_name = Z_OBJ_HANDLER_P(zv, get_class_name)(Z_OBJ_P(zv));
  590. phpdbg_xml("<object refstatus=\"%s\" class=\"%.*s\" id=\"%d\" num=\"%d\">", COMMON, (int) ZSTR_LEN(class_name), ZSTR_VAL(class_name), Z_OBJ_HANDLE_P(zv), myht ? zend_hash_num_elements(myht) : 0);
  591. zend_string_release(class_name);
  592. element_dump_func = phpdbg_xml_object_property_dump;
  593. head_done:
  594. if (myht) {
  595. ZEND_HASH_FOREACH_KEY_VAL_IND(myht, num, key, val) {
  596. element_dump_func(val, key, num);
  597. } ZEND_HASH_FOREACH_END();
  598. zend_hash_apply_with_arguments(myht, (apply_func_args_t) element_dump_func, 0);
  599. GC_UNPROTECT_RECURSION(myht);
  600. if (is_temp) {
  601. zend_hash_destroy(myht);
  602. efree(myht);
  603. }
  604. }
  605. if (Z_TYPE_P(zv) == IS_ARRAY) {
  606. phpdbg_xml("</array>");
  607. } else {
  608. phpdbg_xml("</object>");
  609. }
  610. break;
  611. case IS_RESOURCE: {
  612. const char *type_name = zend_rsrc_list_get_rsrc_type(Z_RES_P(zv));
  613. phpdbg_xml("<resource refstatus=\"%s\" id=\"%pd\" type=\"%s\" />", COMMON, Z_RES_P(zv)->handle, type_name ? type_name : "unknown");
  614. break;
  615. }
  616. default:
  617. break;
  618. }
  619. } phpdbg_end_try_access();
  620. }
  621. PHPDBG_API zend_bool phpdbg_check_caught_ex(zend_execute_data *execute_data, zend_object *exception) {
  622. const zend_op *op;
  623. zend_op *cur;
  624. uint32_t op_num, i;
  625. zend_op_array *op_array = &execute_data->func->op_array;
  626. if (execute_data->opline >= EG(exception_op) && execute_data->opline < EG(exception_op) + 3) {
  627. op = EG(opline_before_exception);
  628. } else {
  629. op = execute_data->opline;
  630. }
  631. op_num = op - op_array->opcodes;
  632. for (i = 0; i < op_array->last_try_catch && op_array->try_catch_array[i].try_op <= op_num; i++) {
  633. uint32_t catch = op_array->try_catch_array[i].catch_op, finally = op_array->try_catch_array[i].finally_op;
  634. if (op_num <= catch || op_num <= finally) {
  635. if (finally) {
  636. return 1;
  637. }
  638. cur = &op_array->opcodes[catch];
  639. while (1) {
  640. zend_class_entry *ce;
  641. if (!(ce = CACHED_PTR(cur->extended_value & ~ZEND_LAST_CATCH))) {
  642. ce = zend_fetch_class_by_name(Z_STR_P(RT_CONSTANT(cur, cur->op1)), RT_CONSTANT(cur, cur->op1) + 1, ZEND_FETCH_CLASS_NO_AUTOLOAD);
  643. CACHE_PTR(cur->extended_value & ~ZEND_LAST_CATCH, ce);
  644. }
  645. if (ce == exception->ce || (ce && instanceof_function(exception->ce, ce))) {
  646. return 1;
  647. }
  648. if (cur->extended_value & ZEND_LAST_CATCH) {
  649. return 0;
  650. }
  651. cur = OP_JMP_ADDR(cur, cur->op2);
  652. }
  653. return 0;
  654. }
  655. }
  656. return op->opcode == ZEND_CATCH;
  657. }
  658. char *phpdbg_short_zval_print(zval *zv, int maxlen) /* {{{ */
  659. {
  660. char *decode = NULL;
  661. switch (Z_TYPE_P(zv)) {
  662. case IS_UNDEF:
  663. decode = estrdup("");
  664. break;
  665. case IS_NULL:
  666. decode = estrdup("null");
  667. break;
  668. case IS_FALSE:
  669. decode = estrdup("false");
  670. break;
  671. case IS_TRUE:
  672. decode = estrdup("true");
  673. break;
  674. case IS_LONG:
  675. spprintf(&decode, 0, ZEND_LONG_FMT, Z_LVAL_P(zv));
  676. break;
  677. case IS_DOUBLE:
  678. spprintf(&decode, 0, "%.*G", 14, Z_DVAL_P(zv));
  679. /* Make sure it looks like a float */
  680. if (zend_finite(Z_DVAL_P(zv)) && !strchr(decode, '.')) {
  681. size_t len = strlen(decode);
  682. char *decode2 = emalloc(len + strlen(".0") + 1);
  683. memcpy(decode2, decode, len);
  684. decode2[len] = '.';
  685. decode2[len+1] = '0';
  686. decode2[len+2] = '\0';
  687. efree(decode);
  688. decode = decode2;
  689. }
  690. break;
  691. case IS_STRING: {
  692. int i;
  693. zend_string *str = php_addcslashes(Z_STR_P(zv), "\\\"\n\t\0", 5);
  694. for (i = 0; i < ZSTR_LEN(str); i++) {
  695. if (ZSTR_VAL(str)[i] < 32) {
  696. ZSTR_VAL(str)[i] = ' ';
  697. }
  698. }
  699. spprintf(&decode, 0, "\"%.*s\"%c",
  700. ZSTR_LEN(str) <= maxlen - 2 ? (int) ZSTR_LEN(str) : (maxlen - 3),
  701. ZSTR_VAL(str), ZSTR_LEN(str) <= maxlen - 2 ? 0 : '+');
  702. zend_string_release(str);
  703. } break;
  704. case IS_RESOURCE:
  705. spprintf(&decode, 0, "Rsrc #%d", Z_RES_HANDLE_P(zv));
  706. break;
  707. case IS_ARRAY:
  708. spprintf(&decode, 0, "array(%d)", zend_hash_num_elements(Z_ARR_P(zv)));
  709. break;
  710. case IS_OBJECT: {
  711. zend_string *str = Z_OBJCE_P(zv)->name;
  712. spprintf(&decode, 0, "%.*s%c",
  713. ZSTR_LEN(str) <= maxlen ? (int) ZSTR_LEN(str) : maxlen - 1,
  714. ZSTR_VAL(str), ZSTR_LEN(str) <= maxlen ? 0 : '+');
  715. break;
  716. }
  717. case IS_CONSTANT_AST: {
  718. zend_ast *ast = Z_ASTVAL_P(zv);
  719. if (ast->kind == ZEND_AST_CONSTANT
  720. || ast->kind == ZEND_AST_CONSTANT_CLASS) {
  721. decode = estrdup("<constant>");
  722. } else {
  723. decode = estrdup("<ast>");
  724. }
  725. break;
  726. }
  727. default:
  728. spprintf(&decode, 0, "unknown type: %d", Z_TYPE_P(zv));
  729. break;
  730. }
  731. return decode;
  732. } /* }}} */