在Windows系统中使用VC9、VC11编译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文件夹进行编译即可。
实践过程中,主要参考如下三篇文章,仅向原作者致谢: