首页 > 后端开发 > 正文

如何调试PHP的Core之获取基本信息

2024-09-06 10:11:14 | 我爱编程网

我爱编程网小编给大家带来了如何调试PHP的Core之获取基本信息相关文章,一起来看一下吧。

本文目录一览:

如何调试PHP的Core之获取基本信息

如何调试PHP的Core之获取基本信息

在这个过程中, 会涉及到对PHP的函数调用, PHP的传参, PHP的一些全局变量的知识.

首先, 让我们生成一个供我们举例子的Core文件:

<?php
function recurse($num) {
recurse(++$num);
}

recurse(0);
运行这个PHP文件:

$ php test.php
Segmentation fault (core dumped)
这个PHP因为无线递归, 会导致爆栈, 从而造成 segment fault而在PHP的当前工作目录产生Coredump文件(如果你的系统没有产生Coredump文件, 那请查询ulimit的相关设置).

现在删除掉这个test.php, 忘掉上面的代码, 我们现在仅有的是这个Core文件, 任务是, 找出这个Core产生的原因, 以及发生时候的状态.

首先, 让我们用gdb打开这个core文件:

$ gdb php -c core.31656
会看到很多的信息, 首先让我们注意这段:

Core was generated by `php test.php'.
Program terminated with signal 11, Segmentation fault.
他告诉我们Core发生的原因:”Segmentation fault”.

一般来说, 这种Core是最常见的, 解引用空指针, double free, 以及爆栈等等, 都会触发SIGSEGV, 继而默认的产生Coredump.

现在让我们看看Core发生时刻的堆栈:

#0 execute (op_array=0xdc9a70) at /home/laruence/package/php-5.2.14/Zend/zend_vm_execute.h:53
53 memset(EX(CVs), 0, sizeof(zval**) * op_array->last_var);
(gdb) bt
#0 execute (op_array=0xdc9a70) at /home/laruence/package/php-5.2.14/Zend/zend_vm_execute.h:53
#1 0x00000000006ea263 in zend_do_fcall_common_helper_SPEC (execute_data=0x7fbf400210) at /home/laruence/package/php-5.2.14/Zend/zend_vm_execute.h:234
#2 0x00000000006e9f61 in execute (op_array=0xdc9a70) at /home/laruence/package/php-5.2.14/Zend/zend_vm_execute.h:92
#3 0x00000000006ea263 in zend_do_fcall_common_helper_SPEC (execute_data=0x7fbf400440) at /home/laruence/package/php-5.2.14/Zend/zend_vm_execute.h:234
#4 0x00000000006e9f61 in execute (op_array=0xdc9a70) at /home/laruence/package/php-5.2.14/Zend/zend_vm_execute.h:92
#5 0x00000000006ea263 in zend_do_fcall_common_helper_SPEC (execute_data=0x7fbf400670) at /home/laruence/package/php-5.2.14/Zend/zend_vm_execute.h:234
.....
不停的按回车, 可以看到堆栈很深, 不停的是zend_do_fcall_common_helper_SPEC和execute的重复, 那么这基本就能断定是因为产生了无穷大的递归(不能一定说是无穷递归, 比如我之前文章中介绍深悉正则(pcre)最大回溯/递归限制). 从而造成爆栈产生的Core.

Ok, 那么现在让我们看看, Core发生在PHP的什么函数中, 在PHP中, 对于FCALL_* Opcode的handler来说, execute_data代表了当前函数调用的一个State, 这个State中包含了信息:

(gdb)f 1
#1 0x00000000006ea263 in zend_do_fcall_common_helper_SPEC (execute_data=0x7fbf400210) at /home/laruence/package/php-5.2.14/Zend/zend_vm_execute.h:234
234 zend_execute(EG(active_op_array) TSRMLS_CC);
(gdb) p execute_data->function_state.function->common->function_name
$3 = 0x2a95b65a78 "recurse"
(gdb) p execute_data->function_state.function->op_array->filename
$4 = 0x2a95b632a0 "/home/laruence/test.php"
(gdb) p execute_data->function_state.function->op_array->line_start
$5 = 2
现在我们得到, 在调用的PHP函数是recurse, 这个函数定义在/home/laruence/test.php的第二行

经过重复验证几个frame, 我们可以看出, 一直是在重复调用这个PHP函数.

要注意的是, 为了介绍查看执行信息的原理, 我才采用原生的gdb的print来查看, 其实我们还可以使用PHP源代码中提供的.gdbinit(gdb命令编写脚本), 来简单的获取到上面的信息:

(gdb) source /home/laruence/package/php-5.2.14/.gdbinit
(gdb) zbacktrace
[0xbf400210] recurse() /home/laruence/test.php:3
[0xbf400440] recurse() /home/laruence/test.php:3
[0xbf400670] recurse() /home/laruence/test.php:3
[0xbf4008a0] recurse() /home/laruence/test.php:3
[0xbf400ad0] recurse() /home/laruence/test.php:3
[0xbf400d00] recurse() /home/laruence/test.php:3
[0xbf400f30] recurse() /home/laruence/test.php:3
[0xbf401160] recurse() /home/laruence/test.php:3
.....
关于.gdbinit, 是一段小小的脚本文件, 定义了一些方便我们去调试PHP的Core, 大家也可以用文本编辑器打开, 看看里面定义的一些快捷的命令, 一般来说, 我常用的有:

zbacktrace
print_ht**系列
zmemcheck
OK, 回归正题, 我们现在知道, 问题发生在/home/laruence/test.php的recurse函数的递归调用上了.

现在, 让我们来看看, 在调用这个函数的时候的参数是什么?

PHP的参数传递是依靠一个全局Stack来完成的, 也就是EG(argument_stack), EG在非多线程情况下就是executor_globals, 它保持了很多执行状态. 而argument_statck就是参数的传递栈, 保存着对应PHP函数调用层数相当的调用参数.

要注意的是, 这个PHP函数调用堆栈(层数)不和gdb所看到的backtrace简单的一一对应, 所以参数也不能直接和gdb的backtrace对应起来, 需要单独分析:

//先看看, 最后一次函数调用的参数数目是多少
(gdb) p (int )*(executor_globals->argument_stack->top_element - 2)
$13 = 1

//再看看, 最后一次函数调用的参数是什么
(gdb) p **(zval **)(executor_globals->argument_stack->top_element - 3)
$2 = {value = {lval = 22445, dval = 1.1089303420906779e-319, str = {val = 0x57ad <Address 0x57ad out of bounds>, len = 7}, ht = 0x57ad, obj = {handle = 22445, handlers = 0x7}},
refcount = 2, type = 1 '\001', is_ref = 0 '\0'}
好, 我们现在得到, 最后一次调用的参数是一个整数, 数值是22445

到了这一步, 我们就得到了这个Core发生的时刻的PHP层面的相关信息, 接下来, 就可以交给对应的PHP开发工程师来排查, 这个参数下, 可能造成的无穷大递归的原因, 从而修复这个问题..

如何调试PHP的Core之获取基本信息

想问这段PHP代码为什么一开始就会显示+-*/不是要匹配$_POST["op"]的值才能输出吗,哪里有触发点的?

一开始$_POST["op"]不存在,显示的+-*是最后一个else语句里的。然后你按计算按钮时,相当于跳转到这个页面,这次$_POST["op"]有值了然后进行switch匹配计算,然后再打印出页面。

如何调试PHP的Core之获取基本信息我爱编程网

php底层原理 php是如何运行的

我爱编程网(https://www.52biancheng.com)小编还为大家带来php底层原理 php是如何运行的的相关内容。

1、PHP动态语言执行过程:拿到一段代码后,经过词法解析、语法解析等阶段后,源程序会被翻译成一个个指令(opcodes),然后ZEND虚拟机顺次执行这些指令完成操作。PHP本身是用C实现的,因此最终调用的也是C的函数,实际上,我们可以把PHP看做一个C开发的软件。

2、PHP的4层运行体系:

(1)Zend引擎:Zend整体用纯C实现,是PHP的内核部分,他将PHP代码翻译(词法、语法解析等一系列编译过程)为可执行opcode的处理并实现相应的处理方法、实现了基本的数据结构(如:hashtable、OO)、内存分配机制及管理、提供了相应的api方法供外部调用,是一切的核心,所有的外围功能均围绕Zend实现。

(2)Extensions:围绕着Zend引擎,extensions通过组件式的方式提供各种基础服务,我们常见的各种内置函数(array系列)、标准库等都是通过extension来实现,用户也可以根据需要实现自己的extension的典型应用)。

(3)Sapi:Sapi全称ServerApplicationProgrammingInterface,也就是服务端应用编程接口,Sapi通过一系列钩子函数,使得PHP可以和外围交互数据,这是PHP非常优雅和成功的设计,通过sapi成功的将PHP本身和上层应用解耦隔离,PHP可以不再考虑如何针对不同应用进行兼容,而应用本身也可以针对自己的特点实现不同的处理方式。

(4)上层应用:这就是我们平时编写的PHP程序,通过不同的spai方式得到各种各样的应用模式,如何通过webserver实现web应用、在命令行下已脚本方式运行等等。

以上就是我爱编程网小编给大家带来的如何调试PHP的Core之获取基本信息,希望能对大家有所帮助。更多相关文章关注我爱编程网:www.52biancheng.com

免责声明:文章内容来自网络,如有侵权请及时联系删除。
与“如何调试PHP的Core之获取基本信息”相关推荐
php 如何获取服务器的信息
php 如何获取服务器的信息

php如何获取服务器的信息服务器变量$_SERVER详解:1、$_SESSION['PHP_SELF']—获取当前正在执行脚本的文件名2、$_SERVER['SERVER_PROTOCOL']—请求页面时通信协议的名称和版本。例如,“HTTP/1.0”。3、$_SERVER['REQUEST_TIME']—请求开始时的时间戳。从PHP5.1.0起有效。和time函数效果一样

2024-10-12 20:46:41
jsp中request.getHeader()获取客户端的各种信息,那php中什么方法来获取
jsp中request.getHeader()获取客户端的各种信息,那php中什么方法来获取

thinkphprequest和param的区别request默认可以接受$_GET,$_POST和$_COOKIE(这是php内置的),$_REQUEST(超全局变量),只是把值获取到,但并不判断是什么类型的数据。param是thinkphp自己的类型,自动判断是什么类型的数据,接收即可。无论什么类型都可以接受,包括php://input这种原生数据流。php中get和reques

2024-09-24 01:24:31
如何创建phpinfo查看php信息
如何创建phpinfo查看php信息

如何创建phpinfo查看php信息1、方法一:命令行查询,如果已经配置好环境变量,直接在命令行中输入php-v,将会显示php的版本信息。如果没有配置环境变量,直接在命令行中进入到php的安装目录后,再输入命令php-v,在安装目录下输入查询命令,可以看到输出的版本信息为PHP5.3.29。2、方法二:(1)使用预定义常量PHP_VERSION查询,新建一个php文件,

2024-07-22 16:21:15
java中 在更改密码界面中,需要获取当前登录的用户信息,请问在程序中是如何获?
java中 在更改密码界面中,需要获取当前登录的用户信息,请问在程序中是如何获?

怎样在一个java程序中获得另一个程序的运行状态先把2个文件放在同一个包下面,这样就可以访问另一个java程序了。一般的方法是在a程序里实例化b类,然后通过b.方法名去调用b类里的方法runtime.getruntime().exec("外部程序");相当于你在cmd控制台中输入"外部程序"并回车执行java中在更改密码界面中,需要获取当前登录的用户信息,请问在程序中是如

2024-04-26 06:28:00
php如何获取函数被调用位置
php如何获取函数被调用位置

PHP直接调用函数和调用另外一个文件中的函数的区别//如果是PHP文件中的函数,唯一的区别是:先引入外部文件,然后直接调用。//如果是外部类文件中的函数,需要引入,然后new 对象后,再调用该对象的函数。$msg=test_msg();function test_msg(){echo "";}//调用外部文件中的函数//首先引入文件include_once 'file.php';$msg=t

2024-08-09 16:32:20
如何用php获取rabbitmq指定队列中的未处理消息数量
如何用php获取rabbitmq指定队列中的未处理消息数量

如何用php获取rabbitmq指定队列中的未处理消息数量lare一个队列,置AMQP_PASSIVE标志位,就不会影响服务端状态,并返回消息计数。$conn=newAMQPConnection();//...$queue=newAMQPQueue($conn);$queue-&gt;setFlags(AMQP_PASSIVE);$messageCount=$queue-

2024-08-01 13:38:35
java程序如何调试
java程序如何调试

java程序如何调试其实最好是在DOS环境下来进行调试的。这样的话。你会对JAVA的运行机制会有很深刻的了解的。当然你也可以使用象eclipse和myeclipse这样的软件的。这样的软件集成度会有点高的。但是就是说呢比较傻瓜化,不利于学习的。怎么搭建的java开发,测试,生产环境的Java编译运行过程程序员所编写的是以.java为后缀的文件,此文件操作系统不能正确识别,因此,

2024-04-11 18:45:53
PHP如何获取本地时间与服务器时间
PHP如何获取本地时间与服务器时间

PHP如何获取本地时间与服务器时间PHP获取服务器时间和本地时间很简单,可以用内置函数实现,具体代码如下:记得给分!!!&lt;?php //获取服务器时间    echo date("Y-m-d h:i:s"); //获取本地时间(也可以说是你所在时区时间)    date_default_timezone_set('地区'); //地区:中国是PRC,或shanghai

2024-08-11 01:52:48