如何在扩展里调用PHP函数呢?
作者 斯人 | 发布于 2012 年 6 月 6 日
PHP PHP内核

在写扩展的时候,

肯定不能什么功能都自己实现,原因有很多,开发效率、性能问题、维护成本等。

这就避免不了要在扩展里调用PHP内核函数。

那么如何在扩展里面使用我们的内置函数呢?

PHP既然有函数,肯定提供了调用函数的接口,

这里有两个函数:

ZEND_API int call_user_function(HashTable *function_table, zval **object_pp, zval *function_name, zval *retval_ptr, zend_uint param_count, zval *params[] TSRMLS_DC);

ZEND_API int call_user_function_ex(HashTable *function_table, zval **object_pp, zval *function_name, zval **retval_ptr_ptr, zend_uint param_count, zval **params[], int no_separation, HashTable *symbol_table TSRMLS_DC);

很可惜,只找到声明的地方,没有找到注释的地方,

从声明来看,call_user_function 封装了call_user_function_ex,

少了两个传参:

int no_separation:这个的用意就是是否对zval进行分离,不过此功能现在已经不用了,如果设为1则直接会出错,分离的作用是为了优化空间。

symbol_table :是干吗用的,确实目前还不知道。

HashTable* function_table: 函数表,我们都知道 用户函数也好,其他内置函数也好都会存在hashtable里面,function_table会保存所有内置函数和用户函数,用CG来获取,因为函数表属于是 编译全局变量。

zval **object_pp:这个是用来我们调用类里的某个方法的对象,看到这里,你恍然大悟没有,调用普通函数和调用类的方法是同一个,
function_table和object_pp只需要一个就可以了。

zval **retval_ptr_ptr:是函数的返回值。

zend_uint param_count:函数/方法 的参数个数

zval **params[] :函数/方法的参数指针。

那么清楚了两个API的参数及调用方法之后,我们来测试一下,究竟如何来使用。

首先 创建一个 扩展文件
/ext/ext_skel –extname=call_func

修改config.m4

在php_call_func.h中添加
PHP_FUNCTION(siren_call);做一个声明

打开call_func.c
重点代码:

PHP_FUNCTION(siren_call){
        zval *function_name;
        zval* args;
        zval* retval;
        args=(zval*)malloc(sizeof(zval));
        if(zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC,"zz",&function_name,&args)==FAILURE){
                zend_error(E_ERROR,"function requires args");
        }   
        if(function_name->type!=IS_STRING){
                zend_error(E_ERROR,"function name must be a string");
        }   
        //call_user_function(`<HashTable *function_table>`,`< zval **object_pp>`,`< zval *function_name>`,`< zval *retval_ptr>`,`< zend_uint param_count>`,`< zval *params[] TSRMLS_DC>`)
        zval**params=(zval**)malloc(sizeof(zval));
        params[0]=args;
        if(call_user_function(CG(function_table),NULL,function_name,retval,1,params TSRMLS_DC)==FAILURE){
                zend_error(E_ERROR,"call function failed");
        }   
        *return_value=*retval;
        zval_copy_ctor(return_value);
        zval_ptr_dtor(&retval);
}

这样,我们编译之后,就建好了一个扩展,
siren_call接收两个参数,第一个是需要调用的函数名,第二个是传给函数的参数。这里只是一个简单的zval类型

使用方法如下

function test($a){
        echo "my name is :".$a;
}
siren_call("test"," siren!\r\n");

这样 就会输出 my name is siren;

怎么样 是不是很简单。

原文出处:http://www.imsiren.com/archives/606