在Windows系统中使用VC9VC11编译32位64位PHP及其扩展
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了在Windows系统中使用VC9VC11编译32位64位PHP及其扩展相关的知识,希望对你有一定的参考价值。
项目中需要使用runkit模块实现AOP,但是团队成员的开发环境都是Windows,而runkit模块官方没有提供Windows环境下的dll扩展,只能自己编译。
下面是编译过程的分类总结。(操作系统环境为Windows 10 64位中文旗舰版)
- php的编译版本
这里的版本指的不是PHP的发行版本,如5.3、7.0,而是指编译时所使用的编译器、程序架构和是否为线程安全。
这些信息可以在phpinfo中打印出来。
上图中所使用的PHP,是使用Miscrosoft Visual 2012中的MSVC11编译的,程序架构为x86(即32位),非线程安全(NTS)。
上图中的PHP,是使用Miscrosoft Visual 2012中的MSVC11编译的,程序架构为x64(即64位),线程安全(TS)。
上图略有不同,没有专门的列表项说明编译器版本、程序架构,只能从编译时的配置中看到相关信息。这里的PHP,是使用Miscrosoft Visual 6中的MSVC6编译的,程序架构为x86(即32位),线程安全(Thread Safety = enabled)。
最后这一张,是使用Miscrosoft Visual 2008中的MSVC9编译的,程序架构为x86(即32位),线程安全(TS)。
本文只讨论MSVC9和MSVC11编译两种情况。 - 准备PHP-SDK和PHP源码
以下步骤无论编译哪种版本的PHP,都是必须的。
- 下载PHP-SDK:
下载地址在http://windows.php.net/downloads/php-sdk/中可以找到,下载其中的php-sdk-binary-tools-20110915.zip - 解压到C:\\php-sdk文件夹
- 打开命令行,执行如下命令:
cd c:\\php-sdk\\
bin\\phpsdk_buildtree.bat phpdev
会在c:\\php-sdk文件夹下生成phpdev文件夹,其中包含vc6、vc8、vc9子文件夹。
如果使用MSVC11编译PHP,就复制c:\\php-sdk\\phpdev\\vc9到c:\\php-sdk\\phpdev\\vc11。
如果使用MSVC14编译PHP,就复制c:\\php-sdk\\phpdev\\vc9到c:\\php-sdk\\phpdev\\vc14。 - 下载相应版本的PHP的编译时依赖包,下载地址见http://windows.php.net/downloads/php-sdk/。
将其解压至相应编译文件夹下的x86或x64文件夹下,覆盖其中的deps文件夹。注意要与要编译的PHP发行版本和程序架构一致,如编译php5.4.x的32位版本则解压deps-5.4-vc9-x86.7z,如编译php5.5.x的64位版本则解压deps-5.6-vc11-x64.7z,依此类推。
php5.4.x和php5.3.x通常需要使用vc9编译,因此其依赖文件解压到c:\\php-sdk\\phpdev\\vc9\\x86或c:\\php-sdk\\phpdev\\vc9\\x64。
php5.5.x和php5.6.x通常需要使用vc11编译,因此其依赖文件解压到c:\\php-sdk\\phpdev\\vc11\\x86或c:\\php-sdk\\phpdev\\vc11\\x64。
而php7.0.x则需要vc14编译,因此其依赖文件解压到c:\\php-sdk\\phpdev\\vc14\\x86或c:\\php-sdk\\phpdev\\vc14\\x64。 - 下载所需要的PHP源文件,下载地址在http://windows.php.net/download/、http://php.net/downloads.php、http://php.net/releases/、http://windows.php.net/downloads/releases/archives/几处都可以找到。
将其解压制相应的编译文件夹下。
例如:
将php-5.4.45.tar.gz解压到c:\\php-sdk\\phpdev\\vc9\\x86\\php-5.4.45或c:\\php-sdk\\phpdev\\vc9\\x64\\php-5.4.45下,以分别编译其32位和64位版本。
将php-5.6.16.tar.gz则解压到c:\\php-sdk\\phpdev\\vc11\\x86\\php-5.6.16或c:\\php-sdk\\phpdev\\vc11\\x64\\php-5.6.16下。 - 如果是编译PHP扩展,则到相应的地址下载源码,本文是以runkit为例,下载地址在http://pecl.php.net/package/runkit。
将下载到的源码解压到PHP源码的ext文件夹下,本例是分别解压到c:\\php-sdk\\phpdev\\vc9\\x86\\php-5.4.45\\ext\\runkit、c:\\php-sdk\\phpdev\\vc9\\x64\\php-5.4.45\\ext\\runkit、c:\\php-sdk\\phpdev\\vc11\\x86\\php-5.6.16\\ext\\runkit、c:\\php-sdk\\phpdev\\vc11\\x64\\php-5.6.16\\ext\\runkit。
- 下载PHP-SDK:
- 准备编译环境:
- 使用MSVC9编译PHP,需要下载安装Windows SDK 6.1,此地址下载的是在线安装版。也可以在http://download.microsoft.com/download/a/4/2/a4282359-1d35-4648-a7a7-d85e3bfe81ac/6.0.6001.16621.148.WindowsSDK_LonghornServer_IDS04_idw.WindowsSDK.DVD.Release.iso下载离线安装包。
如需编译32位版本,在安装Windows SDK 6.1时,要注意选择安装x86编译器,否则会出现找不到cl.exe的问题。 - 使用MSVC11编译PHP,可以下载Visual Studio 2012 Express for Windows Desktop,有在线安装版和离线安装版(ISO)。
- 两个工具不冲突,可以同时安装在同一系统下。
- 使用MSVC9编译PHP,需要下载安装Windows SDK 6.1,此地址下载的是在线安装版。也可以在http://download.microsoft.com/download/a/4/2/a4282359-1d35-4648-a7a7-d85e3bfe81ac/6.0.6001.16621.148.WindowsSDK_LonghornServer_IDS04_idw.WindowsSDK.DVD.Release.iso下载离线安装包。
下面是具体的编译步骤,有些内容会出现重复,是为了让偷懒的同学按图索骥,有样学样:)
- 使用MSVC9编译PHP5.4.33的32位版(本例为编译runkit.dll)
- 安装Windows SDK 6.1,注意安装时选择x86编译器。
- 将下载的php-sdk-binary-tools-20110915.zip解压到c:\\php-sdk。
- 打开Windows命令行,输入如下命令:
cd c:\\php-sdk
bin\\phpsdk_buildtree.bat phpdev - 将下载的php-5.4.45.tar.gz解压到c:\\php-sdk\\phpdev\\vc9\\x86\\php-5.4.45。
- 将下载的deps-5.4-vc9-x86.7z解压到c:\\php-sdk\\phpdev\\vc9\\x86\\deps覆盖原deps文件夹。
- 将下载的http://pecl.php.net/get/runkit-1.0.4.tgz解压到c:\\php-sdk\\phpdev\\vc9\\x86\\php-5.4.45\\ext文件夹,注意文件夹的名称不要带版本号,本例应为runkit,而非runkit-1.0.4。
- 在c:\\php-sdk\\phpdev\\vc9\\x86下建立obj文件夹,用于存放最终的php.exe及相关的dll文件。
- 点击开始菜单,选择Microsoft Windows SDK V6.1>CMD Shell,启动编译命令行。
- 输入如下指令:
setenv /x86 /xp /release(此行命令执行成功,命令行的文字将变为绿色)
cd c:\\php-sdk
bin\\phpsdk_setvars.bat
cd phpdev\\vc9\\x86\\php-5.4.45
buildconf
configure --disable-all --enable-cli --enable-runkit=shared --enable-object-out-dir=..\\obj
如需编译非线程安全版 (nts版),将上面最后一行换成
configure --disable-all --enable-cli --enable-runkit=shared --disable-zts --enable-object-out-dir=..\\obj
最终执行
nmake - 在obj中有Release(线程安全版-ts版)或Release_nts(非线程安全版-nts版)文件夹,其中的php.exe及php-runkit.dll即为编译结果。
- 将php-runkit.dll文件放入PHP安装文件夹的ext文件夹中,修改php.ini,加入extension=php_runkit.dll,重新启动PHP服务即可。
- 使用MSVC9编译PHP5.4.33的64位版(本例为编译runkit.dll)
- 安装Windows SDK 6.1,注意安装时选择x86编译器。
- 将下载的php-sdk-binary-tools-20110915.zip解压到c:\\php-sdk。
- 打开Windows命令行,输入如下命令:
cd c:\\php-sdk
bin\\phpsdk_buildtree.bat phpdev - 将下载的php-5.4.45.tar.gz解压到c:\\php-sdk\\phpdev\\vc9\\x64\\php-5.4.45。
- 将下载的deps-5.4-vc9-x86.7z解压到c:\\php-sdk\\phpdev\\vc9\\x64\\deps覆盖原deps文件夹。
- 将下载的http://pecl.php.net/get/runkit-1.0.4.tgz解压到c:\\php-sdk\\phpdev\\vc9\\x64\\php-5.4.45\\ext文件夹,注意文件夹的名称不要带版本号,本例应为runkit,而非runkit-1.0.4。
- 在c:\\php-sdk\\phpdev\\vc9\\x64下建立obj文件夹,用于存放最终的php.exe及相关的dll文件。
- 点击开始菜单,选择Microsoft Windows SDK V6.1>CMD Shell,启动编译命令行。
- 输入如下指令:与上面编译32位程序明显不同的是,不需要执行 setenv /x86 /xp /release
cd c:\\php-sdk
bin\\phpsdk_setvars.bat
cd phpdev\\vc9\\x64\\php-5.4.45
buildconf
configure --disable-all --enable-cli --enable-runkit=shared --enable-object-out-dir=..\\obj
如需编译非线程安全版 (nts版),将上面最后一行换成
configure --disable-all --enable-cli --enable-runkit=shared --disable-zts --enable-object-out-dir=..\\obj
最终执行
nmake - 在obj中有Release(线程安全版-ts版)或Release_nts(非线程安全版-nts版)文件夹,其中的php.exe及php-runkit.dll即为编译结果。
- 将php-runkit.dll文件放入PHP安装文件夹的ext文件夹中,修改php.ini,加入extension=php_runkit.dll,重新启动PHP服务即可。
- 使用MSVC11编译PHP5.4.33的32位版(本例为编译runkit.dll)
- 安装Visual Studio 2012 Express for Windows Desktop。
- 将下载的php-sdk-binary-tools-20110915.zip解压到c:\\php-sdk。
- 打开Windows命令行,输入如下命令:
cd c:\\php-sdk
bin\\phpsdk_buildtree.bat phpdev - 将下载的php-5.6.16.tar.gz解压到c:\\php-sdk\\phpdev\\vc11\\x64\\php-5.6.16。
- 将下载的deps-5.6-vc11-x64.7z解压到c:\\php-sdk\\phpdev\\vc11\\x64\\deps覆盖原deps文件夹。
- 将下载的http://pecl.php.net/get/runkit-1.0.4.tgz解压到c:\\php-sdk\\phpdev\\vc11\\x64\\php-5.6.16\\ext文件夹,注意文件夹的名称不要带版本号,本例应为runkit,而非runkit-1.0.4。
- 在c:\\php-sdk\\phpdev\\vc11\\x64下建立obj文件夹,用于存放最终的php.exe及相关的dll文件。
- 点击开始菜单,选择Microsoft Visual Studio 2012>Visual Studio Tools>VS2012 x64 Cross Tools Command Prompt,启动编译命令行。
- 输入如下指令:
cd c:\\php-sdk
bin\\phpsdk_setvars.bat
cd phpdev\\vc11\\x64\\php-5.6.16
buildconf
configure --disable-all --enable-cli --enable-runkit=shared --enable-object-out-dir=..\\obj
如需编译非线程安全版 (nts版),将上面最后一行换成
configure --disable-all --enable-cli --enable-runkit=shared --disable-zts --enable-object-out-dir=..\\obj
最终执行
nmake - 在obj中有Release(线程安全版-ts版)或Release_nts(非线程安全版-nts版)文件夹,其中的php.exe及php-runkit.dll即为编译结果。
- 将php-runkit.dll文件放入PHP安装文件夹的ext文件夹中,修改php.ini,加入extension=php_runkit.dll,重新启动PHP服务即可。
- 使用MSVC11编译PHP5.4.33的64位版(本例为编译runkit.dll)
- 安装Visual Studio 2012 Express for Windows Desktop。
- 将下载的php-sdk-binary-tools-20110915.zip解压到c:\\php-sdk。
- 打开Windows命令行,输入如下命令:
cd c:\\php-sdk
bin\\phpsdk_buildtree.bat phpdev - 将下载的php-5.6.16.tar.gz解压到c:\\php-sdk\\phpdev\\vc11\\x86\\php-5.6.16。
- 将下载的deps-5.6-vc11-x86.7z解压到c:\\php-sdk\\phpdev\\vc11\\x86\\deps覆盖原deps文件夹。
- 将下载的http://pecl.php.net/get/runkit-1.0.4.tgz解压到c:\\php-sdk\\phpdev\\vc11\\x86\\php-5.6.16\\ext文件夹,注意文件夹的名称不要带版本号,本例应为runkit,而非runkit-1.0.4。
- 在c:\\php-sdk\\phpdev\\vc11\\x86下建立obj文件夹,用于存放最终的php.exe及相关的dll文件。
- 点击开始菜单,选择Microsoft Visual Studio 2012>Visual Studio Tools>VS2012 x86 Native Tools Command Prompt,启动编译命令行。
- 输入如下指令:
cd c:\\php-sdk
bin\\phpsdk_setvars.bat
cd phpdev\\vc11\\x86\\php-5.6.16
buildconf
configure --disable-all --enable-cli --enable-runkit=shared --enable-object-out-dir=..\\obj
如需编译非线程安全版 (nts版),将上面最后一行换成
configure --disable-all --enable-cli --enable-runkit=shared --disable-zts --enable-object-out-dir=..\\obj
最终执行
nmake - 在obj中有Release(线程安全版-ts版)或Release_nts(非线程安全版-nts版)文件夹,其中的php.exe及php-runkit.dll即为编译结果。
- 将php-runkit.dll文件放入PHP安装文件夹的ext文件夹中,修改php.ini,加入extension=php_runkit.dll,重新启动PHP服务即可。
可能遇到的问题:
- 如果遇到There is no script engine for file extension ".js"错误,说明Windows的jscript引擎没有正确注册,可按如下步骤解决:
- 以管理员身份打开命令行:
- 执行如下指令:
regsvr32 %WINDIR%\\System32\\jscript.dll
regsvr32 %WINDIR%\\SysWOW64\\jscript.dll - 生成一个jscript.reg的文件,内容如下:
Windows Registry Editor Version 5.00
[HKEY_CLASSES_ROOT\\.js]
@="jsfile"[HKEY_CLASSES_ROOT\\.js\\PersistentHandler]
@="{5e941d80-bf96-11cd-b579-08002b30bfeb}"[HKEY_CLASSES_ROOT\\JSFile]
"FriendlyTypeName"=hex(2):40,00,25,00,53,00,79,00,73,00,74,00,65,00,6d,00,52,\\
00,6f,00,6f,00,74,00,25,00,5c,00,53,00,79,00,73,00,74,00,65,00,6d,00,33,00,\\
32,00,5c,00,77,00,73,00,68,00,65,00,78,00,74,00,2e,00,64,00,6c,00,6c,00,2c,\\
00,2d,00,34,00,38,00,30,00,34,00,00,00[HKEY_CLASSES_ROOT\\JSFile\\ScriptEngine]
@="JScript"[HKEY_CLASSES_ROOT\\JSFile\\ScriptHostEncode]
@="{85131630-480C-11D2-B1F9-00C04F86C324}"[HKEY_CLASSES_ROOT\\JSFile\\Shell]
@="Open"[HKEY_CLASSES_ROOT\\JSFile\\Shell\\Open\\Command]
@=hex(2):25,00,53,00,79,00,73,00,74,00,65,00,6d,00,52,00,6f,00,6f,00,74,00,25,\\
00,5c,00,53,00,79,00,73,00,74,00,65,00,6d,00,33,00,32,00,5c,00,57,00,53,00,\\
63,00,72,00,69,00,70,00,74,00,2e,00,65,00,78,00,65,00,20,00,22,00,25,00,31,\\
00,22,00,20,00,25,00,2a,00,00,00[HKEY_CLASSES_ROOT\\JSFile\\Shell\\Open2]
@=hex(2):4f,00,70,00,65,00,6e,00,20,00,26,00,77,00,69,00,74,00,68,00,20,00,43,\\
00,6f,00,6d,00,6d,00,61,00,6e,00,64,00,20,00,50,00,72,00,6f,00,6d,00,70,00,\\
74,00,00,00
"MUIVerb"=hex(2):40,00,25,00,53,00,79,00,73,00,74,00,65,00,6d,00,52,00,6f,00,\\
6f,00,74,00,25,00,5c,00,53,00,79,00,73,00,74,00,65,00,6d,00,33,00,32,00,5c,\\
00,77,00,73,00,68,00,65,00,78,00,74,00,2e,00,64,00,6c,00,6c,00,2c,00,2d,00,\\
34,00,35,00,31,00,31,00,00,00[HKEY_CLASSES_ROOT\\JSFile\\Shell\\Open2\\Command]
@=hex(2):25,00,53,00,79,00,73,00,74,00,65,00,6d,00,52,00,6f,00,6f,00,74,00,25,\\
00,5c,00,53,00,79,00,73,00,74,00,65,00,6d,00,33,00,32,00,5c,00,43,00,53,00,\\
63,00,72,00,69,00,70,00,74,00,2e,00,65,00,78,00,65,00,20,00,22,00,25,00,31,\\
00,22,00,20,00,25,00,2a,00,00,00[HKEY_CLASSES_ROOT\\JSFile\\Shell\\Print\\Command]
@=hex(2):25,00,53,00,79,00,73,00,74,00,65,00,6d,00,52,00,6f,00,6f,00,74,00,25,\\
00,5c,00,53,00,79,00,73,00,74,00,65,00,6d,00,33,00,32,00,5c,00,4e,00,6f,00,\\
74,00,65,00,70,00,61,00,64,00,2e,00,65,00,78,00,65,00,20,00,2f,00,70,00,20,\\
00,25,00,31,00,00,00[HKEY_CLASSES_ROOT\\JSFile\\ShellEx\\DropHandler]
@="{60254CA5-953B-11CF-8C96-00AA00B8708C}"[HKEY_CLASSES_ROOT\\JSFile\\ShellEx\\PropertySheetHandlers\\WSHProps]
@="{60254CA5-953B-11CF-8C96-00AA00B8708C}"[HKEY_CLASSES_ROOT\\JScript]
@="JScript Language"[HKEY_CLASSES_ROOT\\JScript\\CLSID]
@="{f414c260-6ac0-11cf-b6d1-00aa00bbbb58}"[HKEY_CLASSES_ROOT\\JScript\\OLEScript]
- 双击该文件,点击确定,将其导入注册表即可。
- 如果在使用MSVC9编译32位PHP时,遇到Checking for cl.exe ... <not found>错误,就表示在安装Windows SDK 6.1时,忘记选择x86编译器。
PS:
- 以上的步骤中,其实有很多是不需要按部就班的,尤其是文件夹结构。c:\\php-sdk,其实可以放在任意盘符下。只要在开始编译前,执行一上:x:\\php-sdk\\bin\\php-setvars.bat就可以了(x:为php-sdk所在盘符)。
- phpdev及其下的各级文件夹也是不需要的,尤其是x86和x64文件夹,并不是放错了就不行的,那只是方便辨识最后编译的结果。只要将deps和php源码放在同一个文件夹下,比如在d:建立php-source文件夹,其下放置desp、obj和php-5.6.16子文件夹(嗯,obj放哪儿也无所谓,在编译时直接将输出结果的位置指向到obj就行了),进入php-5.6.16文件夹进行编译即可。
实践过程中,主要参考如下三篇文章,仅向原作者致谢:
- Build your own PHP on Windows
- windows环境下php和Php扩展编译,扩展dll文件编译
- 解决Win7下There is no script engine for file extension ".js"
以上是关于在Windows系统中使用VC9VC11编译32位64位PHP及其扩展的主要内容,如果未能解决你的问题,请参考以下文章
php版本选择问题.VC6和VC9的区别,TS和NTS如何选择