想要深入瞭解JVM,就必須瞭解其實現機制。瞭解JVM實現的最好方法便是自己動手編譯JDK。好了,讓我們開始吧! 1. 準備工作 獲取OpenJDK源碼 獲取OpenJDK源碼 本次編譯選擇的是OpenJDK7u,官方源碼包:https://jdk7.java.net/source.html 系統需求 ...
想要深入瞭解JVM,就必須瞭解其實現機制。瞭解JVM實現的最好方法便是自己動手編譯JDK。好了,讓我們開始吧!
1. 準備工作
- 獲取OpenJDK源碼
本次編譯選擇的是OpenJDK7u,官方源碼包:https://jdk7.java.net/source.html
- 系統需求
為了提高效率,儘量選擇Linux 或 MacOS作為編譯平臺。本次使用Ubuntu12.04進行編譯。仔細閱讀源碼包中README-builds.html文檔,就可以構建編譯環境了。
2. 配置編譯環境
- 編譯依賴
OpenJDK包括虛擬機Hotsport | JDK API | JAXWS | JAXP等。需要各種編譯依賴,包括C++,C的編譯環境,編譯Java的JDK(稱為Bootstrap JDK),還有用於執行java代碼的Ant腳本等等。這些依賴在Linux中都可以通過命令一次安裝完成。
sudo apt-get install build-essential gawk m4 libasound2-dev libcups-dev libxrender-dev xorg-dev xutils-dev x11proto-print-dev binutils libmotif3 libmotif-dev ant
當然,也可以在命令裡面加上openjdk-6-jdk,但是由於openjdk在後面的編譯中出現了bug,所以還是建議大家安裝Oracle jdk。註意,bootstrap JDK版本必須在6以上。
- 環境變數
OpenJDK在編譯時會讀取許多環境變數,所以必須對Linux的環境變數進行配置。使用VIM編輯/etc/profile。 vim /etc/profile
具體在profile中添加的環境變數如下
export LANG=C #BootStrap-JDK的安裝路徑,替換為自己bootstrap-JDK的路徑 export ALT_BOOTDIR=/usr/lib/jvm/java-6-openjdk-i386 #同上,我之前使用的是openjdk編譯的,後面運行hotspot時出現問題替換為oracleJDK,讀者可以直接替換為OracleJDK export ALT_JDK_IMPORT_PATH=/usr/lib/jvm/java-6-openjdk-i386 #要編譯的內容,讀者可以根據需要自行選擇 export BUILD_LANGTOOLS=true #export BUILD_JAXWS=false #export BUILD_JAXP=false #export BUILD_CORBA=false export BUILD_HOTSPOT=true export BUILD_JDK=true export SKIP_COMPARE_IMAGES=true BUILD_DEPLOY=false BUILD_INSTALL=false #編譯結果存放的路徑,建議存放在openjdk源碼中build文件夾 export ALT_OUTPUTDIR=/usr/dev/jvm/openjdk/build export ALLOW_DOWNLOADS=true #這兩個環境變數需要去掉,不然會出問題 unset JAVA_HOME unset CLASSPATH
添加完成後,進入openjdk源碼路徑,通過make sanity命令來檢查設置是否正確,如果正確,會返回Sanity check passed。
3. 開始編譯OpenJDK
- 使用make命令
在openjdk目錄下,輸入make命令,正常情況下大概需要30分鐘左右,具體速度根據機器性能決定。編譯正常結束後,會出現日誌清單展示內容,如圖
- 查看編譯結果
進入build/j2sdk-image目錄下,查看整個JDK的編譯結果,運行java –version
4. 運行HotSpot
- 編輯env.sh
虛擬機的輸出結果存放在build/hotspot/outputdir/linux_i486_compiler2路徑下,如圖
使用VIM編輯product目錄下的env.sh
我們發現裡面已經有了JAVA_HOME CLASSPATH HOTSPOT_BUILD_USER等環境變數,我們還需要添加一個LD_LIBRARY_PATH,否則在運行時還會出現問題。
LD_LIBRARY_PATH=.:${JAVA_HOME}/jre/lib/i386/native_threads:${JAVA_HOME}/jre/lib/i386:
export LD_LIBRARY_PATH
- 執行命令啟動JVM
使用如下命令,啟動虛擬機,輸出版本號
. ./env.sh ./gamma –vesion
成功結果如圖
至此成功編譯運行OpenJDK7,下麵講講過程中遇到的問題。
5. 編譯運行過程中可能會遇到的問題
- OpenJDK6.0的bug
使用OpenJDK6.0作為bootstrap JDK的話,在編譯及運行過程中可能會出現類似於這樣的錯誤,運行./gamma時也可能出現,這類錯誤都屬於OpenJDK-6中的bug
relocation error: /usr/lib/jvm/java-6-openjdk-i386/jre/lib/i386/libjava.so: symbol JVM_FindClassFromCaller, version SUNWprivate_1.1 not defined in file libjvm.so with link time reference
網上提供的一種解決方案是通過find –name 'CurrencyData.properties' 找到CurrencyData文件,把文件中的時間全部修改為10年以內。然而我嘗試了這種方法並不能解決問題,於是嘗試把Bootstrap JDK由openjdk-6更換為OracleJDK6,終於解決問題。
- 未取消掉JAVA_HOME變數
如果沒有在環境變數中添加unset JAVA_HOME,make Sanity時會出現以下錯誤
ERROR: Your JAVA_HOME environment variable is set. This will most likely cause the build to fail. Please unset it and start your build again. Exiting because of the above error(s). make: *** [post-sanity] Error 1
在/etc/profile中添加unset JAVA_HOME以解決問題
還有許多在編譯過程中遇到的問題,在前文中已經進行彌補和完善,相信大家按照這些步驟進行編譯,會省去不少麻煩。
6. 總結
通過自己動手編譯OpenJDK-7的源碼,我們可以深入瞭解JVM的編譯環境以及運行過程。通過解決編譯過程中遇到的問題,為後續對JVM的深入探索打下了良好的基礎。在此基礎上,我們還可以通過NetBeans對Hotspot進行運行和調試,進一步瞭解JVM源碼的結構與細節。