virtualbox编译记录

参考文档

机器环境及之前已有库

  • windows 10 x64
  • Visual Studio 2015 Community 版本
  • Windows Driver Development Kit (WDK) v7.1
  • Qt5.7.1
  • Python 2.7.8
  • Nsis 3.0b1
  • JDK7 JDK8
  • zlib 在 D:\thridparty\zlib
  • VirtualBox 69891版本

下载工具和库

下载Virtualbox推荐的链接

  • mingw64下载解压到D:\projects\virtualbox\mingw64

mingw32

以下文件都解压D:\projects\virtualbox\mingw32

openssl

其实下Light版本应该就够了

其它库

  • libxml2 virtualbox已含
  • zlib virtualbox已含
  • 其它可选库都不要

check out代码

编译curl

将以下指令保存到curl根目录的批处理文件中,并执行:

build_dll.bat

@echo off
@cd winbuild
@echo 正在使用release模式编译libcurl~~~
@nmake /f Makefile.vc WITH_DEVEL=D:\VC_INCLUDE\OpenSSL-Win32 WITH_ZLIB=D:\thridparty\zlib mode=dll VC=14 WITH_SSL=dll ENABLE_IDN=no RTLIBCFG=dll DEBUG=no MACHINE=x86
@REM @echo on
@cd ..

build_dll_x64.bat

@echo off
@cd winbuild
@echo 正在使用release模式编译libcurl~~~
@nmake /f Makefile.vc WITH_DEVEL=D:\VC_INCLUDE\OpenSSL-Win64 WITH_ZLIB=D:\thridparty\zlib mode=dll VC=14 WITH_SSL=dll ENABLE_IDN=no RTLIBCFG=dll DEBUG=no MACHINE=x64
@REM @echo on
@cd ..

编译输出在

  • D:\thridparty\curl\builds\libcurl-vc14-x64-release-dll-ssl-dll-ipv6-sspi
  • D:\thridparty\curl\builds\libcurl-vc14-x86-release-dll-ssl-dll-ipv6-sspi

因为virtualbox查找curl的lib和dll时只在指定目录查找,所以还需要将他们子目录下的bin和lib目录里的文件拷贝到父级目录。

编译Virtulbox

LocalConfig.kmk

代码根目录新建文件LocalConfig.kmk,内容:

VBOX_WITH_ADDITIONS=

VBOX_WITH_ADDITIONS_PACKING=
#Don't build + package the VirtualBox Guest Additions. If only VBOX_WITH_ADDITIONS= is specified then the Guest Additions are not built but the VBoxGuestAdditions.iso file is expected to be present in the 'out\win.ARCH\release\bin\additions' directory.

VBOX_ONLY_ADDITIONS=1
#Build the Guest Additions exclusively.

VBOX_WITH_VALIDATIONKIT=
#Don't build + package the VirtualBox validation kit. The validation kit is not part of the final .msi package anyway.

VBOX_WITH_WEBSERVICES=
#Don't build + package the webservices API server.

VBOX_WITHOUT_HARDENING=1

VBOX_INF2CAT=D:\WinDDK\7600.16385.1\bin\selfsign\Inf2Cat.exe
VBOX_PATH_SIGN_TOOLS=D:\WinDDK\7600.16385.1\bin\amd64\SignTool.exe

# 以下的我不用,只是机器上有,就放上来
VBOX_PATH_NSIS=D:\Program Files\NSIS
VBOX_ZIP=D:\VC_INCLUDE\GetGnuWin32\gnuwin32\bin\zip.exe
VBOX_PATH_WISUMINFO=D:\projects\Microsoft SDKs\Windows\v7.1\Samples\sysmgmt\msi\scripts\WiSumInf.vbs
VBOX_MKISOFS=C:\Program Files (x86)\VMware\VMware Workstation\mkisofs.exe

configure.vbs

configure.vbs中修改代码

if   (InStr(1, g_strShellOutput, "Version 16.") <= 0) _
And (InStr(1, g_strShellOutput, "Version 17.") <= 0) then
    MsgError "The Visual C++ compiler we found ('" & strPathVC & "') isn't 10.0 or 11.0. Check the build requirements."
    exit sub
end if

if   (InStr(1, g_strShellOutput, "Version 16.") <= 0) _
    And (InStr(1, g_strShellOutput, "Version 17.") <= 0) _
    And (InStr(1, g_strShellOutput, "Version 18.") <= 0) _
    And (InStr(1, g_strShellOutput, "19.00") <= 0) then
    MsgError "Check Version failed: The Visual C++ compiler we found ('" & strPathVC & "') isn't 10.0 - 14.0. Check the build requirements."
    exit sub
end if

用来支持vs2015

build.bat

根目录新建build_x64.bat:

@echo off
set cur_dir= %~dp0%
call D:\projects\virtualbox\kBuild\envwin.cmd

cscript %cur_dir%configure.vbs --with-vc="C:\Program Files (x86)\Microsoft Visual Studio 14.0" --with-MinGW-w64=D:\projects\virtualbox\mingw64 --with-SDK="C:\Program Files (x86)\Microsoft SDKs\Windows\v7.1A" --with-DDK=D:\VC_INCLUDE\Windows\WinDDK  --with-Qt5=E:\Qt\Qt5.7.1\5.7\msvc2015_64 --with-python=D:\Python27 --with-libcurl32=D:\thridparty\curl\builds\libcurl-vc14-x86-release-dll-ssl-dll-ipv6-sspi --with-libcurl=D:\thridparty\curl\builds\libcurl-vc14-x64-release-dll-ssl-dll-ipv6-sspi --with-openssl32="D:\VC_INCLUDE\OpenSSL-Win32" --with-openssl="D:\VC_INCLUDE\OpenSSL-Win32" --disable-SDL --target-arch=amd64   
pause

执行之,然后再在命令行中分别执行:

env
kmk

问题

kmk: *** Invalid number "ev". Stop.

执行kmk后,出现这个问题,然后我无论是切换virtualbox的版本还是切换kbuild的版本,最后安装了vs2010版本。结果还是报这个错误。
最后挨个文件夹执行kmk,最后发现在virtualbox\trunk\src\recompiler中kmk提示这个错误。
打开这个目录的Makefile.kmk后搜索ev,只搜索到DVBOX_SVN_REV,就想起来既然makefile中使用到了svn版本号,而我用的这个库又是git的,那问题会不会出现在svn版本号生成的地方。后面又在kmk输出的out\revision.kmk里发现了export VBOX_SVN_REV=ev这样一来问题就很明显了。我只需要在生成svn版本号的地方改一下代码即可,但是手改config.kmk中的svn版本号分析实在是太烦了,所以就直接在LocalConfig.kmk中添加一行VBOX_SVN_REV=69807即可解决。

找不到 <k/kLdr.h>

D:\projects\virtualbox\trunk\src\VBox\Runtime\common\ldr\ldrkStuff.cpp中有#include <k/kLdr.h>,但是代码库里没有。在老版本的文件列表里倒是能找到。
最后没办法,先屏蔽掉它吧。在LocalConfig.kmk中添加一行IPRT_WITH_KSTUFF=这样D:\projects\virtualbox\trunk\src\VBox\Runtime\Makefile.kmk中就不会包含kdrkStuff.cpp了。

ifdef IPRT_WITH_KSTUFF
RuntimeR3_SOURCES += \
    common/ldr/ldrkStuff.cpp
endif

后面发现,可以从http://www.virtualbox.org/svn/kstuff-mirror下载kstuff的源码,所以这个又能编译的过

spu_dispatch_table.h

D:\projects\virtualbox\trunk\src\VBox\GuestHost\OpenGL\include\cr_spu.h(16) : fatal error C1083: 无法打开包括文件:“spu_dispatch_table.h”: No such file or directory

查看MakeFile发现D:\projects\virtualbox\trunk\src\VBox\Additions\common\crOpenGL\Makefile.kmk中用到D:\projects\virtualbox\trunk\src\VBox\GuestHost\OpenGL\Makefile.kmk中生成的spu_dispatch_table.h。而OpenGL\Makefile.kmk还没有执行。

并且在https://www.virtualbox.org/wiki/Windows%20build%20instructions中有提到:
This step will fail for a 64-bit (amd64) target if the Guest Additions are part of the build process (which is the default, disable by adding VBOX_WITH_ADDITIONS= and VBOX_WITH_ADDITIONS_PACKING= to LocalConfig.kmk, see below). It will complain about a dependency to VBoxOGL* libraries in out\win.x86\release\bin\additions. In that case, create the 32-bit Guest Additions by executing。

所以,先在LocalConfig.kmk中加入 VBOX_WITH_ADDITIONS= and VBOX_WITH_ADDITIONS_PACKING= 跳过addtions的编译

无法解析的外部符号 isprint

用2015编译过程中,大量的link错误,报错都是:

bin2c.obj : error LNK2019: 无法解析的外部符号 isprint,该符号在函数 main 中被引用
bin2c.obj : error LNK2019: 无法解析的外部符号 __acrt_iob_func,该符号在函数 main 中被引用
bin2c.obj : error LNK2019: 无法解析的外部符号 fclose,该符号在函数 main 中被引用

