前言
一直是在source insight上直接看android源碼,可是一直沒有調(diào)試過,而調(diào)試才能對(duì)代碼了解的更加深刻,所以就想到了自己編譯rom,然后動(dòng)態(tài)調(diào)試各個(gè)系統(tǒng)Service。這篇文章記錄了2天的忐忑的編譯歷程。
環(huán)境
手機(jī):nexus6p(已經(jīng)解鎖)
電腦:ubuntu 16.04(ubuntu-16.04.2-desktop-amd64.iso),
提醒:
硬盤剩余容量100G以上(第一次安裝時(shí)linux分配了80G,結(jié)果編譯到90%左右時(shí)提示 backend i/o error,最后只好格掉硬盤重新安裝linux)。
準(zhǔn)備工作
1 翻墻上網(wǎng),公司網(wǎng)絡(luò)提供了翻墻服務(wù),所以這一步我就省了,公司網(wǎng)絡(luò)還不錯(cuò),一個(gè)晚上就把代碼下載下來.
2 gmail郵箱,在下載源碼的過程中,谷歌為了防止匿名連接過多,對(duì)同一個(gè)ip的訪問次數(shù)進(jìn)行了限制,這就導(dǎo)致我們下載過程中很可能會(huì)失敗,因此我們需要用一個(gè)谷歌賬號(hào)(也就是Gmail郵箱)來進(jìn)行驗(yàn)證。
下載源碼
1. 安裝git
sudo apt-get install git
2. 配置git
這里用戶名可以隨便寫,但是郵箱請(qǐng)務(wù)必使用你的Gmail郵箱,因?yàn)樯婕暗较旅娴脑L問驗(yàn)證
git config --global user.name "Your Name" git config --global user.email "Your Gmail@gmail.com"
3. google賬號(hào)驗(yàn)證:
對(duì)于匿名訪問,為了防止連接過多,谷歌對(duì)同一個(gè)ip的訪問次數(shù)進(jìn)行了限制,所以我們需要添加一個(gè)下載驗(yàn)證,打開下面的鏈接:
https://android.googlesource.com/new-password
登錄谷歌賬號(hào),會(huì)看到下面的界面:
通過復(fù)制下面框中的腳本代碼到shell中,為Git配置cookie,用于訪問谷歌服務(wù)。
我們使用的是Ubuntu,所以我們打開shell窗口,將下面框框的代碼一行一行的復(fù)制到shell中執(zhí)行,復(fù)制一行執(zhí)行一行就可以了。
4. 安裝repo
android源碼包括很多模塊,每個(gè)模塊都是一個(gè)個(gè)工程,不同版本依賴不同模塊,總不能一個(gè)一個(gè)去下載這些工程吧,所以為了管理這些工程,谷歌開發(fā)了一個(gè)管理Git倉庫的工具叫repo,官網(wǎng)的安裝教程是這樣的:
首先創(chuàng)建bin文件夾:
$ mkdir ~/bin $ PATH=~/bin:$PATH
下載repo工具:
$ curl https://storage.googleapis.com/git-repo-downloads/repo > ~/bin/repo $ chmod a+x ~/bin/repo
5. 選擇合適的源碼版本
android源碼除了大版本外還有很多小版本,例如:android-7.1.1_r31,7.1.1就是大版本號(hào),r31就是小版本號(hào),因此我們需要先確定好需要下載的版本號(hào),具體有哪些版本請(qǐng)看下面的鏈接:
https://source.android.com/source/build-numbers.html#source-code-tags-and-builds
由于我的手機(jī)是nexus 6p 所以我選擇了以下版本:
第一列是編譯的代號(hào),第二列是版本號(hào),第三列是版本名稱,最后一列是該版本支持的手機(jī)型號(hào),請(qǐng)下載與你手機(jī)對(duì)應(yīng)的版本。
5. 開始下載源碼
確定了版本號(hào)之后,首先創(chuàng)建文件夾,這里以版本號(hào)命名:
$ mkdir android-7.1.1_r31 $ cd android-7.1.1_r31
初始化repo,注意把 -b 后面改為你要下載的版本號(hào):
$ repo init -u https://android.googlesource.com/platform/manifest -b android-7.1.1_r31
使用下面的命令就可以開始下載了:
$ repo sync
但是,請(qǐng)先不要著急下載,由于下載過程中還會(huì)遇到各種各樣的問題,所以我們需要?jiǎng)?chuàng)建一個(gè)自動(dòng)下載腳本,確保出錯(cuò)了之后會(huì)自動(dòng)執(zhí)行repo sync,在 android-7.1.1_r31下創(chuàng)建一個(gè)腳本down.sh,代碼如下:
#!/bin/bash repo sync -j16 while [ $? = 1 ]; do echo “======sync failed, re-sync again======” sleep 3 repo sync -j16 done
這樣就可以不用執(zhí)行repo sync命令了,直接執(zhí)行以下命令:
chmod a+x down.sh ./down.sh
好了,開始下載源碼了,android 7.1.1整個(gè)源碼大概有50G左右,建議在晚上下載。在沒有下載完畢之前,源碼的文件夾中只有一個(gè).repo文件夾,這是正常的,等到所有源碼下載完畢,其余文件夾就會(huì)出現(xiàn),不要著急。下載完畢后如下圖:
看到syncing work tree: 100%..... 就說明下載成功了,道路又前進(jìn)了一大步,道行且阻,行則將至。
全部下載完后android-7.1.1_r62中文件如下:
源碼下載完畢后,先不要著急著編譯,我們先備份一下源碼,萬一玩壞了還可以推倒重來,否則又要辛辛苦苦下載一遍。源碼怎么這么大呢,后來發(fā)現(xiàn)都是.repo這個(gè)文件夾太大了,這個(gè)文件夾只是與代碼同步有關(guān),并不影響編譯,刪掉之后,壓縮打包源碼,這次只有7G左右,而且刪掉之后編譯成功。
首先刪掉 .repo 文件夾,然后使用下面的命令備份源碼:
cd ~ tar -zcvf android-7.1.1_r31.tar.gz android-7.1.1_r31/
下載完源碼請(qǐng)務(wù)必刪掉 .repo 文件夾,以便備份源碼,同時(shí)防止編譯時(shí)磁盤空間不夠用.
編譯源碼
1. 安裝依賴
sudo apt-get install -y git flex bison gperf build-essential libncurses5-dev:i386 sudo apt-get install libx11-dev:i386 libreadline6-dev:i386 libgl1-mesa-dev g++-multilib sudo apt-get install tofrodos python-markdown libxml2-utils xsltproc zlib1g-dev:i386 sudo apt-get install dpkg-dev libsdl1.2-dev libesd0-dev sudo apt-get install git-core gnupg flex bison gperf build-essential sudo apt-get install zip curl zlib1g-dev gcc-multilib g++-multilib sudo apt-get install libc6-dev-i386 sudo apt-get install lib32ncurses5-dev x11proto-core-dev libx11-dev sudo apt-get install lib32z-dev ccache sudo apt-get install libgl1-mesa-dev libxml2-utils xsltproc unzip m4
Ubuntu 16.04請(qǐng)務(wù)必使用上面的依賴,Ubuntu 16.04需要的依賴和Ubuntu 14.04所需要的依賴是不同的.
2. 安裝open JDK8
從Android 6.0開始Android源碼的編譯需要安裝OpenJDK,不能使用Oracle JDK,而且Android 7.1.1只能使用OpenJDK8, 剛開始按照其他blog安裝了open-jdk7,編譯時(shí)提示require java_1.8_XXXX, 果斷又重新安裝了一次open-jdk8。
由于Ubuntu 16.04沒有OpenJDK7的源,因此在16.04上安裝OpenJDK7需要執(zhí)行下面的命令:
sudo add-apt-repository ppa:openjdk-r/ppa sudo apt-get update sudo apt-get install openjdk-8-jdk
檢查OpenJDK配置是否正確:
java -version
配置正確則如下圖:
3. 設(shè)置jack服務(wù)內(nèi)存
export ANDROID_JACK_VM_ARGS="-Dfile.encoding=UTF-8 -XX:+TieredCompilation -Xmx6g"
4. 開始編譯
4.1導(dǎo)入編譯Android源碼所需的環(huán)境變量和其它參數(shù):
source build/envsetup.sh
4.2運(yùn)行l(wèi)unch命令選擇編譯目標(biāo):
lunch
選擇aosp_angler-userdebug版本,這個(gè)版本才是debug 版本,可以調(diào)試。當(dāng)然eng也可以。
解釋一下,angler是nexus 6p的代號(hào)(code name),每個(gè)設(shè)備對(duì)應(yīng)的代號(hào)如下:
userdebug 是編譯類型,含義如下:
user 用于正式產(chǎn)品
userdebug 和user類似,但是有root權(quán)限,并且可以調(diào)試,主要用于調(diào)試
eng 開發(fā)用的選項(xiàng),配有額外的調(diào)試工具。
如果編譯后只在模擬器上運(yùn)行,則lunch時(shí)選擇:1
如果你不知道lunch該選哪一個(gè),請(qǐng)參考下面的鏈接,根據(jù)第三列選擇編譯選項(xiàng):
https://source.android.com/source/running.html
4.3開始執(zhí)行編譯:
執(zhí)行make -j8, 8為同時(shí)編譯的線程數(shù),一般google推薦這個(gè)數(shù)字為2倍的cpu個(gè)數(shù)再加上2,比如4核,就是10。博主使用的是8:
make -j8
關(guān)于cpu個(gè)數(shù),可以用下面命令查看:
cat /proc/cpuinfo
好了,開始編譯了, 晚上開始編譯,大約2個(gè)小時(shí),編譯成功了。(當(dāng)然了上面這個(gè)過程是成功的步驟,我自己編譯的時(shí)候,參考幾個(gè)blog,每個(gè)blog寫的不一樣,所以就很痛苦, 并且有些都是過時(shí)的或者錯(cuò)的,但是只要相信自己能解決,當(dāng)編譯出錯(cuò)的時(shí)候不要慌張,仔細(xì)查看log提示,然后google,總能找到解決問題的答案的。比如說上面我提到的 backend i/o error,我就是通過看log才發(fā)現(xiàn)的。還有一個(gè)提示是源碼肯定是完備的,在編譯之前盡量不修改任何文件,先編譯一次試試。我在編譯之前看某個(gè)blog修改了一個(gè)文件,導(dǎo)致編譯不過,我又趕緊修改了回來。)
如果在編譯過程中遇到了錯(cuò)誤,請(qǐng)先執(zhí)行以下命令:
make clean
然后再重新編譯:
source build/envsetup.sh lunch make -j8
編譯成功的截圖如下:
哈哈,到此為止,又前進(jìn)了一大步~, 下面就是怎么把rom刷入手機(jī)了。
驅(qū)動(dòng)
(1) 為何還要下載工廠鏡像呢?
前面我們提到了,只有nexus系列手機(jī)放出了驅(qū)動(dòng),所以我們需要去谷歌官網(wǎng)獲取驅(qū)動(dòng)。
(2) 驅(qū)動(dòng)又是以什么形式放出的呢?
在nexus 6p(5x)之前,例如nexus 5,驅(qū)動(dòng)的二進(jìn)制代碼(當(dāng)然不可能有源碼啦)是壓縮包的形式,下載后在aosp源碼根目錄下面解壓,然后執(zhí)行里面的sh腳本文件,接著編譯全部源碼就能編譯出驅(qū)動(dòng)鏡像,谷歌一下都是這種方法,然而關(guān)于nexus 6p編譯AOSP后刷機(jī)時(shí)驅(qū)動(dòng)問題的資料就很少了,博主在這里卡了很長時(shí)間才搞清楚了nexus 6p刷入AOSP時(shí)如何刷入驅(qū)動(dòng)的方法。
(3) 如何獲取nexus 6p的驅(qū)動(dòng)?
從nexus 6p(5x)起,驅(qū)動(dòng)都放在了專門的vender分區(qū)中,谷歌官網(wǎng)上只有工廠鏡像,沒有二進(jìn)制文件的壓縮包了。我們需要下載工廠鏡像,然后提取出里面的vender.img鏡像,刷入nexus 6p中就可以了。當(dāng)然,經(jīng)過實(shí)踐發(fā)現(xiàn),這android7.1.1 nexus6p手機(jī)上是可以的,但是其他博主說不行,可能和版本有關(guān)系吧。
首先到谷歌官網(wǎng)下載工廠鏡像,下載地址如下:
https://developers.google.com/android/nexus/images
根據(jù)前面下載源碼時(shí)候的編譯代號(hào)(Build)來選擇工廠鏡像:
根據(jù)源碼下載那一節(jié),我們選擇的android-7.1.1_r31版本對(duì)應(yīng)的build是N4F26U,所以我們下載N4F26U的工廠鏡像,點(diǎn)擊link下載鏡像, 下載完成后解壓:
再解壓image-angler-n4f26u.zip
vender.img就是我們的驅(qū)動(dòng)文件。
刷機(jī)就是將我們編譯出來的系統(tǒng)鏡像和工廠鏡像里面的vernder.img用fastboot命令寫入手機(jī)中,那么我們可以將image-angler-n4f26u中的鏡像除vender.img以外全部替換成我們自己的,然后打包,用flash-all.sh腳本一次性刷入,省時(shí)省力。在image-angler-n4f26u下將所有文件壓縮打成包image-angler-n4f26u.zip。然后將手機(jī)進(jìn)入fastboot模式。執(zhí)行flash-all.sh/flash_all.bat, 等待2分鐘,刷機(jī)就完成了。 哈哈,刷機(jī)到此結(jié)束。
刷機(jī)完畢,開機(jī),原生android就呈現(xiàn)在你面前,沒有Google全家桶。電話藍(lán)牙都正常。
就是每次開機(jī)都會(huì)彈出個(gè)對(duì)話框:
There’s an internal problem with your device. Contact your manufacturer for details
不影響使用,這個(gè)網(wǎng)址有解決方案。
http://blog.csdn.net/jerrywu145/article/details/56342863
下一篇文章講講如何基于自己編譯的rom,進(jìn)行Android源碼級(jí)調(diào)試。
參考文章:
http://blog.csdn.net/jerrywu145/article/details/56342863
http://www.jb51.net/article/101929.htm