JVM之编译OpenJDK

Posted zhengshuangxi

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JVM之编译OpenJDK相关的知识,希望对你有一定的参考价值。

学习JVM时看到书里讲到自己编译OpenJDK。记录一下过程

  • Mac系统版本:High Sierra 10.13.6
  • 源码版本:jdk8u-dev

一、准备源码

我是从官网下载网站的,openJDK源码是用mercurial进行管理的,所以首先使用homebrew安装mercurial

brew install mercurial

安装完成之后,再运行命令克隆jdk源码,之后通过运行脚本get_source.sh获取所有的源代码

hg clone http://hg.openjdk.java.net/jdk8u/jdk8u-dev/
cd jdk8u-dev
sh ./get_source.sh 

在获取源码的过程中可能会遇到一些问题,如下所示。我遇到的问题主要是网络的问题,后来通过切换网络,使用手机热点解决了。

abort: stream ended unexpectedly (got 1596 bytes, expected 10164)

二、设置环境变量

在编译开始之前需要设置一些变量。这里与书中所讲的有一些不一样,只因为我用的是Mac平台,所以编译器之类的可能会有一些不一样。可以将下面的内容拷贝建立一个脚本运行。

这里的环境变量设置采用了博客https://www.jianshu.com/p/d9a1e1072f37中的设置

# 语言选项
export LANG=C
# Mac平台,C编译器不再是GCC,是clang
export CC=clang
# 跳过clang的一些严格的语法检查,不然会将N多的警告作为Error
export COMPILER_WARNINGS_FATAL=false
# 链接时使用的参数
export LFLAGS=-Xlinker -lstdc++
# 是否使用clang
export USE_CLANG=true
# 使用64位数据模型
export LP64=1
# 告诉编译平台是64位,不然会按32位来编译
export ARCH_DATA_MODEL=64
# 允许自动下载依赖
export ALLOW_DOWNLOADS=true
# 并行编译的线程数,编译时间长,为了不影响其他工作,我选择为2
export HOTSPOT_BUILD_JOBS=2
# 是否跳过与先前版本的比较
export SKIP_COMPARE_IMAGES=true
# 是否使用预编译头文件,加快编译速度
export USE_PRECOMPILED_HEADER=true
# 是否使用增量编译
export INCREMENTAL_BUILD=true
# 编译内容
export BUILD_LANGTOOLS=true
export BUILD_JAXP=true
export BUILD_JAXWS=true
export BUILD_CORBA=true
export BUILD_HOTSPOT=true
export BUILD_JDK=true
# 编译版本
export SKIP_DEBUG_BUILD=true
export SKIP_FASTDEBUG_BUILD=false
export DEBUG_NAME=debug
# 避开javaws和浏览器Java插件之类的部分的build
export BUILD_DEPLOY=false
export BUILD_INSTALL=false
unset JAVA_HOME

三、编译

设置好环境变量之后,就可以开始编译了,首先运行configure,这个脚本会调用很多其他的脚本进行配置并检查编译环境。

bash ./configure

运行成功后的显示如下:

A new configuration has been successfully created in
/Users/zhengshuangxi/mercurial/temp/build/macosx-x86_64-normal-server-release
using default settings.

Configuration summary:
* Debug level:    release
* JDK variant:    normal
* JVM variants:   server
* OpenJDK target: OS: macosx, CPU architecture: x86, address length: 64

Tools summary:
* Boot JDK:       java version "1.8.0_212" Java(TM) SE Runtime Environment (build 1.8.0_212-b10) Java HotSpot(TM) 64-Bit Server VM (build 25.212-b10, mixed mode)  (at /Library/Java/JavaVirtualMachines/jdk1.8.0_212.jdk/Contents/Home)
* Toolchain:      gcc (GNU Compiler Collection)
* C Compiler:     Version Apple LLVM version 10.0.0 (clang-1000.11.45.5) Target: x86_64-apple-darwin17.7.0 Thread model: posix InstalledDir: /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin (at /usr/bin/clang)
* C++ Compiler:   Version Configured with: --prefix=/Applications/Xcode.app/Contents/Developer/usr --with-gxx-include-dir=/usr/include/c++/4.2.1 Apple LLVM version 10.0.0 (clang-1000.11.45.5) Target: x86_64-apple-darwin17.7.0 Thread model: posix InstalledDir: /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin (at /usr/bin/g++)

