Heim  >  Artikel  >  Backend-Entwicklung  >  PHP7-Kernelanalyse 5 Kompilierung von PHP-Code

PHP7-Kernelanalyse 5 Kompilierung von PHP-Code

不言
不言Original
2018-04-13 15:02:201679Durchsuche

Der Inhalt dieses Artikels befasst sich mit der Kompilierung von PHP-Code in PHP7 Kernel Analysis 5. Jetzt können Freunde in Not darauf verweisen

1

Die Aufgabe des PHP-Parsing-Prozesses besteht darin, den PHP-Code in ein Opcode-Array umzuwandeln. Alle Informationen im Code werden im Opcode-Array gespeichert und dann wird das Opcode-Array an Zend übergeben Der Opcode wird speziell vom Kernel ausgeführt. Befehle wie Zuweisungs-, Additions- und Subtraktionsoperationen, Funktionsaufrufe usw. Diese Handler sind im Voraus definierte C-Funktionen.

PHP7-Kernelanalyse 5 Kompilierung von PHP-Code


2.PHP-Code->Abstract Syntax Tree (AST)

PHP使用re2c、bison完成这个阶段的工作:
    re2c: 词法分析器,将输入分割为一个个有意义的词块,称为token
    bison: 语法分析器,确定词法分析器分割出的token是如何彼此关联的

PHP7-Kernelanalyse 5 Kompilierung von PHP-Code

词法、语法解析过程

1.yyparse(zendparse)调用yylex(zendlex),当读取到一个合法的token时,返回值为token类型
2.yylex调用lex_scan读取合法的token值
3.yyparse将token类型与token值构造抽象语法树,最后将根节点保存到CG(compiler_globals ,Zend编译器相关的全局变量)的ast中


3.AST-Knotenstruktur

typedef struct _zend_ast   zend_ast;

//普通节点类型
struct _zend_ast {
    zend_ast_kind kind;  //节点类型
    zend_ast_attr attr;  //节点附加属性
    uint32_t lineno;    //行号
    zend_ast *child[1];  //子节点数组
};

//普通节点类型,但有子节点的个数
typedef struct _zend_ast_list {
    zend_ast_kind kind; //节点类型
    zend_ast_attr attr; //节点附加属性
    uint32_t lineno; //行号
    uint32_t children; //子节点数量
    zend_ast *child[1];//子节点数组
} zend_ast_list;

//函数、类的ast节点结构
typedef struct _zend_ast_decl {
    zend_ast_kind kind; //节点类型
    zend_ast_attr attr; //节点附加属性
    uint32_t start_lineno; //开始行号
    uint32_t end_lineno;   //结束行号
    uint32_t flags;
    unsigned char *lex_pos;
    zend_string *doc_comment;
    zend_string *name;
    zend_ast *child[4]; //类中会将继承的父类、实现的接口以及类中的语句解析保存在child中
} zend_ast_decl;
实例:
$a = 123;
$b = "hi~";

echo $a,$b;

PHP7-Kernelanalyse 5 Kompilierung von PHP-Code


4.zend_op_array

PHP7-Kernelanalyse 5 Kompilierung von PHP-Code

zend_op *opcodes; //opcode指令数组
zval *literals; //字面量(常量)数组,这些都是在PHP代码定义的一些值
zend_string **vars; //PHP变量名数组,根据变量编号可以获取相应的变量
//opcode指令结构
struct _zend_op {
    const void *handler; //指令执行handler
    znode_op op1;   //操作数1
    znode_op op2;   //操作数2
    znode_op result; //返回值
    uint32_t extended_value; 
    uint32_t lineno; 
    zend_uchar opcode;  //opcode指令
    zend_uchar op1_type; //操作数1类型
    zend_uchar op2_type; //操作数2类型
    zend_uchar result_type; //返回值类型
};


5.Handler-Verarbeitungsfunktion

Handler entspricht jedem Opcode Verarbeitungsprozess in C-Sprache geschrieben, die Verarbeitungsprozesse, die allen Opcodes entsprechen, sind in zend_vm_def.h definiert, der Opcode-Verarbeitungsprozess hat drei verschiedene Formen: CALL, SWITCH, GOTO, die Standardmethode ist CALL
CALL:把每种opcode负责的工作封装成不同的function,然后执行器循环调用执行
SWITCH:把所有的处理方式写到一个switch下,然后通过case不同的opcode执行具体的操作
GOTO:把所有opcode的处理方式通过C语言里面的label标签区分开,然后执行器执行的时候goto到相应的位置处理


6. Abstrakter Syntaxbaum->Opcodes

void zend_compile_top_stmt(zend_ast *ast){
    ....
    if (ast->kind == ZEND_AST_STMT_LIST) { //第一次进来一定是这种类型
        zend_ast_list *list = zend_ast_get_list(ast);
        uint32_t i;
        for (i = 0; i < list->children; ++i) {
            zend_compile_top_stmt(list->child[i]);//list各child语句相互独立,递归编译
        }
        return;
    }
    //各语句编译入口
    zend_compile_stmt(ast);
    ....
}

1.zend_compile_top_stmt接收语法树,首先判断节点类型是否为ZEND_AST_STMT_LIST(表示当前节点下
有多个独立的节点),若是则进行递归
2.当递归结束后,调用zend_compile_stmt进行编译成opcodes

Verwandte Empfehlungen:

PHP7 Kernel Analysis 4 Local Variablen, globale Variablen, Konstanten

Variablen von PHP7 Kernel Analysis 3

I/O-Modell von PHP7 Kernel Analysis 2

                                                                                 

Das obige ist der detaillierte Inhalt vonPHP7-Kernelanalyse 5 Kompilierung von PHP-Code. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn