diff options
| author | Bob Weinand <bobwei9@hotmail.com> | 2014-10-24 19:29:50 +0200 | 
|---|---|---|
| committer | Bob Weinand <bobwei9@hotmail.com> | 2014-10-24 19:29:50 +0200 | 
| commit | 2bcac53bca8ea82d661f057b6d9ff3c7c84f05a7 (patch) | |
| tree | 681b32fe3d9d342c0461c2641b2d651f095b06f8 /sapi/phpdbg/phpdbg_frame.c | |
| parent | 53560ca06b333b71883269091f7d74c0a25e087b (diff) | |
| parent | c03ac47bafd0ea55055a2f3d4de0bc6bb4d98d8d (diff) | |
| download | php-git-2bcac53bca8ea82d661f057b6d9ff3c7c84f05a7.tar.gz | |
Made phpdbg compatible with new engine
Diffstat (limited to 'sapi/phpdbg/phpdbg_frame.c')
| -rw-r--r-- | sapi/phpdbg/phpdbg_frame.c | 201 | 
1 files changed, 108 insertions, 93 deletions
| diff --git a/sapi/phpdbg/phpdbg_frame.c b/sapi/phpdbg/phpdbg_frame.c index 5678534005..4429a7fb73 100644 --- a/sapi/phpdbg/phpdbg_frame.c +++ b/sapi/phpdbg/phpdbg_frame.c @@ -1,6 +1,6 @@  /*     +----------------------------------------------------------------------+ -   | PHP Version 7                                                        | +   | PHP Version 5                                                        |     +----------------------------------------------------------------------+     | Copyright (c) 1997-2014 The PHP Group                                |     +----------------------------------------------------------------------+ @@ -37,13 +37,7 @@ void phpdbg_restore_frame(TSRMLS_D) /* {{{ */  	/* move things back */  	EG(current_execute_data) = PHPDBG_FRAME(execute_data); -	EG(opline_ptr) = &PHPDBG_EX(opline); -	EG(active_op_array) = PHPDBG_EX(op_array); -	EG(return_value_ptr_ptr) = PHPDBG_EX(original_return_value); -	EG(active_symbol_table) = PHPDBG_EX(symbol_table); -	EG(This) = PHPDBG_EX(current_this); -	EG(scope) = PHPDBG_EX(current_scope); -	EG(called_scope) = PHPDBG_EX(current_called_scope); +	EG(scope) = PHPDBG_EX(scope);  } /* }}} */  void phpdbg_switch_frame(int frame TSRMLS_DC) /* {{{ */ @@ -52,22 +46,27 @@ void phpdbg_switch_frame(int frame TSRMLS_DC) /* {{{ */  	int i = 0;  	if (PHPDBG_FRAME(num) == frame) { -		phpdbg_notice("Already in frame #%d", frame); +		phpdbg_notice("frame", "id=\"%d\"", "Already in frame #%d", frame);  		return;  	} -	while (execute_data) { -		if (i++ == frame) { -			break; -		} +	phpdbg_try_access { +		while (execute_data) { +			if (i++ == frame) { +				break; +			} -		do { -			execute_data = execute_data->prev_execute_data; -		} while (execute_data && execute_data->opline == NULL); -	} +			do { +				execute_data = execute_data->prev_execute_data; +			} while (execute_data && execute_data->opline == NULL); +		} +	} phpdbg_catch_access { +		phpdbg_error("signalsegv", "", "Couldn't switch frames, invalid data source"); +		return; +	} phpdbg_end_try_access();  	if (execute_data == NULL) { -		phpdbg_error("No frame #%d", frame); +		phpdbg_error("frame", "type=\"maxnum\" id=\"%d\"", "No frame #%d", frame);  		return;  	} @@ -80,127 +79,143 @@ void phpdbg_switch_frame(int frame TSRMLS_DC) /* {{{ */  		PHPDBG_FRAME(execute_data) = EG(current_execute_data);  		EG(current_execute_data) = execute_data; -		EG(opline_ptr) = &PHPDBG_EX(opline); -		EG(active_op_array) = PHPDBG_EX(op_array); -		PHPDBG_FRAME(execute_data)->original_return_value = EG(return_value_ptr_ptr); -		EG(return_value_ptr_ptr) = PHPDBG_EX(original_return_value); -		EG(active_symbol_table) = PHPDBG_EX(symbol_table); -		EG(This) = PHPDBG_EX(current_this); -		EG(scope) = PHPDBG_EX(current_scope); -		EG(called_scope) = PHPDBG_EX(current_called_scope); +		EG(scope) = PHPDBG_EX(scope);  	} -	phpdbg_notice("Switched to frame #%d", frame); -	phpdbg_list_file( -		zend_get_executed_filename(TSRMLS_C), -		3, -		zend_get_executed_lineno(TSRMLS_C)-1, -		zend_get_executed_lineno(TSRMLS_C) -		TSRMLS_CC -	); +	phpdbg_notice("frame", "id=\"%d\"", "Switched to frame #%d", frame); + +	{ +		const char *file_chr = zend_get_executed_filename(TSRMLS_C); +		zend_string *file = zend_string_init(file_chr, strlen(file_chr), 0); +		phpdbg_list_file(file, 3, zend_get_executed_lineno(TSRMLS_C) - 1, zend_get_executed_lineno(TSRMLS_C) TSRMLS_CC); +		efree(file); +	}  } /* }}} */ -static void phpdbg_dump_prototype(zval **tmp TSRMLS_DC) /* {{{ */ +static void phpdbg_dump_prototype(zval *tmp TSRMLS_DC) /* {{{ */  { -	zval **funcname, **class, **type, **args, **argstmp; -	char is_class; +	zval *funcname, *class, class_zv, *type, *args, *argstmp; -	zend_hash_find(Z_ARRVAL_PP(tmp), "function", sizeof("function"), -		(void **)&funcname); +	funcname = zend_hash_str_find(Z_ARRVAL_P(tmp), ZEND_STRL("function")); -	if ((is_class = zend_hash_find(Z_ARRVAL_PP(tmp), -		"object", sizeof("object"), (void **)&class)) == FAILURE) { -		is_class = zend_hash_find(Z_ARRVAL_PP(tmp), "class", sizeof("class"), -			(void **)&class); +	if ((class = zend_hash_str_find(Z_ARRVAL_P(tmp), ZEND_STRL("object")))) { +		ZVAL_NEW_STR(&class_zv, Z_OBJCE_P(class)->name); +		class = &class_zv;  	} else { -		zend_get_object_classname(*class, (const char **)&Z_STRVAL_PP(class), -			(uint32_t *)&Z_STRLEN_PP(class) TSRMLS_CC); +		class = zend_hash_str_find(Z_ARRVAL_P(tmp), ZEND_STRL("class")); +	} + +	if (class) { +		type = zend_hash_str_find(Z_ARRVAL_P(tmp), ZEND_STRL("type"));  	} -	if (is_class == SUCCESS) { -		zend_hash_find(Z_ARRVAL_PP(tmp), "type", sizeof("type"), (void **)&type); +	args = zend_hash_str_find(Z_ARRVAL_P(tmp), ZEND_STRL("args")); + +	phpdbg_xml(" symbol=\"%s%s%s\"", class ? Z_STRVAL_P(class) : "", class ? Z_STRVAL_P(type) : "", Z_STRVAL_P(funcname)); + +	if (args) { +		phpdbg_xml(">"); +	} else { +		phpdbg_xml(" />");  	} -	phpdbg_write("%s%s%s(", -		is_class == FAILURE?"":Z_STRVAL_PP(class), -		is_class == FAILURE?"":Z_STRVAL_PP(type), -		Z_STRVAL_PP(funcname) -	); - -	if (zend_hash_find(Z_ARRVAL_PP(tmp), "args", sizeof("args"), -		(void **)&args) == SUCCESS) { -		HashPosition iterator; -		const zend_function *func = phpdbg_get_function( -			Z_STRVAL_PP(funcname), is_class == FAILURE ? NULL : Z_STRVAL_PP(class) TSRMLS_CC); -		const zend_arg_info *arginfo = func ? func->common.arg_info : NULL; -		int j = 0, m = func ? func->common.num_args : 0; +	phpdbg_out("%s%s%s(", class ? Z_STRVAL_P(class) : "", class ? Z_STRVAL_P(type) : "", Z_STRVAL_P(funcname)); + +	if (args) { +		const zend_function *func = NULL; +		const zend_arg_info *arginfo = NULL;  		zend_bool is_variadic = 0; +		int j = 0, m; + +		phpdbg_try_access { +			/* assuming no autoloader call is necessary, class should have been loaded if it's in backtrace ... */ +			if ((func = phpdbg_get_function(Z_STRVAL_P(funcname), class ? Z_STRVAL_P(class) : NULL TSRMLS_CC))) { +				arginfo = func->common.arg_info; +			} +		} phpdbg_end_try_access(); + +		m = func ? func->common.num_args : 0; -		zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(args), &iterator); -		while (zend_hash_get_current_data_ex(Z_ARRVAL_PP(args), -			(void **) &argstmp, &iterator) == SUCCESS) { +		ZEND_HASH_FOREACH_VAL(Z_ARRVAL_P(args), argstmp) {  			if (j) { -				phpdbg_write(", "); +				phpdbg_out(", ");  			} +			phpdbg_xml("<arg %r");  			if (m && j < m) { -#if PHP_VERSION_ID >= 50600 -				is_variadic = arginfo[j].is_variadic; -#endif -				phpdbg_write("%s=%s", -					arginfo[j].name, is_variadic ? "[": ""); +				if (!is_variadic) { +					is_variadic = arginfo ? arginfo[j].is_variadic : 0; +				} +				phpdbg_xml(" variadic=\"%s\" name=\"%s\">", is_variadic ? "variadic" : "", arginfo ? arginfo[j].name : ""); +				phpdbg_out("%s=%s", arginfo ? arginfo[j].name : "?", is_variadic ? "[": ""); + +			} else { +				phpdbg_xml(">");  			}  			++j; -			zend_print_flat_zval_r(*argstmp TSRMLS_CC); -			zend_hash_move_forward_ex(Z_ARRVAL_PP(args), &iterator); -		} +			zend_print_flat_zval_r(argstmp TSRMLS_CC); + +			phpdbg_xml("</arg>"); +		} ZEND_HASH_FOREACH_END(); +  		if (is_variadic) { -			phpdbg_write("]"); +			phpdbg_out("]");  		} +		phpdbg_xml("</frame>");  	} -	phpdbg_write(")"); +	phpdbg_out(")");  }  void phpdbg_dump_backtrace(size_t num TSRMLS_DC) /* {{{ */  { -	zval zbacktrace; -	zval **tmp; -	zval **file, **line;  	HashPosition position; +	zval zbacktrace; +	zval *tmp; +	zval *file, *line;  	int i = 0, limit = num; -	int user_defined;  	if (limit < 0) { -		phpdbg_error("Invalid backtrace size %d", limit); +		phpdbg_error("backtrace", "type=\"minnum\"", "Invalid backtrace size %d", limit); +		return;  	} -	zend_fetch_debug_backtrace( -		&zbacktrace, 0, 0, limit TSRMLS_CC); +	phpdbg_try_access { +		zend_fetch_debug_backtrace(&zbacktrace, 0, 0, limit TSRMLS_CC); +	} phpdbg_catch_access { +		phpdbg_error("signalsegv", "", "Couldn't fetch backtrace, invalid data source"); +		return; +	} phpdbg_end_try_access(); + +	phpdbg_xml("<backtrace %r>");  	zend_hash_internal_pointer_reset_ex(Z_ARRVAL(zbacktrace), &position); -	zend_hash_get_current_data_ex(Z_ARRVAL(zbacktrace), (void**)&tmp, &position); +	tmp = zend_hash_get_current_data_ex(Z_ARRVAL(zbacktrace), &position);  	while (1) { -		user_defined = zend_hash_find(Z_ARRVAL_PP(tmp), "file", sizeof("file"), (void **)&file); -		zend_hash_find(Z_ARRVAL_PP(tmp), "line", sizeof("line"), (void **)&line); +		file = zend_hash_str_find(Z_ARRVAL_P(tmp), ZEND_STRL("file")); +		line = zend_hash_str_find(Z_ARRVAL_P(tmp), ZEND_STRL("line"));  		zend_hash_move_forward_ex(Z_ARRVAL(zbacktrace), &position); -		if (zend_hash_get_current_data_ex(Z_ARRVAL(zbacktrace), -			(void**)&tmp, &position) == FAILURE) { -			phpdbg_write("frame #%d: {main} at %s:%ld", i, Z_STRVAL_PP(file), Z_LVAL_PP(line)); +		if (!(tmp = zend_hash_get_current_data_ex(Z_ARRVAL(zbacktrace), &position))) { +			phpdbg_write("frame", "id=\"%d\" symbol=\"{main}\" file=\"%s\" line=\"%d\"", "frame #%d: {main} at %s:%ld", i, Z_STRVAL_P(file), Z_LVAL_P(line));  			break;  		} -		if (user_defined == SUCCESS) { -			phpdbg_write("frame #%d: ", i++); +		if (file) { /* userland */ +			phpdbg_out("frame #%d: ", i); +			phpdbg_xml("<frame %r id=\"%d\" file=\"%s\" line=\"%d\"", i, Z_STRVAL_P(file), Z_LVAL_P(line));  			phpdbg_dump_prototype(tmp TSRMLS_CC); -			phpdbg_writeln(" at %s:%ld", Z_STRVAL_PP(file), Z_LVAL_PP(line)); +			phpdbg_out(" at %s:%ld\n", Z_STRVAL_P(file), Z_LVAL_P(line)); +			i++;  		} else { -			phpdbg_write(" => "); +			phpdbg_out(" => "); +			phpdbg_xml("<frame %r id=\"%d\" internal=\"internal\"", i);  			phpdbg_dump_prototype(tmp TSRMLS_CC); -			phpdbg_writeln(" (internal function)"); +			phpdbg_out(" (internal function)\n");  		}  	} -	phpdbg_writeln(EMPTY); +	phpdbg_out("\n"); +	phpdbg_xml("</backtrace>"); +  	zval_dtor(&zbacktrace);  } /* }}} */ | 
