diff options
Diffstat (limited to 'sapi/phpdbg/phpdbg_bp.c')
| -rw-r--r-- | sapi/phpdbg/phpdbg_bp.c | 131 | 
1 files changed, 104 insertions, 27 deletions
| diff --git a/sapi/phpdbg/phpdbg_bp.c b/sapi/phpdbg/phpdbg_bp.c index aaeaee13fd..f6610cd0ca 100644 --- a/sapi/phpdbg/phpdbg_bp.c +++ b/sapi/phpdbg/phpdbg_bp.c @@ -207,48 +207,111 @@ PHPDBG_API void phpdbg_set_breakpoint_file(const char *path, long line_num TSRML  {  	php_stream_statbuf ssb;  	char realpath[MAXPATHLEN]; +	const char *original_path = path; +	zend_bool pending; -	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; +	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; +	if (VCWD_REALPATH(path, realpath)) { +		path = realpath; +	} +	path_len = strlen(path); + +	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); +			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; +		} +	} -			if (!(broken = zend_hash_str_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], path, path_len))) { -				HashTable breaks; +	if (!(broken = zend_hash_str_find_ptr(file_breaks, path, path_len))) { +		HashTable breaks; +		zend_hash_init(&breaks, 8, NULL, phpdbg_file_breaks_dtor, 0); -				zend_hash_init(&breaks, 8, NULL, phpdbg_file_breaks_dtor, 0); +		broken = zend_hash_str_add_mem(file_breaks, path, path_len, &breaks, sizeof(HashTable)); +	} -				broken = zend_hash_str_update_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], path, path_len, &breaks, sizeof(HashTable)); -			} +	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_mem(broken, line_num, &new_break, sizeof(phpdbg_breakfile_t)); -			if (!zend_hash_index_exists(broken, line_num)) { -				PHPDBG_G(flags) |= PHPDBG_HAS_FILE_BP; +		if (pending) { +			PHPDBG_G(flags) |= PHPDBG_HAS_PENDING_FILE_BP; -				PHPDBG_BREAK_INIT(new_break, PHPDBG_BREAK_FILE); -				new_break.filename = estrndup(path, path_len); -				new_break.line = line_num; +			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_G(flags) |= PHPDBG_HAS_FILE_BP; -				zend_hash_index_update_mem(broken, line_num, &new_break, sizeof(phpdbg_breakfile_t)); +			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); +		} -				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); +		PHPDBG_BREAK_MAPPING(new_break.id, broken); +	} else { +		phpdbg_error("breakpoint", "type=\"exists\" add=\"fail\" file=\"%s\" line=\"%ld\"", "Breakpoint at %s:%ld exists", path, line_num); +	} +} /* }}} */ -				PHPDBG_BREAK_MAPPING(new_break.id, broken); -			} else { -				phpdbg_error("breakpoint", "type=\"exists\" add=\"fail\" file=\"%s\" line=\"%ld\"", "Breakpoint at %s:%ld exists", path, line_num); +PHPDBG_API void phpdbg_resolve_pending_file_break(const char *file TSRMLS_DC) /* {{{ */ +{ +	HashTable *fileht; +	uint filelen = strlen(file); +	zend_string *cur; + +	ZEND_HASH_FOREACH_STR_KEY_PTR(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING], cur, fileht) { +		if (cur->len < filelen && file[filelen - cur->len - 1] == '/' && !memcmp(file + filelen - cur->len, cur->val, cur->len)) { +			phpdbg_breakfile_t *brake, new_brake; +			HashTable *master; +			dtor_func_t dtor; + +			PHPDBG_G(flags) |= PHPDBG_HAS_FILE_BP; + +			if (!(master = zend_hash_str_find_ptr(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], file, filelen))) { +				dtor = PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING].pDestructor; +				PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING].pDestructor = NULL; +				fileht = zend_hash_str_add_mem(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE], file, filelen, fileht, sizeof(HashTable));  			} -		} else { -			phpdbg_error("breakpoint", "type=\"notregular\" add=\"fail\" file=\"%s\"", "Cannot set breakpoint in %s, it is not a regular file", path); +			ZEND_HASH_FOREACH_PTR(fileht, brake) { +				new_brake = *brake; +				new_brake.filename = estrndup(file, filelen); +				PHPDBG_BREAK_UNMAPPING(brake->id); + +				if (master) { +					zend_hash_index_update_mem(master, brake->line, &new_brake, sizeof(phpdbg_breakfile_t)); +					PHPDBG_BREAK_MAPPING(brake->id, master); +				} else { +					efree((char *) brake->filename); +					*brake = new_brake; +					PHPDBG_BREAK_MAPPING(brake->id, fileht); +				} +			} ZEND_HASH_FOREACH_END(); + +			zend_hash_del(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING], cur); + +			if (!master) { +				PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING].pDestructor = dtor; +			} + +			if (!zend_hash_num_elements(&PHPDBG_G(bp)[PHPDBG_BREAK_FILE_PENDING])) { +				PHPDBG_G(flags) &= ~PHPDBG_HAS_PENDING_FILE_BP; +			}  		} -	} else { -		phpdbg_error("breakpoint", "type=\"nofile\" add=\"fail\" file=\"%s\"", "Cannot stat %s, it does not exist", path); -	} +	} ZEND_HASH_FOREACH_END();  } /* }}} */  PHPDBG_API void phpdbg_set_breakpoint_symbol(const char *name, size_t name_len TSRMLS_DC) /* {{{ */ @@ -1048,6 +1111,7 @@ PHPDBG_API void phpdbg_delete_breakpoint(zend_ulong num TSRMLS_DC) /* {{{ */  PHPDBG_API void phpdbg_clear_breakpoints(TSRMLS_D) /* {{{ */  {  	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]); @@ -1283,7 +1347,20 @@ PHPDBG_API void phpdbg_print_breakpoints(zend_ulong type TSRMLS_DC) /* {{{ */  						((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)) { | 
