diff options
Diffstat (limited to 'sapi/phpdbg/phpdbg_bp.c')
| -rw-r--r-- | sapi/phpdbg/phpdbg_bp.c | 1264 | 
1 files changed, 627 insertions, 637 deletions
| diff --git a/sapi/phpdbg/phpdbg_bp.c b/sapi/phpdbg/phpdbg_bp.c index 06a6ad72be..4cb1dae42c 100644 --- a/sapi/phpdbg/phpdbg_bp.c +++ b/sapi/phpdbg/phpdbg_bp.c @@ -1,6 +1,6 @@  /*     +----------------------------------------------------------------------+ -   | PHP Version 5                                                        | +   | PHP Version 7                                                        |     +----------------------------------------------------------------------+     | Copyright (c) 1997-2016 The PHP Group                                |     +----------------------------------------------------------------------+ @@ -26,28 +26,28 @@  #include "phpdbg_opcode.h"  #include "zend_globals.h" -ZEND_EXTERN_MODULE_GLOBALS(phpdbg); +ZEND_EXTERN_MODULE_GLOBALS(phpdbg)  /* {{{ private api functions */ -static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_file(zend_op_array* TSRMLS_DC); -static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_symbol(zend_function* TSRMLS_DC); -static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_method(zend_op_array* TSRMLS_DC); -static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_opline(phpdbg_opline_ptr_t TSRMLS_DC); -static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_opcode(zend_uchar TSRMLS_DC); -static inline phpdbg_breakbase_t *phpdbg_find_conditional_breakpoint(zend_execute_data *execute_data TSRMLS_DC); /* }}} */ +static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_file(zend_op_array*); +static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_symbol(zend_function*); +static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_method(zend_op_array*); +static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_opline(phpdbg_opline_ptr_t); +static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_opcode(zend_uchar); +static inline phpdbg_breakbase_t *phpdbg_find_conditional_breakpoint(zend_execute_data *execute_data); /* }}} */  /*  * Note:  *	A break point must always set the correct id and type  *	A set breakpoint function must always map new points  */ -static inline void _phpdbg_break_mapping(int id, HashTable *table TSRMLS_DC) +static inline void _phpdbg_break_mapping(int id, HashTable *table) /* {{{ */  { -	zend_hash_index_update( -		&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], (id), (void**) &table, sizeof(void*), NULL); +	zend_hash_index_update_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], id, table);  } +/* }}} */ -#define PHPDBG_BREAK_MAPPING(id, table) _phpdbg_break_mapping(id, table TSRMLS_CC) +#define PHPDBG_BREAK_MAPPING(id, table) _phpdbg_break_mapping(id, table)  #define PHPDBG_BREAK_UNMAPPING(id) \  	zend_hash_index_del(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], (id)) @@ -58,29 +58,32 @@ static inline void _phpdbg_break_mapping(int id, HashTable *table TSRMLS_DC)  	b.hits = 0; \  } while(0) -static void phpdbg_file_breaks_dtor(void *data) /* {{{ */ +static void phpdbg_file_breaks_dtor(zval *data) /* {{{ */  { -	phpdbg_breakfile_t *bp = (phpdbg_breakfile_t*) data; +	phpdbg_breakfile_t *bp = (phpdbg_breakfile_t*) Z_PTR_P(data);  	efree((char*)bp->filename); +	efree(bp);  } /* }}} */ -static void phpdbg_class_breaks_dtor(void *data) /* {{{ */ +static void phpdbg_class_breaks_dtor(zval *data) /* {{{ */  { -	phpdbg_breakmethod_t *bp = (phpdbg_breakmethod_t*) data; +	phpdbg_breakmethod_t *bp = (phpdbg_breakmethod_t *) Z_PTR_P(data);  	efree((char*)bp->class_name);  	efree((char*)bp->func_name); +	efree(bp);  } /* }}} */ -static void phpdbg_opline_class_breaks_dtor(void *data) /* {{{ */ +static void phpdbg_opline_class_breaks_dtor(zval *data) /* {{{ */  { -	zend_hash_destroy((HashTable *)data); +	zend_hash_destroy(Z_ARRVAL_P(data)); +	efree(Z_ARRVAL_P(data));  } /* }}} */ -static void phpdbg_opline_breaks_dtor(void *data) /* {{{ */ +static void phpdbg_opline_breaks_dtor(zval *data) /* {{{ */  { -	phpdbg_breakopline_t *bp = (phpdbg_breakopline_t *) data; +	phpdbg_breakopline_t *bp = (phpdbg_breakopline_t *) Z_PTR_P(data);  	if (bp->class_name) {  		efree((char*)bp->class_name); @@ -88,96 +91,94 @@ static void phpdbg_opline_breaks_dtor(void *data) /* {{{ */  	if (bp->func_name) {  		efree((char*)bp->func_name);  	} +	efree(bp);  } /* }}} */ -PHPDBG_API void phpdbg_reset_breakpoints(TSRMLS_D) /* {{{ */ +PHPDBG_API void phpdbg_reset_breakpoints(void) /* {{{ */  { -	if (zend_hash_num_elements(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP])) { -		HashPosition position[2]; -		HashTable **table = NULL; +	HashTable *table; -		for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], &position[0]); -			zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], (void**)&table, &position[0]) == SUCCESS; -			zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], &position[0])) { -			phpdbg_breakbase_t *brake; +	ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], table) { +		phpdbg_breakbase_t *brake; -			for (zend_hash_internal_pointer_reset_ex((*table), &position[1]); -				zend_hash_get_current_data_ex((*table), (void**)&brake, &position[1]) == SUCCESS; -				zend_hash_move_forward_ex((*table), &position[1])) { -				brake->hits = 0; -			} -		} -	} +		ZEND_HASH_FOREACH_PTR(table, brake) { +			brake->hits = 0; +		} ZEND_HASH_FOREACH_END(); +	} ZEND_HASH_FOREACH_END();  } /* }}} */ -PHPDBG_API void phpdbg_export_breakpoints(FILE *handle TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_export_breakpoints(FILE *handle) /* {{{ */  { -	HashPosition position[2]; -	HashTable **table = NULL; +	char *string; +	phpdbg_export_breakpoints_to_string(&string); +	fputs(string, handle); +} +/* }}} */ + +PHPDBG_API void phpdbg_export_breakpoints_to_string(char **str) /* {{{ */ +{ +	HashTable *table;  	zend_ulong id = 0L; +	*str = ""; +  	if (zend_hash_num_elements(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP])) { -		phpdbg_notice( -			"Exporting %d breakpoints", -			zend_hash_num_elements(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP])); +		phpdbg_notice("exportbreakpoint", "count=\"%d\"", "Exporting %d breakpoints", zend_hash_num_elements(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP])); +  		/* this only looks like magic, it isn't */ -		for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], &position[0]); -			zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], (void**)&table, &position[0]) == SUCCESS; -			zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], &position[0])) { +		ZEND_HASH_FOREACH_NUM_KEY_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], id, table) {  			phpdbg_breakbase_t *brake; -			zend_hash_get_current_key_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], NULL, NULL, &id, 0, &position[0]); - -			for (zend_hash_internal_pointer_reset_ex((*table), &position[1]); -				zend_hash_get_current_data_ex((*table), (void**)&brake, &position[1]) == SUCCESS; -				zend_hash_move_forward_ex((*table), &position[1])) { +			ZEND_HASH_FOREACH_PTR(table, brake) {  				if (brake->id == id) { +					char *new_str = NULL; +  					switch (brake->type) {  						case PHPDBG_BREAK_FILE: { -							fprintf(handle, -								"break %s:%lu\n", +							phpdbg_asprintf(&new_str, +								"%sbreak %s:%lu\n", *str,  								((phpdbg_breakfile_t*)brake)->filename,  								((phpdbg_breakfile_t*)brake)->line);  						} break;  						case PHPDBG_BREAK_SYM: { -							fprintf(handle, -								"break %s\n", +							phpdbg_asprintf(&new_str, +								"%sbreak %s\n", *str,  								((phpdbg_breaksymbol_t*)brake)->symbol);  						} break;  						case PHPDBG_BREAK_METHOD: { -							fprintf(handle, -								"break %s::%s\n", +							phpdbg_asprintf(&new_str, +								"%sbreak %s::%s\n", *str,  								((phpdbg_breakmethod_t*)brake)->class_name,  								((phpdbg_breakmethod_t*)brake)->func_name);  						} break;  						case PHPDBG_BREAK_METHOD_OPLINE: { -							fprintf(handle, -								"break %s::%s#%ld\n", +							phpdbg_asprintf(&new_str, +								"%sbreak %s::%s#%llu\n", *str,  								((phpdbg_breakopline_t*)brake)->class_name,  								((phpdbg_breakopline_t*)brake)->func_name,  								((phpdbg_breakopline_t*)brake)->opline_num);  						} break;  						case PHPDBG_BREAK_FUNCTION_OPLINE: { -							fprintf(handle, -								"break %s#%ld\n", +							phpdbg_asprintf(&new_str, +								"%sbreak %s#%llu\n", *str,  								((phpdbg_breakopline_t*)brake)->func_name,  								((phpdbg_breakopline_t*)brake)->opline_num);  						} break;  						case PHPDBG_BREAK_FILE_OPLINE: { -							fprintf(handle, -								"break %s:#%ld\n", +							phpdbg_asprintf(&new_str, +								"%sbreak %s:#%llu\n", *str,  								((phpdbg_breakopline_t*)brake)->class_name,  								((phpdbg_breakopline_t*)brake)->opline_num);  						} break;  						case PHPDBG_BREAK_OPCODE: { -							fprintf(handle, -								"break %s\n", +							phpdbg_asprintf(&new_str, +								"%sbreak %s\n", *str,  								((phpdbg_breakop_t*)brake)->name);  						} break; @@ -187,20 +188,20 @@ PHPDBG_API void phpdbg_export_breakpoints(FILE *handle TSRMLS_DC) /* {{{ */  							if (conditional->paramed) {  								switch (conditional->param.type) {  									case STR_PARAM: -										fprintf(handle, -											"break at %s if %s\n", conditional->param.str, conditional->code); +										phpdbg_asprintf(&new_str, +											"%sbreak at %s if %s\n", *str, conditional->param.str, conditional->code);  									break;  									case METHOD_PARAM: -										fprintf(handle, -											"break at %s::%s if %s\n", +										phpdbg_asprintf(&new_str, +											"%sbreak at %s::%s if %s\n", *str,  											conditional->param.method.class, conditional->param.method.name,  											conditional->code);  									break;  									case FILE_PARAM: -										fprintf(handle, -											"break at %s:%lu if %s\n", +										phpdbg_asprintf(&new_str, +											"%sbreak at %s:%lu if %s\n", *str,  											conditional->param.file.name, conditional->param.file.line,  											conditional->code);  									break; @@ -208,73 +209,186 @@ PHPDBG_API void phpdbg_export_breakpoints(FILE *handle TSRMLS_DC) /* {{{ */  									default: { /* do nothing */ } break;  								}  							} else { -								fprintf( -									handle, "break if %s\n", conditional->code); +								phpdbg_asprintf(&new_str, "%sbreak if %s\n", str, conditional->code);  							}  						} break; + +						default: continue;  					} + +					if ((*str)[0]) { +						efree(*str); +					} +					*str = new_str;  				} -			} -		} +			} ZEND_HASH_FOREACH_END(); +		} ZEND_HASH_FOREACH_END(); +	} + +	if (!(*str)[0]) { +		*str = NULL;  	}  } /* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_file(const char *path, long line_num TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_set_breakpoint_file(const char *path, long line_num) /* {{{ */  {  	php_stream_statbuf ssb;  	char realpath[MAXPATHLEN]; -	 -	if (php_stream_stat_path(path, &ssb) != FAILURE) { -		if (ssb.sb.st_mode & (S_IFREG|S_IFLNK)) { -			HashTable *broken; -			phpdbg_breakfile_t new_break; -			size_t path_len = 0L; -			 -			if (VCWD_REALPATH(path, realpath)) { -				path = realpath; +	const char *original_path = path; +	zend_bool pending = 0; +	zend_string *path_str; + +	HashTable *broken, *file_breaks = &PHPDBG_G(bp)[PHPDBG_BREAK_FILE]; +	phpdbg_breakfile_t new_break; +	size_t path_len = 0L; + +	if (VCWD_REALPATH(path, realpath)) { +		path = realpath; +	} +	path_len = strlen(path); + +	phpdbg_debug("file path: %s, resolved path: %s, was compiled: %d\n", original_path, path, zend_hash_str_exists(&PHPDBG_G(file_sources), path, path_len)); + +	if (!zend_hash_str_exists(&PHPDBG_G(file_sources), path, path_len)) { +		if (php_stream_stat_path(path, &ssb) == FAILURE) { +			if (original_path[0] == '/') { +				phpdbg_error("breakpoint", "type=\"nofile\" add=\"fail\" file=\"%s\"", "Cannot stat %s, it does not exist", original_path); +				return;  			} + +			file_breaks = &PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING]; +			path = original_path;  			path_len = strlen(path); -			 -			if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], -				path, path_len, (void**)&broken) == FAILURE) { -				HashTable breaks; +			pending = 1; +		} else if (!(ssb.sb.st_mode & (S_IFREG|S_IFLNK))) { +			phpdbg_error("breakpoint", "type=\"notregular\" add=\"fail\" file=\"%s\"", "Cannot set breakpoint in %s, it is not a regular file", path); +			return; +		} else { +			phpdbg_debug("File exists, but not compiled\n"); +		} +	} -				zend_hash_init(&breaks, 8, NULL, phpdbg_file_breaks_dtor, 0); +	path_str = zend_string_init(path, path_len, 0); -				zend_hash_update(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], -					path, path_len, &breaks, sizeof(HashTable), -					(void**)&broken); -			} +	if (!(broken = zend_hash_find_ptr(file_breaks, path_str))) { +		HashTable breaks; +		zend_hash_init(&breaks, 8, NULL, phpdbg_file_breaks_dtor, 0); -			if (!zend_hash_index_exists(broken, line_num)) { -				PHPDBG_G(flags) |= PHPDBG_HAS_FILE_BP; +		broken = zend_hash_add_mem(file_breaks, path_str, &breaks, sizeof(HashTable)); +	} -				PHPDBG_BREAK_INIT(new_break, PHPDBG_BREAK_FILE); -				new_break.filename = estrndup(path, path_len); -				new_break.line = line_num; +	if (!zend_hash_index_exists(broken, line_num)) { +		PHPDBG_BREAK_INIT(new_break, PHPDBG_BREAK_FILE); +		new_break.filename = estrndup(path, path_len); +		new_break.line = line_num; -				zend_hash_index_update( -					broken, line_num, (void**)&new_break, sizeof(phpdbg_breakfile_t), NULL); +		zend_hash_index_update_mem(broken, line_num, &new_break, sizeof(phpdbg_breakfile_t)); -				phpdbg_notice("Breakpoint #%d added at %s:%ld", -					new_break.id, new_break.filename, new_break.line); +		PHPDBG_BREAK_MAPPING(new_break.id, broken); -				PHPDBG_BREAK_MAPPING(new_break.id, broken); -			} else { -				phpdbg_error("Breakpoint at %s:%ld exists", path, line_num); -			} +		if (pending) { +			zend_string *file; +			ZEND_HASH_FOREACH_STR_KEY(&PHPDBG_G(file_sources), file) { +				HashTable *fileht; + +				phpdbg_debug("Compare against loaded %s\n", file); + +				if (!(pending = ((fileht = phpdbg_resolve_pending_file_break_ex(ZSTR_VAL(file), ZSTR_LEN(file), path_str, broken)) == NULL))) { +					new_break = *(phpdbg_breakfile_t *) zend_hash_index_find_ptr(fileht, line_num); +					break; +				} +			} ZEND_HASH_FOREACH_END(); +		} +		if (pending) { +			PHPDBG_G(flags) |= PHPDBG_HAS_PENDING_FILE_BP; + +			phpdbg_notice("breakpoint", "add=\"success\" id=\"%d\" file=\"%s\" line=\"%ld\" pending=\"pending\"", "Pending breakpoint #%d added at %s:%ld", new_break.id, new_break.filename, new_break.line);  		} else { -			phpdbg_error("Cannot set breakpoint in %s, it is not a regular file", path); +			PHPDBG_G(flags) |= PHPDBG_HAS_FILE_BP; + +			phpdbg_notice("breakpoint", "add=\"success\" id=\"%d\" file=\"%s\" line=\"%ld\"", "Breakpoint #%d added at %s:%ld", new_break.id, new_break.filename, new_break.line);  		}  	} else { -		phpdbg_error("Cannot stat %s, it does not exist", path); +		phpdbg_error("breakpoint", "type=\"exists\" add=\"fail\" file=\"%s\" line=\"%ld\"", "Breakpoint at %s:%ld exists", path, line_num);  	} + +	zend_string_release(path_str);  } /* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_symbol(const char *name, size_t name_len TSRMLS_DC) /* {{{ */ +PHPDBG_API HashTable *phpdbg_resolve_pending_file_break_ex(const char *file, uint filelen, zend_string *cur, HashTable *fileht) /* {{{ */  { -	if (!zend_hash_exists(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], name, name_len)) { +	phpdbg_debug("file: %s, filelen: %u, cur: %s, curlen %u, pos: %c, memcmp: %d\n", file, filelen, ZSTR_VAL(cur), ZSTR_LEN(cur), filelen > ZSTR_LEN(cur) ? file[filelen - ZSTR_LEN(cur) - 1] : '?', filelen > ZSTR_LEN(cur) ? memcmp(file + filelen - ZSTR_LEN(cur), ZSTR_VAL(cur), ZSTR_LEN(cur)) : 0); + +#ifdef _WIN32 +# define WIN32_PATH_CHECK file[filelen - ZSTR_LEN(cur) - 1] == '\\' +#else +# define WIN32_PATH_CHECK 0 +#endif + +	if (((ZSTR_LEN(cur) < filelen && (file[filelen - ZSTR_LEN(cur) - 1] == '/' || WIN32_PATH_CHECK)) || filelen == ZSTR_LEN(cur)) && !memcmp(file + filelen - ZSTR_LEN(cur), ZSTR_VAL(cur), ZSTR_LEN(cur))) { +		phpdbg_breakfile_t *brake, new_brake; +		HashTable *master; + +		PHPDBG_G(flags) |= PHPDBG_HAS_FILE_BP; + +		if (!(master = zend_hash_str_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], file, filelen))) { +			HashTable new_ht; +			zend_hash_init(&new_ht, 8, NULL, phpdbg_file_breaks_dtor, 0); +			master = zend_hash_str_add_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], file, filelen, &new_ht, sizeof(HashTable)); +		} + +		ZEND_HASH_FOREACH_PTR(fileht, brake) { +			new_brake = *brake; +			new_brake.filename = estrndup(file, filelen); +			PHPDBG_BREAK_UNMAPPING(brake->id); + +			if (zend_hash_index_add_mem(master, brake->line, &new_brake, sizeof(phpdbg_breakfile_t))) { +				PHPDBG_BREAK_MAPPING(brake->id, master); +			} +		} ZEND_HASH_FOREACH_END(); + +		zend_hash_del(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING], cur); + +		if (!zend_hash_num_elements(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING])) { +			PHPDBG_G(flags) &= ~PHPDBG_HAS_PENDING_FILE_BP; +		} + +		phpdbg_debug("compiled file: %s, cur bp file: %s\n", file, cur); + +		return master; +	} + +	return NULL; +} /* }}} */ + +PHPDBG_API void phpdbg_resolve_pending_file_break(const char *file) /* {{{ */ +{ +	HashTable *fileht; +	uint filelen = strlen(file); +	zend_string *cur; + +	phpdbg_debug("was compiled: %s\n", file); + +	ZEND_HASH_FOREACH_STR_KEY_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING], cur, fileht) { +		phpdbg_debug("check bp: %s\n", cur); + +		phpdbg_resolve_pending_file_break_ex(file, filelen, cur, fileht); +	} ZEND_HASH_FOREACH_END(); +} /* }}} */ + +PHPDBG_API void phpdbg_set_breakpoint_symbol(const char *name, size_t name_len) /* {{{ */ +{ +	char *lcname; + +	if (*name == '\\') { +		name++; +		name_len--; +	} + +	lcname = zend_str_tolower_dup(name, name_len); + +	if (!zend_hash_str_exists(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], name, name_len)) {  		phpdbg_breaksymbol_t new_break;  		PHPDBG_G(flags) |= PHPDBG_HAS_SYM_BP; @@ -282,35 +396,39 @@ PHPDBG_API void phpdbg_set_breakpoint_symbol(const char *name, size_t name_len T  		PHPDBG_BREAK_INIT(new_break, PHPDBG_BREAK_SYM);  		new_break.symbol = estrndup(name, name_len); -		zend_hash_update(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], new_break.symbol, -			name_len, &new_break, sizeof(phpdbg_breaksymbol_t), NULL); +		zend_hash_str_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], lcname, name_len, &new_break, sizeof(phpdbg_breaksymbol_t)); -		phpdbg_notice("Breakpoint #%d added at %s", -			new_break.id, new_break.symbol); +		phpdbg_notice("breakpoint", "add=\"success\" id=\"%d\" function=\"%s\"", "Breakpoint #%d added at %s", new_break.id, new_break.symbol);  		PHPDBG_BREAK_MAPPING(new_break.id, &PHPDBG_G(bp)[PHPDBG_BREAK_SYM]);  	} else { -		phpdbg_notice("Breakpoint exists at %s", name); +		phpdbg_error("breakpoint", "type=\"exists\" add=\"fail\" function=\"%s\"", "Breakpoint exists at %s", name);  	} + +	efree(lcname);  } /* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_method(const char *class_name, const char *func_name TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_set_breakpoint_method(const char *class_name, const char *func_name) /* {{{ */  {  	HashTable class_breaks, *class_table;  	size_t class_len = strlen(class_name);  	size_t func_len = strlen(func_name); -	char *lcname = zend_str_tolower_dup(func_name, func_len); +	char *func_lcname, *class_lcname; -	if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], class_name, -		class_len, (void**)&class_table) != SUCCESS) { +	if (*class_name == '\\') { +		class_name++; +		class_len--; +	} + +	func_lcname = zend_str_tolower_dup(func_name, func_len); +	class_lcname = zend_str_tolower_dup(class_name, class_len); + +	if (!(class_table = zend_hash_str_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], class_lcname, class_len))) {  		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); +		class_table = zend_hash_str_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], class_lcname, class_len, &class_breaks, sizeof(HashTable));  	} -	if (!zend_hash_exists(class_table, lcname, func_len)) { +	if (!zend_hash_str_exists(class_table, func_lcname, func_len)) {  		phpdbg_breakmethod_t new_break;  		PHPDBG_G(flags) |= PHPDBG_HAS_METHOD_BP; @@ -321,21 +439,20 @@ PHPDBG_API void phpdbg_set_breakpoint_method(const char *class_name, const char  		new_break.func_name = estrndup(func_name, func_len);  		new_break.func_len = func_len; -		zend_hash_update(class_table, lcname, func_len, -			&new_break, sizeof(phpdbg_breakmethod_t), NULL); +		zend_hash_str_update_mem(class_table, func_lcname, func_len, &new_break, sizeof(phpdbg_breakmethod_t)); -		phpdbg_notice("Breakpoint #%d added at %s::%s", -			new_break.id, class_name, func_name); +		phpdbg_notice("breakpoint", "add=\"success\" id=\"%d\" method=\"%s::%s\"", "Breakpoint #%d added at %s::%s", new_break.id, class_name, func_name);  		PHPDBG_BREAK_MAPPING(new_break.id, class_table);  	} else { -		phpdbg_notice("Breakpoint exists at %s::%s", class_name, func_name); +		phpdbg_error("breakpoint", "type=\"exists\" add=\"fail\" method=\"%s::%s\"", "Breakpoint exists at %s::%s", class_name, func_name);  	} -	efree(lcname); +	efree(func_lcname); +	efree(class_lcname);  } /* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_opline(zend_ulong opline TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_set_breakpoint_opline(zend_ulong opline) /* {{{ */  {  	if (!zend_hash_index_exists(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], opline)) {  		phpdbg_breakline_t new_break; @@ -347,27 +464,25 @@ PHPDBG_API void phpdbg_set_breakpoint_opline(zend_ulong opline TSRMLS_DC) /* {{{  		new_break.opline = opline;  		new_break.base = NULL; -		zend_hash_index_update(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], opline, -			&new_break, sizeof(phpdbg_breakline_t), NULL); +		zend_hash_index_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], opline, &new_break, sizeof(phpdbg_breakline_t)); -		phpdbg_notice("Breakpoint #%d added at %#lx", -			new_break.id, new_break.opline); +		phpdbg_notice("breakpoint", "add=\"success\" id=\"%d\" opline=\"%#lx\"", "Breakpoint #%d added at %#lx", new_break.id, new_break.opline);  		PHPDBG_BREAK_MAPPING(new_break.id, &PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE]);  	} else { -		phpdbg_notice("Breakpoint exists at %#lx", opline); +		phpdbg_error("breakpoint", "type=\"exists\" add=\"fail\" opline=\"%#lx\"", "Breakpoint exists at %#lx", opline);  	}  } /* }}} */ -PHPDBG_API int phpdbg_resolve_op_array_break(phpdbg_breakopline_t *brake, zend_op_array *op_array TSRMLS_DC) /* {{{ */ +PHPDBG_API int phpdbg_resolve_op_array_break(phpdbg_breakopline_t *brake, zend_op_array *op_array) /* {{{ */  {  	phpdbg_breakline_t opline_break;  	if (op_array->last <= brake->opline_num) {  		if (brake->class_name == NULL) { -			phpdbg_error("There are only %d oplines in function %s (breaking at opline %ld impossible)", op_array->last, brake->func_name, brake->opline_num); +			phpdbg_error("breakpoint", "type=\"maxoplines\" add=\"fail\" maxoplinenum=\"%d\" function=\"%s\" usedoplinenum=\"%ld\"", "There are only %d oplines in function %s (breaking at opline %ld impossible)", op_array->last, brake->func_name, brake->opline_num);  		} else if (brake->func_name == NULL) { -			phpdbg_error("There are only %d oplines in file %s (breaking at opline %ld impossible)", op_array->last, brake->class_name, brake->opline_num); +			phpdbg_error("breakpoint", "type=\"maxoplines\" add=\"fail\" maxoplinenum=\"%d\" file=\"%s\" usedoplinenum=\"%ld\"", "There are only %d oplines in file %s (breaking at opline %ld impossible)", op_array->last, brake->class_name, brake->opline_num);  		} else { -			phpdbg_error("There are only %d oplines in method %s::%s (breaking at opline %ld impossible)", op_array->last, brake->class_name, brake->func_name, brake->opline_num); +			phpdbg_error("breakpoint", "type=\"maxoplines\" add=\"fail\" maxoplinenum=\"%d\" method=\"%s::%s\" usedoplinenum=\"%ld\"", "There are only %d oplines in method %s::%s (breaking at opline %ld impossible)", op_array->last, brake->class_name, brake->func_name, brake->opline_num);  		}  		return FAILURE; @@ -389,52 +504,48 @@ PHPDBG_API int phpdbg_resolve_op_array_break(phpdbg_breakopline_t *brake, zend_o  	PHPDBG_G(flags) |= PHPDBG_HAS_OPLINE_BP; -	zend_hash_index_update(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], opline_break.opline, &opline_break, sizeof(phpdbg_breakline_t), NULL); +	zend_hash_index_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], opline_break.opline, &opline_break, sizeof(phpdbg_breakline_t));  	return SUCCESS;  } /* }}} */ -PHPDBG_API void phpdbg_resolve_op_array_breaks(zend_op_array *op_array TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_resolve_op_array_breaks(zend_op_array *op_array) /* {{{ */  {  	HashTable *func_table = &PHPDBG_G(bp)[PHPDBG_BREAK_FUNCTION_OPLINE];  	HashTable *oplines_table; -	HashPosition position;  	phpdbg_breakopline_t *brake; -	if (op_array->scope != NULL && -	    zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], op_array->scope->name, op_array->scope->name_length, (void **)&func_table) == FAILURE) { +	if (op_array->scope != NULL && !(func_table = zend_hash_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], op_array->scope->name))) {  		return;  	}  	if (op_array->function_name == NULL) { -		if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], op_array->filename, strlen(op_array->filename), (void **)&oplines_table) == FAILURE) { +		if (!(oplines_table = zend_hash_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], op_array->filename))) {  			return;  		} -	} else if (zend_hash_find(func_table, op_array->function_name?op_array->function_name:"", op_array->function_name?strlen(op_array->function_name):0, (void **)&oplines_table) == FAILURE) { +	} else if (!op_array->function_name || !(oplines_table = zend_hash_find_ptr(func_table, op_array->function_name))) {  		return;  	} -	for (zend_hash_internal_pointer_reset_ex(oplines_table, &position); -	     zend_hash_get_current_data_ex(oplines_table, (void**) &brake, &position) == SUCCESS; -	     zend_hash_move_forward_ex(oplines_table, &position)) { -		if (phpdbg_resolve_op_array_break(brake, op_array TSRMLS_CC) == SUCCESS) { +	ZEND_HASH_FOREACH_PTR(oplines_table, brake) { +		if (phpdbg_resolve_op_array_break(brake, op_array) == SUCCESS) {  			phpdbg_breakline_t *opline_break;  			zend_hash_internal_pointer_end(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE]); -			zend_hash_get_current_data(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], (void **)&opline_break); +			opline_break = zend_hash_get_current_data_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE]); -			phpdbg_notice("Breakpoint #%d resolved at %s%s%s#%ld (opline %#lx)", -				brake->id, -				brake->class_name?brake->class_name:"", -				brake->class_name&&brake->func_name?"::":"", -				brake->func_name?brake->func_name:"", +			phpdbg_notice("breakpoint", "add=\"success\" id=\"%d\" symbol=\"%s\" num=\"%ld\" opline=\"%#lx\"", "Breakpoint #%d resolved at %s%s%s#%ld (opline %#lx)", +				opline_break->id, +				brake->class_name ? brake->class_name : "", +				brake->class_name && brake->func_name ? "::" : "", +				brake->func_name ? brake->func_name : "",  				brake->opline_num, -				brake->opline); +				opline_break->opline);  		} -	} +	} ZEND_HASH_FOREACH_END();  } /* }}} */ -PHPDBG_API int phpdbg_resolve_opline_break(phpdbg_breakopline_t *new_break TSRMLS_DC) /* {{{ */ +PHPDBG_API int phpdbg_resolve_opline_break(phpdbg_breakopline_t *new_break) /* {{{ */  {  	HashTable *func_table = EG(function_table);  	zend_function *func; @@ -442,7 +553,7 @@ PHPDBG_API int phpdbg_resolve_opline_break(phpdbg_breakopline_t *new_break TSRML  	if (new_break->func_name == NULL) {  		if (EG(current_execute_data) == NULL) {  			if (PHPDBG_G(ops) != NULL && !memcmp(PHPDBG_G(ops)->filename, new_break->class_name, new_break->class_len)) { -				if (phpdbg_resolve_op_array_break(new_break, PHPDBG_G(ops) TSRMLS_CC) == SUCCESS) { +				if (phpdbg_resolve_op_array_break(new_break, PHPDBG_G(ops)) == SUCCESS) {  					return SUCCESS;  				} else {  					return 2; @@ -452,11 +563,14 @@ PHPDBG_API int phpdbg_resolve_opline_break(phpdbg_breakopline_t *new_break TSRML  		} else {  			zend_execute_data *execute_data = EG(current_execute_data);  			do { -				if (execute_data->op_array->function_name == NULL && execute_data->op_array->scope == NULL && !memcmp(execute_data->op_array->filename, new_break->class_name, new_break->class_len)) { -					if (phpdbg_resolve_op_array_break(new_break, execute_data->op_array TSRMLS_CC) == SUCCESS) { -						return SUCCESS; -					} else { -						return 2; +				if (ZEND_USER_CODE(execute_data->func->common.type)) { +					zend_op_array *op_array = &execute_data->func->op_array; +					if (op_array->function_name == NULL && op_array->scope == NULL && new_break->class_len == ZSTR_LEN(op_array->filename) && !memcmp(ZSTR_VAL(op_array->filename), new_break->class_name, new_break->class_len)) { +						if (phpdbg_resolve_op_array_break(new_break, op_array) == SUCCESS) { +							return SUCCESS; +						} else { +							return 2; +						}  					}  				}  			} while ((execute_data = execute_data->prev_execute_data) != NULL); @@ -465,16 +579,16 @@ PHPDBG_API int phpdbg_resolve_opline_break(phpdbg_breakopline_t *new_break TSRML  	}  	if (new_break->class_name != NULL) { -		zend_class_entry **ce; -		if (zend_hash_find(EG(class_table), zend_str_tolower_dup(new_break->class_name, new_break->class_len), new_break->class_len + 1, (void **)&ce) == FAILURE) { +		zend_class_entry *ce; +		if (!(ce = zend_hash_str_find_ptr(EG(class_table), zend_str_tolower_dup(new_break->class_name, new_break->class_len), new_break->class_len))) {  			return FAILURE;  		} -		func_table = &(*ce)->function_table; +		func_table = &ce->function_table;  	} -	if (zend_hash_find(func_table, zend_str_tolower_dup(new_break->func_name, new_break->func_len), new_break->func_len + 1, (void **)&func) == FAILURE) { +	if (!(func = zend_hash_str_find_ptr(func_table, zend_str_tolower_dup(new_break->func_name, new_break->func_len), new_break->func_len))) {  		if (new_break->class_name != NULL && new_break->func_name != NULL) { -			phpdbg_error("Method %s doesn't exist in class %s", new_break->func_name, new_break->class_name); +			phpdbg_error("breakpoint", "type=\"nomethod\" method=\"%s::%s\"", "Method %s doesn't exist in class %s", new_break->func_name, new_break->class_name);  			return 2;  		}  		return FAILURE; @@ -482,21 +596,23 @@ PHPDBG_API int phpdbg_resolve_opline_break(phpdbg_breakopline_t *new_break TSRML  	if (func->type != ZEND_USER_FUNCTION) {  		if (new_break->class_name == NULL) { -			phpdbg_error("%s is not an user defined function, no oplines exist", new_break->func_name); +			phpdbg_error("breakpoint", "type=\"internalfunction\" function=\"%s\"", "%s is not a user defined function, no oplines exist", new_break->func_name);  		} else { -			phpdbg_error("%s::%s is not an user defined method, no oplines exist", new_break->class_name, new_break->func_name); +			phpdbg_error("breakpoint", "type=\"internalfunction\" method=\"%s::%s\"", "%s::%s is not a user defined method, no oplines exist", new_break->class_name, new_break->func_name);  		}  		return 2;  	} -	if (phpdbg_resolve_op_array_break(new_break, &func->op_array TSRMLS_CC) == FAILURE) { +	if (phpdbg_resolve_op_array_break(new_break, &func->op_array) == FAILURE) {  		return 2;  	}  	return SUCCESS;  } /* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_method_opline(const char *class, const char *method, zend_ulong opline TSRMLS_DC) /* {{{ */ +/* TODO ... method/function oplines need to be normalized (leading backslash, lowercase) and file oplines need to be resolved properly */ + +PHPDBG_API void phpdbg_set_breakpoint_method_opline(const char *class, const char *method, zend_ulong opline) /* {{{ */  {  	phpdbg_breakopline_t new_break;  	HashTable class_breaks, *class_table; @@ -510,39 +626,31 @@ PHPDBG_API void phpdbg_set_breakpoint_method_opline(const char *class, const cha  	new_break.opline_num = opline;  	new_break.opline = 0; -	switch (phpdbg_resolve_opline_break(&new_break TSRMLS_CC)) { +	switch (phpdbg_resolve_opline_break(&new_break)) {  		case FAILURE: -			phpdbg_notice("Pending breakpoint #%d at %s::%s#%ld", new_break.id, new_break.class_name, new_break.func_name, opline); +			phpdbg_notice("breakpoint", "pending=\"pending\" id=\"%d\" method=\"%::%s\" num=\"%ld\"", "Pending breakpoint #%d at %s::%s#%ld", new_break.id, new_break.class_name, new_break.func_name, opline);  			break;  		case SUCCESS: -			phpdbg_notice("Breakpoint #%d added at %s::%s#%ld", new_break.id, new_break.class_name, new_break.func_name, opline); +			phpdbg_notice("breakpoint", "id=\"%d\" method=\"%::%s\" num=\"%ld\"", "Breakpoint #%d added at %s::%s#%ld", new_break.id, new_break.class_name, new_break.func_name, opline);  			break;  		case 2:  			return;  	} -	if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], new_break.class_name, new_break.class_len, (void **)&class_table) == FAILURE) { +	if (!(class_table = zend_hash_str_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], new_break.class_name, new_break.class_len))) {  		zend_hash_init(&class_breaks, 8, NULL, phpdbg_opline_class_breaks_dtor, 0); -		zend_hash_update( -			&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], -			new_break.class_name, -			new_break.class_len, -			(void **)&class_breaks, sizeof(HashTable), (void **)&class_table); +		class_table = zend_hash_str_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], new_break.class_name, new_break.class_len, &class_breaks, sizeof(HashTable));  	} -	if (zend_hash_find(class_table, new_break.func_name, new_break.func_len, (void **)&method_table) == FAILURE) { +	if (!(method_table = zend_hash_str_find_ptr(class_table, new_break.func_name, new_break.func_len))) {  		zend_hash_init(&method_breaks, 8, NULL, phpdbg_opline_breaks_dtor, 0); -		zend_hash_update( -			class_table, -			new_break.func_name, -			new_break.func_len, -			(void **)&method_breaks, sizeof(HashTable), (void **)&method_table); +		method_table = zend_hash_str_update_mem(class_table, new_break.func_name, new_break.func_len, &method_breaks, sizeof(HashTable));  	}  	if (zend_hash_index_exists(method_table, opline)) { -		phpdbg_notice("Breakpoint already exists for %s::%s#%ld", new_break.class_name, new_break.func_name, opline); +		phpdbg_error("breakpoint", "type=\"exists\" method=\"%s\" num=\"%ld\"", "Breakpoint already exists for %s::%s#%ld", new_break.class_name, new_break.func_name, opline);  		efree((char*)new_break.func_name);  		efree((char*)new_break.class_name);  		PHPDBG_G(bp_count)--; @@ -553,10 +661,11 @@ PHPDBG_API void phpdbg_set_breakpoint_method_opline(const char *class, const cha  	PHPDBG_BREAK_MAPPING(new_break.id, method_table); -	zend_hash_index_update(method_table, opline, &new_break, sizeof(phpdbg_breakopline_t), NULL); +	zend_hash_index_update_mem(method_table, opline, &new_break, sizeof(phpdbg_breakopline_t));  } +/* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_function_opline(const char *function, zend_ulong opline TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_set_breakpoint_function_opline(const char *function, zend_ulong opline) /* {{{ */  {  	phpdbg_breakopline_t new_break;  	HashTable func_breaks, *func_table; @@ -569,30 +678,26 @@ PHPDBG_API void phpdbg_set_breakpoint_function_opline(const char *function, zend  	new_break.opline_num = opline;  	new_break.opline = 0; -	switch (phpdbg_resolve_opline_break(&new_break TSRMLS_CC)) { +	switch (phpdbg_resolve_opline_break(&new_break)) {  		case FAILURE: -			phpdbg_notice("Pending breakpoint #%d at %s#%ld", new_break.id, new_break.func_name, opline); +			phpdbg_notice("breakpoint", "pending=\"pending\" id=\"%d\" function=\"%s\" num=\"%ld\"", "Pending breakpoint #%d at %s#%ld", new_break.id, new_break.func_name, opline);  			break;  		case SUCCESS: -			phpdbg_notice("Breakpoint #%d added at %s#%ld", new_break.id, new_break.func_name, opline); +			phpdbg_notice("breakpoint", "id=\"%d\" function=\"%s\" num=\"%ld\"", "Breakpoint #%d added at %s#%ld", new_break.id, new_break.func_name, opline);  			break;  		case 2:  			return;  	} -	if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_FUNCTION_OPLINE], new_break.func_name, new_break.func_len, (void **)&func_table) == FAILURE) { +	if (!(func_table = zend_hash_str_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_FUNCTION_OPLINE], new_break.func_name, new_break.func_len))) {  		zend_hash_init(&func_breaks, 8, NULL, phpdbg_opline_breaks_dtor, 0); -		zend_hash_update( -			&PHPDBG_G(bp)[PHPDBG_BREAK_FUNCTION_OPLINE], -			new_break.func_name, -			new_break.func_len, -			(void **)&func_breaks, sizeof(HashTable), (void **)&func_table); +		func_table = zend_hash_str_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_FUNCTION_OPLINE], new_break.func_name, new_break.func_len, &func_breaks, sizeof(HashTable));  	}  	if (zend_hash_index_exists(func_table, opline)) { -		phpdbg_notice("Breakpoint already exists for %s#%ld", new_break.func_name, opline); +		phpdbg_error("breakpoint", "type=\"exists\" function=\"%s\" num=\"%ld\"", "Breakpoint already exists for %s#%ld", new_break.func_name, opline);  		efree((char*)new_break.func_name);  		PHPDBG_G(bp_count)--;  		return; @@ -602,10 +707,11 @@ PHPDBG_API void phpdbg_set_breakpoint_function_opline(const char *function, zend  	PHPDBG_G(flags) |= PHPDBG_HAS_FUNCTION_OPLINE_BP; -	zend_hash_index_update(func_table, opline, &new_break, sizeof(phpdbg_breakopline_t), NULL); +	zend_hash_index_update_mem(func_table, opline, &new_break, sizeof(phpdbg_breakopline_t));  } +/* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_file_opline(const char *file, zend_ulong opline TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_set_breakpoint_file_opline(const char *file, zend_ulong opline) /* {{{ */  {  	phpdbg_breakopline_t new_break;  	HashTable file_breaks, *file_table; @@ -618,30 +724,26 @@ PHPDBG_API void phpdbg_set_breakpoint_file_opline(const char *file, zend_ulong o  	new_break.opline_num = opline;  	new_break.opline = 0; -	switch (phpdbg_resolve_opline_break(&new_break TSRMLS_CC)) { +	switch (phpdbg_resolve_opline_break(&new_break)) {  		case FAILURE: -			phpdbg_notice("Pending breakpoint #%d at %s:%ld", new_break.id, new_break.class_name, opline); +			phpdbg_notice("breakpoint", "pending=\"pending\" id=\"%d\" file=\"%s\" num=\"%ld\"", "Pending breakpoint #%d at %s:%ld", new_break.id, new_break.class_name, opline);  			break;  		case SUCCESS: -			phpdbg_notice("Breakpoint #%d added at %s:%ld", new_break.id, new_break.class_name, opline); +			phpdbg_notice("breakpoint", "id=\"%d\" file=\"%s\" num=\"%ld\"", "Breakpoint #%d added at %s:%ld", new_break.id, new_break.class_name, opline);  			break;  		case 2:  			return;  	} -	if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], new_break.class_name, new_break.class_len, (void **)&file_table) == FAILURE) { +	if (!(file_table = zend_hash_str_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], new_break.class_name, new_break.class_len))) {  		zend_hash_init(&file_breaks, 8, NULL, phpdbg_opline_breaks_dtor, 0); -		zend_hash_update( -			&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], -			new_break.class_name, -			new_break.class_len, -			(void **)&file_breaks, sizeof(HashTable), (void **)&file_table); +		file_table = zend_hash_str_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], new_break.class_name, new_break.class_len, &file_breaks, sizeof(HashTable));  	}  	if (zend_hash_index_exists(file_table, opline)) { -		phpdbg_notice("Breakpoint already exists for %s:%ld", new_break.class_name, opline); +		phpdbg_error("breakpoint", "type=\"exists\" file=\"%s\" num=\"%d\"", "Breakpoint already exists for %s:%ld", new_break.class_name, opline);  		efree((char*)new_break.class_name);  		PHPDBG_G(bp_count)--;  		return; @@ -651,17 +753,17 @@ PHPDBG_API void phpdbg_set_breakpoint_file_opline(const char *file, zend_ulong o  	PHPDBG_G(flags) |= PHPDBG_HAS_FILE_OPLINE_BP; -	zend_hash_index_update(file_table, opline, &new_break, sizeof(phpdbg_breakopline_t), NULL); +	zend_hash_index_update_mem(file_table, opline, &new_break, sizeof(phpdbg_breakopline_t));  } +/* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_opcode(const char *name, size_t name_len TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_set_breakpoint_opcode(const char *name, size_t name_len) /* {{{ */  {  	phpdbg_breakop_t new_break;  	zend_ulong hash = zend_hash_func(name, name_len);  	if (zend_hash_index_exists(&PHPDBG_G(bp)[PHPDBG_BREAK_OPCODE], hash)) { -		phpdbg_notice( -			"Breakpoint exists for %s", name); +		phpdbg_error("breakpoint", "type=\"exists\" opcode=\"%s\"", "Breakpoint exists for %s", name);  		return;  	} @@ -669,16 +771,15 @@ PHPDBG_API void phpdbg_set_breakpoint_opcode(const char *name, size_t name_len T  	new_break.hash = hash;  	new_break.name = estrndup(name, name_len); -	zend_hash_index_update(&PHPDBG_G(bp)[PHPDBG_BREAK_OPCODE], hash, -		&new_break, sizeof(phpdbg_breakop_t), NULL); +	zend_hash_index_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_OPCODE], hash, &new_break, sizeof(phpdbg_breakop_t));  	PHPDBG_G(flags) |= PHPDBG_HAS_OPCODE_BP; -	phpdbg_notice("Breakpoint #%d added at %s", new_break.id, name); +	phpdbg_notice("breakpoint", "id=\"%d\" opcode=\"%s\"", "Breakpoint #%d added at %s", new_break.id, name);  	PHPDBG_BREAK_MAPPING(new_break.id, &PHPDBG_G(bp)[PHPDBG_BREAK_OPCODE]);  } /* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_opline_ex(phpdbg_opline_ptr_t opline TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_set_breakpoint_opline_ex(phpdbg_opline_ptr_t opline) /* {{{ */  {  	if (!zend_hash_index_exists(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], (zend_ulong) opline)) {  		phpdbg_breakline_t new_break; @@ -689,19 +790,19 @@ PHPDBG_API void phpdbg_set_breakpoint_opline_ex(phpdbg_opline_ptr_t opline TSRML  		new_break.opline = (zend_ulong) opline;  		new_break.base = NULL; -		zend_hash_index_update(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], -			(zend_ulong) opline, &new_break, sizeof(phpdbg_breakline_t), NULL); +		zend_hash_index_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], (zend_ulong) opline, &new_break, sizeof(phpdbg_breakline_t)); -		phpdbg_notice("Breakpoint #%d added at %#lx", -			new_break.id, new_break.opline); +		phpdbg_notice("breakpoint", "id=\"%d\" opline=\"%#lx\"", "Breakpoint #%d added at %#lx", new_break.id, new_break.opline);  		PHPDBG_BREAK_MAPPING(new_break.id, &PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE]); +	} else { +		phpdbg_error("breakpoint", "type=\"exists\" opline=\"%#lx\"", "Breakpoint exists for opline %#lx", (zend_ulong) opline);  	}  } /* }}} */ -static inline void phpdbg_create_conditional_break(phpdbg_breakcond_t *brake, const phpdbg_param_t *param, const char *expr, size_t expr_len, zend_ulong hash TSRMLS_DC) /* {{{ */ +static inline void phpdbg_create_conditional_break(phpdbg_breakcond_t *brake, const phpdbg_param_t *param, const char *expr, size_t expr_len, zend_ulong hash) /* {{{ */  {  	phpdbg_breakcond_t new_break; -	zend_uint cops = CG(compiler_options); +	uint32_t cops = CG(compiler_options);  	zval pv;  	PHPDBG_BREAK_INIT(new_break, PHPDBG_BREAK_COND); @@ -710,7 +811,7 @@ static inline void phpdbg_create_conditional_break(phpdbg_breakcond_t *brake, co  	if (param) {  		new_break.paramed = 1;  		phpdbg_copy_param( -			param, &new_break.param TSRMLS_CC); +			param, &new_break.param);  	} else {  		new_break.paramed = 0;  	} @@ -722,176 +823,165 @@ static inline void phpdbg_create_conditional_break(phpdbg_breakcond_t *brake, co  	new_break.code = estrndup(expr, expr_len);  	new_break.code_len = expr_len; -	Z_STRLEN(pv) = expr_len + sizeof("return ;") - 1; -	Z_STRVAL(pv) = emalloc(Z_STRLEN(pv) + 1); +	Z_STR(pv) = zend_string_alloc(expr_len + sizeof("return ;") - 1, 0);  	memcpy(Z_STRVAL(pv), "return ", sizeof("return ") - 1);  	memcpy(Z_STRVAL(pv) + sizeof("return ") - 1, expr, expr_len);  	Z_STRVAL(pv)[Z_STRLEN(pv) - 1] = ';';  	Z_STRVAL(pv)[Z_STRLEN(pv)] = '\0'; -	Z_TYPE(pv) = IS_STRING; +	Z_TYPE_INFO(pv) = IS_STRING; -	new_break.ops = zend_compile_string( -		&pv, "Conditional Breakpoint Code" TSRMLS_CC); +	new_break.ops = zend_compile_string(&pv, "Conditional Breakpoint Code");  	zval_dtor(&pv);  	if (new_break.ops) { -		zend_hash_index_update( -			&PHPDBG_G(bp)[PHPDBG_BREAK_COND], hash, &new_break, -			sizeof(phpdbg_breakcond_t), (void**)&brake); +		brake = zend_hash_index_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], hash, &new_break, sizeof(phpdbg_breakcond_t)); -		phpdbg_notice("Conditional breakpoint #%d added %s/%p", -			brake->id, brake->code, brake->ops); +		phpdbg_notice("breakpoint", "id=\"%d\" expression=\"%s\" ptr=\"%p\"", "Conditional breakpoint #%d added %s/%p", brake->id, brake->code, brake->ops);  		PHPDBG_G(flags) |= PHPDBG_HAS_COND_BP;  		PHPDBG_BREAK_MAPPING(new_break.id, &PHPDBG_G(bp)[PHPDBG_BREAK_COND]);  	} else { -		 phpdbg_error( -			"Failed to compile code for expression %s", expr); +		 phpdbg_error("compile", "expression=\"%s\"", "Failed to compile code for expression %s", expr);  		 efree((char*)new_break.code);  		 PHPDBG_G(bp_count)--;  	} +  	CG(compiler_options) = cops;  } /* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_expression(const char *expr, size_t expr_len TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_set_breakpoint_expression(const char *expr, size_t expr_len) /* {{{ */  {  	zend_ulong expr_hash = zend_inline_hash_func(expr, expr_len);  	phpdbg_breakcond_t new_break;  	if (!zend_hash_index_exists(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], expr_hash)) {  		phpdbg_create_conditional_break( -			&new_break, NULL, expr, expr_len, expr_hash TSRMLS_CC); +			&new_break, NULL, expr, expr_len, expr_hash);  	} else { -		phpdbg_notice("Conditional break %s exists", expr); +		phpdbg_error("breakpoint", "type=\"exists\" expression=\"%s\"", "Conditional break %s exists", expr);  	}  } /* }}} */ -PHPDBG_API void phpdbg_set_breakpoint_at(const phpdbg_param_t *param TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_set_breakpoint_at(const phpdbg_param_t *param) /* {{{ */  {  	phpdbg_breakcond_t new_break;  	phpdbg_param_t *condition;  	zend_ulong hash = 0L; -	 +  	if (param->next) {  		condition = param->next;  		hash = zend_inline_hash_func(condition->str, condition->len); -		 +  		if (!zend_hash_index_exists(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], hash)) { -			phpdbg_create_conditional_break( -				&new_break, param,  -					condition->str, condition->len, hash TSRMLS_CC); +			phpdbg_create_conditional_break(&new_break, param, condition->str, condition->len, hash);  		} else { -			phpdbg_notice( -				"Conditional break %s exists at the specified location", condition->str); -		}	 +			phpdbg_notice("breakpoint", "type=\"exists\" arg=\"%s\"", "Conditional break %s exists at the specified location", condition->str); +		}  	} -	 +  } /* }}} */ -static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_file(zend_op_array *op_array TSRMLS_DC) /* {{{ */ +static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_file(zend_op_array *op_array) /* {{{ */  {  	HashTable *breaks;  	phpdbg_breakbase_t *brake; -	size_t name_len = strlen(op_array->filename); +	size_t path_len; +	char realpath[MAXPATHLEN]; +	const char *path = ZSTR_VAL(op_array->filename); + +	if (VCWD_REALPATH(path, realpath)) { +		path = realpath; +	} + +	path_len = strlen(path); -	if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], op_array->filename, -		name_len, (void**)&breaks) == FAILURE) { +#if 0 +	phpdbg_debug("Op at: %.*s %d\n", path_len, path, (*EG(opline_ptr))->lineno); +#endif + +	if (!(breaks = zend_hash_str_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], path, path_len))) {  		return NULL;  	} -	if (zend_hash_index_find(breaks, (*EG(opline_ptr))->lineno, (void**)&brake) == SUCCESS) { +	if (EG(current_execute_data) && (brake = zend_hash_index_find_ptr(breaks, EG(current_execute_data)->opline->lineno))) {  		return brake;  	}  	return NULL;  } /* }}} */ -static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_symbol(zend_function *fbc TSRMLS_DC) /* {{{ */ +static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_symbol(zend_function *fbc) /* {{{ */  { -	const char *fname;  	zend_op_array *ops; -	phpdbg_breakbase_t *brake;  	if (fbc->type != ZEND_USER_FUNCTION) {  		return NULL;  	} -	ops = (zend_op_array*)fbc; +	ops = (zend_op_array *) fbc;  	if (ops->scope) {  		/* find method breaks here */ -		return phpdbg_find_breakpoint_method(ops TSRMLS_CC); +		return phpdbg_find_breakpoint_method(ops);  	} -	fname = ops->function_name; +	if (ops->function_name) { +		phpdbg_breakbase_t *brake; +		zend_string *fname = zend_string_tolower(ops->function_name); -	if (!fname) { -		fname = "main"; -	} +		brake = zend_hash_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], fname); -	if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], fname, strlen(fname), (void**)&brake) == SUCCESS) { +		zend_string_release(fname);  		return brake; +	} else { +		return zend_hash_str_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], ZEND_STRL("main"));  	} - -	return NULL;  } /* }}} */ -static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_method(zend_op_array *ops TSRMLS_DC) /* {{{ */ +static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_method(zend_op_array *ops) /* {{{ */  {  	HashTable *class_table; -	phpdbg_breakbase_t *brake; +	phpdbg_breakbase_t *brake = NULL; +	zend_string *class_lcname = zend_string_tolower(ops->scope->name); -	if (zend_hash_find(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], ops->scope->name, -		ops->scope->name_length, (void**)&class_table) == SUCCESS) { -		char *lcname = zend_str_tolower_dup(ops->function_name, strlen(ops->function_name)); -		size_t lcname_len = strlen(lcname); - -		if (zend_hash_find( -		        class_table, -		        lcname, -		        lcname_len, (void**)&brake) == SUCCESS) { -			efree(lcname); -			return brake; -		} +	if ((class_table = zend_hash_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], class_lcname))) { +		zend_string *lcname = zend_string_tolower(ops->function_name); -		efree(lcname); +		brake = zend_hash_find_ptr(class_table, lcname); + +		zend_string_release(lcname);  	} -	return NULL; +	zend_string_release(class_lcname); +	return brake;  } /* }}} */ -static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_opline(phpdbg_opline_ptr_t opline TSRMLS_DC) /* {{{ */ +static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_opline(phpdbg_opline_ptr_t opline) /* {{{ */  {  	phpdbg_breakline_t *brake; -	if (zend_hash_index_find(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], -		(zend_ulong) opline, (void**)&brake) == SUCCESS) { -		return (brake->base?(phpdbg_breakbase_t *)brake->base:(phpdbg_breakbase_t *)brake); +	if ((brake = zend_hash_index_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], (zend_ulong) opline)) && brake->base) { +		return (phpdbg_breakbase_t *)brake->base;  	} -	return NULL; +	return (phpdbg_breakbase_t *) brake;  } /* }}} */ -static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_opcode(zend_uchar opcode TSRMLS_DC) /* {{{ */ +static inline phpdbg_breakbase_t *phpdbg_find_breakpoint_opcode(zend_uchar opcode) /* {{{ */  { -	phpdbg_breakbase_t *brake; -	const char *opname = phpdbg_decode_opcode(opcode); +	const char *opname = zend_get_opcode_name(opcode); -	if (memcmp(opname, PHPDBG_STRL("UNKNOWN")) == 0) { +	if (!opname) {  		return NULL;  	} -	if (zend_hash_index_find(&PHPDBG_G(bp)[PHPDBG_BREAK_OPCODE], -		zend_hash_func(opname, strlen(opname)), (void**)&brake) == SUCCESS) { -		return brake; -	} -	return NULL; +	return zend_hash_index_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_OPCODE], zend_hash_func(opname, strlen(opname)));  } /* }}} */ -static inline zend_bool phpdbg_find_breakpoint_param(phpdbg_param_t *param, zend_execute_data *execute_data TSRMLS_DC) /* {{{ */ +static inline zend_bool phpdbg_find_breakpoint_param(phpdbg_param_t *param, zend_execute_data *execute_data) /* {{{ */  { -	zend_function *function = (zend_function*) execute_data->function_state.function; +	zend_function *function = execute_data->func;  	switch (param->type) {  		case NUMERIC_FUNCTION_PARAM: @@ -906,8 +996,8 @@ static inline zend_bool phpdbg_find_breakpoint_param(phpdbg_param_t *param, zend  				const char *str = NULL;  				size_t len = 0L;  				zend_op_array *ops = (zend_op_array*)function; -				str = ops->function_name ? ops->function_name : "main"; -				len = strlen(str); +				str = ops->function_name ? ZSTR_VAL(ops->function_name) : "main"; +				len = ops->function_name ? ZSTR_LEN(ops->function_name) : strlen(str);  				if (len == param->len && memcmp(param->str, str, len) == SUCCESS) {  					return param->type == STR_PARAM || execute_data->opline - ops->opcodes == param->num; @@ -916,8 +1006,8 @@ static inline zend_bool phpdbg_find_breakpoint_param(phpdbg_param_t *param, zend  		} break;  		case FILE_PARAM: { -			if (param->file.line == zend_get_executed_lineno(TSRMLS_C)) { -				const char *str = zend_get_executed_filename(TSRMLS_C); +			if (param->file.line == zend_get_executed_lineno()) { +				const char *str = zend_get_executed_filename();  				size_t lengths[2] = {strlen(param->file.name), strlen(str)};  				if (lengths[0] == lengths[1]) { @@ -937,10 +1027,10 @@ static inline zend_bool phpdbg_find_breakpoint_param(phpdbg_param_t *param, zend  				zend_op_array *ops = (zend_op_array*) function;  				if (ops->scope) { -					size_t lengths[2] = {strlen(param->method.class), ops->scope->name_length}; +					size_t lengths[2] = { strlen(param->method.class), ZSTR_LEN(ops->scope->name) };  					if (lengths[0] == lengths[1] && memcmp(param->method.class, ops->scope->name, lengths[0]) == SUCCESS) {  						lengths[0] = strlen(param->method.name); -						lengths[1] = strlen(ops->function_name); +						lengths[1] = ZSTR_LEN(ops->function_name);  						if (lengths[0] == lengths[1] && memcmp(param->method.name, ops->function_name, lengths[0]) == SUCCESS) {  							return param->type == METHOD_PARAM || (execute_data->opline - ops->opcodes) == param->num; @@ -961,80 +1051,54 @@ static inline zend_bool phpdbg_find_breakpoint_param(phpdbg_param_t *param, zend  	return 0;  } /* }}} */ -static inline phpdbg_breakbase_t *phpdbg_find_conditional_breakpoint(zend_execute_data *execute_data TSRMLS_DC) /* {{{ */ +static inline phpdbg_breakbase_t *phpdbg_find_conditional_breakpoint(zend_execute_data *execute_data) /* {{{ */  {  	phpdbg_breakcond_t *bp; -	HashPosition position;  	int breakpoint = FAILURE; -	for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], &position); -	     zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], (void*)&bp, &position) == SUCCESS; -	     zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], &position)) { -		zval *retval = NULL; -		int orig_interactive = CG(interactive); -		zval **orig_retval = EG(return_value_ptr_ptr); -		zend_op_array *orig_ops = EG(active_op_array); -		zend_op **orig_opline = EG(opline_ptr); +	ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], bp) { +		zval retval; +		const zend_op *orig_opline = EG(current_execute_data)->opline; +		zend_function *orig_func = EG(current_execute_data)->func; +		zval *orig_retval = EG(current_execute_data)->return_value;  		if (((phpdbg_breakbase_t*)bp)->disabled) {  			continue;  		}  		if (bp->paramed) { -			if (!phpdbg_find_breakpoint_param(&bp->param, execute_data TSRMLS_CC)) { +			if (!phpdbg_find_breakpoint_param(&bp->param, execute_data)) {  				continue;  			}  		} -		ALLOC_INIT_ZVAL(retval); - -		EG(return_value_ptr_ptr) = &retval; -		EG(active_op_array) = bp->ops;  		EG(no_extensions) = 1; -		if (!EG(active_symbol_table)) { -			zend_rebuild_symbol_table(TSRMLS_C); -		} - -		CG(interactive) = 0; +		zend_rebuild_symbol_table();  		zend_try {  			PHPDBG_G(flags) |= PHPDBG_IN_COND_BP; -			zend_execute(EG(active_op_array) TSRMLS_CC); -#if PHP_VERSION_ID >= 50700 -			if (zend_is_true(retval TSRMLS_CC)) { -#else -			if (zend_is_true(retval)) { -#endif +			zend_execute(bp->ops, &retval); +			if (zend_is_true(&retval)) {  				breakpoint = SUCCESS;  			} -		} zend_catch { -			CG(interactive) = orig_interactive; - -			EG(no_extensions)=1; -			EG(return_value_ptr_ptr) = orig_retval; -			EG(active_op_array) = orig_ops; -			EG(opline_ptr) = orig_opline; -			PHPDBG_G(flags) &= ~PHPDBG_IN_COND_BP;   		} zend_end_try(); -		CG(interactive) = orig_interactive; - -		EG(no_extensions)=1; -		EG(return_value_ptr_ptr) = orig_retval; -		EG(active_op_array) = orig_ops; -		EG(opline_ptr) = orig_opline; +		EG(no_extensions) = 1; +		EG(current_execute_data)->opline = orig_opline; +		EG(current_execute_data)->func = orig_func; +		EG(current_execute_data)->return_value = orig_retval;  		PHPDBG_G(flags) &= ~PHPDBG_IN_COND_BP;  		if (breakpoint == SUCCESS) {  			break;  		} -	} +	} ZEND_HASH_FOREACH_END(); -	return (breakpoint == SUCCESS) ? ((phpdbg_breakbase_t*)bp) : NULL; +	return (breakpoint == SUCCESS) ? ((phpdbg_breakbase_t *) bp) : NULL;  } /* }}} */ -PHPDBG_API phpdbg_breakbase_t *phpdbg_find_breakpoint(zend_execute_data* execute_data TSRMLS_DC) /* {{{ */ +PHPDBG_API phpdbg_breakbase_t *phpdbg_find_breakpoint(zend_execute_data *execute_data) /* {{{ */  {  	phpdbg_breakbase_t *base = NULL; @@ -1045,32 +1109,29 @@ PHPDBG_API phpdbg_breakbase_t *phpdbg_find_breakpoint(zend_execute_data* execute  	/* conditions cannot be executed by eval()'d code */  	if (!(PHPDBG_G(flags) & PHPDBG_IN_EVAL) &&  		(PHPDBG_G(flags) & PHPDBG_HAS_COND_BP) && -		(base = phpdbg_find_conditional_breakpoint(execute_data TSRMLS_CC))) { +		(base = phpdbg_find_conditional_breakpoint(execute_data))) {  		goto result;  	} -	if ((PHPDBG_G(flags) & PHPDBG_HAS_FILE_BP) && -		(base = phpdbg_find_breakpoint_file(execute_data->op_array TSRMLS_CC))) { +	if ((PHPDBG_G(flags) & PHPDBG_HAS_FILE_BP) && (base = phpdbg_find_breakpoint_file(&execute_data->func->op_array))) {  		goto result;  	}  	if (PHPDBG_G(flags) & (PHPDBG_HAS_METHOD_BP|PHPDBG_HAS_SYM_BP)) { -		/* check we are at the beginning of the stack */ -		if (execute_data->opline == EG(active_op_array)->opcodes) { -			if ((base = phpdbg_find_breakpoint_symbol( -					execute_data->function_state.function TSRMLS_CC))) { +		zend_op_array *op_array = &execute_data->func->op_array; +		/* check we are at the beginning of the stack, but after argument RECV */ +		if (execute_data->opline == op_array->opcodes + op_array->num_args + !!(op_array->fn_flags & ZEND_ACC_VARIADIC)) { +			if ((base = phpdbg_find_breakpoint_symbol(execute_data->func))) {  				goto result;  			}  		}  	} -	if ((PHPDBG_G(flags) & PHPDBG_HAS_OPLINE_BP) && -		(base = phpdbg_find_breakpoint_opline(execute_data->opline TSRMLS_CC))) { +	if ((PHPDBG_G(flags) & PHPDBG_HAS_OPLINE_BP) && (base = phpdbg_find_breakpoint_opline((phpdbg_opline_ptr_t) execute_data->opline))) {  		goto result;  	} -	if ((PHPDBG_G(flags) & PHPDBG_HAS_OPCODE_BP) && -		(base = phpdbg_find_breakpoint_opcode(execute_data->opline->opcode TSRMLS_CC))) { +	if ((PHPDBG_G(flags) & PHPDBG_HAS_OPCODE_BP) && (base = phpdbg_find_breakpoint_opcode(execute_data->opline->opcode))) {  		goto result;  	} @@ -1085,16 +1146,14 @@ result:  	return base;  } /* }}} */ -PHPDBG_API void phpdbg_delete_breakpoint(zend_ulong num TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_delete_breakpoint(zend_ulong num) /* {{{ */  { -	HashTable **table; -	HashPosition position; +	HashTable *table;  	phpdbg_breakbase_t *brake; +	zend_string *strkey; +	zend_ulong numkey; -	if ((brake = phpdbg_find_breakbase_ex(num, &table, &position TSRMLS_CC))) { -		char *key; -		zend_uint klen; -		zend_ulong idx; +	if ((brake = phpdbg_find_breakbase_ex(num, &table, &numkey, &strkey))) {  		int type = brake->type;  		char *name = NULL;  		size_t name_len = 0L; @@ -1102,7 +1161,7 @@ PHPDBG_API void phpdbg_delete_breakpoint(zend_ulong num TSRMLS_DC) /* {{{ */  		switch (type) {  			case PHPDBG_BREAK_FILE:  			case PHPDBG_BREAK_METHOD: -				if (zend_hash_num_elements((*table)) == 1) { +				if (zend_hash_num_elements(table) == 1) {  					name = estrdup(brake->name);  					name_len = strlen(name);  					if (zend_hash_num_elements(&PHPDBG_G(bp)[type]) == 1) { @@ -1112,7 +1171,7 @@ PHPDBG_API void phpdbg_delete_breakpoint(zend_ulong num TSRMLS_DC) /* {{{ */  			break;  			default: { -				if (zend_hash_num_elements((*table)) == 1) { +				if (zend_hash_num_elements(table) == 1) {  					PHPDBG_G(flags) &= ~(1<<(brake->type+1));  				}  			} @@ -1125,40 +1184,36 @@ PHPDBG_API void phpdbg_delete_breakpoint(zend_ulong num TSRMLS_DC) /* {{{ */  				if (zend_hash_num_elements(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE]) == 1) {  					PHPDBG_G(flags) &= PHPDBG_HAS_OPLINE_BP;  				} -				zend_hash_index_del(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], ((phpdbg_breakopline_t*)brake)->opline); +				zend_hash_index_del(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], ((phpdbg_breakopline_t *) brake)->opline);  		} -		switch (zend_hash_get_current_key_ex( -			(*table), &key, &klen, &idx, 0, &position)) { - -			case HASH_KEY_IS_STRING: -				zend_hash_del((*table), key, klen); -			break; - -			default: -				zend_hash_index_del((*table), idx); +		if (strkey) { +			zend_hash_del(table, strkey); +		} else { +			zend_hash_index_del(table, numkey);  		}  		switch (type) {  			case PHPDBG_BREAK_FILE:  			case PHPDBG_BREAK_METHOD:  				if (name) { -					zend_hash_del(&PHPDBG_G(bp)[type], name, name_len); +					zend_hash_str_del(&PHPDBG_G(bp)[type], name, name_len);  					efree(name);  				}  			break;  		} -		phpdbg_notice("Deleted breakpoint #%ld", num); +		phpdbg_notice("breakpoint", "deleted=\"success\" id=\"%ld\"", "Deleted breakpoint #%ld", num);  		PHPDBG_BREAK_UNMAPPING(num);  	} else { -		phpdbg_error("Failed to find breakpoint #%ld", num); +		phpdbg_error("breakpoint", "type=\"nobreakpoint\" deleted=\"fail\" id=\"%ld\"", "Failed to find breakpoint #%ld", num);  	}  } /* }}} */ -PHPDBG_API void phpdbg_clear_breakpoints(TSRMLS_D) /* {{{ */ +PHPDBG_API void phpdbg_clear_breakpoints(void) /* {{{ */  {  	zend_hash_clean(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE]); +	zend_hash_clean(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING]);  	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_OPLINE]); @@ -1174,23 +1229,23 @@ PHPDBG_API void phpdbg_clear_breakpoints(TSRMLS_D) /* {{{ */  	PHPDBG_G(bp_count) = 0;  } /* }}} */ -PHPDBG_API void phpdbg_hit_breakpoint(phpdbg_breakbase_t *brake, zend_bool output TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_hit_breakpoint(phpdbg_breakbase_t *brake, zend_bool output) /* {{{ */  {  	brake->hits++;  	if (output) { -		phpdbg_print_breakpoint(brake TSRMLS_CC); +		phpdbg_print_breakpoint(brake);  	}  } /* }}} */ -PHPDBG_API void phpdbg_print_breakpoint(phpdbg_breakbase_t *brake TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_print_breakpoint(phpdbg_breakbase_t *brake) /* {{{ */  {  	if (!brake)  		goto unknown;  	switch (brake->type) {  		case PHPDBG_BREAK_FILE: { -			phpdbg_notice("Breakpoint #%d at %s:%ld, hits: %lu", +			phpdbg_notice("breakpoint", "id=\"%d\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Breakpoint #%d at %s:%ld, hits: %lu",  				((phpdbg_breakfile_t*)brake)->id,  				((phpdbg_breakfile_t*)brake)->filename,  				((phpdbg_breakfile_t*)brake)->line, @@ -1198,91 +1253,90 @@ PHPDBG_API void phpdbg_print_breakpoint(phpdbg_breakbase_t *brake TSRMLS_DC) /*  		} break;  		case PHPDBG_BREAK_SYM: { -			phpdbg_notice("Breakpoint #%d in %s() at %s:%u, hits: %lu", +			phpdbg_notice("breakpoint", "id=\"%d\" function=\"%s\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Breakpoint #%d in %s() at %s:%u, hits: %lu",  				((phpdbg_breaksymbol_t*)brake)->id,  				((phpdbg_breaksymbol_t*)brake)->symbol, -				zend_get_executed_filename(TSRMLS_C), -				zend_get_executed_lineno(TSRMLS_C), +				zend_get_executed_filename(), +				zend_get_executed_lineno(),  				((phpdbg_breakfile_t*)brake)->hits);  		} break;  		case PHPDBG_BREAK_OPLINE: { -			phpdbg_notice("Breakpoint #%d in %#lx at %s:%u, hits: %lu", +			phpdbg_notice("breakpoint", "id=\"%d\" opline=\"%#lx\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Breakpoint #%d in %#lx at %s:%u, hits: %lu",  				((phpdbg_breakline_t*)brake)->id,  				((phpdbg_breakline_t*)brake)->opline, -				zend_get_executed_filename(TSRMLS_C), -				zend_get_executed_lineno(TSRMLS_C), +				zend_get_executed_filename(), +				zend_get_executed_lineno(),  				((phpdbg_breakline_t*)brake)->hits);  		} break;  		case PHPDBG_BREAK_METHOD_OPLINE: { -			 phpdbg_notice("Breakpoint #%d in %s::%s()#%lu at %s:%u, hits: %lu", +			 phpdbg_notice("breakpoint", "id=\"%d\" method=\"%s::%s\" num=\"%lu\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Breakpoint #%d in %s::%s()#%lu at %s:%u, hits: %lu",  				((phpdbg_breakopline_t*)brake)->id,  				((phpdbg_breakopline_t*)brake)->class_name,  				((phpdbg_breakopline_t*)brake)->func_name,  				((phpdbg_breakopline_t*)brake)->opline_num, -				zend_get_executed_filename(TSRMLS_C), -				zend_get_executed_lineno(TSRMLS_C), +				zend_get_executed_filename(), +				zend_get_executed_lineno(),  				((phpdbg_breakopline_t*)brake)->hits);  		} break;  		case PHPDBG_BREAK_FUNCTION_OPLINE: { -			 phpdbg_notice("Breakpoint #%d in %s()#%lu at %s:%u, hits: %lu", +			 phpdbg_notice("breakpoint", "id=\"%d\" num=\"%lu\" function=\"%s\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Breakpoint #%d in %s()#%lu at %s:%u, hits: %lu",  				((phpdbg_breakopline_t*)brake)->id,  				((phpdbg_breakopline_t*)brake)->func_name,  				((phpdbg_breakopline_t*)brake)->opline_num, -				zend_get_executed_filename(TSRMLS_C), -				zend_get_executed_lineno(TSRMLS_C), +				zend_get_executed_filename(), +				zend_get_executed_lineno(),  				((phpdbg_breakopline_t*)brake)->hits);  		} break;  		case PHPDBG_BREAK_FILE_OPLINE: { -			 phpdbg_notice("Breakpoint #%d in %s:%lu at %s:%u, hits: %lu", +			 phpdbg_notice("breakpoint", "id=\"%d\" num=\"%lu\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Breakpoint #%d in #%lu at %s:%u, hits: %lu",  				((phpdbg_breakopline_t*)brake)->id, -				((phpdbg_breakopline_t*)brake)->class_name,  				((phpdbg_breakopline_t*)brake)->opline_num, -				zend_get_executed_filename(TSRMLS_C), -				zend_get_executed_lineno(TSRMLS_C), +				zend_get_executed_filename(), +				zend_get_executed_lineno(),  				((phpdbg_breakopline_t*)brake)->hits);  		} break;  		case PHPDBG_BREAK_OPCODE: { -			 phpdbg_notice("Breakpoint #%d in %s at %s:%u, hits: %lu", +			 phpdbg_notice("breakpoint", "id=\"%d\" opcode=\"%s\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Breakpoint #%d in %s at %s:%u, hits: %lu",  				((phpdbg_breakop_t*)brake)->id,  				((phpdbg_breakop_t*)brake)->name, -				zend_get_executed_filename(TSRMLS_C), -				zend_get_executed_lineno(TSRMLS_C), +				zend_get_executed_filename(), +				zend_get_executed_lineno(),  				((phpdbg_breakop_t*)brake)->hits);  		} break;  		case PHPDBG_BREAK_METHOD: { -			 phpdbg_notice("Breakpoint #%d in %s::%s() at %s:%u, hits: %lu", +			 phpdbg_notice("breakpoint", "id=\"%d\" method=\"%s::%s\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Breakpoint #%d in %s::%s() at %s:%u, hits: %lu",  				((phpdbg_breakmethod_t*)brake)->id,  				((phpdbg_breakmethod_t*)brake)->class_name,  				((phpdbg_breakmethod_t*)brake)->func_name, -				zend_get_executed_filename(TSRMLS_C), -				zend_get_executed_lineno(TSRMLS_C), +				zend_get_executed_filename(), +				zend_get_executed_lineno(),  				((phpdbg_breakmethod_t*)brake)->hits);  		} break;  		case PHPDBG_BREAK_COND: {  			if (((phpdbg_breakcond_t*)brake)->paramed) {  				char *param; -				phpdbg_notice("Conditional breakpoint #%d: at %s if %s %s:%u, hits: %lu", +				phpdbg_notice("breakpoint", "id=\"%d\" location=\"%s\" eval=\"%s\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Conditional breakpoint #%d: at %s if %s at %s:%u, hits: %lu",  					((phpdbg_breakcond_t*)brake)->id, -					phpdbg_param_tostring(&((phpdbg_breakcond_t*)brake)->param, ¶m TSRMLS_CC), +					phpdbg_param_tostring(&((phpdbg_breakcond_t*)brake)->param, ¶m),  					((phpdbg_breakcond_t*)brake)->code, -					zend_get_executed_filename(TSRMLS_C), -					zend_get_executed_lineno(TSRMLS_C), +					zend_get_executed_filename(), +					zend_get_executed_lineno(),  					((phpdbg_breakcond_t*)brake)->hits);  				if (param)  					free(param);  			} else { -				phpdbg_notice("Conditional breakpoint #%d: on %s == true %s:%u, hits: %lu", +				phpdbg_notice("breakpoint", "id=\"%d\" eval=\"%s\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Conditional breakpoint #%d: on %s == true at %s:%u, hits: %lu",  					((phpdbg_breakcond_t*)brake)->id,  					((phpdbg_breakcond_t*)brake)->code, -					zend_get_executed_filename(TSRMLS_C), -					zend_get_executed_lineno(TSRMLS_C), +					zend_get_executed_filename(), +					zend_get_executed_lineno(),  					((phpdbg_breakcond_t*)brake)->hits);  			} @@ -1290,349 +1344,285 @@ PHPDBG_API void phpdbg_print_breakpoint(phpdbg_breakbase_t *brake TSRMLS_DC) /*  		default: {  unknown: -			phpdbg_notice("Unknown breakpoint at %s:%u", -				zend_get_executed_filename(TSRMLS_C), -				zend_get_executed_lineno(TSRMLS_C)); +			phpdbg_notice("breakpoint", "id=\"\" file=\"%s\" line=\"%ld\" hits=\"%lu\"", "Unknown breakpoint at %s:%u", +				zend_get_executed_filename(), +				zend_get_executed_lineno());  		}  	}  } /* }}} */ -PHPDBG_API void phpdbg_enable_breakpoint(zend_ulong id TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_enable_breakpoint(zend_ulong id) /* {{{ */  { -	phpdbg_breakbase_t *brake = phpdbg_find_breakbase(id TSRMLS_CC); +	phpdbg_breakbase_t *brake = phpdbg_find_breakbase(id);  	if (brake) {  		brake->disabled = 0;  	}  } /* }}} */ -PHPDBG_API void phpdbg_disable_breakpoint(zend_ulong id TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_disable_breakpoint(zend_ulong id) /* {{{ */  { -	phpdbg_breakbase_t *brake = phpdbg_find_breakbase(id TSRMLS_CC); +	phpdbg_breakbase_t *brake = phpdbg_find_breakbase(id);  	if (brake) {  		brake->disabled = 1;  	}  } /* }}} */ -PHPDBG_API void phpdbg_enable_breakpoints(TSRMLS_D) /* {{{ */ +PHPDBG_API void phpdbg_enable_breakpoints(void) /* {{{ */  {  	PHPDBG_G(flags) |= PHPDBG_IS_BP_ENABLED;  } /* }}} */ -PHPDBG_API void phpdbg_disable_breakpoints(TSRMLS_D) { /* {{{ */ +PHPDBG_API void phpdbg_disable_breakpoints(void) { /* {{{ */  	PHPDBG_G(flags) &= ~PHPDBG_IS_BP_ENABLED;  } /* }}} */ -PHPDBG_API phpdbg_breakbase_t *phpdbg_find_breakbase(zend_ulong id TSRMLS_DC) /* {{{ */ +PHPDBG_API phpdbg_breakbase_t *phpdbg_find_breakbase(zend_ulong id) /* {{{ */  { -	HashTable **table; -	HashPosition position; +	HashTable *table; +	zend_string *strkey; +	zend_ulong numkey; -	return phpdbg_find_breakbase_ex(id, &table, &position TSRMLS_CC); +	return phpdbg_find_breakbase_ex(id, &table, &numkey, &strkey);  } /* }}} */ -PHPDBG_API phpdbg_breakbase_t *phpdbg_find_breakbase_ex(zend_ulong id, HashTable ***table, HashPosition *position TSRMLS_DC) /* {{{ */ +PHPDBG_API phpdbg_breakbase_t *phpdbg_find_breakbase_ex(zend_ulong id, HashTable **table, zend_ulong *numkey, zend_string **strkey) /* {{{ */  { -	if (zend_hash_index_find(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], id, (void**)table) == SUCCESS) { +	if ((*table = zend_hash_index_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_MAP], id))) {  		phpdbg_breakbase_t *brake; -		for (zend_hash_internal_pointer_reset_ex((**table), position); -			zend_hash_get_current_data_ex((**table), (void**)&brake, position) == SUCCESS; -			zend_hash_move_forward_ex((**table), position)) { - +		ZEND_HASH_FOREACH_KEY_PTR(*table, *numkey, *strkey, brake) {  			if (brake->id == id) {  				return brake;  			} -		} +		} ZEND_HASH_FOREACH_END();  	} +  	return NULL;  } /* }}} */ -PHPDBG_API void phpdbg_print_breakpoints(zend_ulong type TSRMLS_DC) /* {{{ */ +PHPDBG_API void phpdbg_print_breakpoints(zend_ulong type) /* {{{ */  { +	phpdbg_xml("<breakpoints %r>"); +  	switch (type) {  		case PHPDBG_BREAK_SYM: if ((PHPDBG_G(flags) & PHPDBG_HAS_SYM_BP)) { -			HashPosition position;  			phpdbg_breaksymbol_t *brake; -			phpdbg_writeln(SEPARATE); -			phpdbg_writeln("Function Breakpoints:"); -			for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], &position); -			     zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], (void**) &brake, &position) == SUCCESS; -			     zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], &position)) { -				phpdbg_writeln("#%d\t\t%s%s", +			phpdbg_out(SEPARATE "\n"); +			phpdbg_out("Function Breakpoints:\n"); +			ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_SYM], brake) { +				phpdbg_writeln("function", "id=\"%d\" name=\"%s\" disabled=\"%s\"", "#%d\t\t%s%s",  					brake->id, brake->symbol, -					((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); -			} +					((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : ""); +			} ZEND_HASH_FOREACH_END();  		} break;  		case PHPDBG_BREAK_METHOD: if ((PHPDBG_G(flags) & PHPDBG_HAS_METHOD_BP)) { -			HashPosition position[2];  			HashTable *class_table; -			char *class_name = NULL; -			zend_uint class_len = 0; -			zend_ulong class_idx = 0L; - -			phpdbg_writeln(SEPARATE); -			phpdbg_writeln("Method Breakpoints:"); -			for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], &position[0]); -			     zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], (void**) &class_table, &position[0]) == SUCCESS; -			     zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], &position[0])) { - -				if (zend_hash_get_current_key_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], -					&class_name, &class_len, &class_idx, 0, &position[0]) == HASH_KEY_IS_STRING) { -					phpdbg_breakmethod_t *brake; - -					for (zend_hash_internal_pointer_reset_ex(class_table, &position[1]); -					     zend_hash_get_current_data_ex(class_table, (void**)&brake, &position[1]) == SUCCESS; -					     zend_hash_move_forward_ex(class_table, &position[1])) { -						phpdbg_writeln("#%d\t\t%s::%s%s", -							brake->id, brake->class_name, brake->func_name, -							((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); -					} -				} -			} +			phpdbg_out(SEPARATE "\n"); +			phpdbg_out("Method Breakpoints:\n"); +			ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD], class_table) { +				phpdbg_breakmethod_t *brake; + +				ZEND_HASH_FOREACH_PTR(class_table, brake) { +					phpdbg_writeln("method", "id=\"%d\" name=\"%s::%s\" disabled=\"%s\"", "#%d\t\t%s::%s%s", +						brake->id, brake->class_name, brake->func_name, +						((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : ""); +				} ZEND_HASH_FOREACH_END(); +			} ZEND_HASH_FOREACH_END();  		} break;  		case PHPDBG_BREAK_FILE: if ((PHPDBG_G(flags) & PHPDBG_HAS_FILE_BP)) { -			HashPosition position[2];  			HashTable *points; -			phpdbg_writeln(SEPARATE); -			phpdbg_writeln("File Breakpoints:"); -			for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], &position[0]); -			     zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], (void**) &points, &position[0]) == SUCCESS; -			     zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], &position[0])) { +			phpdbg_out(SEPARATE "\n"); +			phpdbg_out("File Breakpoints:\n"); +			ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], points) {  				phpdbg_breakfile_t *brake; -				for (zend_hash_internal_pointer_reset_ex(points, &position[1]); -				     zend_hash_get_current_data_ex(points, (void**)&brake, &position[1]) == SUCCESS; -				     zend_hash_move_forward_ex(points, &position[1])) { -					phpdbg_writeln("#%d\t\t%s:%lu%s", +				ZEND_HASH_FOREACH_PTR(points, brake) { +					phpdbg_writeln("file", "id=\"%d\" name=\"%s\" line=\"%lu\" disabled=\"%s\"", "#%d\t\t%s:%lu%s",  						brake->id, brake->filename, brake->line, -						((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); -				} -			} +						((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : ""); +				} ZEND_HASH_FOREACH_END(); +			} ZEND_HASH_FOREACH_END(); +		}  if ((PHPDBG_G(flags) & PHPDBG_HAS_PENDING_FILE_BP)) { +			HashTable *points; +			phpdbg_out(SEPARATE "\n"); +			phpdbg_out("Pending File Breakpoints:\n"); +			ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING], points) { +				phpdbg_breakfile_t *brake; + +				ZEND_HASH_FOREACH_PTR(points, brake) { +					phpdbg_writeln("file", "id=\"%d\" name=\"%s\" line=\"%lu\" disabled=\"%s\" pending=\"pending\"", "#%d\t\t%s:%lu%s", +						brake->id, brake->filename, brake->line, +						((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : ""); +				} ZEND_HASH_FOREACH_END(); +			} ZEND_HASH_FOREACH_END();  		} break;  		case PHPDBG_BREAK_OPLINE: if ((PHPDBG_G(flags) & PHPDBG_HAS_OPLINE_BP)) { -			HashPosition position;  			phpdbg_breakline_t *brake; -			phpdbg_writeln(SEPARATE); -			phpdbg_writeln("Opline Breakpoints:"); -			for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], &position); -			     zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], (void**) &brake, &position) == SUCCESS; -			     zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], &position)) { +			phpdbg_out(SEPARATE "\n"); +			phpdbg_out("Opline Breakpoints:\n"); +			ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_OPLINE], brake) { +				const char *type;  				switch (brake->type) {  					case PHPDBG_BREAK_METHOD_OPLINE: +						type = "method"; +						goto print_opline;  					case PHPDBG_BREAK_FUNCTION_OPLINE: +						type = "function"; +						goto print_opline;  					case PHPDBG_BREAK_FILE_OPLINE: -						phpdbg_writeln("#%d\t\t%#lx\t\t(%s breakpoint)%s", brake->id, brake->opline, -							brake->type == PHPDBG_BREAK_METHOD_OPLINE?"method": -								brake->type == PHPDBG_BREAK_FUNCTION_OPLINE?"function": -									brake->type == PHPDBG_BREAK_FILE_OPLINE?"file": -										"--- error ---", -							((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); -						break; +						type = "method"; + +					print_opline: { +						if (brake->type == PHPDBG_BREAK_METHOD_OPLINE) { +							type = "method"; +						} else if (brake->type == PHPDBG_BREAK_FUNCTION_OPLINE) { +							type = "function"; +						} else if (brake->type == PHPDBG_BREAK_FILE_OPLINE) { +							type = "file"; +						} + +						phpdbg_writeln("opline", "id=\"%d\" num=\"%#lx\" type=\"%s\" disabled=\"%s\"", "#%d\t\t%#lx\t\t(%s breakpoint)%s", +							brake->id, brake->opline, type, +							((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : ""); +					} break;  					default: -						phpdbg_writeln("#%d\t\t%#lx", brake->id, brake->opline); +						phpdbg_writeln("opline", "id=\"%d\" num=\"%#lx\" disabled=\"%s\"", "#%d\t\t%#lx%s", +							brake->id, brake->opline, +							((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : "");  						break;  				} -			} +			} ZEND_HASH_FOREACH_END();  		} break;  		case PHPDBG_BREAK_METHOD_OPLINE: if ((PHPDBG_G(flags) & PHPDBG_HAS_METHOD_OPLINE_BP)) { -			HashPosition position[3];  			HashTable *class_table, *method_table; -			char *class_name = NULL, *method_name = NULL; -			zend_uint class_len = 0, method_len = 0; -			zend_ulong class_idx = 0L, method_idx = 0L; - -			phpdbg_writeln(SEPARATE); -			phpdbg_writeln("Method opline Breakpoints:"); -			for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], &position[0]); -			     zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], (void**) &class_table, &position[0]) == SUCCESS; -			     zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], &position[0])) { - -				if (zend_hash_get_current_key_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], -					&class_name, &class_len, &class_idx, 0, &position[0]) == HASH_KEY_IS_STRING) { - -					for (zend_hash_internal_pointer_reset_ex(class_table, &position[1]); -					     zend_hash_get_current_data_ex(class_table, (void**) &method_table, &position[1]) == SUCCESS; -					     zend_hash_move_forward_ex(class_table, &position[1])) { - -						if (zend_hash_get_current_key_ex(class_table, -							&method_name, &method_len, &method_idx, 0, &position[0]) == HASH_KEY_IS_STRING) { - -							phpdbg_breakopline_t *brake; - -							for (zend_hash_internal_pointer_reset_ex(method_table, &position[2]); -							     zend_hash_get_current_data_ex(method_table, (void**)&brake, &position[2]) == SUCCESS; -							     zend_hash_move_forward_ex(method_table, &position[2])) { -								phpdbg_writeln("#%d\t\t%s::%s opline %ld%s", -									brake->id, brake->class_name, brake->func_name, brake->opline_num, -									((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); -							} -						} -					} -				} -			} +			phpdbg_out(SEPARATE "\n"); +			phpdbg_out("Method opline Breakpoints:\n"); +			ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_METHOD_OPLINE], class_table) { +				ZEND_HASH_FOREACH_PTR(class_table, method_table) { +					phpdbg_breakopline_t *brake; + +					ZEND_HASH_FOREACH_PTR(method_table, brake) { +						phpdbg_writeln("methodopline", "id=\"%d\" name=\"%s::%s\" num=\"%ld\" disabled=\"%s\"", "#%d\t\t%s::%s opline %ld%s", +							brake->id, brake->class_name, brake->func_name, brake->opline_num, +							((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : ""); +					} ZEND_HASH_FOREACH_END(); +				} ZEND_HASH_FOREACH_END(); +			} ZEND_HASH_FOREACH_END();  		} break;  		case PHPDBG_BREAK_FUNCTION_OPLINE: if ((PHPDBG_G(flags) & PHPDBG_HAS_FUNCTION_OPLINE_BP)) { -			HashPosition position[2];  			HashTable *function_table; -			char *function_name = NULL; -			zend_uint function_len = 0; -			zend_ulong function_idx = 0L; - -			phpdbg_writeln(SEPARATE); -			phpdbg_writeln("Function opline Breakpoints:"); -			for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FUNCTION_OPLINE], &position[0]); -			     zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FUNCTION_OPLINE], (void**) &function_table, &position[0]) == SUCCESS; -			     zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FUNCTION_OPLINE], &position[0])) { - -				if (zend_hash_get_current_key_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FUNCTION_OPLINE], -					&function_name, &function_len, &function_idx, 0, &position[0]) == HASH_KEY_IS_STRING) { -					phpdbg_breakopline_t *brake; - -					for (zend_hash_internal_pointer_reset_ex(function_table, &position[1]); -					     zend_hash_get_current_data_ex(function_table, (void**)&brake, &position[1]) == SUCCESS; -					     zend_hash_move_forward_ex(function_table, &position[1])) { -						phpdbg_writeln("#%d\t\t%s opline %ld%s", -							brake->id, brake->func_name, brake->opline_num, -							((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); -					} -				} - -			} +			phpdbg_out(SEPARATE "\n"); +			phpdbg_out("Function opline Breakpoints:\n"); +			ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_FUNCTION_OPLINE], function_table) { +				phpdbg_breakopline_t *brake; + +				ZEND_HASH_FOREACH_PTR(function_table, brake) { +					phpdbg_writeln("functionopline", "id=\"%d\" name=\"%s\" num=\"%ld\" disabled=\"%s\"", "#%d\t\t%s opline %ld%s", +						brake->id, brake->func_name, brake->opline_num, +						((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : ""); +				} ZEND_HASH_FOREACH_END(); +			} ZEND_HASH_FOREACH_END();  		} break;  		case PHPDBG_BREAK_FILE_OPLINE: if ((PHPDBG_G(flags) & PHPDBG_HAS_FILE_OPLINE_BP)) { -			HashPosition position[2];  			HashTable *file_table; -			char *file_name = NULL; -			zend_uint file_len = 0; -			zend_ulong file_idx = 0L; -			phpdbg_writeln(SEPARATE); -			phpdbg_writeln("File opline Breakpoints:"); -			for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], &position[0]); -			     zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], (void**) &file_table, &position[0]) == SUCCESS; -			     zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], &position[0])) { - -				if (zend_hash_get_current_key_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], -					&file_name, &file_len, &file_idx, 0, &position[0]) == HASH_KEY_IS_STRING) { - -					phpdbg_breakopline_t *brake; - -					for (zend_hash_internal_pointer_reset_ex(file_table, &position[1]); -					     zend_hash_get_current_data_ex(file_table, (void**)&brake, &position[1]) == SUCCESS; -					     zend_hash_move_forward_ex(file_table, &position[1])) { -						phpdbg_writeln("#%d\t\t%s opline %ld%s", -							brake->id, brake->class_name, brake->opline_num, -							((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); -					} -				} -			} +			phpdbg_out(SEPARATE "\n"); +			phpdbg_out("File opline Breakpoints:\n"); +			ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_OPLINE], file_table) { +				phpdbg_breakopline_t *brake; + +				ZEND_HASH_FOREACH_PTR(file_table, brake) { +					phpdbg_writeln("fileopline", "id=\"%d\" name=\"%s\" num=\"%ld\" disabled=\"%s\"", "#%d\t\t%s opline %ld%s", +						brake->id, brake->class_name, brake->opline_num, +						((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : ""); +				} ZEND_HASH_FOREACH_END(); +			} ZEND_HASH_FOREACH_END();  		} break;  		case PHPDBG_BREAK_COND: if ((PHPDBG_G(flags) & PHPDBG_HAS_COND_BP)) { -			HashPosition position;  			phpdbg_breakcond_t *brake; -			phpdbg_writeln(SEPARATE); -			phpdbg_writeln("Conditional Breakpoints:"); -			for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], &position); -			     zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], (void**) &brake, &position) == SUCCESS; -			     zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], &position)) { +			phpdbg_out(SEPARATE "\n"); +			phpdbg_out("Conditional Breakpoints:\n"); +			ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_COND], brake) {  				if (brake->paramed) {  					switch (brake->param.type) {  						case STR_PARAM: -							phpdbg_writeln("#%d\t\tat %s if %s%s", -				 				brake->id, -				 				brake->param.str, -				 				brake->code, -				 				((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); +							phpdbg_writeln("evalfunction", "id=\"%d\" name=\"%s\" eval=\"%s\" disabled=\"%s\"", "#%d\t\tat %s if %s%s", +				 				brake->id, brake->param.str, brake->code, +				 				((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : "");  						break;  						case NUMERIC_FUNCTION_PARAM: -							phpdbg_writeln("#%d\t\tat %s#%ld if %s%s", -				 				brake->id, -				 				brake->param.str, -								brake->param.num, -				 				brake->code, -				 				((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); +							phpdbg_writeln("evalfunctionopline", "id=\"%d\" name=\"%s\" num=\"%ld\" eval=\"%s\" disabled=\"%s\"", "#%d\t\tat %s#%ld if %s%s", +				 				brake->id, brake->param.str, brake->param.num, brake->code, +				 				((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : "");  						break;  						case METHOD_PARAM: -							phpdbg_writeln("#%d\t\tat %s::%s if %s%s", -				 				brake->id, -				 				brake->param.method.class, -				 				brake->param.method.name, -				 				brake->code, +							phpdbg_writeln("evalmethod", "id=\"%d\" name=\"%s::%s\" eval=\"%s\" disabled=\"%s\"", "#%d\t\tat %s::%s if %s%s", +				 				brake->id, brake->param.method.class, brake->param.method.name, brake->code,  				 				((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : "");  						break;  						case NUMERIC_METHOD_PARAM: -							phpdbg_writeln("#%d\t\tat %s::%s#%ld if %s%s", -				 				brake->id, -				 				brake->param.method.class, -				 				brake->param.method.name, -								brake->param.num, -				 				brake->code, -				 				((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); +							phpdbg_writeln("evalmethodopline", "id=\"%d\" name=\"%s::%s\" num=\"%d\" eval=\"%s\" disabled=\"%s\"", "#%d\t\tat %s::%s#%ld if %s%s", +				 				brake->id, brake->param.method.class, brake->param.method.name, brake->param.num, brake->code, +				 				((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : "");  						break;  						case FILE_PARAM: -							phpdbg_writeln("#%d\t\tat %s:%lu if %s%s", -				 				brake->id, -				 				brake->param.file.name, -				 				brake->param.file.line, -				 				brake->code, -				 				((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); +							phpdbg_writeln("evalfile", "id=\"%d\" name=\"%s\" line=\"%d\" eval=\"%s\" disabled=\"%s\"", "#%d\t\tat %s:%lu if %s%s", +				 				brake->id, brake->param.file.name, brake->param.file.line, brake->code, +				 				((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : "");  						break;  						case ADDR_PARAM: -							phpdbg_writeln("#%d\t\tat #%lx if %s%s", -				 				brake->id, -				 				brake->param.addr, -				 				brake->code, -				 				((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); +							phpdbg_writeln("evalopline", "id=\"%d\" opline=\"%#lx\" eval=\"%s\" disabled=\"%s\"", "#%d\t\tat #%lx if %s%s", +				 				brake->id, brake->param.addr, brake->code, +				 				((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : "");  						break;  						default: -							phpdbg_error("Invalid parameter type for conditional breakpoint"); +							phpdbg_error("eval", "type=\"invalidparameter\"", "Invalid parameter type for conditional breakpoint");  						return;  					}  				} else { -					phpdbg_writeln("#%d\t\tif %s%s", +					phpdbg_writeln("eval", "id=\"%d\" eval=\"%s\" disabled=\"%s\"", "#%d\t\tif %s%s",  				 		brake->id, brake->code, -				 		((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); +				 		((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : "");  				} -			} +			} ZEND_HASH_FOREACH_END();  		} break;  		case PHPDBG_BREAK_OPCODE: if (PHPDBG_G(flags) & PHPDBG_HAS_OPCODE_BP) { -			HashPosition position;  			phpdbg_breakop_t *brake; -			phpdbg_writeln(SEPARATE); -			phpdbg_writeln("Opcode Breakpoints:"); -			for (zend_hash_internal_pointer_reset_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_OPCODE], &position); -			     zend_hash_get_current_data_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_OPCODE], (void**) &brake, &position) == SUCCESS; -			     zend_hash_move_forward_ex(&PHPDBG_G(bp)[PHPDBG_BREAK_OPCODE], &position)) { -				phpdbg_writeln("#%d\t\t%s%s", +			phpdbg_out(SEPARATE "\n"); +			phpdbg_out("Opcode Breakpoints:\n"); +			ZEND_HASH_FOREACH_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_OPCODE], brake) { +				phpdbg_writeln("opcode", "id=\"%d\" name=\"%s\" disabled=\"%s\"", "#%d\t\t%s%s",  					brake->id, brake->name, -					((phpdbg_breakbase_t*)brake)->disabled ? " [disabled]" : ""); -			} +					((phpdbg_breakbase_t *) brake)->disabled ? " [disabled]" : ""); +			} ZEND_HASH_FOREACH_END();  		} break;  	} + +	phpdbg_xml("</breakpoints>");  } /* }}} */ | 
