PHP 大会日本 2024

使用 PHP 和 DTrace

PHP 可以配置为在支持 DTrace 动态跟踪的平台上使用 DTrace 静态探针。

配置 PHP 以使用 DTrace 静态探针

请参考外部平台特定的文档以启用操作系统 DTrace 支持。例如,在 Oracle Linux 上启动 UEK3 内核并执行以下操作:

# modprobe fasttrap
# chmod 666 /dev/dtrace/helper

您可以使用 ACL 包规则来限制特定用户的设备访问权限,而不是使用 chmod

使用 --enable-dtrace 配置参数构建 PHP

# ./configure --enable-dtrace ...
# make
# make install

这将在核心 PHP 中启用静态探针。任何提供自身探针的 PHP 扩展都应作为共享扩展单独构建。

核心 PHP 中的 DTrace 静态探针

PHP 中提供了以下静态探针:
探针名称 探针描述 探针参数
request-startup 请求开始时触发。 char *file, char *request_uri, char *request_method
request-shutdown 请求关闭时触发。 char *file, char *request_uri, char *request_method
compile-file-entry 脚本编译开始时触发。 char *compile_file, char *compile_file_translated
compile-file-return 脚本编译完成后触发。 char *compile_file, char *compile_file_translated
execute-entry 要执行操作码数组时触发。例如,在函数调用、包含和生成器恢复时触发。 char *request_file, int lineno
execute-return 操作码数组执行后触发。 char *request_file, int lineno
function-entry PHP 引擎进入 PHP 函数或方法调用时触发。 char *function_name, char *request_file, int lineno, char *classname, char *scope
function-return PHP 引擎从 PHP 函数或方法调用返回时触发。 char *function_name, char *request_file, int lineno, char *classname, char *scope
exception-thrown 抛出异常时触发。 char *classname
exception-caught 捕获异常时触发。 char *classname
error 发生错误时触发,与 error_reporting 级别无关。 char *errormsg, char *request_file, int lineno

PHP 扩展也可能具有其他静态探针。

列出 PHP 中的 DTrace 静态探针

要列出可用的探针,请启动一个 PHP 进程,然后运行:

# dtrace -l

输出将类似于:

   ID   PROVIDER            MODULE                          FUNCTION NAME
   [ . . . ]
    4   php15271               php               dtrace_compile_file compile-file-entry
    5   php15271               php               dtrace_compile_file compile-file-return
    6   php15271               php                        zend_error error
    7   php15271               php  ZEND_CATCH_SPEC_CONST_CV_HANDLER exception-caught
    8   php15271               php     zend_throw_exception_internal exception-thrown
    9   php15271               php                 dtrace_execute_ex execute-entry
   10   php15271               php           dtrace_execute_internal execute-entry
   11   php15271               php                 dtrace_execute_ex execute-return
   12   php15271               php           dtrace_execute_internal execute-return
   13   php15271               php                 dtrace_execute_ex function-entry
   14   php15271               php                 dtrace_execute_ex function-return
   15   php15271               php              php_request_shutdown request-shutdown
   16   php15271               php               php_request_startup request-startup

Provider 列的值由 php 和当前正在运行的 PHP 进程的进程 ID 组成。

如果 Apache Web 服务器正在运行,则模块名称可能为 libphp5.so,并且将有多个列表块,每个正在运行的 Apache 进程一个。

Function 列指的是 PHP 的内部 C 实现函数名称,每个提供程序都位于其中。

如果未运行 PHP 进程,则不会显示任何 PHP 探针。

PHP 示例中的 DTrace

此示例展示了 DTrace D 脚本语言的基础知识。

示例 #1 all_probes.d 用于使用 DTrace 跟踪所有 PHP 静态探针

#!/usr/sbin/dtrace -Zs

#pragma D option quiet

php*:::compile-file-entry
{
    printf("PHP compile-file-entry\n");
    printf("  compile_file              %s\n", copyinstr(arg0));
    printf("  compile_file_translated   %s\n", copyinstr(arg1));
}

php*:::compile-file-return
{
    printf("PHP compile-file-return\n");
    printf("  compile_file              %s\n", copyinstr(arg0));
    printf("  compile_file_translated   %s\n", copyinstr(arg1));
}

php*:::error
{
    printf("PHP error\n");
    printf("  errormsg                  %s\n", copyinstr(arg0));
    printf("  request_file              %s\n", copyinstr(arg1));
    printf("  lineno                    %d\n", (int)arg2);
}

php*:::exception-caught
{
    printf("PHP exception-caught\n");
    printf("  classname                 %s\n", copyinstr(arg0));
}

php*:::exception-thrown
{
    printf("PHP exception-thrown\n");
    printf("  classname                 %s\n", copyinstr(arg0));
}

php*:::execute-entry
{
    printf("PHP execute-entry\n");
    printf("  request_file              %s\n", copyinstr(arg0));
    printf("  lineno                    %d\n", (int)arg1);
}

php*:::execute-return
{
    printf("PHP execute-return\n");
    printf("  request_file              %s\n", copyinstr(arg0));
    printf("  lineno                    %d\n", (int)arg1);
}

php*:::function-entry
{
    printf("PHP function-entry\n");
    printf("  function_name             %s\n", copyinstr(arg0));
    printf("  request_file              %s\n", copyinstr(arg1));
    printf("  lineno                    %d\n", (int)arg2);
    printf("  classname                 %s\n", copyinstr(arg3));
    printf("  scope                     %s\n", copyinstr(arg4));
}

php*:::function-return
{
    printf("PHP function-return\n");
    printf("  function_name             %s\n", copyinstr(arg0));
    printf("  request_file              %s\n", copyinstr(arg1));
    printf("  lineno                    %d\n", (int)arg2);
    printf("  classname                 %s\n", copyinstr(arg3));
    printf("  scope                     %s\n", copyinstr(arg4));
}

php*:::request-shutdown
{
    printf("PHP request-shutdown\n");
    printf("  file                      %s\n", copyinstr(arg0));
    printf("  request_uri               %s\n", copyinstr(arg1));
    printf("  request_method            %s\n", copyinstr(arg2));
}

php*:::request-startup
{
    printf("PHP request-startup\n");
    printf("  file                      %s\n", copyinstr(arg0));
    printf("  request_uri               %s\n", copyinstr(arg1));
    printf("  request_method            %s\n", copyinstr(arg2));
}

此脚本使用 -Z 选项传递给 dtrace,允许它在没有 PHP 进程执行时运行。如果省略此选项,则脚本将立即终止,因为它知道没有要监视的探针存在。

该脚本在运行的 PHP 脚本的整个持续时间内跟踪所有核心 PHP 静态探针点。运行 D 脚本:

# ./all_probes.d

运行 PHP 脚本或应用程序。监视 D 脚本将在探针触发时输出每个探针的参数。

监视完成后,可以使用 CTRL+C 终止 D 脚本。

在多 CPU 机器上,探针顺序可能看起来不连续。这取决于哪个 CPU 正在处理探针,以及线程如何在 CPU 之间迁移。显示探针时间戳将有助于减少混淆,例如:

php*:::function-entry
{
      printf("%lld: PHP function-entry ", walltimestamp);
      [ . . .]
}

添加注释

用户贡献的注释

此页面没有用户贡献的注释。
To Top