背景
由于最近想分析几个native方法,所以需要手头有一个可以debug的JDK,因此,这两天折腾了折腾,踩了10+个坑,看了10+篇文章,尝试了10+次,最后总算把JDK给编出来了,当在自己编译出来的JDK上运行javac -version输出了那熟悉的文字后,感觉已是老泪纵横还是很有成就感的。
前期准备
- 了解一下OpenJDK的相关知识: 我们要编译的是
Open JDK 9u - 安装
Homebrew(可选) - 番羽土啬: 因为假如编译出错需要查询错误的解决方案时需要通过一个不存在的网站来进行
- 准备一个好用的源码编辑器: 因为运气好的话[滑稽]我们可能需要修改
JDK源码中的八阿哥(我用的 mscode) - 安装一个
Boot JDK: 因为JDK中有不少代码是Java写的,所以想要编译JDK就需要用到另一个JDK来辅助(鸡生蛋、蛋生鸡/自己拽自己的头发把自己拽起来),我安装的是Oracle官方的JDK - 安装
freetype:这是个依赖项brew install freetype - 安装
Xcode
环境描述
- 操作系统:
macOS High Sierra Version: 10.13.3 (17D47) Boot JDK:1.8.0_161freetype:2.9Xcode:Version 9.2 (9C40b)Open JDK源码:jdk9u-2e265b4b8622corba-2ef36e70f490hotspot-bb73b31e70e3jaxp-95a71f690b44jaxws-f4f878b5f01cjdk-a779673ab57dlangtools-e2bf77b3f002nashorn-fb3f7ae74bf6
前人蹚过的坑/我蹚过的坑
- 不要编译
JDK 8: 因为编译8需要Xcode 4现在Xcode版本已远高于4了(这个坑是前人蹚出来的,当然我不信邪,然后自己蹚了一遍,然后发现确实是个坑) - 尽量不要使用
Mercurial来下Open JDK的源码,而用浏览器直接下载打包好的源码: 因为你要用Mercurial下载了Open JDK的顶层工程后,还需要执行其中的get_source.sh来下载其子工程的源码,这个过程漫长而且失败率高(关键还没有执行百分比提示,干等),用浏览器自己下载所有子工程的压缩包要快很多,而且成功率100%(我自己蹚出来的,前人的几篇文章中全告诉我用Mercurial,结果坑了) - 编译
JDK 9u而不是JDK 9因为我当时编译9的时候出了一堆error,所以我想带个u的是不是会好些,当然这个有很大程度是我臆测的,我后期就都拿9u搞了,没试9,如果各位有兴趣可以自己蹚蹚这个 configure的时候一定要带上--disable-warnings-as-errors这个参数,否则编译过程中的warning也会中断编译的进程,实际上这些warning并不影响编译后的目标JDK的运行
走流程
如果前面几点 该准备的你都准备了,该注意的你也注意了,那么开始走流程编译吧,如果报错(有很大几率会)不要摔键盘,用前面我们提到的不可描述存在的网站来找答案,我自己摸着石头过河,反复试了2天才搞定,鲁迅先生说的好:“只要功夫深,铁杵磨成针”,祝你好运!!
step0
所有前文提到的准备工作
step1



step2
在源码顶层目录上执行sh configure --with-debug-level=slowdebug --disable-warnings-as-errors --with-freetype-include=/usr/local/Cellar/freetype/2.9/include/freetype2 --with-freetype-lib=/usr/local/Cellar/freetype/2.9/lib
(带版本号的地方自己注意根据实际情况换)如果这一步没问题的话应当看到这样的输出:

step3
如果你的编译环境以及源码版本跟我的完全一样,那么先别make all,先来修一个bug:(patch在此)
这个bug会导致make报这个:error: ordered comparison between pointer and zero ('char *' and 'int') 。。。此处省略
打开hotspot目录中的
src/share/vm/memory/virtualspace.cpp搜索其中if (base() > 0) {改为if (base() != NULL) {src/share/vm/opto/lcm.cpp搜索其中if (Universe::narrow_oop_base() > 0) {改为if (Universe::narrow_oop_base() != NULL) {src/share/vm/opto/loopPredicate.cpp搜索其中assert(rng->Opcode() == Op_LoadRange || _igvn.type(rng)->is_int() >= 0, "must be");改为assert(rng->Opcode() == Op_LoadRange || iff->is_RangeCheck() || _igvn.type(rng)->is_int()->_lo >= 0, "must be");
step4
make all
编译时间略长,我的机子是2017款的 MacBook Pro大概用了20分钟+(没仔细计),而且编译过程中风扇狂转,笔记本的话注意剩余电量以及散热,编译若成功,指定javac全路径运行一下javac -version看看

如果报了error,具体error具体分析,因为编译环境不同,可能error也不同,去不存在的网站找找解决方案,耐心+运气你就一定能成功的。
总结
原以为编一个JDK能有多难,事实证明:是不难,但是坑多,麻烦,前前后后试了10多次,一共花了两天时间,从尝试编译JDK 8再到9中间出了各样问题,之中一度想放弃,但是由于想用它来实现更远的目标,所以还是坚持弄下来了,最终在搜索引擎和前人的帮助下成功了,所以还是那句话:“只要功夫深铁杵磨成针”,按正确的方向,反复尝试,终能成功。
参考资料
Building OpenJDK 8 on Mac OS X Mavericks
Building and Packaging OpenJDK9 for OSX
Compilation errors with clang-4.0
OpenJDK / jdk10 / jdk10 / hotspot changeset 13502:316854ef2fa2
Ordered comparison between pointer and zero ('char *' and 'int') error #5
openjdk code compilation/ IDE setup
xcode-select active developer directory error
Mac OsX Sierra - Stuck on binaryTreeDictionary.hpp compilation