这里阅读的php版本为PHP-7.1.0 RC3,阅读代码的平台为linux

查看opcode

php是先把源码解析成opcode,然后再把opcode传递给zend_vm进行执行的。

  1. // 一个opcode的结构
  2. struct _zend_op {
  3. const void *handler; // opcode对应的执行函数,每个opcode都有一个对应的执行函数
  4. znode_op op1; // 执行参数的第一个元素
  5. znode_op op2; // 执行参数的第二个元素
  6. znode_op result; // 执行结果
  7. uint32_t extended_value; // 额外扩展的字段和值
  8. uint32_t lineno; // 行数
  9. 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代码:

  1. $ bin/phpdbg -f /home/xiaoju/software/php7/demo/echo.php
  2. prompt> list 100
  3. 00001: <?php
  4. 00002:
  5. 00003: $a = 1;
  6. 00004: $b = $a;
  7. 00005: $b = $b + 1;
  8. 00006: echo $b;
  9. 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. 1 <?php
  2. 2
  3. 3 $a = 1;
  4. 4 $b = $a;
  5. 5 $b = $b + 1;
  6. 6 echo $b;

我的php安装路径在:

  1. /home/xiaoju/software/php7/bin/php

php源码路径在:

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

运行gdb

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

加载gdbinit:

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

设置断点:

  1. (gdb) b zend_execute_scripts

运行:

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

我想在1459这行设置个断点:

  1. 1452 for (i = 0; i < file_count; i++) {
  2. 1453 file_handle = va_arg(files, zend_file_handle *);
  3. 1454 if (!file_handle) {
  4. 1455 continue;
  5. 1456 }
  6. 1457
  7. 1458 op_array = zend_compile_file(file_handle, type);
  8. 1459 if (file_handle->opened_path) {
  9. 1460 zend_hash_add_empty_element(&EG(included_files), file_handle->opened_path);
  10. 1461 }
  11. (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}}

我可以优化输出:

  1. (gdb) set print pretty on
  2. (gdb) p *op_array
  3. $5 = {
  4. type = 2 '\002',
  5. arg_flags = "\000\000",
  6. fn_flags = 134217728,
  7. function_name = 0x0,
  8. scope = 0x0,
  9. 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这个结构:

  1. // opcode组成的数组,编译的时候就是生成这个结构
  2. struct _zend_op_array {
  3. zend_uchar type; // op array的类型,比如 ZEND_EVAL_CODE
  4. zend_uchar arg_flags[3]; /* bitset of arg_info.pass_by_reference */
  5. uint32_t fn_flags;
  6. zend_string *function_name;
  7. zend_class_entry *scope;
  8. zend_function *prototype;
  9. uint32_t num_args; // 脚本的参数
  10. uint32_t required_num_args;
  11. zend_arg_info *arg_info;
  12. /* END of common elements */
  13. uint32_t *refcount; // 这个结构的引用次数
  14. uint32_t last; // opcode的个数
  15. zend_op *opcodes; // 存储所有的opcode
  16. int last_var; // php变量的个数
  17. uint32_t T;
  18. zend_string **vars; // 被编译的php变量的个数
  19. int last_live_range;
  20. int last_try_catch; // try_catch的个数
  21. zend_live_range *live_range;
  22. zend_try_catch_element *try_catch_array; //
  23. /* static variables support */
  24. HashTable *static_variables; // 静态变量
  25. zend_string *filename; // 执行的脚本的文件
  26. uint32_t line_start; // 开始于第几行
  27. uint32_t line_end; // 结束于第几行
  28. zend_string *doc_comment; // 文档的注释
  29. uint32_t early_binding; /* the linked list of delayed declarations */
  30. int last_literal;
  31. zval *literals;
  32. int cache_size;
  33. void **run_time_cache;
  34. void *reserved[ZEND_MAX_RESERVED_RESOURCES]; // 保留字段
  35. };

php内核分析(六)-opcode的更多相关文章

  1. Linux内核分析(六)----字符设备控制方法实现|揭秘系统调用本质

    原文:Linux内核分析(六)----字符设备控制方法实现|揭秘系统调用本质 Linux内核分析(六) 昨天我们对字符设备进行了初步的了解,并且实现了简单的字符设备驱动,今天我们继续对字符设备的某些方 ...

  2. 《Linux内核分析》第六周学习总结

    <Linux内核分析>第六周学习总结                         ——进程的描述和进程的创建 姓名:王玮怡  学号:20135116 一.理论部分 (一)进程的描述 1 ...

  3. 《Linux内核分析》第六周学习笔记

    <Linux内核分析>第六周学习笔记 进程的描述和创建 郭垚 原创作品转载请注明出处 <Linux内核分析>MOOC课程http://mooc.study.163.com/co ...

  4. LINUX内核分析第六周学习总结——进程的描述与创建

    LINUX内核分析第六周学习总结--进程的描述与创建 标签(空格分隔): 20135321余佳源 余佳源 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc ...

  5. linux内核分析第六周学习笔记

    LINUX内核分析第六周学习总结 标签(空格分隔): 20135328陈都 陈都 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.c ...

  6. LINUX内核分析第六周学习总结——进程的描述和进程的创建

    LINUX内核分析第六周学习总结——进程的描述和进程的创建 张忻(原创作品转载请注明出处) <Linux内核分析>MOOC课程http://mooc.study.163.com/cours ...

  7. 《Linux内核分析》 第六节 进程的描述和进程的创建

    <Linux内核分析> 第六节 进程的描述和进程的创建 20135307 张嘉琪 原创作品转载请注明出处 +<Linux内核分析>MOOC课程http://mooc.study ...

  8. Linux内核分析实验六

    Linux内核分析实验六 进程控制块PCB——task_struct(进程描述符) 为了管理进程,内核必须对每个进程进行清晰的描述,进程描述符提供了内核所需了解的进程信息. struct task_s ...

  9. Linux内核分析第六周学习笔记——分析Linux内核创建一个新进程的过程

    Linux内核分析第六周学习笔记--分析Linux内核创建一个新进程的过程 zl + <Linux内核分析>MOOC课程http://mooc.study.163.com/course/U ...

随机推荐

  1. ExtJS 4.2 组件的查找方式

    组件创建了,就有方法找到这些组件.在DOM.Jquery都有各自的方法查找元素/组件,ExtJS也有自己独特的方式查找组件.元素.本次从全局查找.容器内查找.form表单查找.通用组件等4个方面介绍组 ...

  2. 初探Vue

    Vue.js(读音/vju:/,类似于view),是近来比较火的前端框架,但一直没有怎么具体了解.实现过,就知道个啥的MVVM啦,数据驱动啦,等这些关于Vue的虚概念. 由于最近,小生在公司中,负责开 ...

  3. Oracle学习之路-- 案例分析实现行列转换的几种方式

    注:本文使用的数据库表为oracle自带scott用户下的emp,dept等表结构. 通过一个例子来说明行列转换: 需求:查询每个部门中各个职位的总工资 按我们最原始的思路可能会这么写:       ...

  4. 【知识必备】RxJava+Retrofit二次封装最佳结合体验,打造懒人封装框架~

    一.写在前面 相信各位看官对retrofit和rxjava已经耳熟能详了,最近一直在学习retrofit+rxjava的各种封装姿势,也结合自己的理解,一步一步的做起来. 骚年,如果你还没有掌握ret ...

  5. 编写高质量代码:改善Java程序的151个建议(第6章:枚举和注解___建议88~92)

    建议88:用枚举实现工厂方法模式更简洁 工厂方法模式(Factory Method Pattern)是" 创建对象的接口,让子类决定实例化哪一个类,并使一个类的实例化延迟到其它子类" ...

  6. C# 自定义控件VS用户控件

    1 自定义控件与用户控件区别 WinForm中, 用户控件(User Control):继承自 UserControl,主要用于开发 Container 控件,Container控件可以添加其他Con ...

  7. 水平可见直线 bzoj 1007

    水平可见直线 (1s 128M) lines [问题描述] 在xoy直角坐标平面上有n条直线L1,L2,...Ln,若在y值为正无穷大处往下看,能见到Li的某个子线段,则称Li为可见的,否则Li为被覆 ...

  8. Spring配置文件标签报错:The prefix "XXX" for element "XXX:XXX" is not bound. .

    例如:The prefix "context" for element "context:annotation-config" is not bound. 这种 ...

  9. BPM生产安全管理解决方案分享

    一.方案概述生产安全管理是企业生产管理的重要组成部分,组织实施好企业安全管理规划.指导.检查和决策,保证生产处于最佳安全状态是安全管理的重要内容和职责.H3 BPM企业生产安全管理解决方案是一套专门为 ...

  10. 初识git版本控制系统

    当下git分布式版本控制系统越来越火,掌握git也是必须的一个技能.因此,对git做了如下学习. Git初级指南 1. 先安装git.(ps:在select cmponents处要勾选Git Bash ...