`

请您先登录,才能继续操作

使用PHP Embed SAPI实现Opcodes查看器

阅读更多

使用PHP Embed SAPI实现Opcodes查看器


·作者:laruence(http://www.laruence.com/)

·本文地址: http://www.laruence.com/2008/09/23/539.html

·转载请注明出处

PHP提供了一个Embed SAPI,也就是说,PHP容许你在C/C++语言中调用PHP/ZE提供的函数。本文就通过基于Embed SAPI实现一个PHP的opcodes查看器。

首先,下载PHP源码以供编译, 我现在使用的是PHP5.3 alpha2

进入源码目录:

./configure --enable-embed 

./make

./make install


最后,记得要将生成的libphp5.so复制到运行时库的目录,我直接拷贝到了/lib/, 否则会在运行你自己的embed程序的时候报错:

./embed: error while loading shared libraries: libphp5.so: cannot open shared object file: No such file or directory


如果你对PHP的SAPI还不熟悉的话,我建议你看看我的这篇文章:深入理解Zend SAPIs(Zend SAPI Internals)

这个时候,你就可以在你的C代码中,嵌入PHP脚本解析器了, 我的例子:

#include "sapi/embed/php_embed.h" 
int main(int argc, char * argv[]){
    PHP_EMBED_START_BLOCK(argc,argv);
    char * script = " print 'Hello World!';";
    zend_eval_string(script, NULL,  "Simple Hello World App" TSRMLS_CC);
    PHP_EMBED_END_BLOCK();
    return 0;
}

然后就是要指明include path了,一个简单的Makefile

CC = gcc

CFLAGS = -I/usr/local/include/php/ \

            -I/usr/local/include/php/main \

            -I/usr/local/include/php/Zend \

            -I/usr/local/include/php/TSRM \

            -Wall -g

LDFLAGS = -lstdc++ -L/usr/local/lib -lphp5

ALL:

    $(CC) -o embed embed.cpp $(CFLAGS) $(LDFLAGS)


编译成功以后, 运行,我们可以看到, stdout输出 Hello World!

基于这个,我们就可以很容易的实现一个类似于vld的Opcodes dumper:

首先我们定义opcode的转换函数(全部的opcodes可以查看Zend/zend_vm_opcodes.h);

 

 

char *opname(zend_uchar opcode){
    switch(opcode) {
        case ZEND_NOP: return "ZEND_NOP"; break;
        case ZEND_ADD: return "ZEND_ADD"; break;
        case ZEND_SUB: return "ZEND_SUB"; break;
        case ZEND_MUL: return "ZEND_MUL"; break;
        case ZEND_DIV: return "ZEND_DIV"; break;
        case ZEND_MOD: return "ZEND_MOD"; break;
        case ZEND_SL: return "ZEND_SL"; break;
        case ZEND_SR: return "ZEND_SR"; break;
        case ZEND_CONCAT: return "ZEND_CONCAT"; break;
        case ZEND_BW_OR: return "ZEND_BW_OR"; break;
        case ZEND_BW_AND: return "ZEND_BW_AND"; break;
        case ZEND_BW_XOR: return "ZEND_BW_XOR"; break;
        case ZEND_BW_NOT: return "ZEND_BW_NOT"; break;
        /*...省略 ....*/
        default : return "UNKNOW"; break;
}
//然后定义zval和znode的输出函数:
char *format_zval(zval *z)
{
    static char buffer[BUFFER_LEN];
    int len;
 
    switch(z->type) {
        case IS_NULL:
            return "NULL";
        case IS_LONG:
        case IS_BOOL:
            snprintf(buffer, BUFFER_LEN, "%d", z->value.lval);
            return buffer;
        case IS_DOUBLE:
            snprintf(buffer, BUFFER_LEN, "%f", z->value.dval);
            return buffer;
        case IS_STRING:
            snprintf(buffer, BUFFER_LEN, "\"%s\"", z->value.str.val);
            return buffer;
        case IS_ARRAY:
        case IS_OBJECT:
        case IS_RESOURCE:
        case IS_CONSTANT:
        case IS_CONSTANT_ARRAY:
            return "";
        default:
            return "unknown";
    }
}
 
char * format_znode(znode *n){
    static char buffer[BUFFER_LEN];
 
    switch (n->op_type) {
        case IS_CONST:
            return format_zval(&n->u.constant);
            break;
        case IS_VAR:
            snprintf(buffer, BUFFER_LEN, "$%d",  n->u.var/sizeof(temp_variable));
            return buffer;
            break;
        case IS_TMP_VAR:
            snprintf(buffer, BUFFER_LEN, "~%d",  n->u.var/sizeof(temp_variable));
            return buffer;
            break;
        default:
            return "";
            break;
    }
}

//然后定义op_array的输出函数:
void dump_op(zend_op *op, int num){
    printf("%5d  %5d %30s %040s %040s %040s\n", num, op->lineno,
            opname(op->opcode),
            format_znode(&op->op1),
            format_znode(&op->op2),
            format_znode(&op->result)) ;
}
 
void dump_op_array(zend_op_array *op_array){
    if(op_array) {
        int i;
        printf("%5s  %5s %30s %040s %040s %040s\n", "opnum", "line", "opcode", "op1", "op2", "result");
        for(i = 0; i < op_array->last; i++) {
            dump_op(&op_array->opcodes[i], i);
        }
    }
}

//最后,就是程序的主函数了:
int main(int argc, char **argv){
    zend_op_array *op_array;
    zend_file_handle file_handle;
 
    if(argc != 2) {
        printf("usage:  op_dumper <script>\n");
        return 1;
    }
    PHP_EMBED_START_BLOCK(argc,argv);
    printf("Script: %s\n", argv[1]);
    file_handle.filename = argv[1];
    file_handle.free_filename = 0;
    file_handle.type = ZEND_HANDLE_FILENAME;
    file_handle.opened_path = NULL;
    op_array =  zend_compile_file(&file_handle, ZEND_INCLUDE TSRMLS_CC);
    if(!op_array) {
        printf("Error parsing script: %s\n", file_handle.filename);
        return 1;
    }
    dump_op_array(op_array);
    PHP_EMBED_END_BLOCK();
    return 0;
}
 

 

编译,运行测试脚本(sample.php):

sample.php:

echo "laruence";


命令:

./opcodes_dumper  sample.php


得到输出结果(如果你对下面的结果很迷惑,那么建议你再看看我的这篇文章:深入理解PHP原理之Opcodes):

Script: sample.php

opnum   line                         opcode                                      op1                                      op2                                   result

    0      2                      ZEND_ECHO                               "laruence"                                          

    1      4                    ZEND_RETURN                                        1


呵呵,怎么样,是不是很好玩呢?

 

分享到:
评论

相关推荐

    如何使用PHP Embed SAPI实现Opcodes查看器

    标题中的“如何使用PHP Embed SAPI实现Opcodes查看器”是指在C或C++程序中利用PHP的嵌入式SAPI(Server Application Programming Interface)来创建一个能够查看PHP脚本编译后生成的Opcodes(操作码)的工具。...

    为PHP5.4开启Zend OPCode缓存

    - 如何使用PHP Embed SAPI实现Opcodes查看器,了解如何通过PHP Embed SAPI来查看和分析OPCode。 - PHP中opcode缓存简单用法分析,介绍如何在实际应用中设置和使用OPCode缓存。 - PHP7如何开启Opcode打造强悍性能详解...

    embed2-sapi:允许在多线程应用程序中嵌入 PHP 和并发 PHP 脚本使用的替代 PHP SAPI

    如果您需要在多线程环境中使用嵌入式 PHP,在处理多个脚本运行时需要更高的速度,或者您希望操作属于 PHP_INI_SYSTEM 或 PHP_INI_PERDIR 类别的 INI 设置,您将需要 EMBED2 SAPI 模块。 如何在多线程模式下运行并发...

    php-embed-windows:PHP在Windows上嵌入SAPI

    php-embed-windows PHP在Windows上嵌入SAPI 在Windows上构建PHP7 您可以阅读有关如何在Windows上构建PHP7.2的。 在Windows上构建嵌入式SAPI git克隆这个项目 ...

    embed标签使用详解

    - **兼容性**:`embed`标签主要针对非IE浏览器设计,而`object`标签适用于所有浏览器,尤其是IE浏览器。 - **使用建议**:为了确保页面在不同浏览器中的兼容性,建议同时使用`&lt;embed&gt;`和`&lt;object&gt;`标签。通常情况下...

    html用的图片查看器

    描述中提到,该图片查看器可以直接插入到网页中使用,这意味着它可能是一个自包含的组件,可以通过HTML代码插入到网页的`&lt;object&gt;`或`&lt;embed&gt;`标签中,或者使用JavaScript库如SWFObject来加载和控制。 标签“图片...

    embed使用,embed播放多媒体

    本文将深入探讨`embed`元素的使用方法、属性以及其在多媒体播放中的作用。 ### 一、embed的基本结构 `embed`标签通常用于插入外部资源,例如: ```html &lt;embed src="path_to_your_media_file" width="320" height...

    HTML5 embed 标签使用方法介绍

    要使用`&lt;embed&gt;`标签嵌入内容,必须通过其`src`属性来指定要嵌入的资源的URL地址,而且该URL地址必须包含具体的文件扩展名。这一点在尝试播放MP3音乐文件时尤为重要。例如,如果要嵌入一个MP3文件,`src`属性的值就...

    html使用embed 播放flv 视频

    在本案例中,我们关注的是如何使用`&lt;embed&gt;`标签来播放FLV(Flash Video)格式的视频。FLV是Adobe Flash平台广泛采用的一种视频格式,尤其在早期互联网上,它因为良好的跨平台性和浏览器兼容性而非常流行。 首先,...

    Go-embed:另一个Golang静态内容嵌入器

    2. **使用文件系统接口**: `embed.FS`是一个实现了标准的`io/fs`接口的类型,这意味着你可以使用各种标准的文件系统操作函数来访问嵌入的文件。例如,读取一个HTML文件: ```go html, err := staticFiles.ReadFile...

    embed标签的使用(在网页中播放各种音频视频的插件的使用)

    ### embed标签的使用详解 #### 一、embed标签概述 `&lt;embed&gt;`标签是一种用于嵌入多媒体内容到HTML文档中的标记。它支持多种媒体格式,包括音频、视频以及Flash动画等。通过设置不同的属性,我们可以控制媒体文件的...

    HTML-embed代码详解

    embed src=url 其中,src 是音频或视频文件的路径,可以是相对路径或绝对路径。 属性设置 1. 自动播放 autostart=true、false 如果 autostart=true,音频或视频文件将在下载完毕后自动播放;如果 autostart=...

    动态修改Embed的src属性

    在实际应用中,为了确保平滑过渡,可以添加加载指示器,或者在新资源加载期间隐藏 `&lt;embed&gt;` 元素。同时,处理错误情况也很重要,例如当新资源无法加载时,可以显示错误信息或回退到旧资源。 6. **应用场景**: -...

    embed embed

    在实际应用中,embed标签通常与其他属性结合使用,以提供更丰富的用户体验。然而,考虑到兼容性和标准化,开发者应考虑使用HTML5的video和audio标签,它们提供了更多的控制选项和更好的跨浏览器支持。例如,对于音频...

    AS3 Embed用法

    ### AS3 Embed 用法详解 #### 一、引言 ...`Embed`不仅可以帮助我们更高效地管理和使用资源,还提供了更多的灵活性和扩展性。在未来的工作中,熟练掌握`Embed`的使用将会使我们的开发工作更加高效。

    Embed嵌入图片

    下面我们将深入探讨`Embed`元标签的使用方法以及它在加载图片资源中的应用。 一、`Embed`元标签基础 `Embed`元标签是AS3中用于内联嵌入资源的关键字,通常与`[Embed]`语法一起使用。它的基本结构如下: ```as3 ...

    Go-go-embed-生成的Go代码来嵌入资源文件到你的库或可执行文件中

    在`pyros2097/go-embed-4274f34` 这个压缩包中,包含了`go-embed` 某一版本的源码,如果你需要深入了解其内部实现或进行二次开发,可以解压并查看源代码。通常,源码中会有更详细的文档和示例来指导你如何使用和扩展...

    Html 标签embed 动态显示

    `&lt;embed&gt;`标签的使用方式如下: ```html &lt;embed src="资源URL" width="宽度" height="高度" allowfullscreen="true" /&gt; ``` 其中,`src`属性指定了要嵌入的外部资源的URL,`width`和`height`分别定义了嵌入内容的...

    EMBED用法介绍在网页中播放视频的jsp标签

    "EMBED用法介绍在网页中播放视频的jsp&lt;embed&gt;标签" EMBED标签是HTML中一种常用的标签,用于在网页中播放音频、视频文件。它可以播放多种格式的文件,包括Windows Media Player支持的格式,如WMA、WMV、ASF、MPG、...

Global site tag (gtag.js) - Google Analytics