报错的程序太多,构建脚本又太复杂,经过摸索使用bin2c这个cpp来突破,bin2c不依赖于其它virtualbox或三方库代码,单独使用命令行cl即可编译生成出exe。

经研究,virtualbox在编译命令行中加入了-Zl,于是用cl -Zl bin2c.c 编译,果然直接失败,但输出obj后在link时添加c运行时库能成功。

// -Zl https://msdn.microsoft.com/zh-cn/library/f1tbxcxh.aspx。 ps:-Zl这是小写L,不是数字1.
// 可以使用 /Zl 来编译打算放入库中的 .obj 文件。 虽然省略库名只为单个 .obj 文件节省了少量空间,但在包含许多对象模块的库中节省的总空间是很多的。
cl.exe  -Zl -Fobin2c.obj bin2c.c
然后使用 link.exe bin2c.obj libcmt.lib 可以链接通过。

但从Config.kmk中看到,编译时还是带上了libcmt.lib的,比如:

VBoxRT_LIBS.win.x86 = \
    $(PATH_TOOL_$(VBOX_VCC_TOOL_STEM)X86_LIB)/oldnames.lib \
    $(PATH_TOOL_$(VBOX_VCC_TOOL_STEM)X86_LIB)/libcmt.lib \
    $(PATH_TOOL_$(VBOX_VCC_TOOL_STEM)X86_LIB)/libcpmt.lib

但还是链接,从日志文件中提取出来最后的line命令行,并且精简如下:link.exe /NOD bin2c.obj libcmt.lib,其中去掉这个/NOD就会链接成功。msdn上并没有该选项,仅有一个/NODEFAULTLIB 在解析外部引用时忽略所有(或指定的)默认库。,通过实验它们两是效果相等的。

但同样是添加了/NOD选项,用vs2010编译就不会失败,而用2015就失败。只好先去掉/NOD这个选项。

kmk_builtin_kDepObj 删除sanity-c相关文件失败

kmk: *** [D:/projects/virtualbox/trunk/out/obj/RuntimeBldProg/common/misc/sanity-c.obj] Error 512 (0x200)
The failing command:
@kmk_builtin_kDepObj -f -s -q -o D:/projects/virtualbox/trunk/out/obj/RuntimeBldProg/common/misc/sanity-c.obj.dep -t D:/projects/virtualbox/trunk/out/obj/RuntimeBldProg/common/misc/sanity-c.obj D:/projects/virtualbox/trunk/out/obj/RuntimeBldProg/common/misc/sanity-c.obj
kmk: *** Deleting file `D:/projects/virtualbox/trunk/out/obj/RuntimeBldProg/common/misc/sanity-c.obj'
kmk: *** Waiting for unfinished jobs....
kmk: *** Exiting with status 2

vs2015下才有此bug,原因未知,先换用2010继续编译

资源编译失败

D:/projects/virtualbox/mingw32/bin/windres: unknown format type `D:/projects/virtualbox/trunk/include'
D:/projects/virtualbox/mingw32/bin/windres: supported formats: rc res coff

原始命令行:

D:/projects/virtualbox/mingw32/bin/windres    -ID:/projects/virtualbox/trunk/include -ID:/projects/virtualbox/trunk/out_vs2010 -IC:/PROGRA~2/MICROS~1/Windows/v7.1A/Include -IC:/PROGRA~2/MICROS~4.0/VC/include     -DVBOX_SVN_REV=69891     -DVBOX_SVN_REV_MOD_5K=19891     D:/projects/virtualbox/trunk/src/recompiler/VBoxREM.rc D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxREMImp/VBoxREMRes.o

查看windres的帮助,发现下载的virtualbox推荐mingw32里的windres是GNU windres (GNU Binutils) 2.25,-I参数不是输入文件,而是指格式。

Usage: D:\projects\virtualbox\mingw32\bin\windres.exe [option(s)] [input-file] [output-file]
 The options are:
  -i --input=<file>            Name input file
  -o --output=<file>           Name output file
  -I --input-format=<format>   Specify input format
  -O --output-format=<format>  Specify output format    

再查看其它几个windres发现是-I --input-format=<format> Specify input format,于是就从其它地方找了windres替换掉这个。