Build performance summary:
* Cores to use:   2
* Memory limit:   8192 MB

也有可能会遇到一些问题,这里我遇到的问题有:

  • Xcode版本不符合,需要Xcode 4
  • 无法找到gcc编译器

具体解决办法在本文最后提到。如果没有遇到问题就可以进一步编译了。

直接使用make all命令进行编译

make all

编译成功之后显示如下:

## Finished docs (build time 00:02:31)

----- Build times -------
Start 2019-06-21 16:43:05
End   2019-06-21 16:57:48
00:00:27 corba
00:00:55 demos
00:02:31 docs
00:03:43 hotspot
00:01:34 images
00:00:15 jaxp
00:00:24 jaxws
00:04:10 jdk
00:00:31 langtools
00:00:12 nashorn
00:14:43 TOTAL
-------------------------
Finished building OpenJDK for target all

在编译过程中还可能出现一些问题,目前我只遇到这一个问题:

  • 参数错误“-std=gnu++98”,具体解决办法在本文最后提到。

四、运行

编译完成之后,在build文件夹中会生成一个名为macosx-x86_64-normal-server-release的文件夹,这里就是我们编译生成的结果。在这个文件夹中jdk就是OpenJDK编译成功的产物。

我们可以通过如下命令运行虚拟机,查看版本号,如果没什么问题终端会显示如下信息:

cd ./build/macosx-x86_64-normal-server-release/jdk/bin
./java -version
openjdk version "1.8.0-internal"
OpenJDK Runtime Environment (build 1.8.0-internal-zhengshuangxi_2019_06_21_17_25-b00)
OpenJDK 64-Bit Server VM (build 25.71-b00, mixed mode)

运行之后我又遇到一个问题,在运行过程中会报错,提示 libjvm.dylib 动态链接库相关的错误。具体解决在本文最后。

五、遇到的问题及解决办法

5.1、运行configure之后,提示Xcode版本问题

checking for xcodebuild... /usr/bin/xcodebuild
configure: error: Xcode 4 is required to build JDK 8, the version found was 10.1. Use --with-xcode-path to specify the location of Xcode 4 or make Xcode 4 active by using xcode-select.
configure exiting with result code 1

我们只需要将文件 jdk8u-dev/common/autoconf/generated-configure.sh 进行一些修改,找到以下代码并进行注释,代码大约在26780行左右。注释之后就不会提示版本问题了

    # Fail-fast: verify were building on Xcode 4, we cannot build with Xcode 5 or later
    XCODE_VERSION=`$XCODEBUILD -version | grep ^Xcode  | sed s/Xcode //`
    XC_VERSION_PARTS=( $XCODE_VERSION//./  )
    if test ! "$XC_VERSION_PARTS[0]" = "4"; then
      as_fn_error $? "Xcode 4 is required to build JDK 8, the version found was $XCODE_VERSION. Use --with-xcode-path to specify the location of Xcode 4 or make Xcode 4 active by using xcode-select." "$LINENO" 5
    fi

5.2、gcc编译器找不到

configure: Will use user supplied compiler CC=clang
checking for clang... /usr/bin/clang
checking resolved symbolic links for CC... no symlink
configure: The C compiler (located as /usr/bin/clang) does not seem to be the required gcc compiler.
configure: The result from running with --version was: ""
configure: error: A gcc compiler is required. Try setting --with-tools-dir.
configure exiting with result code 1

解决办法也是在generated-configure.sh文件中注释掉以下的条件代码。注释的地方一共有两处:

# 第一处代码,27936行左右
elif test "x$TOOLCHAIN_TYPE" = xgcc; then # gcc --version output typically looks like # gcc (Ubuntu/Linaro 4.8.1-10ubuntu9) 4.8.1 # Copyright (C) 2013 Free Software Foundation, Inc. # This is free software; see the source for copying conditions. There is NO # warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. COMPILER_VERSION_OUTPUT=`$COMPILER --version 2>&1` # Check that this is likely to be GCC. $ECHO "$COMPILER_VERSION_OUTPUT" | $GREP "Free Software Foundation" > /dev/null # 条件语句注释掉 # if test $? -ne 0; then # $as_echo "$as_me:$as_lineno-$LINENO: The $COMPILER_NAME compiler (located as $COMPILER) does not seem to be the required $TOOLCHAIN_TYPE compiler." >&5 # $as_echo "$as_me: The $COMPILER_NAME compiler (located as $COMPILER) does not seem to be the required $TOOLCHAIN_TYPE compiler." >&6; # $as_echo "$as_me:$as_lineno-$LINENO: The result from running with --version was: \"$COMPILER_VERSION\"" >&5 # $as_echo "$as_me: The result from running with --version was: \"$COMPILER_VERSION\"" >&6; # as_fn_error $? "A $TOOLCHAIN_TYPE compiler is required. Try setting --with-tools-dir." "$LINENO" 5 # fi
# 第二处代码,29677行左右
  elif test  "x$TOOLCHAIN_TYPE" = xgcc; then
    # gcc --version output typically looks like
    #     gcc (Ubuntu/Linaro 4.8.1-10ubuntu9) 4.8.1
    #     Copyright (C) 2013 Free Software Foundation, Inc.
    #     This is free software; see the source for copying conditions.  There is NO
    #     warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
    COMPILER_VERSION_OUTPUT=`$COMPILER --version 2>&1`
    # Check that this is likely to be GCC.
    $ECHO "$COMPILER_VERSION_OUTPUT" | $GREP "Free Software Foundation" > /dev/null
# 条件语句注释掉
#     if test $? -ne 0; then
#        $as_echo "$as_me:$as_lineno-$LINENO: The $COMPILER_NAME compiler (located as $COMPILER) does not seem to be the required $TOOLCHAIN_TYPE compiler." >&5
# $as_echo "$as_me: The $COMPILER_NAME compiler (located as $COMPILER) does not seem to be the required $TOOLCHAIN_TYPE compiler." >&6;
#        $as_echo "$as_me:$as_lineno-$LINENO: The result from running with --version was: \"$COMPILER_VERSION\"" >&5
# $as_echo "$as_me: The result from running with --version was: \"$COMPILER_VERSION\"" >&6;
#       as_fn_error $? "A $TOOLCHAIN_TYPE compiler is required. Try setting --with-tools-dir." "$LINENO" 5
#     fi

注释完之后,重新运行configure

5.3、make过程中遇到参数错误“-std=gnu++98”

Making signal interposition lib...
error: invalid argument -std=gnu++98 not allowed with C
make[6]: *** [libjsig.dylib] Error 1
make[6]: *** Waiting for unfinished jobs....
make[5]: *** [the_vm] Error 2
make[4]: *** [product] Error 2
make[3]: *** [generic_build2] Error 2
make[2]: *** [product] Error 2
make[1]: *** [/Users/zhengshuangxi/mercurial/temp/build/macosx-x86_64-normal-server-release/hotspot/_hotspot.timestamp] Error 2
make: *** [hotspot-only] Error 2

generated-configure.sh文件中找到参数“-std=gnu++98”,将代码注释如下:

      LDFLAGS_JDK="$LDFLAGS_JDK -Wl,-z,relro"
      LEGACY_EXTRA_LDFLAGS="$LEGACY_EXTRA_LDFLAGS -Wl,-z,relro"
    fi
    此处注释掉
    #CXXSTD_CXXFLAG="-std=gnu++98"

之后重新运行configure,再执行make all

5.4、编译时遇到 lstdc++ 库无法找到

ld: library not found for -lstdc++
clang: error: linker command failed with exit code 1 (use -v to see invocation)
make[6]: *** [libsaproc.dylib] Error 1
make[6]: *** Waiting for unfinished jobs....
make[5]: *** [the_vm] Error 2
make[4]: *** [product] Error 2
make[3]: *** [generic_build2] Error 2
make[2]: *** [product] Error 2
make[1]: *** [/Users/zhengshuangxi/mercurial/temp/build/macosx-x86_64-normal-server-release/hotspot/_hotspot.timestamp] Error 2
make: *** [hotspot-only] Error 2

这个原因是Xcode升级到10以后就没有包含lstdc++库了,解决办法是将/usr/lib中的库文件拷贝到Xcode相应的文件夹中,并设置相关的软连接。

sudo cp /usr/lib/libstdc++.6.0.9.dylib /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX.sdk/usr/lib/
cd /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX.sdk/usr/lib/
sudo ln -s libstdc++.6.0.9.dylib libstdc++.6.dylib
sudo ln -s libstdc++.6.dylib libstdc++.dylib

然后重新运行make all命令进行编译。

5.5、编译完成之后,运行虚拟机时出现错误

#
# A fatal error has been detected by the Java Runtime Environment:
#
#  SIGILL (0x4) at pc=0x000000010fe801bf, pid=48116, tid=0x0000000000002203
#
# JRE version: OpenJDK Runtime Environment (8.0) (build 1.8.0-internal-zhengshuangxi_2019_06_21_16_42-b00)
# Java VM: OpenJDK 64-Bit Server VM (25.71-b00 mixed mode bsd-amd64 compressed oops)
# Problematic frame:
# V  [libjvm.dylib+0x4801bf]  PerfDataManager::destroy()+0xab
#
# Failed to write core dump. Core dumps have been disabled. To enable core dumping, try "ulimit -c unlimited" before starting Java again
#
# An error report file with more information is saved as:
# /Users/zhengshuangxi/mercurial/temp/build/macosx-x86_64-normal-server-release/jdk/bin/hs_err_pid48116.log
#
# If you would like to submit a bug report, please visit:
#   http://bugreport.java.com/bugreport/crash.jsp
#

[error occurred during error reporting , id 0x4]

Abort trap: 6

根据这个错误提示,我们可以知道错误发生在动态链接库 libjvm.dylib中,根据提示,错误代码时PerDataManager::destroy()函数中。因此我们需要对这个函数进行修改,即文件 hotspot/src/share/vm/runtime/perfData.cpp,在这个文件中将代码"delete p"注释掉。

问题的解决办法是在博客https://www.jianshu.com/p/34c8a8c37169中看到的。

// 大约在287行
void PerfDataManager::destroy() 

  if (_all == NULL)
    // destroy already called, or initialization never happened
    return;

  for (int index = 0; index < _all->length(); index++) 
    PerfData* p = _all->at(index);
    // 将delete p注释掉
    //delete p;
  

  delete(_all);
  delete(_sampled);
  delete(_constants);

  _all = NULL;
  _sampled = NULL;
  _constants = NULL;

然后重新运行configure以及make all进行编译。之后就不会出现问题了。

以上是关于JVM之编译OpenJDK的主要内容,如果未能解决你的问题,请参考以下文章

JVM之编译OpenJDK

JVM之JIT编译器实战

JVM系列之Java是解释性语言还是编译型语言?

java之JDK,Jvm,JRE

Java 学习笔记之 JVM初识

Clojure 运行原理之编译器剖析