Mac OSX 10.11 编译Openjdk 9
以下只是个人记录仅供参考
编译时间:2016年05月12日
系统版本:
Boot JDK: java version "1.8.0_40" Java(TM) SE Runtime Environment (build 1.8.0_40-b27) Java HotSpot(TM) 64-Bit Server VM
Toolchain: clang (clang/LLVM from Xcode 7.0.1)
C Compiler: Version 7.0.0 (at /usr/bin/clang)
C++ Compiler: Version 7.0.0 (at /usr/bin/clang++)
编译前准备
先大致看一下官网的要求介绍
https://wiki.openjdk.java.net/display/Build/Supported+Build+Platforms
这里xcode显示6.1的会后需要安装freetype,我查了一下FreeType库是一个完全免费(开源)的、高质量的且可移植的字体引擎。
Other JDK 9 build platforms
开始下载一系列准备工具
1、openjdk代码由hg管理,因此需要下载一个mercurial工具
$brew install mercurial
如果你不知道brew,只能说mac用的不够溜。
$sudo brew link mercurial
记住要保证hg version不低于2.6.3
2、安装ccache提高编译速度
$brew install ccache
3、安装freetype,上文有提到
$brew install freetype
获取源码
$hg clone http://hg.openjdk.java.net/jdk9/jdk9 YourOpenJDK
$cd YourOpenJDK
$bash ./get_source.sh
由于国内对hg的网络支持不太好,失败的几率很高,一直下载失败,又懒得弄VPN(所以我没尝试VPN)。只能改下get_source.sh脚本,让它失败了自动重新获取直到成功。将脚本最后部分改成如下就可以了。
# Get clones of all absent nested repositories (harmless if already exist)
sh ./common/bin/hgforest.sh clone "$@"
while [ $? -ne 0 ]
do
sh ./common/bin/hgforest.sh clone "$@"
done
# Update all existing repositories to the latest sources
sh ./common/bin/hgforest.sh pull -u
while [ $? -ne 0 ]
do
sh ./common/bin/hgforest.sh pull -u
done
然后断断续续下了几天终于下下来了。
最后一步编译
输入一下命令激活后会生成一个编译配置。
$ bash ./configure --enable-debug --with-target-bits=64
然后执行make就可以了
$make
这之中可能会遇到一些奇奇怪怪的问题,我的情况很简单。
error这之中提到了两个文件
/usr/local/openjdk/hotspot/src/jdk.hotspot.agent/macosx/native/libsaproc/MacosxDebuggerLocal.m和/usr/include/sys/ptrace.h
一个是openjdk里的,一个是sys里的。定位到代码。
ptrace.h
#ifndef _SYS_PTRACE_H_
#define _SYS_PTRACE_H_
#include <sys/appleapiopts.h>
#include <sys/cdefs.h>
enum {
ePtAttachDeprecated __deprecated_enum_msg("PT_ATTACH is deprecated. See PT_ATTACHEXC") = 10
};
#define PT_TRACE_ME 0 /* child declares it's being traced */
#define PT_READ_I 1 /* read word in child's I space */
#define PT_READ_D 2 /* read word in child's D space */
#define PT_READ_U 3 /* read word in child's user structure */
#define PT_WRITE_I 4 /* write word in child's I space */
#define PT_WRITE_D 5 /* write word in child's D space */
#define PT_WRITE_U 6 /* write word in child's user structure */
#define PT_CONTINUE 7 /* continue the child */
#define PT_KILL 8 /* kill the child process */
#define PT_STEP 9 /* single step the child */
#define PT_ATTACH ePtAttachDeprecated /* trace some running process */
#define PT_DETACH 11 /* stop tracing a process */
#define PT_SIGEXC 12 /* signals as exceptions for current_proc */
#define PT_THUPDATE 13 /* signal for thread# */
#define PT_ATTACHEXC 14 /* attach to running process with signal exception */
#define PT_FORCEQUOTA 30 /* Enforce quota for root */
#define PT_DENY_ATTACH 31
#define PT_FIRSTMACH 32 /* for machine-specific requests */
__BEGIN_DECLS
int ptrace(int _request, pid_t _pid, caddr_t _addr, int _data);
__END_DECLS
#endif /* !_SYS_PTRACE_H_ */
MacosxDebuggerLocal.m
static bool ptrace_attach(pid_t pid) {
int res;
if ((res = ptrace(PT_ATTACH, pid, 0, 0)) < 0) {
print_error("ptrace(PT_ATTACH, %d) failed with %d\n", pid, res);
return false;
} else {
return ptrace_waitpid(pid);
}
这个错误很简单,就是macosDebuggerLocal.m在调用ptrace_attach函数是报了一个错,出错代码为
res = trace(PT_ATTACH, pid, 0, 0) > 0
再看一眼声明
PT_ATTACH这个值是一个枚举值,唯独这个不一样如下
#define PT_ATTACH ePtAttachDeprecated /* trace some running process */
enum {
ePtAttachDeprecated __deprecated_enum_msg("PT_ATTACH is deprecated. See PT_ATTACHEXC") = 10
};
从意思来看,这个是用来追踪线程的,而报的错也跟上面的字符串一样。因此我把bool函数直接返回true试了一个(系统希望返回的值不是true就是false,试试嘛),结果是可以了,也没报其他错了,但是这样改有点虚,退了一步既然这个枚举值应该为10,那我就改成10吧,就是不发送按个消息了。结果是可以了。编译通过。
验证一下
openjdk版本现在还没用到openjdk,不知道这样的验证方法是不是严谨,如果不,用到的时候再处理吧。。。
后来我验证了一下,那个错误消息虽然发出了,但是好像还是能够获得java -version的结果,好像没什么影响