<address id="ousso"></address>
<form id="ousso"><track id="ousso"><big id="ousso"></big></track></form>
  1. php語言

    php內核分析之sapi-module-struct

    時間:2025-05-23 17:16:09 php語言 我要投稿
    • 相關推薦

    php內核分析之sapi-module-struct

      為了幫助同學們學習PHP內核技術,小編整理了php內核分析之sapi_module_struct,希望對大家有所幫助!更多內容請關注應屆畢業生網!

      這里閱讀的php版本為PHP-7.1.0 RC3,閱讀代碼的平臺為linux

      首先是尋找php的入口,php有很多種模式,apache,php-fpm, cli模式,我要入手的話,只能先從最簡單的cli模型開始。

      那么,我需要先尋找

      php -r 'echo 12;'

      這個命令是如何執行的。

      首先還是尋找main入口,由于我們看的是命令行的php程序。所以,這個入口在sapi/cli/php_cli.c中。

      首先是定義一系列的變量

      int c;

      zend_file_handle file_handle;

      int behavior = PHP_MODE_STANDARD;

      char *reflection_what = NULL;

      volatile int request_started = 0;

      volatile int exit_status = 0;

      char *php_optarg = NULL, *orig_optarg = NULL;

      int php_optind = 1, orig_optind = 1;

      char *exec_direct=NULL, *exec_run=NULL, *exec_begin=NULL, *exec_end=NULL;

      char *arg_free=NULL, **arg_excp=&arg_free;

      char *script_file=NULL, *translated_path = NULL;

      int interactive=0;

      int lineno = 0;

      const char *param_error=NULL;

      int hide_argv = 0;

      然后是這個

      sapi_module_struct *sapi_module = &cli_sapi_module;

      這是一個sapi_module_struct結構,這個結構是sapi中最重要的數據結構。它的定義在main/SAPI.h中。

      下面是增加了注釋的代碼:

      struct _sapi_module_struct { // SAPI模塊結構

      char *name; // 應用層名稱,比如cli,cgi等

      char *pretty_name; // 應用層更易讀的名字,比如cli對應的就是Command Line Interface

      int (*startup)(struct _sapi_module_struct *sapi_module); // 當一個應用要調用php的時候,這個模塊啟動的時候會調用的函數

      int (*shutdown)(struct _sapi_module_struct *sapi_module); // 當一個應用要調用php的時候,這個模塊結束的時候會調用的函數

      int (*activate)(void); // 在處理每個request的時候,激活需要調用的函數

      int (*deactivate)(void); // 在處理完每個request的時候,收尾時候要調用的函數

      size_t (*ub_write)(const char *str, size_t str_length); // 這個函數告訴php如何輸出數據

      void (*flush)(void *server_context); // 提供給php的刷新緩存的函數指針

      zend_stat_t *(*get_stat)(void); // 用來判斷要執行文件的權限,來判斷是否有執行權限

      char *(*getenv)(char *name, size_t name_len); // 獲取環境變量的方法

      void (*sapi_error)(int type, const char *error_msg, ...) ZEND_ATTRIBUTE_FORMAT(printf, 2, 3); // 錯誤處理方法

      int (*header_handler)(sapi_header_struct *sapi_header, sapi_header_op_enum op, sapi_headers_struct *sapi_headers); // 這個函數會在我們調用header()的時候被調用

      int (*send_headers)(sapi_headers_struct *sapi_headers); // 發送所有的header

      void (*send_header)(sapi_header_struct *sapi_header, void *server_context); // 單獨發送某一個header

      size_t (*read_post)(char *buffer, size_t count_bytes); // 如何獲取HTTP POST中的數據

      char *(*read_cookies)(void); // 如何獲取cookie中的數據

      void (*register_server_variables)(zval *track_vars_array); // 這個函數可以給$_SERVER中獲取變量

      void (*log_message)(char *message, int syslog_type_int); // 輸出錯誤信息函數

      double (*get_request_time)(void); // 獲取請求時間的函數

      void (*terminate_process)(void); // TODO: 調用exit的時候調用的方法

      char *php_ini_path_override; // PHP的ini文件被復寫了所復寫的地址

      void (*default_post_reader)(void); // 這里和前面的read_post有個差別,read_post負責如何獲取POST數據,而這里的函數負責如何解析POST數據

      void (*treat_data)(int arg, char *str, zval *destArray); // 對數據進行處理,比如進行安全過濾等。 default_post_reader/tread_data/input_filter是三個能對輸入進行過濾和處理的函數

      char *executable_location; // 執行的地理位置

      int php_ini_ignore; // 是否不使用任何ini配置文件,比如php -n 就將這個位置設置為1

      int php_ini_ignore_cwd; // 不在當前路徑尋找php.ini

      int (*get_fd)(int *fd); // 獲取執行文件的fd

      int (*force_http_10)(void); // 強制使用http1.0

      int (*get_target_uid)(uid_t *); // 獲取執行程序的uid

      int (*get_target_gid)(gid_t *); // 獲取執行程序的gid

      unsigned int (*input_filter)(int arg, char *var, char **val, size_t val_len, size_t *new_val_len); // 對輸入進行過濾。比如將輸入參數填充到自動全局變量$_GET, $_POST, $_COOKIE中

      void (*ini_defaults)(HashTable *configuration_hash); // 默認的ini配置

      int phpinfo_as_text; // 是否打印phpinfo信息

      char *ini_entries; // 有沒有附帶的ini配置,比如使用php -d date.timezone=America/Adak,可以在命令行中設置時區

      const zend_function_entry *additional_functions; // 每個SAPI模塊特有的一些函數注冊,比如cli的cli_get_process_title

      unsigned int (*input_filter_init)(void); // TODO:

      };

      那么我們看下cli的SAPI的module是什么樣子的呢?

      其中我把里面原先有的STANDARD_SAPI_MODULE_PROPERTIES宏給解出來展示如下:

      static sapi_module_struct cli_sapi_module = {

      "cli", /* name */

      "Command Line Interface", /* pretty name */

      php_cli_startup, /* startup */

      php_module_shutdown_wrapper, /* shutdown */

      NULL, /* activate */

      sapi_cli_deactivate, /* deactivate */

      sapi_cli_ub_write, /* unbuffered write */

      sapi_cli_flush, /* flush */

      NULL, /* get uid */

      NULL, /* getenv */

      php_error, /* error handler */

      sapi_cli_header_handler, /* header handler */

      sapi_cli_send_headers, /* send headers handler */

      sapi_cli_send_header, /* send header handler */

      NULL, /* read POST data */

      sapi_cli_read_cookies, /* read Cookies */

      sapi_cli_register_variables, /* register server variables */

      sapi_cli_log_message, /* Log message */

      NULL, /* Get request time */

      NULL, /* Child terminate */

      NULL, /* php_ini_path_override */ \

      NULL, /* default_post_reader */ \

      NULL, /* treat_data */ \

      NULL, /* executable_location */ \

      0, /* php_ini_ignore */ \

      0, /* php_ini_ignore_cwd */ \

      NULL, /* get_fd */ \

      NULL, /* force_http_10 */ \

      NULL, /* get_target_uid */ \

      NULL, /* get_target_gid */ \

      NULL, /* input_filter */ \

      NULL, /* ini_defaults */ \

      0, /* phpinfo_as_text; */ \

      NULL, /* ini_entries; */ \

      NULL, /* additional_functions */ \

      NULL /* input_filter_init */

      };

      有幾個點可以總結:

      cli模式是不需要發送header的,所以對應header處理的三個函數

      sapi_cli_header_handler

      sapi_cli_send_headers

      sapi_cli_send_header

      實際上都是空實現。

      cookie也是同樣道理

      sapi_cli_read_cookies

      其他的一些定義的函數,等到我們遇到的時候再分析吧。

      main

      回到main函數,根據上面的那個結構,我們就理解了

      argv = save_ps_args(argc, argv); //這里獲取一次當前執行進程的參數,環境變量等。為的是對特定平臺,修正下argv變量以供后續使用。

      cli_sapi_module.additional_functions = additional_functions; // cli模式特有的函數

      signal

      #ifdef HAVE_SIGNAL_H

      #if defined(SIGPIPE) && defined(SIG_IGN)

      // 忽略SIGPIPE是為了如果php是socket的客戶端,那么當服務端關閉的話,會返回一個PIPE的信號,為的是當前的程序不會因為這個而結束

      signal(SIGPIPE, SIG_IGN);

      #endif

      #endif

    【php內核分析之sapi-module-struct】相關文章:

    php內核分析之zval09-07

    php內核分析之擴展10-03

    php內核分析之opcode08-04

    php內核分析之全局變量09-03

    php內核分析之zend-compile09-01

    php內核分析之do-cli05-30

    php內核分析之ZTS和zend-try07-18

    php學習之php配置07-15

    php學習之php預定義變量07-29

    <address id="ousso"></address>
    <form id="ousso"><track id="ousso"><big id="ousso"></big></track></form>
    1. 日日做夜狠狠爱欧美黑人