php核心分析(六)-opcode

技術mix呢發表於2018-01-07

這裡閱讀的php版本為PHP-7.1.0 RC3,閱讀程式碼的平臺為linux

檢視opcode

php是先把原始碼解析成opcode,然後再把opcode傳遞給zend_vm進行執行的。

// 一個opcode的結構
struct _zend_op {
     const void *handler; // opcode對應的執行函式,每個opcode都有一個對應的執行函式
     znode_op op1;  // 執行引數的第一個元素
     znode_op op2;  //  執行引數的第二個元素
     znode_op result; // 執行結果
     uint32_t extended_value; // 額外擴充套件的欄位和值
     uint32_t lineno; // 行數
     zend_uchar opcode;   // 操作碼,具體操作碼列表見 http://cn.php.net/manual/zh/internals2.opcodes.php
     zend_uchar op1_type; // 第一個元素的型別
     zend_uchar op2_type; // 第二個元素的型別
     zend_uchar result_type; // 結果的型別
};

在php7中,我們能很方便用phpdbg來檢視一個檔案或者一個函式的opcode了。至於phpdbg的使用,現在網上介紹不多,不過好在有很詳細的help文件。下面是一個最簡單的opcode程式碼:

$ bin/phpdbg -f /home/xiaoju/software/php7/demo/echo.php
prompt> list 100
00001: <?php
00002:
00003: $a = 1;
00004: $b = $a;
00005: $b = $b + 1;
00006: echo $b;
00007:
prompt> print exec
[Context /home/xiaoju/software/php7/demo/echo.php (6 ops)]
L1-7 {main}() /home/xiaoju/software/php7/demo/echo.php - 0x7fe3fae63300 + 6 ops
L3    #0     ASSIGN                  $a                   1
L4    #1     ASSIGN                  $b                   $a
L5    #2     ADD                     $b                   1                    ~2
L5    #3     ASSIGN                  $b                   ~2
L6    #4     ECHO                    $b
L7    #5     RETURN                  1

這個php檔案就做了一個最簡單的加法操作。生成了6個_zend_op。所展示的每一行代表一個_zend_op

_zendop.lineno  op號   _zend_op.opcode       _zend_op.op1          _zend_op.op2          _zend_op.result
L5              #2     ADD                     $b                   1                    ~2

這裡_zend_op.opcode對應的操作在官網有文件和詳細的例子可以檢視:http://cn.php.net/manual/zh/internals2.opcodes.php

值得一說的是,phpdbg還有一個遠端UI版本,能讓我們在近端診斷服務端的php資訊

gdb

但是我們的目標還是在於研究php原始碼,phpdbg只能分析到opcode這層,還是不夠的,gdb可能是更好的選擇。

gdb的使用和平時使用差不多

比如我現在有個指令碼echo.php:

  1 <?php
  2
  3 $a = 1;
  4 $b = $a;
  5 $b = $b + 1;
  6 echo $b;

我的php安裝路徑在:

/home/xiaoju/software/php7/bin/php

php原始碼路徑在:

/home/xiaoju/webroot/php-src/php-src-master/

執行gdb

$ gdb /home/xiaoju/software/php7/bin/php

載入gdbinit:

(gdb) source /home/xiaoju/webroot/php-src/php-src-master/.gdbinit

設定斷點:

(gdb) b zend_execute_scripts

執行:

(gdb) run -f /home/xiaoju/software/php7/demo/echo.php

我想在1459這行設定個斷點:

1452          for (i = 0; i < file_count; i++) {
1453               file_handle = va_arg(files, zend_file_handle *);
1454               if (!file_handle) {
1455                    continue;
1456               }
1457
1458               op_array = zend_compile_file(file_handle, type);
1459               if (file_handle->opened_path) {
1460                    zend_hash_add_empty_element(&EG(included_files), file_handle->opened_path);
1461               }

(gdb) b 1459

繼續跑

(gdb) continue
(gdb) s
(gdb) s

列印出這個時候的op_array

(gdb) p *op_array
$4 = {type = 2 ` 02`, arg_flags = " 00 00", fn_flags = 134217728, function_name = 0x0, scope = 0x0,
  prototype = 0x0, num_args = 0, required_num_args = 0, arg_info = 0x0, refcount = 0x7ffff6002000, last = 6,
  opcodes = 0x7ffff6076240, last_var = 2, T = 4, vars = 0x7ffff6079030, last_live_range = 0, last_try_catch = 0,
  live_range = 0x0, try_catch_array = 0x0, static_variables = 0x0, filename = 0x7ffff605c2d0, line_start = 1,
  line_end = 7, doc_comment = 0x0, early_binding = 4294967295, last_literal = 3, literals = 0x7ffff60030c0,
  cache_size = 0, run_time_cache = 0x0, reserved = {0x0, 0x0, 0x0, 0x0}}

我可以優化輸出:

(gdb) set print pretty on
(gdb) p *op_array
$5 = {
  type = 2 ` 02`,
  arg_flags = " 00 00",
  fn_flags = 134217728,
  function_name = 0x0,
  scope = 0x0,
  prototype = 0x0,
  num_args = 0,
  required_num_args = 0,
  arg_info = 0x0,
  refcount = 0x7ffff6002000,
  last = 6,
  opcodes = 0x7ffff6076240,
  last_var = 2,
  T = 4,
  vars = 0x7ffff6079030,
  last_live_range = 0,
  last_try_catch = 0,
  live_range = 0x0,
  try_catch_array = 0x0,
  static_variables = 0x0,
  filename = 0x7ffff605c2d0,
  line_start = 1,
  line_end = 7,
  doc_comment = 0x0,
  early_binding = 4294967295,
  last_literal = 3,
  literals = 0x7ffff60030c0,
  cache_size = 0,
  run_time_cache = 0x0,
  reserved = {0x0, 0x0, 0x0, 0x0}
}

我想打出op_array.filename.val的具體值

(gdb) p (op_array.filename.len)
$12 = 40
(gdb) p *(op_array.filename.val)@40
$13 = "/home/xiaoju/software/php7/demo/echo.php"

好了,我們可以順便研究下_zend_op_array這個結構:

// opcode組成的陣列,編譯的時候就是生成這個結構
struct _zend_op_array {
     zend_uchar type;  // op array的型別,比如 ZEND_EVAL_CODE
     zend_uchar arg_flags[3]; /* bitset of arg_info.pass_by_reference */
     uint32_t fn_flags;
     zend_string *function_name;
     zend_class_entry *scope;
     zend_function *prototype;
     uint32_t num_args;  // 指令碼的引數
     uint32_t required_num_args;
     zend_arg_info *arg_info;
     /* END of common elements */

     uint32_t *refcount; // 這個結構的引用次數

     uint32_t last;  // opcode的個數
     zend_op *opcodes;  // 儲存所有的opcode

     int last_var; // php變數的個數
     uint32_t T;
     zend_string **vars; // 被編譯的php變數的個數

     int last_live_range;
     int last_try_catch;  // try_catch的個數
     zend_live_range *live_range;
     zend_try_catch_element *try_catch_array; //

     /* static variables support */
     HashTable *static_variables; // 靜態變數

     zend_string *filename;  // 執行的指令碼的檔案
     uint32_t line_start; // 開始於第幾行
     uint32_t line_end; // 結束於第幾行
     zend_string *doc_comment; // 文件的註釋
     uint32_t early_binding; /* the linked list of delayed declarations */

     int last_literal;
     zval *literals;

     int  cache_size;
     void **run_time_cache;

     void *reserved[ZEND_MAX_RESERVED_RESOURCES]; // 保留欄位
};


本文轉自軒脈刃部落格園部落格,原文連結:http://www.cnblogs.com/yjf512/p/6112634.html,如需轉載請自行聯絡原作者


相關文章