dllwrap.exe 报错 undefined reference to `__dyn_tls_init_callback' 1

/mingw/lib/dllcrt2.o(.text+0xd1): undefined reference to `__dyn_tls_init_callback` 1

# 命令行是
D:/projects/virtualbox/mingw32/bin/dllwrap.exe  -s                         --dllname=D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/VBoxREM32.dll --output-exp=D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/VBoxREM32.exp --output-lib=D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/VBoxREM32.a  --def D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxREMImp/VBoxREMWin.def  D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/VBoxRecompiler.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/cpu-exec.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/exec.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/translate-all.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/host-utils.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/cutils.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/tcg-runtime.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/tcg/tcg.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/tcg/tcg-dyngen.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/fpu/softfloat-native.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/target-i386/op_helper.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/target-i386/helper.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxRemPrimary/target-i386/translate.o D:/projects/virtualbox/trunk/out_vs2010/obj/VBoxREMImp/VBoxREMRes.o  -LD:/projects/virtualbox/mingw32/lib   D:/projects/virtualbox/trunk/out_vs2010/lib/VBoxVMM.lib   D:/projects/virtualbox/trunk/out_vs2010/lib/VBoxRT.lib

神奇的是,去掉-LD:/projects/virtualbox/mingw32/lib或者将目录改成我原来的D:/Mingw/lib就能正常生成dll。所以我觉得是virtualbox推荐的mingwruntime有问题,所以我去下载了mingwrt-3.22.4版本,也能正常编译。

QT“void (__thiscall UISnapshotPane::* )(void)”转换为“const char *”

D:\projects\virtualbox\trunk\src\VBox\Frontends\VirtualBox\src\selector\UISnapshotPane.cpp(1273) : error C2664: “QAction *QToolBar::addAction(const QIcon &,const QString &,const QObject *,const char *)”: 不能将参数 4 从“void (__thiscall UISnapshotPane::* )(void)”转换为“const char *”
        没有使该转换得以执行的上下文

事实证明一开始偷懒下载的Qt5.5.1不支持virtualbox使用的头文件,Qt5.6.x之后有Q_QDOC宏包括起来的几个头文件。于是下载5.6.2版本重新编译。这里不能下载5.7.x版本,因为5.7.x版本的不支持vs2010编译了。

在config.kmk中的第5933行中发现还有对VBOX_WITH_ORACLE_QT宏的判断,这个宏最终编译QT这些DLL的virtualBox版本。但我没有搞懂(也没有时间)怎么才能像官网版本那样编译出来Qt5CoreVBox.dll、Qt5GuiVBox.dll、Qt5OpenGLVBox.dll、Qt5PrintSupportVBox.dll、Qt5WidgetsVBox.dll、Qt5WinExtrasVBox.dll这几个文件。

src/libs/xpcom18a4/python/gen_python_deps.py: No Python development package found!

其实virtualbox已经能运行了,但是在研究VBoxFB时使用了新的宏,编译时提示错误。
kmk.exe VBOX_WITH_VBOXFB=1 VBOX_WITH_XPCOM=1

但看具体代码,似乎VBoxFB在非win系统下使用的。所以这个就跳过了。

安装启动

解决以上bug后,编译都顺利通过,然后我还继续使用vs2010编译了64位的qt。然后编译了64位的virtualbox。

然后将编译中用到的libcurl和openssl相关dll放到bin目录。 libcrypto-1_1-x64.dll、libcurl.dll、libssl-1_1-x64.dll,还有QT相关的6个dll,Qt5Core.dll、Qt5Gui.dll、Qt5OpenGL.dll、Qt5PrintSupport.dll、Qt5Widgets.dll、Qt5WinExtras.dll。然后各种程序启动不报错,但是启动还没有效果。

需要按照官网上build instructions网页上的说法:

  1. 需要先注册com进程,以管理员权限运行:comregister.cmd。
  2. loadall.cmd 安装驱动,但驱动(.sys结尾的7个文件)需要先签名。
  3. 然后就能启动VirtualBox.exe了。

SUPInstall.exe: error: installation failed. rc=VERR_UNRESOLVED_ERROR

在第2步安装驱动时,提示:
+ D:/projects/virtualbox/out_vs2010_x64/bin/SUPInstall.exe
SUPInstall.exe: error: installation failed. rc=VERR_UNRESOLVED_ERROR

将所有驱动文件生成对应的cat文件,并且将cat和sys签名即可。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容

  • 最近因需要尝试编译了一次Vbox,官方论坛上只有32位环境的编译教程,所以这次在64位、特别是中文环境上遇到不少坑...
    MushiUta阅读 5,653评论 0 6
  • SVN版本:1.5 及更新版本 名词说明: WC:Working Copy 你的工作区 Versioned:受控的...
    日风和阅读 4,264评论 1 23
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,594评论 18 139
  • 一般企业SVN配置:TortoiseSVN和VisualSVN Server黄金组合,最主要是免费的Tortois...
    西门丨不吹雪阅读 1,944评论 1 5
  • 秋草萧疏雁南飞, 丹枫迎雨化寒随; 西风残酒余香醉, 叶落空山踏云归。
    月夜秋荷阅读 190评论 1 3