看了深入理解Java虚拟机:JVM高级特性与最佳实践(第2版) 之后决定自己编译一下OpenJDK,但是书中介绍的关于编译和调试部分已经过时了。所以根据搜索的资料,自己编译调试了一下。
IDE:Netbeans 8.2(最新版本即可)
操作系统:Ubuntu 14.04 (内核3.13.0)
make版本:3.81
源代码:OpenJDK 8
我是直接下载打包好的源代码(使用hg下载还需要再安装hg,这个东西相对来说还是太小众了),地址在这里
唯一需要注意一点的就是Bootstrap JDK
的版本需要低于要编译的JDK的版本。即如果要编译OpenJDK 8,那么Bootstrap JDK
的版本需要为7。
依赖安装没有必要复制别人的,你完全可以运行bash ./configure
文件,它会提示你哪些依赖没有安装。反复执行几次,直至成功。
如果你是编译OpenJDK 8,那么环境变量部分也可以直接忽略。这种方式是编译之前版本的方式。从OpenJDK 8开始,改为了"configure && make" style build
。
make all
NetBeans
调试其实OpenJDK 8里边已经有针对NetBeans
生成好的项目文件,我们没有必要像有些文章或者书里边所说的创建新的C++项目然后导入源代码这么麻烦,直接打开即可。
路径在./common/nb_native
打开之后,注意将configuration切换为”Linux_64”
其实我们也可以略过上边的
make all
,用NetBeans
打开后直接编译。效果是一样的。
文章NetBeans 调试 openjdk8里边提到,需要在构建命令后边加上参数DEBUG_BINARIES=true
。经过实际测试发现,加不加这个都不会影响调试,所以可以忽略这个参数。
像很多文章中所举的例子,调试java
程序。这时需要选对可以调试的程序,我们需要选取./build/linux-x86_64-normal-server-release/jdk/bin
下的可执行程序,而不是./build/linux-x86_64-normal-server-release/images
下的子目录里边的可执行程序。因为后者是release模式编译的,无法触发断点调试。
我在Ubuntu 14.04中编译时(make版本为3.81),没有遇到问题,一次成功。
但是在Ubuntu 16.04中,遇到了好几个问题,如下。
这是OpenJDK中的一个bug。它会在编译时检查Linux的内核版本,之前的检查代码没有检查4.x版本(那个时候还没有这个版本的内核),导致出错。我们只需要在对应的检查代码里加上即可。
文件hotspot/make/linux/Makefile
-SUPPORTED_OS_VERSION = 2.4% 2.5% 2.6% 3%
+SUPPORTED_OS_VERSION = 2.4% 2.5% 2.6% 3% 4%
具体可以参考:
adjust-mflags.sh failed build with GNU Make 4.0 with -I
在make 4.x版本中会出现这个问题。
修改hotspot/make/linux/makefiles/adjust-mflags.sh
文件
@@ -64,7 +64,7 @@
echo "$MFLAGS" \
| sed '
s/^-/ -/
- s/ -\([^ ][^ ]*\)j/ -\1 -j/
+ s/ -\([^ I][^ I]*\)j/ -\1 -j/
s/ -j[0-9][0-9]*/ -j/
s/ -j\([^ ]\)/ -j -\1/
s/ -j/ -j'${HOTSPOT_BUILD_JOBS:-${default_build_jobs}}'/
如果mac(bsd)或者solaris平台上编译,则需要修改对应平台的代码。
路径将上面路径中的linux
改为bsd
或者solaris
即可。
具体可以参考:
Running nasgen Exception in thread "main" java.lang.VerifyError
Running nasgen Exception in thread "main" java.lang.VerifyError: class jdk.nashorn.internal.objects.ScriptFunctionImpl overrides final method setPrototype.(Ljava/lang/Object;)V
解决方法,修改nashorn/make/BuildNashorn.gmk
文件,
$(CP) -R -p $(NASHORN_OUTPUTDIR)/nashorn_classes/* $(@D)/
$(FIXPATH) $(JAVA) \
- -cp "$(NASHORN_OUTPUTDIR)/nasgen_classes$(PATH_SEP)$(NASHORN_OUTPUTDIR)/nashorn_classes" \
+ -Xbootclasspath/p:"$(NASHORN_OUTPUTDIR)/nasgen_classes$(PATH_SEP)$(NASHORN_OUTPUTDIR)/nashorn_classes" \
jdk.nashorn.internal.tools.nasgen.Main $(@D) jdk.nashorn.internal.objects $(@D)
具体可以参考:
如果在高版本上编译,即使将上面遇到的问题一一解决,当运行编译出来的Java时还是会提示运行失败,错误信息如下:
fimh@ubuntu:~/Dev/openjdk/build/linux-x86_64-normal-server-release/jdk/bin$ ./java -version
Error: dl failure on line 864
Error: failed /home/fimh/Dev/openjdk/build/linux-x86_64-normal-server-release/jdk/lib/amd64/server/libjvm.so, because /home/fimh/Dev/openjdk/build/linux-x86_64-normal-server-release/jdk/lib/amd64/server/libjvm.so: undefined symbol: _ZN23G1SATBCardTableModRefBS24write_ref_array_pre_workIP7oopDescEEvPT_i
看起来是链接错误…
即使将make版本换为老版本(如3.81)依然是上面的问题
所以目前暂时还是使用旧的环境进行编译: