php-src/phpdbg_bp.c

311 lines
9.4 KiB
C
Raw Normal View History

/*
+----------------------------------------------------------------------+
| PHP Version 5 |
+----------------------------------------------------------------------+
| Copyright (c) 1997-2013 The PHP Group |
+----------------------------------------------------------------------+
| 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. |
+----------------------------------------------------------------------+
| Authors: Felipe Pena <felipe@php.net> |
| Authors: Joe Watkins <joe.watkins@live.co.uk> |
+----------------------------------------------------------------------+
*/
#include "zend.h"
#include "zend_hash.h"
2013-11-10 20:10:54 +00:00
#include "zend_llist.h"
#include "phpdbg.h"
#include "phpdbg_bp.h"
ZEND_EXTERN_MODULE_GLOBALS(phpdbg);
static void phpdbg_llist_breakfile_dtor(void *data) /* {{{ */
{
phpdbg_breakfile_t *bp = (phpdbg_breakfile_t*) data;
efree((char*)bp->filename);
} /* }}} */
static void phpdbg_class_breaks_dtor(void *data) /* {{{ */
{
phpdbg_breakmethod_t *bp = (phpdbg_breakmethod_t*) data;
2013-11-12 01:41:14 +00:00
efree((char*)bp->class_name);
efree((char*)bp->func_name);
} /* }}} */
void phpdbg_set_breakpoint_file(const char *path, long line_num TSRMLS_DC) /* {{{ */
{
phpdbg_breakfile_t new_break;
zend_llist *break_files_ptr;
size_t path_len = strlen(path);
new_break.filename = estrndup(path, path_len);
new_break.line = line_num;
2013-11-11 22:06:36 +00:00
PHPDBG_G(flags) |= PHPDBG_HAS_FILE_BP;
2013-11-12 01:51:50 +00:00
if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE],
new_break.filename, path_len, (void**)&break_files_ptr) == FAILURE) {
zend_llist break_files;
zend_llist_init(&break_files, sizeof(phpdbg_breakfile_t),
phpdbg_llist_breakfile_dtor, 0);
2013-11-12 01:51:50 +00:00
zend_hash_update(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE],
new_break.filename, path_len, &break_files, sizeof(zend_llist),
(void**)&break_files_ptr);
}
2013-11-10 19:15:32 +00:00
new_break.id = PHPDBG_G(bp_count)++;
zend_llist_add_element(break_files_ptr, &new_break);
2013-11-11 22:06:36 +00:00
printf(
"%sBreakpoint #%d added at %s:%ld%s\n",
PHPDBG_BOLD_LINE(TSRMLS_C),
2013-11-12 06:03:37 +00:00
new_break.id, new_break.filename, new_break.line,
PHPDBG_END_LINE(TSRMLS_C));
} /* }}} */
2013-11-10 21:38:58 +00:00
void phpdbg_set_breakpoint_symbol(const char *name TSRMLS_DC) /* {{{ */
{
size_t name_len = strlen(name);
2013-11-12 01:51:50 +00:00
if (!zend_hash_exists(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], name, name_len)) {
2013-11-10 21:38:58 +00:00
phpdbg_breaksymbol_t new_break;
PHPDBG_G(flags) |= PHPDBG_HAS_SYM_BP;
2013-11-10 21:38:58 +00:00
new_break.symbol = estrndup(name, name_len + 1);
new_break.id = PHPDBG_G(bp_count)++;
2013-11-12 01:51:50 +00:00
zend_hash_update(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], new_break.symbol,
2013-11-10 21:38:58 +00:00
name_len, &new_break, sizeof(phpdbg_breaksymbol_t), NULL);
2013-11-11 22:06:36 +00:00
2013-11-12 06:03:37 +00:00
printf(
"%sBreakpoint #%d added at %s%s\n",
PHPDBG_BOLD_LINE(TSRMLS_C),
2013-11-12 06:03:37 +00:00
new_break.id, new_break.symbol,
PHPDBG_END_LINE(TSRMLS_C));
} else {
2013-11-12 06:03:37 +00:00
printf(
"%sBreakpoint exists at %s%s\n",
2013-11-12 06:03:37 +00:00
PHPDBG_BOLD_LINE(TSRMLS_C), name, PHPDBG_END_LINE(TSRMLS_C));
}
} /* }}} */
2013-11-12 01:41:14 +00:00
void phpdbg_set_breakpoint_method(const char* class_name,
size_t class_len,
2013-11-12 01:41:14 +00:00
const char* func_name,
size_t func_len TSRMLS_DC) /* {{{ */
{
HashTable class_breaks, *class_table;
2013-11-12 01:41:14 +00:00
2013-11-12 01:51:50 +00:00
if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], class_name, class_len, (void**)&class_table) != SUCCESS) {
zend_hash_init(
&class_breaks, 8, NULL, phpdbg_class_breaks_dtor, 0);
zend_hash_update(
&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD],
class_name, class_len,
(void**)&class_breaks, sizeof(HashTable), (void**)&class_table);
}
if (!zend_hash_exists(class_table, func_name, func_len)) {
phpdbg_breakmethod_t new_break;
2013-11-12 01:41:14 +00:00
PHPDBG_G(flags) |= PHPDBG_HAS_METHOD_BP;
2013-11-12 01:41:14 +00:00
new_break.class_name = class_name;
new_break.class_len = class_len;
new_break.func_name = func_name;
new_break.func_len = func_len;
new_break.id = PHPDBG_G(bp_count)++;
2013-11-12 01:41:14 +00:00
zend_hash_update(class_table, func_name, func_len, &new_break, sizeof(phpdbg_breakmethod_t), NULL);
printf(
"%sBreakpoint #%d added at %s::%s%s\n",
2013-11-12 06:03:37 +00:00
PHPDBG_BOLD_LINE(TSRMLS_C),
new_break.id, class_name, func_name,
PHPDBG_END_LINE(TSRMLS_C));
} else {
printf(
"%sBreakpoint exists at %s::%s%s\n",
PHPDBG_BOLD_LINE(TSRMLS_C),
class_name, func_name,
2013-11-12 06:03:37 +00:00
PHPDBG_END_LINE(TSRMLS_C));
}
} /* }}} */
void phpdbg_set_breakpoint_opline(zend_ulong opline TSRMLS_DC) /* {{{ */
2013-11-11 11:54:41 +00:00
{
2013-11-12 01:51:50 +00:00
if (!zend_hash_index_exists(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], opline)) {
2013-11-11 11:54:41 +00:00
phpdbg_breakline_t new_break;
PHPDBG_G(flags) |= PHPDBG_HAS_OPLINE_BP;
2013-11-11 22:06:36 +00:00
new_break.name = NULL;
2013-11-11 13:04:21 +00:00
new_break.opline = opline;
2013-11-11 11:54:41 +00:00
new_break.id = PHPDBG_G(bp_count)++;
2013-11-11 22:06:36 +00:00
2013-11-12 01:51:50 +00:00
zend_hash_index_update(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], opline, &new_break, sizeof(phpdbg_breakline_t), NULL);
2013-11-11 22:06:36 +00:00
printf("%sBreakpoint #%d added at %#lx%s\n",
PHPDBG_BOLD_LINE(TSRMLS_C),
new_break.id, new_break.opline,
2013-11-12 06:03:37 +00:00
PHPDBG_END_LINE(TSRMLS_C));
2013-11-11 11:54:41 +00:00
} else {
2013-11-12 06:03:37 +00:00
printf(
"%sBreakpoint exists at %#lx%s\n",
PHPDBG_BOLD_LINE(TSRMLS_C), opline, PHPDBG_END_LINE(TSRMLS_C));
2013-11-11 11:54:41 +00:00
}
} /* }}} */
2013-11-11 14:33:53 +00:00
void phpdbg_set_breakpoint_opline_ex(phpdbg_opline_ptr_t opline TSRMLS_DC) /* {{{ */
{
2013-11-12 01:51:50 +00:00
if (!zend_hash_index_exists(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], (zend_ulong) opline)) {
2013-11-11 14:33:53 +00:00
phpdbg_breakline_t new_break;
PHPDBG_G(flags) |= PHPDBG_HAS_OPLINE_BP;
2013-11-11 22:06:36 +00:00
new_break.opline = (zend_ulong) opline;
2013-11-11 14:33:53 +00:00
new_break.id = PHPDBG_G(bp_count)++;
2013-11-11 22:06:36 +00:00
2013-11-12 01:51:50 +00:00
zend_hash_index_update(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], (zend_ulong) opline, &new_break, sizeof(phpdbg_breakline_t), NULL);
2013-11-11 22:06:36 +00:00
2013-11-12 06:03:37 +00:00
printf(
"%sBreakpoint #%d added at %#lx%s\n",
PHPDBG_BOLD_LINE(TSRMLS_C),
new_break.id, new_break.opline,
2013-11-12 06:03:37 +00:00
PHPDBG_END_LINE(TSRMLS_C));
2013-11-11 14:33:53 +00:00
}
} /* }}} */
2013-11-10 20:06:19 +00:00
int phpdbg_find_breakpoint_file(zend_op_array *op_array TSRMLS_DC) /* {{{ */
{
size_t name_len = strlen(op_array->filename);
zend_llist *break_list;
2013-11-10 19:15:32 +00:00
zend_llist_element *le;
2013-11-12 01:51:50 +00:00
if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], op_array->filename, name_len,
2013-11-10 19:15:32 +00:00
(void**)&break_list) == FAILURE) {
return FAILURE;
}
2013-11-10 19:15:32 +00:00
for (le = break_list->head; le; le = le->next) {
const phpdbg_breakfile_t *bp = (phpdbg_breakfile_t*)le->data;
2013-11-10 19:15:32 +00:00
if (bp->line == (*EG(opline_ptr))->lineno) {
printf("%sBreakpoint #%d at %s:%ld%s\n",
PHPDBG_BOLD_LINE(TSRMLS_C),
2013-11-12 06:03:37 +00:00
bp->id, bp->filename, bp->line,
PHPDBG_END_LINE(TSRMLS_C));
2013-11-10 19:15:32 +00:00
return SUCCESS;
}
}
return FAILURE;
} /* }}} */
2013-11-10 20:06:19 +00:00
int phpdbg_find_breakpoint_symbol(zend_function *fbc TSRMLS_DC) /* {{{ */
{
const char *fname;
2013-11-12 00:40:40 +00:00
zend_op_array *ops;
2013-11-10 21:38:58 +00:00
phpdbg_breaksymbol_t *bp;
if (fbc->type != ZEND_USER_FUNCTION) {
return FAILURE;
}
2013-11-12 01:41:14 +00:00
2013-11-12 00:40:40 +00:00
ops = (zend_op_array*)fbc;
2013-11-12 01:41:14 +00:00
2013-11-12 00:40:40 +00:00
if (ops->scope) {
2013-11-12 00:47:27 +00:00
/* find method breaks here */
return phpdbg_find_breakpoint_method(
ops TSRMLS_CC);
2013-11-12 00:40:40 +00:00
}
2013-11-12 00:47:27 +00:00
fname = ops->function_name;
if (!fname) {
fname = "main";
}
2013-11-10 21:38:58 +00:00
2013-11-12 01:51:50 +00:00
if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], fname, strlen(fname),
2013-11-10 21:38:58 +00:00
(void**)&bp) == SUCCESS) {
printf("%sBreakpoint #%d in %s() at %s:%u%s\n",
2013-11-12 06:03:37 +00:00
PHPDBG_BOLD_LINE(TSRMLS_C),
bp->id, bp->symbol,
2013-11-11 22:06:36 +00:00
zend_get_executed_filename(TSRMLS_C),
2013-11-12 06:03:37 +00:00
zend_get_executed_lineno(TSRMLS_C),
PHPDBG_END_LINE(TSRMLS_C));
return SUCCESS;
}
return FAILURE;
2013-11-10 21:57:43 +00:00
} /* }}} */
2013-11-11 11:54:41 +00:00
2013-11-12 00:47:27 +00:00
int phpdbg_find_breakpoint_method(zend_op_array *ops TSRMLS_DC) /* {{{ */
{
HashTable *class_table;
phpdbg_breakmethod_t *bp;
if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], ops->scope->name, ops->scope->name_length,
(void**)&class_table) == SUCCESS) {
if (zend_hash_find(
class_table,
2013-11-12 01:41:14 +00:00
ops->function_name,
strlen(ops->function_name), (void**)&bp) == SUCCESS) {
2013-11-12 01:41:14 +00:00
printf(
"%sBreakpoint #%d in %s::%s() at %s:%u%s\n",
2013-11-12 06:03:37 +00:00
PHPDBG_BOLD_LINE(TSRMLS_C),
bp->id, bp->class_name, bp->func_name,
zend_get_executed_filename(TSRMLS_C),
2013-11-12 06:03:37 +00:00
zend_get_executed_lineno(TSRMLS_C),
PHPDBG_END_LINE(TSRMLS_C));
return SUCCESS;
}
}
return FAILURE;
} /* }}} */
2013-11-11 13:07:02 +00:00
int phpdbg_find_breakpoint_opline(phpdbg_opline_ptr_t opline TSRMLS_DC) /* {{{ */
2013-11-11 11:54:41 +00:00
{
phpdbg_breakline_t *bp;
2013-11-12 01:51:50 +00:00
if (zend_hash_index_find(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], (zend_ulong) opline,
2013-11-11 13:04:21 +00:00
(void**)&bp) == SUCCESS) {
printf("%sBreakpoint #%d in %#lx at %s:%u%s\n",
2013-11-12 06:03:37 +00:00
PHPDBG_BOLD_LINE(TSRMLS_C),
bp->id, bp->opline,
2013-11-11 22:06:36 +00:00
zend_get_executed_filename(TSRMLS_C),
2013-11-12 06:03:37 +00:00
zend_get_executed_lineno(TSRMLS_C),
PHPDBG_END_LINE(TSRMLS_C));
2013-11-11 22:06:36 +00:00
2013-11-11 11:54:41 +00:00
return SUCCESS;
}
2013-11-11 22:06:36 +00:00
2013-11-11 11:54:41 +00:00
return FAILURE;
} /* }}} */
2013-11-11 14:33:53 +00:00
2013-11-11 17:19:05 +00:00
void phpdbg_clear_breakpoints(TSRMLS_D) /* {{{ */
{
2013-11-12 01:51:50 +00:00
zend_hash_clean(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE]);
zend_hash_clean(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM]);
zend_hash_clean(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE]);
zend_hash_clean(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD]);
2013-11-12 01:41:14 +00:00
PHPDBG_G(flags) &= ~PHPDBG_BP_MASK;
2013-11-11 17:19:05 +00:00
PHPDBG_G(bp_count) = 0;
} /* }}} */