PHPカーネル解析(6)-オペコード
摘要:这里阅读的php版本为PHP-7.1.0 RC3,阅读代码的平台为linux查看opcodephp是先把源码解析成opcode,然后再把opcode传递给zend_vm进行执行的。// 一个opcode的结构 struct _zend_op { const void *handler; // opcode ...
这里阅读的php版本为PHP-7.1.0 RC3,阅读代码的平台为linux
查看opcode
php是先把源码解析成opcode,然后再把opcode传递给zend_vm进行执行的。
01 // 一个opcode的结构 02 struct _zend_op { 03 const void *handler; // opcode对应的执行函数,每个opcode都有一个对应的执行函数 04 znode_op op1; // 执行参数的第一个元素 05 znode_op op2; // 执行参数的第二个元素 06 znode_op result; // 执行结果 07 uint32_t extended_value; // 额外扩展的字段和值 08 uint32_t lineno; // 行数 09 zend_uchar opcode; // 操作码,具体操作码列表见 http://cn.php.net/manual/zh/internals2.opcodes.php 10 zend_uchar op1_type; // 第一个元素的类型 11 zend_uchar op2_type; // 第二个元素的类型 12 zend_uchar result_type; // 结果的类型 13 };
在php7中,我们能很方便用phpdbg来查看一个文件或者一个函数的opcode了。至于phpdbg的使用,现在网上介绍不多,不过好在有很详细的help文档。下面是一个最简单的opcode代码:
01 $ bin/phpdbg -f /home/xiaoju/software/php7/demo/echo.php 02 prompt> list 100 03 00001: <!--?php 04 00002: 05 00003: $a = 1; 06 00004: $b = $a; 07 00005: $b = $b + 1; 08 00006: echo $b; 09 00007: 10 prompt--> print exec 11 [Context /home/xiaoju/software/php7/demo/echo.php (6 ops)] 12 L1-7 {main}() /home/xiaoju/software/php7/demo/echo.php - 0x7fe3fae63300 + 6 ops 13 L3 #0 ASSIGN $a 1 14 L4 #1 ASSIGN $b $a 15 L5 #2 ADD $b 1 ~2 16 L5 #3 ASSIGN $b ~2 17 L6 #4 ECHO $b 18 L7 #5 RETURN 1
这个php文件就做了一个最简单的加法操作。生成了6个_zend_op。所展示的每一行代表一个_zend_op
1 _zendop.lineno op号 _zend_op.opcode _zend_op.op1 _zend_op.op2 _zend_op.result 2 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这行设置个断点:
01 1452 for (i = 0; i < file_count; i++) { 02 1453 file_handle = va_arg(files, zend_file_handle *); 03 1454 if (!file_handle) { 04 1455 continue; 05 1456 } 06 1457 07 1458 op_array = zend_compile_file(file_handle, type); 08 1459 if (file_handle->opened_path) { 09 1460 zend_hash_add_empty_element(&EG(included_files), file_handle->opened_path); 10 1461 } 11 12 (gdb) b 1459
继续跑
1 (gdb) continue 2 (gdb) s 3 (gdb) s
打印出这个时候的op_array
1 (gdb) p *op_array 2 $4 = {type = 2 '\002', arg_flags = "\000\000", fn_flags = 134217728, function_name = 0x0, scope = 0x0, 3 prototype = 0x0, num_args = 0, required_num_args = 0, arg_info = 0x0, refcount = 0x7ffff6002000, last = 6, 4 opcodes = 0x7ffff6076240, last_var = 2, T = 4, vars = 0x7ffff6079030, last_live_range = 0, last_try_catch = 0, 5 live_range = 0x0, try_catch_array = 0x0, static_variables = 0x0, filename = 0x7ffff605c2d0, line_start = 1, 6 line_end = 7, doc_comment = 0x0, early_binding = 4294967295, last_literal = 3, literals = 0x7ffff60030c0, 7 cache_size = 0, run_time_cache = 0x0, reserved = {0x0, 0x0, 0x0, 0x0}}
我可以优化输出:
01 (gdb) set print pretty on 02 (gdb) p *op_array 03 $5 = { 04 type = 2 '\002', 05 arg_flags = "\000\000", 06 fn_flags = 134217728, 07 function_name = 0x0, 08 scope = 0x0, 09 prototype = 0x0, 10 num_args = 0, 11 required_num_args = 0, 12 arg_info = 0x0, 13 refcount = 0x7ffff6002000, 14 last = 6, 15 opcodes = 0x7ffff6076240, 16 last_var = 2, 17 T = 4, 18 vars = 0x7ffff6079030, 19 last_live_range = 0, 20 last_try_catch = 0, 21 live_range = 0x0, 22 try_catch_array = 0x0, 23 static_variables = 0x0, 24 filename = 0x7ffff605c2d0, 25 line_start = 1, 26 line_end = 7, 27 doc_comment = 0x0, 28 early_binding = 4294967295, 29 last_literal = 3, 30 literals = 0x7ffff60030c0, 31 cache_size = 0, 32 run_time_cache = 0x0, 33 reserved = {0x0, 0x0, 0x0, 0x0} 34 }
我想打出op_array.filename.val的具体值
1 (gdb) p (op_array.filename.len) 2 $12 = 40 3 (gdb) p *(op_array.filename.val)@40 4 $13 = "/home/xiaoju/software/php7/demo/echo.php"
好了,我们可以顺便研究下_zend_op_array这个结构:
01 // opcode组成的数组,编译的时候就是生成这个结构 02 struct _zend_op_array { 03 zend_uchar type; // op array的类型,比如 ZEND_EVAL_CODE 04 zend_uchar arg_flags[3]; /* bitset of arg_info.pass_by_reference */ 05 uint32_t fn_flags; 06 zend_string *function_name; 07 zend_class_entry *scope; 08 zend_function *prototype; 09 uint32_t num_args; // 脚本的参数 10 uint32_t required_num_args; 11 zend_arg_info *arg_info; 12 /* END of common elements */ 13 14 uint32_t *refcount; // 这个结构的引用次数 15 16 uint32_t last; // opcode的个数 17 zend_op *opcodes; // 存储所有的opcode 18 19 int last_var; // php变量的个数 20 uint32_t T; 21 zend_string **vars; // 被编译的php变量的个数 22 23 int last_live_range; 24 int last_try_catch; // try_catch的个数 25 zend_live_range *live_range; 26 zend_try_catch_element *try_catch_array; // 27 28 /* static variables support */ 29 HashTable *static_variables; // 静态变量 30 31 zend_string *filename; // 执行的脚本的文件 32 uint32_t line_start; // 开始于第几行 33 uint32_t line_end; // 结束于第几行 34 zend_string *doc_comment; // 文档的注释 35 uint32_t early_binding; /* the linked list of delayed declarations */ 36 37 int last_literal; 38 zval *literals; 39 40 int cache_size; 41 void **run_time_cache; 42 43 void *reserved[ZEND_MAX_RESERVED_RESOURCES]; // 保留字段 44 };
以上就是php内核分析(六)-opcode的内容,更多相关内容请关注PHP中文网(www.php.cn)!

ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

Video Face Swap
完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

SublimeText3 中国語版
中国語版、とても使いやすい

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

ホットトピック











JWTは、JSONに基づくオープン標準であり、主にアイデンティティ認証と情報交換のために、当事者間で情報を安全に送信するために使用されます。 1。JWTは、ヘッダー、ペイロード、署名の3つの部分で構成されています。 2。JWTの実用的な原則には、JWTの生成、JWTの検証、ペイロードの解析という3つのステップが含まれます。 3. PHPでの認証にJWTを使用する場合、JWTを生成および検証でき、ユーザーの役割と許可情報を高度な使用に含めることができます。 4.一般的なエラーには、署名検証障害、トークンの有効期限、およびペイロードが大きくなります。デバッグスキルには、デバッグツールの使用とロギングが含まれます。 5.パフォーマンスの最適化とベストプラクティスには、適切な署名アルゴリズムの使用、有効期間を合理的に設定することが含まれます。

静的結合(静的::) PHPで後期静的結合(LSB)を実装し、クラスを定義するのではなく、静的コンテキストで呼び出しクラスを参照できるようにします。 1)解析プロセスは実行時に実行されます。2)継承関係のコールクラスを検索します。3)パフォーマンスオーバーヘッドをもたらす可能性があります。

