最近准备阅读JVM源码,打算在本机编译一个Debug版本的JDK,鉴于工作中主要使用JDK7,因此选择OpenJdk7源码进行编译;
源码下载
访问 https://jdk7.java.net/source.html, 获取源码下载地址;
编译准备
解压openjdk-7u40-fcs-src-b43-26_aug_2013.zip包,关于编译的准备可以参考README和 README-builds.html文件;
运行make sanity
检查环境是否准备好,通常而言,要做好如下的工作:
- 按照FreeType,要求2.3.0或更高版本;
brew install freetype
另外需要建立软链接
sudo ln -s /usr/X11/include/X11 /usr/include/X11
在建立软链接时,一直报没权限,郁闷了,都是root用户还没权限,到网上查了资料,发现在EI (10.11)下,引入了SIP(System Integrity Protection)机制,可以在恢复模式下(开机时按Command+R进入)通过csrutil disable命令关闭SIP模式;
关于brew,可以访问http://brew.sh/网站,获取更多信息;
- 安装xquartz,在JDK的文件中需要引用X11的头文件;例如src/solaris/native/sun/xawt/XWindow.c:
#ifdef HEADLESS
#error This file should not be included in headless library
#endif
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/Xos.h>
#include <X11/Xatom.h>
#include <ctype.h>
- 安装Ant,另外需要为ant命令建立链接:
jdk/src/macosx/native/jobjc/JObjC.xcodeproj/project.pbxproj和jdk/make/java/jobjc/Makefile文件中ant路径写死为/usr/bin/ant;
ln -s ant /usr/bin/ant
- 安装JDK,编译源码依赖原有的jdk,默认是jdk6;如果使用jdk7,需要对部分参数做出调整,osx平台对应修改bsd目录下的文件,将JDK版本改为7,编码改为utf-8:
- 修改hotspot/make/bsd/makefiles/rules.make文件
BOOT_SOURCE_LANGUAGE_VERSION = 6
BOOT_TARGET_CLASS_VERSION = 6
JAVAC_FLAGS = -g -encoding ascii
- 修改corba/make/common/shared/Defs-java.gmk文件
JAVACFLAGS += -encoding ascii
- 设置环境变量
unset JAVA_HOME
export LANG=C #设定语言选项,必须设置
export CC=clang #mac平台,c编译器不再是GCC,而是clang
export COMPILER_WARNINGS_FATAL=false#跳过clang一些严格的语法检查
export USE_CLANG=true #使用clang
export LP64=1 #编译平台是64位
export BUILD_DEPLOY=false
export ARCH_DATA_MODEL=64 #使用64位数据模型
export LFLAGS='-Xlinker -lc++ -lstdc++'#链接使用的参数
export ALT_BOOTDIR=/Library/Java/JavaVirtualMachines/jdk1.7.0_79.jdk/Contents/Home
export BOOTDIR=/Library/Java/JavaVirtualMachines/jdk1.7.0_79.jdk/Contents/Home
export ANT_HOME=/Users/allan/Downloads/apache-ant-1.8.1
export SKIP_DEBUG_BUILD=false
export SKIP_FASTDEBUG_BUILD=true
export DEBUG_NAME=debug
export ALT_FREETYPE_HEADERS_PATH=/usr/local/Cellar/freetype/2.6_1/include/freetype2
export ALT_FREETYPE_LIB_PATH=/usr/local/Cellar/freetype/2.6_1/lib
export HOTSPOT_BUILD_JOBS=8
export USE_PRECOMPILED_HEADER=true
export SHOW_ALL_WARNINGS=false
export INCREMENTAL_BUILD=true
-
新的clang编译器不支持fpch-deps选项:
修改hotspot/make/bsd/makefiles/gcc.make文件:#ifneq ("${CC_VER_MAJOR}", "2") #DEPFLAGS = -fpch-deps -MMD -MP -MF $(DEP_DIR)/$(@:%=%.d) #endif DEPFLAGS = -MMD -MP -MF $(DEP_DIR)/$(@:%=%.d) ifeq ($(USE_CLANG),) ifneq ($(CC_VER_MAJOR), 2) DEPFLAGS += -fpch-deps endif endif
链接llvm-gcc和llvm-g++:
我的xcode版本为7.3.1,xcode目录下已经没有llvm-gcc和llvm-g++程序,而build jdk的时候会用到,所以需要link过去
sudo ln -s /usr/bin/llvm-g++ /Applications/Xcode.app/Contents/Developer/usr/bin/llvm-g++
sudo ln -s /usr/bin/llvm-gcc /Applications/Xcode.app/Contents/Developer/usr/bin/llvm-gcc
- 10年问题:
make/tools/src/build.tools.generatecurrencydata/GenerateCurrencyData
有这么一句判断:
if (Math.abs(time - System.currentTimeMillis()) > ((long) 10) * 365 * 24 * 60 * 60 * 1000) {
throw new RuntimeException("time is more than 10 years from present: " + time);
}
需要修改jdk/src/share/classes/java/util/CurrencyData.properties,将类似AZ=AZM;2005-12-31-20-00-00;AZN
的时间修改为距离编译日期小于10年,比如我统一修改为2015-12-31-20-00-00;
- 非空函数不返回值问题:
- 修改
jdk/src/solaris/native/java/net/net_util_md.c
117和119行,将CHECK_NULL(c)改为CHECK_NULL_RETURN(c , 0); - 修改
jdk/src/macosx/native/sun/awt/AWTEvent.m
385行:
将** if (uchr == nil) { return; }修改为 if (uchr == nil) { return 0; }**
- 权限问题:
修改jdk/src/macosx/native/jobjc/src/core/PrimitiveCoder.hs文件权限为为755:
chmod 755 jdk/src/macosx/native/jobjc/src/core/PrimitiveCoder.hs
-
clang编译器不支持garbage collection:
jdk中部分的代码是用OC写的,作为本地库供Java调用。clang做为OC的编译器,早期版本支持垃圾回收,但之后的版本已不再支持,因此编译OC代码的时候出现了问题。解决的方法是在Xcode的工程文件里关掉对垃圾回收的支持:- 进入jdk/src/macosx/native/jobjc/目录,用Xcode打开项目工程文件,在build settings里面,用level视图清除GCC_ENABLE_OBJC_GC项的所有内容。
浮点数超出范围问题:
编译build/macosx-x86_64-debug/JObjC.build/src/jobjc/com/apple/jobjc/appkit/AppKitFramework.java文件出错:
public final float NSEventDurationForever(){ return 1.797693134862316E+308f; }
1.797693134862316E+308f超出了浮点数的范围,发现该文件的产生流程为:
- jdk/src/macosx/native/jobjc/bridgesupport.gmk文件调用gen_bridge_metadata命令生成系统库的XML桥接数据;
- 产生的xml文件为build/macosx-x86_64-debug/stable_bridge_metadata/AppKitFull.bridgesupport;
<enum name='NSEventDurationForever' value='1.797693134862316E+308'/>
- 根据xml文件产生OC代码;
了解流程之后,可以通过如下方式绕过去:
- 首先不做任何修改,进行编译,系统产生xml和oc文件,编译oc报错;
- 修改make文件bridgesupport.gmk,修改54行:
#all: $(GEN_BRIDGESUPPORT)
all:
这样,执行的时候all将什么都不做,也就是说将不会产生xml文件;
- 修改AppKitFull.bridgesupport文件:
<enum name='NSEventDurationForever' value='3.40282E+38'/>
- 输入命令
make debug_build
再次编译
- 形参默认值问题:
修改hotspot/src/share/vm/code/relocInfo.hpp374行,去掉默认值;具体代码修改可以参考这 - test_gamma报错:
没找到原因,个人认为由于我原来的JDK为oracle JDK7,而编译出的为OpenJDK7,所以导致了这个问题;
我采用的做法是注释
build/macosx-x86_64-debug/hotspot/outputdir/bsd_amd64_compiler2/jvmg/test_gamma中的测试代码
#./${GAMMA_PROG} -Xbatch -showversion Queens < /dev/null
Hotspot调试
可以通过GDB调试,进入编译好的
build/macosx-x86_64-debug/hotspot/outputdir/bsd_amd64_compiler2/jvmg目录,通过如下命令进行调试:
./hotspot -gdb Queens
注:Queens是java class文件名称,具体可以参考这篇文章.
另外mac下gdb的安装可以通过brew
sudo brew installl gdb
可以参考这篇文章为gdb建立代码签名;