123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368 |
- /*
- +----------------------------------------------------------------------+
- | Copyright (c) The PHP Group |
- +----------------------------------------------------------------------+
- | This source file is subject to version 3.01 of the PHP license, |
- | that is bundled with this package in the file LICENSE, and is |
- | available through the world-wide-web at the following url: |
- | https://www.php.net/license/3_01.txt |
- | If you did not receive a copy of the PHP license and are unable to |
- | obtain it through the world-wide-web, please send a note to |
- | license@php.net so we can mail you a copy immediately. |
- +----------------------------------------------------------------------+
- | Authors: Felipe Pena <felipe@php.net> |
- | Authors: Joe Watkins <joe.watkins@live.co.uk> |
- | Authors: Bob Weinand <bwoebi@php.net> |
- +----------------------------------------------------------------------+
- */
- #include <stdio.h>
- #include <string.h>
- #include <sys/stat.h>
- #ifndef _WIN32
- # include <sys/mman.h>
- # include <unistd.h>
- #endif
- #include <fcntl.h>
- #include "phpdbg.h"
- #include "phpdbg_list.h"
- #include "phpdbg_utils.h"
- #include "phpdbg_prompt.h"
- #include "php_streams.h"
- #include "zend_exceptions.h"
- ZEND_EXTERN_MODULE_GLOBALS(phpdbg)
- #define PHPDBG_LIST_COMMAND_D(f, h, a, m, l, s, flags) \
- PHPDBG_COMMAND_D_EXP(f, h, a, m, l, s, &phpdbg_prompt_commands[12], flags)
- const phpdbg_command_t phpdbg_list_commands[] = {
- PHPDBG_LIST_COMMAND_D(lines, "lists the specified lines", 'l', list_lines, NULL, "l", PHPDBG_ASYNC_SAFE),
- PHPDBG_LIST_COMMAND_D(class, "lists the specified class", 'c', list_class, NULL, "s", PHPDBG_ASYNC_SAFE),
- PHPDBG_LIST_COMMAND_D(method, "lists the specified method", 'm', list_method, NULL, "m", PHPDBG_ASYNC_SAFE),
- PHPDBG_LIST_COMMAND_D(func, "lists the specified function", 'f', list_func, NULL, "s", PHPDBG_ASYNC_SAFE),
- PHPDBG_END_COMMAND
- };
- PHPDBG_LIST(lines) /* {{{ */
- {
- if (!PHPDBG_G(exec) && !zend_is_executing()) {
- phpdbg_error("Not executing, and execution context not set");
- return SUCCESS;
- }
- switch (param->type) {
- case NUMERIC_PARAM: {
- const char *char_file = phpdbg_current_file();
- zend_string *file = zend_string_init(char_file, strlen(char_file), 0);
- phpdbg_list_file(file, param->num < 0 ? 1 - param->num : param->num, (param->num < 0 ? param->num : 0) + zend_get_executed_lineno(), 0);
- efree(file);
- } break;
- case FILE_PARAM: {
- zend_string *file;
- char resolved_path_buf[MAXPATHLEN];
- const char *abspath = param->file.name;
- if (VCWD_REALPATH(abspath, resolved_path_buf)) {
- abspath = resolved_path_buf;
- }
- file = zend_string_init(abspath, strlen(abspath), 0);
- phpdbg_list_file(file, param->file.line, 0, 0);
- zend_string_release(file);
- } break;
- phpdbg_default_switch_case();
- }
- return SUCCESS;
- } /* }}} */
- PHPDBG_LIST(func) /* {{{ */
- {
- phpdbg_list_function_byname(param->str, param->len);
- return SUCCESS;
- } /* }}} */
- PHPDBG_LIST(method) /* {{{ */
- {
- zend_class_entry *ce;
- if (phpdbg_safe_class_lookup(param->method.class, strlen(param->method.class), &ce) == SUCCESS) {
- zend_function *function;
- char *lcname = zend_str_tolower_dup(param->method.name, strlen(param->method.name));
- if ((function = zend_hash_str_find_ptr(&ce->function_table, lcname, strlen(lcname)))) {
- phpdbg_list_function(function);
- } else {
- phpdbg_error("Could not find %s::%s", param->method.class, param->method.name);
- }
- efree(lcname);
- } else {
- phpdbg_error("Could not find the class %s", param->method.class);
- }
- return SUCCESS;
- } /* }}} */
- PHPDBG_LIST(class) /* {{{ */
- {
- zend_class_entry *ce;
- if (phpdbg_safe_class_lookup(param->str, param->len, &ce) == SUCCESS) {
- if (ce->type == ZEND_USER_CLASS) {
- if (ce->info.user.filename) {
- phpdbg_list_file(ce->info.user.filename, ce->info.user.line_end - ce->info.user.line_start + 1, ce->info.user.line_start, 0);
- } else {
- phpdbg_error("The source of the requested class (%s) cannot be found", ZSTR_VAL(ce->name));
- }
- } else {
- phpdbg_error("The class requested (%s) is not user defined", ZSTR_VAL(ce->name));
- }
- } else {
- phpdbg_error("The requested class (%s) could not be found", param->str);
- }
- return SUCCESS;
- } /* }}} */
- void phpdbg_list_file(zend_string *filename, uint32_t count, int offset, uint32_t highlight) /* {{{ */
- {
- uint32_t line, lastline;
- phpdbg_file_source *data;
- if (!(data = zend_hash_find_ptr(&PHPDBG_G(file_sources), filename))) {
- phpdbg_error("Could not find information about included file...");
- return;
- }
- if (offset < 0) {
- count += offset;
- offset = 0;
- }
- lastline = offset + count;
- if (lastline > data->lines) {
- lastline = data->lines;
- }
- for (line = offset; line < lastline;) {
- uint32_t linestart = data->line[line++];
- uint32_t linelen = data->line[line] - linestart;
- char *buffer = data->buf + linestart;
- if (!highlight) {
- phpdbg_write(" %05u: %.*s", line, linelen, buffer);
- } else {
- if (highlight != line) {
- phpdbg_write(" %05u: %.*s", line, linelen, buffer);
- } else {
- phpdbg_write(">%05u: %.*s", line, linelen, buffer);
- }
- }
- if (*(buffer + linelen - 1) != '\n' || !linelen) {
- phpdbg_out("\n");
- }
- }
- } /* }}} */
- void phpdbg_list_function(const zend_function *fbc) /* {{{ */
- {
- const zend_op_array *ops;
- if (fbc->type != ZEND_USER_FUNCTION) {
- phpdbg_error("The function requested (%s) is not user defined", ZSTR_VAL(fbc->common.function_name));
- return;
- }
- ops = (zend_op_array *) fbc;
- phpdbg_list_file(ops->filename, ops->line_end - ops->line_start + 1, ops->line_start, 0);
- } /* }}} */
- void phpdbg_list_function_byname(const char *str, size_t len) /* {{{ */
- {
- HashTable *func_table = EG(function_table);
- zend_function* fbc;
- char *func_name = (char*) str;
- size_t func_name_len = len;
- /* search active scope if begins with period */
- if (func_name[0] == '.') {
- zend_class_entry *scope = zend_get_executed_scope();
- if (scope) {
- func_name++;
- func_name_len--;
- func_table = &scope->function_table;
- } else {
- phpdbg_error("No active class");
- return;
- }
- } else if (!EG(function_table)) {
- phpdbg_error("No function table loaded");
- return;
- } else {
- func_table = EG(function_table);
- }
- /* use lowercase names, case insensitive */
- func_name = zend_str_tolower_dup(func_name, func_name_len);
- phpdbg_try_access {
- if ((fbc = zend_hash_str_find_ptr(func_table, func_name, func_name_len))) {
- phpdbg_list_function(fbc);
- } else {
- phpdbg_error("Function %s not found", func_name);
- }
- } phpdbg_catch_access {
- phpdbg_error("Could not list function %s, invalid data source", func_name);
- } phpdbg_end_try_access();
- efree(func_name);
- } /* }}} */
- /* Note: do not free the original file handler, let original compile_file() or caller do that. Caller may rely on its value to check success */
- zend_op_array *phpdbg_compile_file(zend_file_handle *file, int type) {
- phpdbg_file_source data, *dataptr;
- zend_op_array *ret;
- uint32_t line;
- char *bufptr, *endptr;
- size_t len;
- /* Copy file contents before calling original compile_file,
- * as it may invalidate the file handle. */
- if (zend_stream_fixup(file, &bufptr, &len) == FAILURE) {
- if (type == ZEND_REQUIRE) {
- zend_message_dispatcher(ZMSG_FAILED_REQUIRE_FOPEN, ZSTR_VAL(file->filename));
- } else {
- zend_message_dispatcher(ZMSG_FAILED_INCLUDE_FOPEN, ZSTR_VAL(file->filename));
- }
- return NULL;
- }
- data.buf = estrndup(bufptr, len);
- data.len = len;
- ret = PHPDBG_G(compile_file)(file, type);
- if (ret == NULL) {
- efree(data.buf);
- return ret;
- }
- data.buf[data.len] = '\0';
- data.line[0] = 0;
- *(dataptr = emalloc(sizeof(phpdbg_file_source) + sizeof(uint32_t) * data.len)) = data;
- for (line = 0, bufptr = data.buf - 1, endptr = data.buf + data.len; ++bufptr < endptr;) {
- if (*bufptr == '\n') {
- dataptr->line[++line] = (uint32_t)(bufptr - data.buf) + 1;
- }
- }
- dataptr->lines = ++line;
- dataptr = erealloc(dataptr, sizeof(phpdbg_file_source) + sizeof(uint32_t) * line);
- dataptr->line[line] = endptr - data.buf;
- zend_hash_del(&PHPDBG_G(file_sources), ret->filename);
- zend_hash_add_ptr(&PHPDBG_G(file_sources), ret->filename, dataptr);
- phpdbg_resolve_pending_file_break(ZSTR_VAL(ret->filename));
- return ret;
- }
- zend_op_array *phpdbg_init_compile_file(zend_file_handle *file, int type) {
- zend_string *filename = file->opened_path ? file->opened_path : file->filename;
- char resolved_path_buf[MAXPATHLEN];
- zend_op_array *op_array;
- phpdbg_file_source *dataptr;
- if (VCWD_REALPATH(ZSTR_VAL(filename), resolved_path_buf)) {
- filename = zend_string_init(resolved_path_buf, strlen(resolved_path_buf), 0);
- if (file->opened_path) {
- zend_string_release(file->opened_path);
- file->opened_path = filename;
- } else {
- zend_string_release(file->filename);
- file->filename = filename;
- }
- }
- op_array = PHPDBG_G(init_compile_file)(file, type);
- if (op_array == NULL) {
- return NULL;
- }
- dataptr = zend_hash_find_ptr(&PHPDBG_G(file_sources), op_array->filename);
- ZEND_ASSERT(dataptr != NULL);
- dataptr->op_array = *op_array;
- if (dataptr->op_array.refcount) {
- ++*dataptr->op_array.refcount;
- }
- return op_array;
- }
- zend_op_array *phpdbg_compile_string(zend_string *source_string, const char *filename) {
- zend_string *fake_name;
- zend_op_array *op_array;
- phpdbg_file_source *dataptr;
- uint32_t line;
- char *bufptr, *endptr;
- if (PHPDBG_G(flags) & PHPDBG_IN_EVAL) {
- return PHPDBG_G(compile_string)(source_string, filename);
- }
- dataptr = emalloc(sizeof(phpdbg_file_source) + sizeof(uint32_t) * ZSTR_LEN(source_string));
- dataptr->buf = estrndup(ZSTR_VAL(source_string), ZSTR_LEN(source_string));
- dataptr->len = ZSTR_LEN(source_string);
- dataptr->line[0] = 0;
- for (line = 0, bufptr = dataptr->buf - 1, endptr = dataptr->buf + dataptr->len; ++bufptr < endptr;) {
- if (*bufptr == '\n') {
- dataptr->line[++line] = (uint32_t)(bufptr - dataptr->buf) + 1;
- }
- }
- dataptr->lines = ++line;
- dataptr->line[line] = endptr - dataptr->buf;
- op_array = PHPDBG_G(compile_string)(source_string, filename);
- if (op_array == NULL) {
- efree(dataptr->buf);
- efree(dataptr);
- return NULL;
- }
- fake_name = strpprintf(0, "%s%c%p", filename, 0, op_array->opcodes);
- dataptr = erealloc(dataptr, sizeof(phpdbg_file_source) + sizeof(uint32_t) * line);
- zend_hash_add_ptr(&PHPDBG_G(file_sources), fake_name, dataptr);
- zend_string_release(fake_name);
- dataptr->op_array = *op_array;
- if (dataptr->op_array.refcount) {
- ++*dataptr->op_array.refcount;
- }
- return op_array;
- }
- void phpdbg_init_list(void) {
- PHPDBG_G(compile_file) = zend_compile_file;
- PHPDBG_G(compile_string) = zend_compile_string;
- zend_compile_file = phpdbg_compile_file;
- zend_compile_string = phpdbg_compile_string;
- }
- void phpdbg_list_update(void) {
- PHPDBG_G(init_compile_file) = zend_compile_file;
- zend_compile_file = phpdbg_init_compile_file;
- }
|