[研究] 深入PHP内核之ZendAPI扩展篇

长平狐 发布于 2012/11/19 15:11
阅读 1K+
收藏 0

PHP_NEW_EXTENSION(php_hello, php_hello.c, $ext_shared)

前面插一个很好笑的报道:“2009/02/27日新浪:首节战罢,火箭命中率27.8%对上骑士17.6%,篮板15对7,助攻4对1”,这是我看过最搞笑的篮球比赛的命中率了:)OK~进入正题,最近忽然有兴趣研究一下PHP的内核和架构,关于PHP架构分析准备在后面的文章里推出,这篇文章主要介绍的是关于如何扩展ZendAPI的问题~

首先给大家介绍一些资料:

http://php.cn/php/zend.html

http://us.php.net/internals2.ze1.zendapi

http://us.php.net/manual/en/internals2.php

下面是一张介绍php系统架构的图片,可以粗略看出php模块和ZendEngine之间的关系:

然后就是实例教程了,动手吧~~

1、首先使用ext_skel建立一个PHP扩展的module骨架:

这里要注意的是ext_skel工具一般在PHP源码包的ext目录下,但是我更愿意把它提出来,也就是不在PHP源码包的ext目录下建立module,假设我现在在/home/php下建立一个module名为php_hello的module

#cd /home/php

#/path/to/ext_skel --extname=php_hello

#cd php_hello

修改config.m4文件为,简单说就是把一些dnl注释去掉即可:

PHP_ARG_WITH(php_hello, for php_hello support,
dnl Make sure that the comment is aligned:
[  --with-php_hello             Include php_hello support])

或者
PHP_ARG_WITH(php_hello, for php_hello support,
dnl Make sure that the comment is aligned:
[  --with-php_hello             Include php_hello support])

这样子一个扩展的module的骨架就搞定了,看看config.m4的最后:PHP_NEW_EXTENSION(php_hello, php_hello.c, $ext_shared) 这行指明了php_hello模块需要编译的目标文件,也就是php_hello.c

2、然后就是写代码,建议边看资料边写,这里我贴上自己的测试代码:

在 php_hello 目录下面新建以下文件:

php_hello.h:

#ifndef PHP_HELLO_H
#define PHP_HELLO_H 1
#define PHP_HELLO_WORLD_VERSION "1.0"
#define PHP_HELLO_WORLD_EXTNAME "php_hello"
 
PHP_MINIT_FUNCTION(hello);
PHP_MSHUTDOWN_FUNCTION(hello);
/* defined your functions */
PHP_FUNCTION(hello_world);
PHP_FUNCTION(hello_float);
PHP_FUNCTION(hello_getobj);
PHP_FUNCTION(hello_callfunc);
extern zend_module_entry php_hello_module_entry;
#define phpext_php_hello_ptr &php_hello_module_entry
#endif

php_hello.c:

#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include "php.h"
#include "php_ini.h"
#include "php_hello.h"
/* add your functions */
static function_entry php_hello_functions[] = {
	PHP_FE(hello_world, NULL)
	PHP_FE(hello_float, NULL)
	PHP_FE(hello_getobj, NULL)
	PHP_FE(hello_callfunc, NULL)
	{NULL, NULL, NULL}
};
zend_module_entry php_hello_module_entry = {
#if ZEND_MODULE_API_NO >= 20010901
	STANDARD_MODULE_HEADER,
#endif
	PHP_HELLO_WORLD_EXTNAME,
	php_hello_functions,
	PHP_MINIT(hello),
	PHP_MSHUTDOWN(hello),
	NULL,
	NULL,
	NULL,
#if ZEND_MODULE_API_NO >= 20010901
	PHP_HELLO_WORLD_VERSION,
#endif
	STANDARD_MODULE_PROPERTIES
};
PHP_INI_BEGIN()
PHP_INI_ENTRY("hello.sayto", "Default", PHP_INI_ALL, NULL)
PHP_INI_END()
#ifdef COMPILE_DL_PHP_HELLO
ZEND_GET_MODULE(php_hello)
#endif
PHP_MINIT_FUNCTION(hello)
{
	REGISTER_INI_ENTRIES();
	return SUCCESS;
}
PHP_MSHUTDOWN_FUNCTION(hello)
{
	UNREGISTER_INI_ENTRIES();
	return SUCCESS;
}
/* implement your functions */
PHP_FUNCTION(hello_world)
{
	zend_printf("Hello World, %s :)", INI_STR("hello.sayto"));
}
PHP_FUNCTION(hello_float)
{
	long params;
	// types : l(long) | d(double) | s(string) | b(boolean) | r(resource) | a(array) | o(object) | O(specified object) | z(the actual zval*)
	// notice : r | a | o | O are stored in zval*
	if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &params) == FAILURE) {
		return;
	}
	RETURN_LONG(params);
}
PHP_FUNCTION(hello_getobj)
{
	zval *params;
	if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "o", &params) == FAILURE) {
		return;
	}
	// types : NULL 0 | LONG 1 | DOUBLE 2 | STRING 3 | ARRAY 4 | OBJECT 5 | BOOL 6 | RESOURCE 7 | CONSTANT 8 | CONSTANT_ARRAY 9
	zend_printf("We get object as type %i ...", params->type);
}
PHP_FUNCTION(hello_callfunc)
{
	zval **function_name;
	zval *function_ret;
	if((ZEND_NUM_ARGS() != 1) || (zend_get_parameters_ex(1, &function_name) != SUCCESS)) {
		WRONG_PARAM_COUNT;
	}
	if((*function_name)->type != IS_STRING) {
		zend_error(E_ERROR, "Function requires string argument");
	}
	TSRMLS_FETCH();
	if(call_user_function_ex(CG(function_table), NULL, *function_name, &function_ret, 0, NULL, 0, NULL TSRMLS_CC) != SUCCESS) {
		zend_error(E_ERROR, "Function call failed");
	}
	zend_printf("We have %i as type ...", function_ret->type);
	*return_value = *function_ret;
	zval_copy_ctor(return_value);
	zval_ptr_dtor(&function_ret);
}

简单分析一下源码:

php_hello.h里面主要定义了一些宏变量以及结构这里需要注意的几个方法:

ZEND_FUNCTION 这个不用说了,一看就知道是定义扩展function的函数了

详细定义请看zend_API.h的45行:

#define ZEND_NAMED_FUNCTION(name)        void name(INTERNAL_FUNCTION_PARAMETERS)
#define ZEND_FUNCTION(name)                ZEND_NAMED_FUNCTION(ZEND_FN(name))
#define ZEND_METHOD(classname, name)    ZEND_NAMED_FUNCTION(ZEND_MN(classname##_##name))

...

至于PHP_MINIT_FUNCTION和PHP_MSHUTDOWN_FUNCTION则分别用来执行php引擎在INIT和SHUTDOWN要做的工作,它们的定义在zend_API.h的88行左右

而PHP_INI_ENTRY则是初始化ini文件的配置的函数,在程序中可以由INI_STR调用。

最后我们分别说一下几个demo方法的作用吧:

hello_world //打印,读取ini

hello_float //处理double

hello_getobj //处理object

hello_callfunc //使用回调函数

然后就是编译模块:

#/path/to/phpize

#./configure --with-php-config=/path/to/php-config

#make install

你会发现module下面出现了php_hello.so把它拷贝到/path/to/phpext目录下面,编辑/etc/php.ini,在最后加上:

[php_hello module]

extension = php_hello.so
hello.sayto = James

就可以了。

3、最后我们写一个php来测试一下吧:)

以下是测试的php的代码:

php_hello.php:

<?php
class Hello {
}
function singing() {
        return 'I am singing :)';
}
echo hello_world()."/n";
echo "Float Num : ".hello_float(3.1415926)."/n";
echo hello_getobj(new Hello())."/n";
echo hello_callfunc("singing")."/n";
?>

运行看看结果吧!

Hello World, James :)
Float Num : 1293080650
We get object as type 5 ...
We have 6 as type ...I am singing :)

看到了吧,至此我们的demo成功执行了,是不是很有趣呢?下几篇我将会分析一下PHP的源码架构,实际上PHP的这套语言架构还是非常强大的,特别是Zend引擎对zval的定义,很有研究价值,慢慢深入,我们将接触到为什么PHP如此好用而流行的核心秘密了,Keep walking  ...

 


原文链接:http://blog.csdn.net/shagoo/article/details/3941584
加载中
返回顶部
顶部