文字列は、文字、数字、シンボルを含む一連の文字です。このチュートリアルでは、さまざまな方法を使用してPHPの特定の文字列内の母音の数を計算する方法を学びます。英語の母音は、a、e、i、o、u、そしてそれらは大文字または小文字である可能性があります。 母音とは何ですか? 母音は、特定の発音を表すアルファベットのある文字です。大文字と小文字など、英語には5つの母音があります。 a、e、i、o、u 例1 入力:string = "tutorialspoint" 出力:6 説明する 文字列「TutorialSpoint」の母音は、u、o、i、a、o、iです。合計で6元があります

PHPの魔法の方法は何ですか? PHPの魔法の方法には次のものが含まれます。1。\ _ \ _コンストラクト、オブジェクトの初期化に使用されます。 2。\ _ \ _リソースのクリーンアップに使用される破壊。 3。\ _ \ _呼び出し、存在しないメソッド呼び出しを処理します。 4。\ _ \ _ get、dynamic属性アクセスを実装します。 5。\ _ \ _セット、動的属性設定を実装します。これらの方法は、特定の状況で自動的に呼び出され、コードの柔軟性と効率を向上させます。

PHPとPythonにはそれぞれ独自の利点があり、プロジェクトの要件に従って選択します。 1.PHPは、特にWebサイトの迅速な開発とメンテナンスに適しています。 2。Pythonは、データサイエンス、機械学習、人工知能に適しており、簡潔な構文を備えており、初心者に適しています。

