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

    php內核分析之do-cli

    時間:2025-01-09 03:58:21 php語言 我要投稿
    • 相關推薦

    php內核分析之do-cli

      下面是百分網小編精心為大家整理的php內核分析之do_cli,不了解do_cli的同學不妨參考學習一下,更多內容請關注應屆畢業生網!

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

      # main

      把剩下的代碼增加了下注釋全部貼出來了(這個是簡化后的main函數,去掉了一些無關緊要的代碼段):

      int main(int argc, char *argv[])

      {

      ...

      sapi_module_struct *sapi_module = &cli_sapi_module;

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

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

      ...

      #ifdef ZTS

      tsrm_startup(1, 1, 0, NULL);

      (void)ts_resource(0);

      ZEND_TSRMLS_CACHE_UPDATE();

      #endif

      zend_signal_startup(); // 設置信號,把一些需要反應的信號位設置為0

      // 獲取參數,做一些對應的初始化行為,或者一些簡單的操作,比如help

      while ((c = php_getopt(argc, argv, OPTIONS, &php_optarg, &php_optind, 0, 2))!=-1) {

      switch (c) { // 這里的c是代表返回的字符串的ascii碼值

      case 'c':

      ...

      case 'n':

      ini_ignore = 1; // 不使用ini文件,通過代碼或者其他指定ini值

      break;

      case 'd': { // 配置ini的key,val值在命令行中,下面的行為都是修改ini_entries這個變量

      ...

      }

      case 'h': /* help & quit */

      case '?':

      php_cli_usage(argv[0]);

      goto out;

      case 'i': case 'v': case 'm':

      sapi_module = &cli_sapi_module;

      goto exit_loop;

      case 'e': /* enable extended info output */

      use_extended_info = 1;

      break;

      }

      }

      exit_loop:

      sapi_module->ini_defaults = sapi_cli_ini_defaults; // 設置初始化的ini值

      sapi_module->php_ini_path_override = ini_path_override; //設置重寫后的ini_path地址,如果是php -c的話,這個就為非null

      sapi_module->phpinfo_as_text = 1; // 打開打印phpinfo的開關,需要的時候可以把phpinfo打印出來

      sapi_module->php_ini_ignore_cwd = 1; // 不在當前路徑尋找php.ini

      sapi_startup(sapi_module); // sapi初始化行為,比如初始化全局變量SG

      sapi_started = 1; // 標記,表示已經調用了startup,關閉的時候需要調用shundown

      ...

      // 開始調用sapi的startup方法,對cli模式,實際上是調用php_cli_startup方法

      if (sapi_module->startup(sapi_module) == FAILURE) {

      exit_status = 1;

      goto out;

      }

      module_started = 1; // 標記位,標記已經調用了module的startup方法

      ...

      zend_first_try {

      exit_status = do_cli(argc, argv); // 這個是實際上調用的內容

      } zend_end_try();

      out: // 這個代碼段已經是要退出了

      if (ini_path_override) {

      free(ini_path_override);

      }

      if (ini_entries) {

      free(ini_entries);

      }

      if (module_started) {

      php_module_shutdown();

      }

      if (sapi_started) {

      sapi_shutdown();

      }

      #ifdef ZTS

      tsrm_shutdown();

      #endif

      cleanup_ps_args(argv);

      exit(exit_status);

      }

      其實看偽碼很簡單:

      tsrm_startup(1, 1, 0, NULL); // TSM啟動

      zend_signal_startup(); // 信號設置

      sapi_startup(sapi_module); // SAPI啟動

      sapi_module->startup(sapi_module); // 當前模塊的startup

      do_cli(argc, argv); // 做實際的行為

      php_module_shutdown(); // 當前模塊的shutdown

      sapi_shutdown(); // SAPI關閉

      tsrm_shutdown(); // TSM關閉

      好了,其實看了一圈,里面最重的函數是do_cli了。

      php參數

      do_cli里面你會看到根據參數的不同,有很多分支,這里你就需要了解這些參數都是什么用的。

      參數

      作用

      實例

      do_cli

      我們把do_cli函數的整個函數去掉多余代碼,僅保留關鍵代碼如下:

      static int do_cli(int argc, char **argv)

      {

      ...

      zend_try {

      // 這里處理了 i-輸出phpinfo內容/ v-輸出php版本 / m-輸出擴展信息

      while ((c = php_getopt(argc, argv, OPTIONS, &php_optarg, &php_optind, 0, 2)) != -1) {

      switch (c) {

      case 'i': // 輸出phpinfo內容

      ...

      php_print_info(0xFFFFFFFF);

      ...

      goto out;

      case 'v': // 輸出php版本信息

      ...

      get_zend_version()

      ...

      goto out;

      case 'm': // 列出所有模塊

      ...

      print_extensions();

      ...

      goto out;

      default:

      break;

      }

      }

      ...

      // 下面的代碼做了幾個事情:

      // 1 根據參數設置了behavior參數

      // 2 有執行文件的就將文件存在script_file

      while ((c = php_getopt(argc, argv, OPTIONS, &php_optarg, &php_optind, 0, 2)) != -1) {

      switch (c) {

      case 'a': // php的交互模式

      ...

      interactive=1;

      ...

      break;

      case 'C': // 不要把cwd目錄變成腳本所在的目錄。這個默認就是cwd是當前執行路徑,所以這里什么都不做。

      break;

      case 'F': // php -F 進入交互模式,每執行一行就執行一次文件

      ...

      behavior=PHP_MODE_PROCESS_STDIN;

      script_file = php_optarg;

      break;

      case 'f': // php -f 解析并執行文件

      ...

      script_file = php_optarg;

      break;

      case 'l': // 檢查文件的語法是否有錯誤

      ...

      behavior=PHP_MODE_LINT;

      break;

      case 'q': // 安靜模式,默認也是安靜模式

      break;

      case 'r': // 從命令行直接執行腳本

      ...

      behavior=PHP_MODE_CLI_DIRECT;

      exec_direct=php_optarg;

      break;

      case 'R': // 每行輸入的時候執行一次code腳本,比如 php -R 'echo 12;'

      ...

      behavior=PHP_MODE_PROCESS_STDIN;

      exec_run=php_optarg;

      break;

      case 'B': // 在每次輸入開始之前執行一次code腳本

      ...

      behavior=PHP_MODE_PROCESS_STDIN;

      exec_begin=php_optarg;

      break;

      case 'E': // 在每次輸入結束之后執行一次code腳本, 上面的 RBE可以參考一個例子:find conf.d | php -B '$l=0;' -R '$l += count(@file($argn));' -E 'echo "Total Lines: $l\n";'

      ...

      behavior=PHP_MODE_PROCESS_STDIN;

      exec_end=php_optarg;

      break;

      case 's': // 使用html高亮方式顯示代碼,這個或許在一些代碼顯示的時候需要用到

      ...

      behavior=PHP_MODE_HIGHLIGHT;

      break;

      case 'w': // php -w 能把中的評論和多余的空格去掉

      ...

      behavior=PHP_MODE_STRIP;

      break;

      case 'z': // 加載外部擴展

      zend_load_extension(php_optarg);

      break;

      case 'H': // 隱藏所有參數

      hide_argv = 1;

      break;

      case 10: // 顯示function定義

      behavior=PHP_MODE_REFLECTION_FUNCTION;

      reflection_what = php_optarg;

      break;

      case 11: // 顯示class定義

      behavior=PHP_MODE_REFLECTION_CLASS;

      reflection_what = php_optarg;

      break;

      case 12: // 顯示擴展定義,注意這里是php擴展

      behavior=PHP_MODE_REFLECTION_EXTENSION;

      reflection_what = php_optarg;

      break;

      case 13: // 顯示zend擴展定義, 比如xdebug

      behavior=PHP_MODE_REFLECTION_ZEND_EXTENSION;

      reflection_what = php_optarg;

      break;

      case 14: // 顯示擴展的對應配置

      behavior=PHP_MODE_REFLECTION_EXT_INFO;

      reflection_what = php_optarg;

      break;

      case 15: // 顯示ini配置

      behavior = PHP_MODE_SHOW_INI_CONFIG;

      break;

      default:

      break;

      }

      }

      ...

      // 初始化request之后,執行了request_startup

      if (php_request_startup()==FAILURE) {

      ...

      goto err;

      }

      ...

      zend_is_auto_global_str(ZEND_STRL("_SERVER"));

      // 根據不同的行為做不同的具體操作,這個是核心方法

      switch (behavior) {

      case PHP_MODE_STANDARD: // 標準,就是執行一個腳本文件

      ...

      php_execute_script(&file_handle);

      ...

      break;

      case PHP_MODE_LINT: // 只檢查文件有沒有語法錯誤

      exit_status = php_lint_script(&file_handle);

      ...

      break;

      case PHP_MODE_STRIP:

      ...

      zend_strip();

      ...

      break;

      case PHP_MODE_HIGHLIGHT:

      ...

      php_get_highlight_struct(&syntax_highlighter_ini);

      zend_highlight(&syntax_highlighter_ini);

      goto out;

      break;

      case PHP_MODE_CLI_DIRECT:

      ...

      if (zend_eval_string_ex(exec_direct, NULL, "Command line code", 1) == FAILURE) {

      exit_status=254;

      }

      break;

      case PHP_MODE_PROCESS_STDIN:

      ...

      zend_eval_string_ex(exec_end, NULL, "Command line end code", 1)

      ...

      break;

      case PHP_MODE_REFLECTION_FUNCTION:

      case PHP_MODE_REFLECTION_CLASS:

      case PHP_MODE_REFLECTION_EXTENSION:

      case PHP_MODE_REFLECTION_ZEND_EXTENSION:

      ...

      ZVAL_STRING(&arg, reflection_what);

      object_init_ex(&ref, pce);

      ...

      zend_call_method_with_1_params(&ref, pce, &pce->constructor, "__construct", NULL, &arg);

      ...

      break;

      case PHP_MODE_REFLECTION_EXT_INFO:

      ...

      if ((module = zend_hash_str_find_ptr(&module_registry, lcname, len)) == NULL) {

      ...

      display_ini_entries(NULL);

      ...

      }

      ...

      break;

      case PHP_MODE_SHOW_INI_CONFIG:

      ...

      break;

      }

      } zend_end_try();

      out:

      ...

      err:

      ...

      }

      整個200行的代碼就很好理解了,整個是包在一個zend_try...zend_catch中的。做了幾步:

      處理-i, -m, -v參數

      對其他的參數設置behavior,script_file等變量

      根據behavior做不同的行為

      回到我們的初步計劃,我們想要了解的事:

      我們的根據-r的參數配置尋找。

      它實際上時調用了

      zend_eval_string_ex(exec_direct, NULL, "Command line code", 1)

      這里的exec_direct是 echo 12字符串

    【php內核分析之do-cli】相關文章:

    php內核分析之zval05-07

    php內核分析之擴展01-31

    php內核分析之opcode08-04

    php內核分析之zend-compile05-17

    php內核分析之全局變量12-14

    php內核分析之sapi-module-struct05-23

    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. 日日做夜狠狠爱欧美黑人