一、前言
最近在看《深入java虚拟机》,看完后,打算自己实际编译一个jvm出来看看,实践一下。
书上提到了Oracle JDK和OpenJdk的关系,Oracle Jdk7 和OpenJdk 7共用了相当多的代码,所以还是很有学习的必要的:
二、环境
我这里的编译的操作系统是CentOS Linux release 7.4.1708 (Core) (通过 cat /etc/centos-release查看)。
编译环境相当重要,因为编译openjdk的过程中,网上的文章里,大家遇到的各个问题都不一样,就是因为操作系统不一致。
我主要参考了某位园友的博客:https://www.cnblogs.com/gotodsp/p/8975387.html
该博客里有他用到的工具的百度网盘链接。大家可自行下载。我是下载了之后,通过centos中安装的lrzsz命令来上传的。
命令为rz(上传),sz(下载)。
没安装的话,执行 yum install lrzsz
下面是另一个参考的博客:
http://blog.51cto.com/wangyahui/1612838
我这里引用了不少原博客的内容,但是中间还是有很多问题。我这里会补充说明。
2.1 卸载原jdk
rpm -qa | grep java
yum -y remove XXXX --XXXX是第一条命令查出来的jdk信息
2.2 编译相关的工具
yum -y install gcc gcc-c++ alsa-lib alsa-lib-devel libXrender libXrender-devel libXi-devel libXt-devel libXtst-devel cups cups-devel
2.3 FreeType
注意:我这边,会把所有工具,最后都解压到/usr/local目录下。先上一张我最后成功了的图:
tar -zxvf freetype-2.4.0.tar.gz -C /usr/local
cd /usr/local/freetype-2.4.0
./configure && make && make install # 编译安装
##注意:如果安装中出现以下错误
rmdir /usr/local/include/freetype2/freetype/internal
rmdir: failed to remove `/usr/local/include/freetype2/freetype/internal': No such file or directory
make: [install] Error 1 (ignored)
/usr/bin/install -c -m 644 ./builds/unix/ft2unix.h \
/usr/local/include/ft2build.h
/usr/bin/install -c -m 644 ./builds/unix/ftconfig.h \
/usr/local/include/freetype2/freetype/config/ftconfig.h
执行以下命令处理:
mkdir -p /usr/local/include/freetype2/freetype/internal
重新安装
./configure && make && make install
2.4 Apache Ant
tar -zxvf apache-ant-1.9.7-bin.tar.gz -C /usr/local # 解压
ln -s /usr/local/apache-ant-1.9.7/bin/ant /usr/bin/ant # 软链接到bin
2.5 BootstrapJDK
上传到你的虚拟机上后,
chmod +x jdk-6u45-linux-x64.bin
./jdk-6u45-linux-x64.bin # 解压
mv jdk1.6.0_45 /usr/local/ # 移动目录
然后 vim /etc/profile,配置:
#java environment
export JAVA_HOME=/usr/local/jdk1.6.0_45
export CLASSPATH=.:${JAVA_HOME}/jre/lib/rt.jar:${JAVA_HOME}/lib/dt.jar:${JAVA_HOME}/lib/tools.jar
export PATH=$PATH:${JAVA_HOME}/bin
然后执行 source /etc/profile.
再执行java -version试试,看看是不是生效。
2.6 OpenJdk
unzip openjdk-7-fcs-src-b147-27_jun_2011.zip # 解压
原文章中说得有点乱,不管怎样,做成下面我这种目录效果就行:
三、准备环境变量
vim /etc/profile #在末尾加以下内容。注意其中的目录的路径,另外,先不要忙着执行,先把这一节看完:
export ANT_HOME=/usr/local/apache-ant-1.9.7
export ALT_FREETYPE_HEADERS_PATH=/usr/local/include/freetype2
export ALT_FREETYPE_LIB_PATH=/usr/local/lib
export ALT_DROPS_DIR=/usr/local/src/openjdk/drop
export ALT_BOOTDIR=/usr/local/jdk1.6.0_45
export ALT_JDK_IMPORT_PATH=/usr/local/jdk1.6.0_45
export ALT_OUTPUTDIR=/usr/local/openjdk_output
export LANG=C
export HOTSPOT_BUILD_JOBS=8
export ALT_PARALLEL_COMPILE_JOBS=8
export SKIP_COMPARE_IMAGES=true
export USE_PRECOMPILED_HEADER=true
export ALLOW_DOWNLOADS=true
export SKIP_DEBUG_BUILD=false
export SKIP_FASTDEBUG_BUILD=true
export DEBUG_NAME=debug
export BUILD_LANGTOOLS=true
export BUILD_HOTSPOT=true
export BUILD_JDK=true
export BUILD_DEPLOY=false
export BUILD_INSTALL=false
unset JAVA_HOME
unset CLASSPATH
unset LD_LIBRARY_PATH
这里依次解释下各个选项:
- ANT_HOME # 编译工具ant的路径
- ALT_FREETYPE_HEADERS_PATH # freetype2头文件安装目录,一般无需替换
- ALT_FREETYPE_LIB_PATH # 这个也无需修改
- ALT_DROPS_DIR #这个目录很重要,下图来自参考的那位园友的博客里的:
- ALT_BOOTDIR #启动jdk的路径
- ALT_JDK_IMPORT_PATH #看文章里是配成一致的,暂时还不知道配成不一致的结果
- ALT_OUTPUTDIR #最终编译输出的位置,来个直观图展示下
- LANG ##语言选项,这个必须设置,否则编译好后会出现一个HashTable的NPE错
- HOTSPOT_BUILD_JOBS #并行编译的线程数,设置为和CPU内核数量一致即可
- ALT_PARALLEL_COMPILE_JOBS #并行编译的线程数,设置为和CPU内核数量一致即可
- SKIP_COMPARE_IMAGES #比较本次build出来的映像与先前版本的差异。这对我们来说没有意义,必须设置为true,否则sanity检查会报缺少先前版本JDK的映像的错误提示。
- USE_PRECOMPILED_HEADER #使用预编译头文件,不加这个编译会更慢一些
- ALLOW_DOWNLOADS #允许自行下载依赖
- SKIP_DEBUG_BUILD #要编译的版本
- SKIP_FASTDEBUG_BUILD 要编译的版本
- BUILD_LANGTOOLS # 是否编译语言工具
- BUILD_HOTSPOT #编译虚拟机
- BUILD_JDK 编译jdk
- BUILD_DEPLOY #把它设置为false可以避开javaws和浏览器Java插件之类的部分的build
BUILD_INSTALL #把它设置为false就不会build出安装包。因为安装包里有些奇怪的依赖,但即便不build出它也已经能得到完整的JDK映像,所以还是别build它好了
unset JAVA_HOME unset CLASSPATH unset LD_LIBRARY_PATH #这三个环境变量必须去掉,不然会有很诡异的事情发生(我没有具体查过这些"诡异的#事情",Makefile脚本检查到有这2个变量就会提示警告)
设置完了后,保存,刷新一下。 source /etc/profile
四、检查
在/usr/local/src/openjdk下运行 make sanity。
如果最后一行显示:
Sanity check passed.
则表示检查通过。
五、编译jvm
在同上目录下,执行 make DISABLE_HOTSPOT_OS_VERSION_CHECK=ok
之所以加DISABLE_HOTSPOT_OS_VERSION_CHECK=ok,是因为遇到了下面的异常:
ERROR:××× recipe for target ‘check_os_version’ failed ×××
为了跳过版本校验,所以加了上述参数。
参考了:https://blog.csdn.net/desiyonan/article/details/80801830
好了,接下来静静等待,我最后那一次成功的编译,花了大概20-30分钟,成功了,会有如下显示(突然发现前面tree了一把,刷屏了,只能网上去找个图了。。。):
(别人家怎么那么快,10分钟。。。)
我这边还是贴一个我这边编译好的虚拟机的version显示吧:
如果你没那么幸运,发生了各种error,导致make中断的话,不要走开。参考下面一节。
六、遇到的问题
ps:如果看到提示某个文件报错,又不知道在哪,大家直接用find / -name abc.txt查找吧。
1、Error: time is more than 10 years from present: 1136059200000
通过修改CurrencyData.properties文件, 把10年之前的时间修改为10年之内即可
Index: /usr/openjdk/jdk/src/share/classes/java/util/CurrencyData.properties
注意,该文件内有5,6处要修改的地方。不要漏了,漏了又是10+分钟。。
2、 /usr/bin/ld: cannot find -lstdc++
Linking vm...
/usr/bin/ld: cannot find -lstdc++
collect2: error: ld returned 1 exit status
/usr/bin/chcon: cannot access 'libjvm.so': No such file or directory
ERROR: Cannot chcon libjvm.so
/usr/bin/objcopy --only-keep-debug libjvm.so libjvm.debuginfo
/usr/bin/objcopy: 'libjvm.so': No such file
make[4]: *** [libjvm.so] Error 1
make[4]: Leaving directory `/usr/src/openjdk/hotspot/build/hotspot_debug/linux_amd64_compiler2/debug'
make[3]: *** [the_vm] Error 2
make[3]: Leaving directory `/usr/src/openjdk/hotspot/build/hotspot_debug/linux_amd64_compiler2/debug'
make[2]: *** [debug] Error 2
make[2]: Leaving directory `/usr/src/openjdk/hotspot/build/hotspot_debug'
make[1]: *** [generic_build2] Error 2
make[1]: Leaving directory `/usr/src/openjdk/hotspot/make'
make: *** [debug] Error 2
解决:
yum search libc++
yum install libstdc++-static
3、不识别'-mimpure-text'参数
报错:gcc: error: unrecognized command line option '-mimpure-text'
make[5]: *** [/openjdk/build/linux-amd64/../linux-amd64-debug/lib/amd64/libverify.so] Error 1
解决:
就在 openjdk源码中的,/make/common/shared/Compiler-gcc.gmk 中去掉 -mimpure-text 命令
4、constantPoolOop.cpp
Error:/openjdk/hotspot/src/share/vm/oops/constantPoolOop.cpp:272:39: error: converting 'false' to pointer type 'methodOop' [-Werror=conversion-null]
将return false 改为return NULL
5、__LEAF 头文件重复定义问题
/openjdk/hotspot/src/share/vm/runtime/interfaceSupport.hpp:430:0: error: "__LEAF" redefined [-Werror]
#define __LEAF(result_type, header)
/usr/include/x86_64-linux-gnu/sys/cdefs.h:42:0: note: this is the location of the previous definition
# define __LEAF , __leaf__
在interfaceSupport.hpp代码的最前面增加:(后面的\也是需要的,宏定义中的换行符)
#ifdef __LEAF
#undef __LEAF
#define __LEAF(result_type, header) \
TRACE_CALL(result_type, header) \
debug_only(NoHandleMark __hm;) \
/* begin of body */
#endif
6、/hotspot/src/share/vm/opto/loopnode.cpp:896:49: error: converting 'fals' to pointer type 'Node*' [-Werror=conversion-null]
解决:修改openjdk/hotspot/src/share/vm/opto/loopnode.cpp: 第896行 return false改为return (Node*)false; 或者 return NULL;
7、无法下载包问题 Redirection detected from https to http. Protocol switch unsafe, not allowed
解决:其实前面已经提到了,网盘里提供了jdk6-jaf-b20.zip、jdk6-jaxp-b20.zip、jdk6-jaxws-b20.zip三个文件即为了解决该问题。将这三个文件复制到openjdk/drop
目录下。
同时要配置ALT_DROPS_DIR指向该目录。
七、总结
今天搞这个+写博客,差不多大半天了。。。
一千个人编译jdk,就会有一千个错误。。。
关键是等得又久,错了又得重来。。。不过再怎么说,还是很有成就感的。
看了书,就得实践,毕竟老祖宗说的:纸上得来终觉浅,绝知此事要躬行。