内容简介:Just for fun——PHP7扩展编写中的宏
- SAPI 是 PHP 的最上层,它是PHP的应用接口层,对于源码目录为sapi
- main 是PHP的主要代码,主要是输入/输出,Web通信,以及PHP框架的初始化操作,对于源码目录为main
- ZendVM 是PHP解释器的主要实现,即ZendVM,对于源码目录为Zend 截一张 php-src 的图,目录都有对应
PHP的生命周期
PHP根据不同SAPI的实现,各阶段的执行情况有些差异。譬如cli模式的话,完整地经历了这些阶段,而Fastcgi模式下则在启动时执行一次模块初始化,然后各个请求只经历请求初始化,执行请求脚本,请求关闭这几个阶段。
PHP扩展
开发者可以通过C/C++实现自定义的功能,通过扩展嵌入到PHP中。 编写扩展的步骤:
- 通过ext目录下 ext_skel 脚本生成扩展的基本框架./ext_skel --extname=module (module is the name of your extension)
- 修改config.m4配置:设置编译配置参数、设置扩展源文件
- 编写扩展源代码
- 生成configure:写完后先phpize(在php的bin目录下)运行一下
- 编译&安装: ./configure、 make、make install,然后改一下php.ini文件,添加一下.so文件
举例
操作系统:CentOS Linux release 7.3.1611 PHP版本:PHP 7.1.11
生成骨架
./ext_skel --extname=my_test --no-help
--no-help是略去注释代码(干净点) 生成目录my_test:
查看C文件
my_test.c
/* $Salamander$ */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "php.h" #include "php_ini.h" #include "ext/standard/info.h" #include "php_my_test.h" static int le_my_test; PHP_MINIT_FUNCTION(my_test) { return SUCCESS; } PHP_MSHUTDOWN_FUNCTION(my_test) { return SUCCESS; } PHP_RINIT_FUNCTION(my_test) { #if defined(COMPILE_DL_MY_TEST) && defined(ZTS) ZEND_TSRMLS_CACHE_UPDATE(); #endif return SUCCESS; } PHP_RSHUTDOWN_FUNCTION(my_test) { return SUCCESS; } PHP_MINFO_FUNCTION(my_test) { php_info_print_table_start(); php_info_print_table_header(2, "my_test support", "enabled"); php_info_print_table_end(); } const zend_function_entry my_test_functions[] = { PHP_FE_END }; zend_module_entry my_test_module_entry = { STANDARD_MODULE_HEADER, "my_test", my_test_functions, PHP_MINIT(my_test), PHP_MSHUTDOWN(my_test), PHP_RINIT(my_test), PHP_RSHUTDOWN(my_test), PHP_MINFO(my_test), PHP_MY_TEST_VERSION, STANDARD_MODULE_PROPERTIES }; #ifdef COMPILE_DL_MY_TEST #ifdef ZTS ZEND_TSRMLS_CACHE_DEFINE() #endif ZEND_GET_MODULE(my_test) #endif
可以注意到这里有一些 宏
- PHP_MINIT_FUNCTION
- PHP_MSHUTDOWN_FUNCTION
- PHP_RINIT_FUNCTION
- PHP_RSHUTDOWN_FUNCTION
- PHP_MINFO_FUNCTION 这些是PHP提供的钩子函数,PHP执行到不同的阶段时 回调 各个扩展定义的钩子函数,定义完成后,最后设置一下zend_module_entry对应的函数指针即可。 回顾之前的PHP的生命周期,也就是说(=>指对应某个阶段): PHP_MINIT_FUNCTION => 模块初始化阶段(M就是module的含义,init就是initial) PHP_MSHUTDOWN_FUNCTION => 模块关闭阶段(M就是module的含义,后面就是shutdown) PHP_RINIT_FUNCTION => 请求初始化(R就是request的含义,init就是initial) PHP_RSHUTDOWN_FUNCTION => 请求关闭阶段(R就是request的含义,后面就是shutdown) PHP_MINFO_FUNCTION 指获取模块信息 最后,设置 zend_module_entry 这个结构体
zend_module_entry my_test_module_entry = { STANDARD_MODULE_HEADER, "my_test", my_test_functions, PHP_MINIT(my_test), PHP_MSHUTDOWN(my_test), PHP_RINIT(my_test), PHP_RSHUTDOWN(my_test), PHP_MINFO(my_test), PHP_MY_TEST_VERSION, STANDARD_MODULE_PROPERTIES };
获取各个钩子函数的指针,有对对应的宏PHP_MINIT,PHP_MSHUTDOWN,PHP_RINIT,PHP_RSHUTDOWN,PHP_MINFO
注册函数
分为两步:
- 定义函数,可以通过PHP_FUNCTION()或ZEND_FUNCTION()宏来完成函数声明
- 注册函数,PHP提供了 zend_function_entry ,扩展只需为每个内部函数生成这样一个结构,然后将所有函数的结构数组提供给zend_module_entry->functions即可 For Example:
PHP_FUNCTION(my_func) { // 具体实现 }
展开后
void zif_my_func(zend_execute_data *execute_data, zval *return_value) { // ... }
zend_function_entry可以通过宏PHP_FE或ZEND_FE生成(FE即function entry)。
const zend_function_entry my_test_functions[] = { PHP_FE(my_func, NULL) PHP_FE_END };
my_test_functions就是这个扩展注册的函数数组。 最后,它设置在了zend_module_entry(第三个参数)
zend_module_entry my_test_module_entry = { STANDARD_MODULE_HEADER, "my_test", my_test_functions, PHP_MINIT(my_test), PHP_MSHUTDOWN(my_test), PHP_RINIT(my_test), PHP_RSHUTDOWN(my_test), PHP_MINFO(my_test), PHP_MY_TEST_VERSION, STANDARD_MODULE_PROPERTIES };
函数参数解析
PHP提供了一个方法将zend_execute_data上的参数解析到指定变量上。
//file: Zend/zend_API.h ZEND_API int zend_parse_parameters(int num_args, const char *type_spec, ...)
- num_args:参数数量,用ZEND_NUM_ARGS()可以获取
- type_spec 为参数解析规则,是一个字符串
- 最后一个是可变参数,指定要解析到的变量地址 举例:
PHP_FUNCTION(my_func) { zval *arr; if(zend_parse_parameters(ZEND_NUM_ARGS(), "a", &a) == FAILURE) { RETURN_FALSE; } ... }
如果有多个变量type_spec可以变为"la",l表示整型,a表示数组(另外还有b:布尔型,s:字符串型,o:对象) ,后面则改为&a, &b
函数返回值
可以设置return_value,但PHP提供了设置了设置返回值的宏
#define RETURN_BOOL(b) { RETVAL_BOOL(b); return; } #define RETURN_NULL() { RETVAL_NULL(); return;} #define RETURN_LONG(l) { RETVAL_LONG(l); return; } #define RETURN_DOUBLE(d) { RETVAL_DOUBLE(d); return; } #define RETURN_STR(s) { RETVAL_STR(s); return; } #define RETURN_INTERNED_STR(s) { RETVAL_INTERNED_STR(s); return; } #define RETURN_NEW_STR(s) { RETVAL_NEW_STR(s); return; } #define RETURN_STR_COPY(s) { RETVAL_STR_COPY(s); return; } #define RETURN_STRING(s) { RETVAL_STRING(s); return; } #define RETURN_STRINGL(s, l) { RETVAL_STRINGL(s, l); return; } #define RETURN_EMPTY_STRING() { RETVAL_EMPTY_STRING(); return; } #define RETURN_RES(r) { RETVAL_RES(r); return; } #define RETURN_ARR(r) { RETVAL_ARR(r); return; } #define RETURN_OBJ(r) { RETVAL_OBJ(r); return; } #define RETURN_ZVAL(zv, copy, dtor) { RETVAL_ZVAL(zv, copy, dtor); return; } #define RETURN_FALSE { RETVAL_FALSE; return; } #define RETURN_TRUE { RETVAL_TRUE; return; }
写个小例子
写一个两个整型变量相加的函数
/* $Salamander$ */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "php.h" #include "php_ini.h" #include "ext/standard/info.h" #include "php_my_test.h" static int le_my_test; PHP_FUNCTION(my_add) { int argc = ZEND_NUM_ARGS(); zend_long a; zend_long b; if (zend_parse_parameters(argc, "ll", &a, &b) == FAILURE) RETURN_FALSE; RETURN_LONG(a + b); } PHP_MINIT_FUNCTION(my_test) { return SUCCESS; } PHP_MSHUTDOWN_FUNCTION(my_test) { return SUCCESS; } PHP_RINIT_FUNCTION(my_test) { #if defined(COMPILE_DL_MY_TEST) && defined(ZTS) ZEND_TSRMLS_CACHE_UPDATE(); #endif return SUCCESS; } PHP_RSHUTDOWN_FUNCTION(my_test) { return SUCCESS; } PHP_MINFO_FUNCTION(my_test) { php_info_print_table_start(); php_info_print_table_header(2, "my_test support", "enabled"); php_info_print_table_end(); } const zend_function_entry my_test_functions[] = { PHP_FE(my_add, NULL) PHP_FE_END }; zend_module_entry my_test_module_entry = { STANDARD_MODULE_HEADER, "my_test", my_test_functions, PHP_MINIT(my_test), PHP_MSHUTDOWN(my_test), PHP_RINIT(my_test), PHP_RSHUTDOWN(my_test), PHP_MINFO(my_test), PHP_MY_TEST_VERSION, STANDARD_MODULE_PROPERTIES }; #ifdef COMPILE_DL_MY_TEST #ifdef ZTS ZEND_TSRMLS_CACHE_DEFINE() #endif ZEND_GET_MODULE(my_test) #endif
config.m4中取消以下注释(删除 dnl 即可)
dnl PHP_ARG_ENABLE(my_test, whether to enable my_test support, dnl Make sure that the comment is aligned: dnl [ --enable-my_test Enable my_test support])
然后在my_test目录下执行
phpize ./configure --with-php-config=/usr/local/php7.1/bin/php-config
php-config这个脚本是获取PHP安装信息的(PHP安装路径,PHP版本,PHP源码的头文件目录,LDFLAGS,依赖的外部库,PHP编译参数),它在php的安装路径的bin目录下,如果你不指定--with-php-config的话,将到默认的PHP的安装路径下搜索( 安装了多个PHP版本时最好指定一下,可能会编译不通过 ) 然后
make && make install
得到
Installing shared extensions: /usr/local/php7.1/lib/php/extensions/no-debug-zts-20160303/
修改php.ini文件,加入.so
date.timezone = "Asia/Shanghai" display_errors = On error_reporting = E_ALL short_open_tag=Off upload_max_filesize = 50M post_max_size = 50M memory_limit=512M extension=my_test.so
函数调用成功。
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网
猜你喜欢:- 为vscode编写扩展
- (全栈学习实践)三、创建php、添加扩展,其他Dockerfile编写
- DockOne微信分享(二六五):如何基于 OAM 编写一个扩展 Trait?
- 使用 Dingo API 扩展包快速构建 Laravel RESTful API(二) —— 编写第一个 API 接口
- 在项目文件 / MSBuild / NuGet 包中编写扩展编译的时候,正确使用 props 文件和 targets 文件
- 基于顺丰同城接口编写sdk,java三方sdk编写思路
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
Linux程序设计
马修 / 陈健 / 人民邮电出版社 / 2007-7 / 89.00元
《Linux 程序设计(第3版)》讲述在Linux系统及其他UNIX风格的操作系统上进行的程序开发,主要内容包括标准Linux C语言函数库和由不同的Linux或UNIX标准指定的各种工具的使用方法,大多数标准Linux开发工具的使用方法,通过DBM和MySQL数据库系统对Linux中的数据进行存储,为X视窗系统建立图形化用户界面等。《Linux 程序设计(第3版)》通过先介绍程序设计理论,再以适......一起来看看 《Linux程序设计》 这本书的介绍吧!