PHPは、電子商取引、コンテンツ管理システム、API開発で広く使用されています。 1)eコマース:ショッピングカート機能と支払い処理に使用。 2)コンテンツ管理システム:動的コンテンツの生成とユーザー管理に使用されます。 3)API開発:RESTFUL API開発とAPIセキュリティに使用されます。パフォーマンスの最適化とベストプラクティスを通じて、PHPアプリケーションの効率と保守性が向上します。

PHPは、サーバー側で広く使用されているスクリプト言語で、特にWeb開発に適しています。 1.PHPは、HTMLを埋め込み、HTTP要求と応答を処理し、さまざまなデータベースをサポートできます。 2.PHPは、ダイナミックWebコンテンツ、プロセスフォームデータ、アクセスデータベースなどを生成するために使用され、強力なコミュニティサポートとオープンソースリソースを備えています。 3。PHPは解釈された言語であり、実行プロセスには語彙分析、文法分析、編集、実行が含まれます。 4.PHPは、ユーザー登録システムなどの高度なアプリケーションについてMySQLと組み合わせることができます。 5。PHPをデバッグするときは、error_reporting()やvar_dump()などの関数を使用できます。 6. PHPコードを最適化して、キャッシュメカニズムを使用し、データベースクエリを最適化し、組み込み関数を使用します。 7

PHP8では、一致式は、式の値に基づいて異なる結果を返す新しい制御構造です。 1)Switchステートメントに似ていますが、実行ステートメントブロックの代わりに値を返します。 2)一致式の式は厳密に比較され、セキュリティが向上します。 3)スイッチステートメントの脱落の可能性を回避し、コードのシンプルさと読みやすさを向上させます。
