diff options
Diffstat (limited to 'sapi/phpdbg/phpdbg_info.c')
| -rw-r--r-- | sapi/phpdbg/phpdbg_info.c | 502 | 
1 files changed, 288 insertions, 214 deletions
| diff --git a/sapi/phpdbg/phpdbg_info.c b/sapi/phpdbg/phpdbg_info.c index 6c5b9b02d5..f34485d6b5 100644 --- a/sapi/phpdbg/phpdbg_info.c +++ b/sapi/phpdbg/phpdbg_info.c @@ -1,6 +1,6 @@  /*     +----------------------------------------------------------------------+ -   | PHP Version 5                                                        | +   | PHP Version 7                                                        |     +----------------------------------------------------------------------+     | Copyright (c) 1997-2015 The PHP Group                                |     +----------------------------------------------------------------------+ @@ -27,50 +27,56 @@  ZEND_EXTERN_MODULE_GLOBALS(phpdbg); -#define PHPDBG_INFO_COMMAND_D(f, h, a, m, l, s) \ -	PHPDBG_COMMAND_D_EXP(f, h, a, m, l, s, &phpdbg_prompt_commands[14]) +#define PHPDBG_INFO_COMMAND_D(f, h, a, m, l, s, flags) \ +	PHPDBG_COMMAND_D_EXP(f, h, a, m, l, s, &phpdbg_prompt_commands[13], flags)  const phpdbg_command_t phpdbg_info_commands[] = { -	PHPDBG_INFO_COMMAND_D(break,    "show breakpoints",              'b', info_break,   NULL, 0), -	PHPDBG_INFO_COMMAND_D(files,    "show included files",           'F', info_files,   NULL, 0), -	PHPDBG_INFO_COMMAND_D(classes,  "show loaded classes",           'c', info_classes, NULL, 0), -	PHPDBG_INFO_COMMAND_D(funcs,    "show loaded classes",           'f', info_funcs,   NULL, 0), -	PHPDBG_INFO_COMMAND_D(error,    "show last error",               'e', info_error,   NULL, 0), -	PHPDBG_INFO_COMMAND_D(vars,     "show active variables",         'v', info_vars,    NULL, 0), -	PHPDBG_INFO_COMMAND_D(literal,  "show active literal constants", 'l', info_literal, NULL, 0), -	PHPDBG_INFO_COMMAND_D(memory,   "show memory manager stats",     'm', info_memory,  NULL, 0), +	PHPDBG_INFO_COMMAND_D(break,     "show breakpoints",              'b', info_break,     NULL, 0, PHPDBG_ASYNC_SAFE), +	PHPDBG_INFO_COMMAND_D(files,     "show included files",           'F', info_files,     NULL, 0, PHPDBG_ASYNC_SAFE), +	PHPDBG_INFO_COMMAND_D(classes,   "show loaded classes",           'c', info_classes,   NULL, 0, PHPDBG_ASYNC_SAFE), +	PHPDBG_INFO_COMMAND_D(funcs,     "show loaded classes",           'f', info_funcs,     NULL, 0, PHPDBG_ASYNC_SAFE), +	PHPDBG_INFO_COMMAND_D(error,     "show last error",               'e', info_error,     NULL, 0, PHPDBG_ASYNC_SAFE), +	PHPDBG_INFO_COMMAND_D(constants, "show user defined constants",   'd', info_constants, NULL, 0, PHPDBG_ASYNC_SAFE), +	PHPDBG_INFO_COMMAND_D(vars,      "show active variables",         'v', info_vars,      NULL, 0, PHPDBG_ASYNC_SAFE), +	PHPDBG_INFO_COMMAND_D(globals,   "show superglobals",             'g', info_globals,   NULL, 0, PHPDBG_ASYNC_SAFE), +	PHPDBG_INFO_COMMAND_D(literal,   "show active literal constants", 'l', info_literal,   NULL, 0, PHPDBG_ASYNC_SAFE), +	PHPDBG_INFO_COMMAND_D(memory,    "show memory manager stats",     'm', info_memory,    NULL, 0, PHPDBG_ASYNC_SAFE),  	PHPDBG_END_COMMAND  };  PHPDBG_INFO(break) /* {{{ */  { -	phpdbg_print_breakpoints(PHPDBG_BREAK_FILE TSRMLS_CC); -	phpdbg_print_breakpoints(PHPDBG_BREAK_SYM TSRMLS_CC); -	phpdbg_print_breakpoints(PHPDBG_BREAK_METHOD TSRMLS_CC); -	phpdbg_print_breakpoints(PHPDBG_BREAK_OPLINE TSRMLS_CC); -	phpdbg_print_breakpoints(PHPDBG_BREAK_FILE_OPLINE TSRMLS_CC); -	phpdbg_print_breakpoints(PHPDBG_BREAK_FUNCTION_OPLINE TSRMLS_CC); -	phpdbg_print_breakpoints(PHPDBG_BREAK_METHOD_OPLINE TSRMLS_CC); -	phpdbg_print_breakpoints(PHPDBG_BREAK_COND TSRMLS_CC); -	phpdbg_print_breakpoints(PHPDBG_BREAK_OPCODE TSRMLS_CC); +	phpdbg_print_breakpoints(PHPDBG_BREAK_FILE); +	phpdbg_print_breakpoints(PHPDBG_BREAK_SYM); +	phpdbg_print_breakpoints(PHPDBG_BREAK_METHOD); +	phpdbg_print_breakpoints(PHPDBG_BREAK_OPLINE); +	phpdbg_print_breakpoints(PHPDBG_BREAK_FILE_OPLINE); +	phpdbg_print_breakpoints(PHPDBG_BREAK_FUNCTION_OPLINE); +	phpdbg_print_breakpoints(PHPDBG_BREAK_METHOD_OPLINE); +	phpdbg_print_breakpoints(PHPDBG_BREAK_COND); +	phpdbg_print_breakpoints(PHPDBG_BREAK_OPCODE);  	return SUCCESS;  } /* }}} */  PHPDBG_INFO(files) /* {{{ */  { -	HashPosition pos; -	char *fname; +	zend_string *fname; -	phpdbg_notice("Included files: %d", -		zend_hash_num_elements(&EG(included_files))); +	phpdbg_try_access { +		phpdbg_notice("includedfilecount", "num=\"%d\"", "Included files: %d", zend_hash_num_elements(&EG(included_files))); +	} phpdbg_catch_access { +		phpdbg_error("signalsegv", "", "Could not fetch included file count, invalid data source"); +		return SUCCESS; +	} phpdbg_end_try_access(); -	zend_hash_internal_pointer_reset_ex(&EG(included_files), &pos); -	while (zend_hash_get_current_key_ex(&EG(included_files), &fname, -		NULL, NULL, 0, &pos) == HASH_KEY_IS_STRING) { -		phpdbg_writeln("File: %s", fname); -		zend_hash_move_forward_ex(&EG(included_files), &pos); -	} +	phpdbg_try_access { +		ZEND_HASH_FOREACH_STR_KEY(&EG(included_files), fname) { +			phpdbg_writeln("includedfile", "name=\"%s\"", "File: %s", fname); +		} ZEND_HASH_FOREACH_END(); +	} phpdbg_catch_access { +		phpdbg_error("signalsegv", "", "Could not fetch file name, invalid data source, aborting included file listing"); +	} phpdbg_end_try_access();  	return SUCCESS;  } /* }}} */ @@ -78,125 +84,189 @@ PHPDBG_INFO(files) /* {{{ */  PHPDBG_INFO(error) /* {{{ */  {  	if (PG(last_error_message)) { -		phpdbg_writeln("Last error: %s at %s line %d", -			PG(last_error_message), PG(last_error_file), PG(last_error_lineno)); +		phpdbg_try_access { +			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)); +		} phpdbg_catch_access { +			phpdbg_notice("lasterror", "error=\"\"", "No error found!"); +		} phpdbg_end_try_access();  	} else { -		phpdbg_notice("No error found!"); +		phpdbg_notice("lasterror", "error=\"\"", "No error found!");  	}  	return SUCCESS;  } /* }}} */ -PHPDBG_INFO(vars) /* {{{ */ +PHPDBG_INFO(constants) /* {{{ */  { -	HashTable vars; -	HashPosition pos; -	char *var; -	zval **data; +	HashTable consts; +	zend_constant *data; -	if (!EG(active_op_array)) { -		phpdbg_error("No active op array!"); -		return SUCCESS; +	zend_hash_init(&consts, 8, NULL, NULL, 0); + +	if (EG(zend_constants)) { +		phpdbg_try_access { +			ZEND_HASH_FOREACH_PTR(EG(zend_constants), data) { +				if (data->module_number == PHP_USER_CONSTANT) { +					zend_hash_update_ptr(&consts, data->name, data); +				} +			} ZEND_HASH_FOREACH_END(); +		} phpdbg_catch_access { +			phpdbg_error("signalsegv", "", "Cannot fetch all the constants, invalid data source"); +		} phpdbg_end_try_access();  	} -	if (!EG(active_symbol_table)) { -		zend_rebuild_symbol_table(TSRMLS_C); +	phpdbg_notice("constantinfo", "num=\"%d\"", "User-defined constants (%d)", zend_hash_num_elements(&consts)); + +	if (zend_hash_num_elements(&consts)) { +		phpdbg_out("Address            Refs    Type      Constant\n"); +		ZEND_HASH_FOREACH_PTR(&consts, data) { + +#define VARIABLEINFO(attrs, msg, ...) phpdbg_writeln("constant", "address=\"%p\" refcount=\"%d\" type=\"%s\" name=\"%.*s\" " attrs, "%-18p %-7d %-9s %.*s" msg, &data->value, Z_REFCOUNT(data->value), zend_zval_type_name(&data->value), data->name->len, data->name->val, ##__VA_ARGS__) + +			switch (Z_TYPE(data->value)) { +				case IS_STRING: +					phpdbg_try_access { +						VARIABLEINFO("length=\"%d\" value=\"%.*s\"", "\nstring (%d) \"%.*s%s\"", Z_STRLEN(data->value), Z_STRLEN(data->value) < 255 ? Z_STRLEN(data->value) : 255, Z_STRVAL(data->value), Z_STRLEN(data->value) > 255 ? "..." : ""); +					} phpdbg_catch_access { +						VARIABLEINFO("", ""); +					} phpdbg_end_try_access(); +					break; +				case IS_TRUE: +					VARIABLEINFO("value=\"true\"", "\nbool (true)"); +					break; +				case IS_FALSE: +					VARIABLEINFO("value=\"false\"", "\nbool (false)"); +					break; +				case IS_LONG: +					VARIABLEINFO("value=\"%ld\"", "\nint (%ld)", Z_LVAL(data->value)); +					break; +				case IS_DOUBLE: +					VARIABLEINFO("value=\"%lf\"", "\ndouble (%lf)", Z_DVAL(data->value)); +					break; +				default: +					VARIABLEINFO("", ""); + +#undef VARIABLEINFO +			} +		} ZEND_HASH_FOREACH_END(); +	} -		if (!EG(active_symbol_table)) { -			phpdbg_error("No active symbol table!"); -			return SUCCESS; +	return SUCCESS; +} /* }}} */ + +static int phpdbg_arm_auto_global(zend_auto_global *auto_global) { +	if (auto_global->armed) { +		if (PHPDBG_G(flags) & PHPDBG_IN_SIGNAL_HANDLER) { +			phpdbg_notice("variableinfo", "unreachable=\"%.*s\"", "Cannot show information about superglobal variable %.*s", auto_global->name->len, auto_global->name->val); +		} else { +			auto_global->armed = auto_global->auto_global_callback(auto_global->name);  		}  	} -	zend_hash_init(&vars, 8, NULL, NULL, 0); +	return 0; +} -	zend_hash_internal_pointer_reset_ex(EG(active_symbol_table), &pos); -	while (zend_hash_get_current_key_ex(EG(active_symbol_table), &var, -		NULL, NULL, 0, &pos) == HASH_KEY_IS_STRING) { -		zend_hash_get_current_data_ex(EG(active_symbol_table), (void **)&data, &pos); -		if (*var != '_') { -			zend_hash_update( -				&vars, var, strlen(var)+1, (void**)data, sizeof(zval*), NULL); -		} -		zend_hash_move_forward_ex(EG(active_symbol_table), &pos); +static int phpdbg_print_symbols(zend_bool show_globals) { +	HashTable vars; +	zend_array *symtable; +	zend_string *var; +	zval *data; + +	if (!EG(current_execute_data) || !EG(current_execute_data)->func) { +		phpdbg_error("inactive", "type=\"op_array\"", "No active op array!"); +		return SUCCESS; +	} + +	if (show_globals) { +		/* that array should only be manipulated during init, so safe for async access during execution */ +		zend_hash_apply(CG(auto_globals), (apply_func_t) phpdbg_arm_auto_global); +		symtable = &EG(symbol_table); +	} else if (!(symtable = zend_rebuild_symbol_table())) { +		phpdbg_error("inactive", "type=\"symbol_table\"", "No active symbol table!"); +		return SUCCESS;  	} -	{ -		zend_op_array *ops = EG(active_op_array); -		 +	zend_hash_init(&vars, 8, NULL, NULL, 0); + +	phpdbg_try_access { +		ZEND_HASH_FOREACH_STR_KEY_VAL(symtable, var, data) { +			if (zend_is_auto_global(var) ^ !show_globals) { +				zend_hash_update(&vars, var, data); +			} +		} ZEND_HASH_FOREACH_END(); +	} phpdbg_catch_access { +		phpdbg_error("signalsegv", "", "Cannot fetch all data from the symbol table, invalid data source"); +	} phpdbg_end_try_access(); + +	if (show_globals) { +		phpdbg_notice("variableinfo", "num=\"%d\"", "Superglobal variables (%d)", zend_hash_num_elements(&vars)); +	} else { +		zend_op_array *ops = &EG(current_execute_data)->func->op_array; +  		if (ops->function_name) {  			if (ops->scope) { -				phpdbg_notice( -				"Variables in %s::%s() (%d)", ops->scope->name, ops->function_name, zend_hash_num_elements(&vars)); +				phpdbg_notice("variableinfo", "method=\"%s::%s\" num=\"%d\"", "Variables in %s::%s() (%d)", ops->scope->name, ops->function_name, zend_hash_num_elements(&vars));  			} else { -				phpdbg_notice( -					"Variables in %s() (%d)", ops->function_name, zend_hash_num_elements(&vars)); +				phpdbg_notice("variableinfo", "function=\"%s\" num=\"%d\"", "Variables in %s() (%d)", ops->function_name, zend_hash_num_elements(&vars));  			}  		} else {  			if (ops->filename) { -				phpdbg_notice( -				"Variables in %s (%d)", ops->filename, zend_hash_num_elements(&vars)); +				phpdbg_notice("variableinfo", "file=\"%s\" num=\"%d\"", "Variables in %s (%d)", ops->filename->val, zend_hash_num_elements(&vars));  			} else { -				phpdbg_notice( -					"Variables @ %p (%d)", ops, zend_hash_num_elements(&vars)); +				phpdbg_notice("variableinfo", "opline=\"%p\" num=\"%d\"", "Variables @ %p (%d)", ops, zend_hash_num_elements(&vars));  			}  		}  	}  	if (zend_hash_num_elements(&vars)) { -		phpdbg_writeln("Address\t\tRefs\tType\t\tVariable"); -		for (zend_hash_internal_pointer_reset_ex(&vars, &pos); -			zend_hash_get_current_data_ex(&vars, (void**) &data, &pos) == SUCCESS; -			zend_hash_move_forward_ex(&vars, &pos)) { -			char *var; - -			zend_hash_get_current_key_ex(&vars, &var, NULL, NULL, 0, &pos); - -			if (*data) { -				phpdbg_write( -				"%p\t%d\t", -					*data, -					Z_REFCOUNT_PP(data)); - -				switch (Z_TYPE_PP(data)) { -					case IS_STRING: 	phpdbg_write("(string)\t"); 	break; -					case IS_LONG: 		phpdbg_write("(integer)\t"); 	break; -					case IS_DOUBLE: 	phpdbg_write("(float)\t"); 		break; -					case IS_RESOURCE:	phpdbg_write("(resource)\t"); 	break; -					case IS_ARRAY:		phpdbg_write("(array)\t"); 		break; -					case IS_OBJECT:		phpdbg_write("(object)\t"); 	break; -					case IS_NULL:		phpdbg_write("(null)\t"); 		break; +		phpdbg_out("Address            Refs    Type      Variable\n"); +		ZEND_HASH_FOREACH_STR_KEY_VAL(&vars, var, data) { +			phpdbg_try_access { +#define VARIABLEINFO(attrs, msg, ...) phpdbg_writeln("variable", "address=\"%p\" refcount=\"%d\" type=\"%s\" refstatus=\"%s\" name=\"%.*s\" " attrs, "%-18p %-7d %-9s %s$%.*s" msg, data, Z_REFCOUNT_P(data), zend_zval_type_name(data), Z_ISREF_P(data) ? "&": "", var->len, var->val, ##__VA_ARGS__) + +				switch (Z_TYPE_P(data)) { +					case IS_RESOURCE: +						phpdbg_try_access { +							const char *type = zend_rsrc_list_get_rsrc_type(Z_RES_P(data)); +							VARIABLEINFO("type=\"%s\"", "\n|-------(typeof)------> (%s)\n", type ? type : "unknown"); +						} phpdbg_catch_access { +							VARIABLEINFO("type=\"unknown\"", "\n|-------(typeof)------> (unknown)\n"); +						} phpdbg_end_try_access(); +						break; +					case IS_OBJECT: +						phpdbg_try_access { +							VARIABLEINFO("instanceof=\"%s\"", "\n|-----(instanceof)----> (%s)\n", Z_OBJCE_P(data)->name); +						} phpdbg_catch_access { +							VARIABLEINFO("instanceof=\"%s\"", "\n|-----(instanceof)----> (unknown)\n"); +						} phpdbg_end_try_access(); +						break; +					case IS_STRING: +						phpdbg_try_access { +							VARIABLEINFO("length=\"%d\" value=\"%.*s\"", "\nstring (%d) \"%.*s%s\"", Z_STRLEN_P(data), Z_STRLEN_P(data) < 255 ? Z_STRLEN_P(data) : 255, Z_STRVAL_P(data), Z_STRLEN_P(data) > 255 ? "..." : ""); +						} phpdbg_catch_access { +							VARIABLEINFO("", ""); +						} phpdbg_end_try_access(); +						break; +					case IS_TRUE: +						VARIABLEINFO("value=\"true\"", "\nbool (true)"); +						break; +					case IS_FALSE: +						VARIABLEINFO("value=\"false\"", "\nbool (false)"); +						break; +					case IS_LONG: +						VARIABLEINFO("value=\"%ld\"", "\nint (%ld)", Z_LVAL_P(data)); +						break; +					case IS_DOUBLE: +						VARIABLEINFO("value=\"%lf\"", "\ndouble (%lf)", Z_DVAL_P(data)); +						break; +					default: +						VARIABLEINFO("", "");  				} -				if (Z_TYPE_PP(data) == IS_RESOURCE) { -					int type; - -					phpdbg_writeln( -						"%s$%s", Z_ISREF_PP(data) ? "&": "", var); -					if (zend_list_find(Z_RESVAL_PP(data), &type)) { -						phpdbg_write( -							"|-------(typeof)------> (%s)", -							zend_rsrc_list_get_rsrc_type(type TSRMLS_CC)); -					} else { -						phpdbg_write( -							"|-------(typeof)------> (unknown)"); -					} -					phpdbg_writeln(EMPTY); -				} else if (Z_TYPE_PP(data) == IS_OBJECT) { -					phpdbg_writeln( -						"%s$%s", Z_ISREF_PP(data) ? "&": "", var); -					phpdbg_write( -						"|-----(instanceof)----> (%s)", Z_OBJCE_PP(data)->name); -					phpdbg_writeln(EMPTY); -				} else { -					phpdbg_write( -						"%s$%s", Z_ISREF_PP(data) ? "&": "", var); -				} -			} else { -				phpdbg_write( -					"n/a\tn/a\tn/a\t$%s", var); -			} -			phpdbg_writeln(EMPTY); -		} +#undef VARIABLEINFO +			} phpdbg_catch_access { +				phpdbg_writeln("variable", "address=\"%p\" name=\"%s\"", "%p\tn/a\tn/a\t$%s", data, var); +			} phpdbg_end_try_access(); +		} ZEND_HASH_FOREACH_END();  	}  	zend_hash_destroy(&vars); @@ -204,42 +274,48 @@ PHPDBG_INFO(vars) /* {{{ */  	return SUCCESS;  } /* }}} */ +PHPDBG_INFO(vars) /* {{{ */ +{ +	return phpdbg_print_symbols(0); +} + +PHPDBG_INFO(globals) /* {{{ */ +{ +	return phpdbg_print_symbols(1); +} +  PHPDBG_INFO(literal) /* {{{ */  { -	if ((EG(in_execution) && EG(active_op_array)) || PHPDBG_G(ops)) { -		zend_op_array *ops = EG(active_op_array) ? EG(active_op_array) : PHPDBG_G(ops); +	/* literals are assumed to not be manipulated during executing of their op_array and as such async safe */ +	zend_bool in_executor = PHPDBG_G(in_execution) && EG(current_execute_data) && EG(current_execute_data)->func; +	if (in_executor || PHPDBG_G(ops)) { +		zend_op_array *ops = in_executor ? &EG(current_execute_data)->func->op_array : PHPDBG_G(ops);  		int literal = 0, count = ops->last_literal-1;  		if (ops->function_name) {  			if (ops->scope) { -				phpdbg_notice( -				"Literal Constants in %s::%s() (%d)", ops->scope->name, ops->function_name, count); +				phpdbg_notice("literalinfo", "method=\"%s::%s\" num=\"%d\"", "Literal Constants in %s::%s() (%d)", ops->scope->name, ops->function_name, count);  			} else { -				phpdbg_notice( -					"Literal Constants in %s() (%d)", ops->function_name, count); +				phpdbg_notice("literalinfo", "function=\"%s\" num=\"%d\"", "Literal Constants in %s() (%d)", ops->function_name, count);  			}  		} else {  			if (ops->filename) { -				phpdbg_notice( -				"Literal Constants in %s (%d)", ops->filename, count); +				phpdbg_notice("literalinfo", "file=\"%s\" num=\"%d\"", "Literal Constants in %s (%d)", ops->filename->val, count);  			} else { -				phpdbg_notice( -					"Literal Constants @ %p (%d)", ops, count); +				phpdbg_notice("literalinfo", "opline=\"%p\" num=\"%d\"", "Literal Constants @ %p (%d)", ops, count);  			}  		}  		while (literal < ops->last_literal) { -			if (Z_TYPE(ops->literals[literal].constant) != IS_NULL) { -				phpdbg_write("|-------- C%u -------> [", literal); -				zend_print_zval( -					&ops->literals[literal].constant, 0); -				phpdbg_write("]"); -				phpdbg_writeln(EMPTY); +			if (Z_TYPE(ops->literals[literal]) != IS_NULL) { +				phpdbg_write("literal", "id=\"%u\"", "|-------- C%u -------> [", literal); +				zend_print_zval(&ops->literals[literal], 0); +				phpdbg_out("]\n");  			}  			literal++;  		}  	} else { -		phpdbg_error("Not executing!"); +		phpdbg_error("inactive", "type=\"execution\"", "Not executing!");  	}  	return SUCCESS; @@ -247,85 +323,85 @@ PHPDBG_INFO(literal) /* {{{ */  PHPDBG_INFO(memory) /* {{{ */  { -	if (is_zend_mm(TSRMLS_C)) { -		phpdbg_notice("Memory Manager Information"); -		phpdbg_notice("Current"); -		phpdbg_writeln("|-------> Used:\t%.3f kB",  -			(float) (zend_memory_usage(0 TSRMLS_CC)/1024)); -		phpdbg_writeln("|-------> Real:\t%.3f kB",  -			(float) (zend_memory_usage(1 TSRMLS_CC)/1024)); -		phpdbg_notice("Peak"); -		phpdbg_writeln("|-------> Used:\t%.3f kB",  -			(float) (zend_memory_peak_usage(0 TSRMLS_CC)/1024)); -		phpdbg_writeln("|-------> Real:\t%.3f kB",  -			(float) (zend_memory_peak_usage(1 TSRMLS_CC)/1024)); +	size_t used, real, peak_used, peak_real; +	zend_mm_heap *heap; +	zend_bool is_mm; + +	if (PHPDBG_G(flags) & PHPDBG_IN_SIGNAL_HANDLER) { +		heap = zend_mm_set_heap(phpdbg_original_heap_sigsafe_mem()); +	} +	if ((is_mm = is_zend_mm())) { +		used = zend_memory_usage(0); +		real = zend_memory_usage(1); +		peak_used = zend_memory_peak_usage(0); +		peak_real = zend_memory_peak_usage(1); +	} +	if (PHPDBG_G(flags) & PHPDBG_IN_SIGNAL_HANDLER) { +		zend_mm_set_heap(heap); +	} + +	if (is_mm) { +		phpdbg_notice("meminfo", "", "Memory Manager Information"); +		phpdbg_notice("current", "", "Current"); +		phpdbg_writeln("used", "mem=\"%.3f\"", "|-------> Used:\t%.3f kB", (float) (used / 1024)); +		phpdbg_writeln("real", "mem=\"%.3f\"", "|-------> Real:\t%.3f kB", (float) (real / 1024)); +		phpdbg_notice("peak", "", "Peak"); +		phpdbg_writeln("used", "mem=\"%.3f\"", "|-------> Used:\t%.3f kB", (float) (peak_used / 1024)); +		phpdbg_writeln("real", "mem=\"%.3f\"", "|-------> Real:\t%.3f kB", (float) (peak_real / 1024));  	} else { -		phpdbg_error("Memory Manager Disabled!"); +		phpdbg_error("inactive", "type=\"memory_manager\"", "Memory Manager Disabled!");  	}  	return SUCCESS;  } /* }}} */ -static inline void phpdbg_print_class_name(zend_class_entry **ce TSRMLS_DC) /* {{{ */ +static inline void phpdbg_print_class_name(zend_class_entry *ce) /* {{{ */  { -	phpdbg_write( -		"%s %s %s (%d)", -		((*ce)->type == ZEND_USER_CLASS) ? -			"User" : "Internal", -		((*ce)->ce_flags & ZEND_ACC_INTERFACE) ? -			"Interface" : -			((*ce)->ce_flags & ZEND_ACC_ABSTRACT) ? -				"Abstract Class" : -					"Class", -		(*ce)->name, zend_hash_num_elements(&(*ce)->function_table)); +	const char *visibility = ce->type == ZEND_USER_CLASS ? "User" : "Internal"; +	const char *type = (ce->ce_flags & ZEND_ACC_INTERFACE) ? "Interface" : (ce->ce_flags & ZEND_ACC_ABSTRACT) ? "Abstract Class" : "Class"; + +	phpdbg_writeln("class", "type=\"%s\" flags=\"%s\" name=\"%.*s\" methodcount=\"%d\"", "%s %s %.*s (%d)", visibility, type, ce->name->len, ce->name->val, zend_hash_num_elements(&ce->function_table));  } /* }}} */  PHPDBG_INFO(classes) /* {{{ */  { -	HashPosition position; -	zend_class_entry **ce; +	zend_class_entry *ce;  	HashTable classes;  	zend_hash_init(&classes, 8, NULL, NULL, 0); -	for (zend_hash_internal_pointer_reset_ex(EG(class_table), &position); -		zend_hash_get_current_data_ex(EG(class_table), (void**)&ce, &position) == SUCCESS; -		zend_hash_move_forward_ex(EG(class_table), &position)) { - -		if ((*ce)->type == ZEND_USER_CLASS) { -			zend_hash_next_index_insert( -				&classes, ce, sizeof(ce), NULL); -		} -	} - -	phpdbg_notice("User Classes (%d)", -		zend_hash_num_elements(&classes)); +	phpdbg_try_access { +		ZEND_HASH_FOREACH_PTR(EG(class_table), ce) { +			if (ce->type == ZEND_USER_CLASS) { +				zend_hash_next_index_insert_ptr(&classes, ce); +			} +		} ZEND_HASH_FOREACH_END(); +	} phpdbg_catch_access { +		phpdbg_notice("signalsegv", "", "Not all classes could be fetched, possibly invalid data source"); +	} phpdbg_end_try_access(); -	for (zend_hash_internal_pointer_reset_ex(&classes, &position); -		zend_hash_get_current_data_ex(&classes, (void**)&ce, &position) == SUCCESS; -		zend_hash_move_forward_ex(&classes, &position)) { +	phpdbg_notice("classinfo", "num=\"%d\"", "User Classes (%d)", zend_hash_num_elements(&classes)); -		phpdbg_print_class_name(ce TSRMLS_CC); -		phpdbg_writeln(EMPTY); +	/* once added, assume that classes are stable... until shutdown. */ +	ZEND_HASH_FOREACH_PTR(&classes, ce) { +		phpdbg_print_class_name(ce); -		if ((*ce)->parent) { -			zend_class_entry *pce = (*ce)->parent; +		if (ce->parent) { +			zend_class_entry *pce; +			phpdbg_xml("<parents %r>"); +			pce = ce->parent;  			do { -				phpdbg_write("|-------- "); -				phpdbg_print_class_name(&pce TSRMLS_CC); -				phpdbg_writeln(EMPTY); +				phpdbg_out("|-------- "); +				phpdbg_print_class_name(pce);  			} while ((pce = pce->parent)); +			phpdbg_xml("</parents>");  		} -		if ((*ce)->info.user.filename) { -			phpdbg_writeln( -				"|---- in %s on line %u", -				(*ce)->info.user.filename, -				(*ce)->info.user.line_start); +		if (ce->info.user.filename) { +			phpdbg_writeln("classsource", "file=\"%s\" line=\"%u\"", "|---- in %s on line %u", ce->info.user.filename->val, ce->info.user.line_start);  		} else { -			phpdbg_writeln("|---- no source code"); +			phpdbg_writeln("classsource", "", "|---- no source code");  		} -	    	phpdbg_writeln(EMPTY); -	} +	} ZEND_HASH_FOREACH_END();  	zend_hash_destroy(&classes); @@ -334,36 +410,34 @@ PHPDBG_INFO(classes) /* {{{ */  PHPDBG_INFO(funcs) /* {{{ */  { -	HashPosition position; -	zend_function *zf, **pzf; +	zend_function *zf;  	HashTable functions;  	zend_hash_init(&functions, 8, NULL, NULL, 0); -	for (zend_hash_internal_pointer_reset_ex(EG(function_table), &position); -		zend_hash_get_current_data_ex(EG(function_table), (void**)&zf, &position) == SUCCESS; -		zend_hash_move_forward_ex(EG(function_table), &position)) { +	phpdbg_try_access { +		ZEND_HASH_FOREACH_PTR(EG(function_table), zf) { +			if (zf->type == ZEND_USER_FUNCTION) { +				zend_hash_next_index_insert_ptr(&functions, zf); +			} +		} ZEND_HASH_FOREACH_END(); +	} phpdbg_catch_access { +		phpdbg_notice("signalsegv", "", "Not all functions could be fetched, possibly invalid data source"); +	} phpdbg_end_try_access(); -		if (zf->type == ZEND_USER_FUNCTION) { -			zend_hash_next_index_insert( -				&functions, (void**) &zf, sizeof(zend_function), NULL); -		} -	} +	phpdbg_notice("functioninfo", "num=\"%d\"", "User Functions (%d)", zend_hash_num_elements(&functions)); -	phpdbg_notice("User Functions (%d)", -		zend_hash_num_elements(&functions)); +	ZEND_HASH_FOREACH_PTR(&functions, zf) { +		zend_op_array *op_array = &zf->op_array; -	for (zend_hash_internal_pointer_reset_ex(&functions, &position); -		zend_hash_get_current_data_ex(&functions, (void**)&pzf, &position) == SUCCESS; -		zend_hash_move_forward_ex(&functions, &position)) { -		zend_op_array *op_array = &((*pzf)->op_array); +		phpdbg_write("function", "name=\"%s\"", "|-------- %s", op_array->function_name ? op_array->function_name->val : "{main}"); -		phpdbg_writeln( -			"|-------- %s in %s on line %d", -			op_array->function_name ? op_array->function_name : "{main}", -			op_array->filename ? op_array->filename : "(no source code)", -			op_array->line_start); -	} +		if (op_array->filename) { +			phpdbg_writeln("functionsource", "file=\"%s\" line=\"%d\"", " in %s on line %d", op_array->filename->val, op_array->line_start); +		} else { +			phpdbg_writeln("functionsource", "", " (no source code)"); +		} +	} ZEND_HASH_FOREACH_END();  	zend_hash_destroy(&functions); | 
