天天看点

android编译全过程

本文主要参考官方文档(http://source.android.com/download)和网上相关资料(http://blog.csdn.net/hkjinzhao/archive/2009/03/18/4002326.aspx,http://www.williamhua.com/2009/04/30/how-to-build-android-15-kernel-image/)。网上的资料对于andorid的最新代码有的已经过时,或者有错误,这份文档本人亲自实验,保证可行。另本人没有使用eclipse的习惯,所以并没有做eclipse的相关配置。

编译环境:ubuntu9.10,widnows平台目前不被支持。

1)安装必要的软件环境

$ sudo apt-get install git-core gnupg sun-java5-jdk flex bison gperf libsdl-dev libesd0-dev libwxgtk2.6-dev build-essential zip curl libncurses5-dev zlib1g-dev

官方推荐的就是上面这些,如果在编译过程中发现某些命令找不到,就apt-get它。可能需要的包还有:

$ sudo apt-get install make

$ sudo apt-get install gcc

$ sudo apt-get install g++

$ sudo apt-get install libc6-dev

$ sudo apt-get install patch

$ sudo apt-get install texinfo

$ sudo apt-get install zlib1g-dev

$ sudo apt-get install valgrind

$ sudo apt-get install python2.5(或者更高版本)

需要注意的是,官方文档说如果用sun-java6-jdk可出问题,得要用sun-java5- jdk。经测试发现,如果仅仅make(make不包括make sdk),用sun-java6-jdk是没有问题的。而make sdk,就会有问题,严格来说是在make doc出问题,它需要的javadoc版本为1.5。

因此,我们安装完sun-java6-jdk后最好再安装sun-java5-jdk,或者只安装sun-java5-jdk。这里sun-java6-jdk和sun-java5-jdk都安装,并只修改javadoc.1.gz和javadoc。因为只有这两个是make sdk用到的。这样的话,除了javadoc工具是用1.5版本,其它均用1.6版本:

$ sudo apt-get install sun-java6-jdk

修改javadoc的link

$ cd /etc/alternatives

$ sudo rm javadoc.1.gz

$ sudo ln -s /usr/lib/jvm/java-1.5.0-sun/man/man1/javadoc.1.gz javadoc.1.gz

$ sudo rm javadoc

$ sudo ln -s /usr/lib/jvm/java-1.5.0-sun/bin/javadoc javadoc

2)设置环境变量

$ emacs ~/.bashrc

在.bashrc中新增或整合path变量,如下:

#java 程序开发/运行的一些环境变量

java_home=/usr/lib/jvm/java-6-sun

jre_home=${java_home}/jre

export android_java_home=$java_home

export classpath=.:${java_home}/lib:$jre_home/lib:$classpath

export java_path=${java_home}/bin:${jre_home}/bin

export java_home;

export jre_home;

export classpath;

home_bin=~/bin/

export path=${path}:${java_path}:${home_bin};

保存后,同步更新:

source ~/.bashrc

3)安装repo(用来更新android源码)

创建~/bin目录,用来存放repo程序,如下:

$ cd ~

$ mkdir bin

并加到环境变量path中,在第2步中已经加入

下载repo脚本并使其可执行:

$ curl http://android.git.kernel.org/repo >~/bin/repo

$ chmod a+x ~/bin/repo

4)初始化repo

repo是android对git的一个封装,简化了一些git的操作。

创建工程目录:

$ mkdir android

$ cd android

repo初始化

$ repo init -u git://android.git.kernel.org/platform/manifest.git

在此过程中需要输入名字和email地址。初始化成功后,会显示:

repo initialized in /android

在~/android下会有一个.repo的隐藏目录。

5)同步源代码

$ repo sync

这一步要很久很久

6)编译android源码,并得到~/android/out目录

$ cd ~/andoird

$ make

这一过程很久

7)在模拟器上运行编译好的android

编译好android之后,emulator在~/android/out/host/linux-x86/bin下,ramdisk.img,system.img和userdata.img则在~/android/out/target/product/generic下

$ cd ~/android/out/host/linux-x86/bin

增加环境变量

在.bashrc中新增环境变量,如下

export android_product_out=~/android/out/target/product/generic

android_product_out_bin=~/android/out/host/linux-x86/bin

export path=${path}:${android_product_out_bin}:${android_product_out};

最后,同步这些变化:

$ source ~/.bashrc

$ cd ~/android/out/target/product/generic

$ emulator -system system.img -data userdata.img -ramdisk ramdisk.img

最后进入android桌面,就说明成功了。

8)编译模块

android中的一个应用程序可以单独编译,编译后要重新生成system.img

在源码目录下执行

$ . build/envsetup.sh (.后面有空格)

就多出一些命令:

- croot:   changes directory to the top of the tree.

- m:       makes from the top of the tree.

- mm:      builds all of the modules in the current directory.

- mmm:     builds all of the modules in the supplied directories.

- cgrep:   greps on all local c/c++ files.

- jgrep:   greps on all local java files.

- resgrep: greps on all local res/*.xml files.

- godir:   go to the directory containing a file.

可以加—help查看用法

我们可以使用mmm来编译指定目录的模块,如编译联系人:

$ mmm packages/apps/contacts/

编完之后生成两个文件:

out/target/product/generic/data/app/contactstests.apk

out/target/product/generic/system/app/contacts.apk

可以使用

$ make snod

重新生成system.img,再运行模拟器

9)编译sdk

直接执行make是不包括make sdk的。make sdk用来生成sdk,这样,我们就可以用与源码同步的sdk来开发android了。

a)修改/frameworks/base/include/utils/asset.h

‘uncompress_data_max = 1 * 1024 * 1024’ 改为 ‘uncompress_data_max = 2 * 1024 * 1024’

原因是eclipse编译工程需要大于1.3m的buffer;

b)编译adt。

由于本人不使用eclipse,所以没有进行这步;

c)执行make sdk。

注意,这里需要的javadoc版本为1.5,所以你需要在步骤1中同时安装sun-java5-jdk

$ make sdk

编译很慢。编译后生成的sdk存放在out/host/linux-x86/sdk/,此目录下有android-sdk_eng.xxx_linux- x86.zip和android-sdk_eng.xxx_linux-x86目录。android-sdk_eng.xxx_linux-x86就是 sdk目录

实际上,当用mmm命令编译模块时,一样会把sdk的输出文件清除,因此,最好把android-sdk_eng.xxx_linux-x86移出来

此后的应用开发,就在该sdk上进行,所以把7)对于~/.bashrc的修改注释掉,增加如下一行:

export path=${path}:~/android/out/host/linux-x86/sdk/android-sdk_eng.xxx_linux-x86/tools

注意要把xxx换成真实的路径;

d)关于环境变量、android工具的选择

目前的android工具有:

a、我们从网上下载的sdk,如果你下载过的话( tools下有许多android工具,lib/images下有img映像)

b、我们用make sdk编译出来的sdk( tools下也有许多android工具,lib/images下有img映像)

c、我们用make编译出来的out目录( tools下也有许多android工具,lib/images下有img映像)

那么我们应该用那些工具和img呢?

首先,我们一般不会用a选项的工具和img,因为一般来说它比较旧,也源码不同步。其次,也不会用c选项的工具和img,因为这些工具和img没有经过sdk的归类处理,会有工具和配置找不到的情况;事实上,make sdk产生的很多工具和img,在make编译出来out目录的时候,已经编译产生了,make sdk只是做了copy而已。

e)安装、配置adt

略过;

f)创建android virtual device

编译出来的sdk是没有avd(android virtual device)的,我们可以通过android工具查看:

$ android list

创建avd:

$ android create avd -t 1 -n myavd

可以android –help来查看上面命令选项的用法。创建中有一些选项,默认就行了

再执行android list,可以看到avd存放的位置

以后每次运行emulator都要加-avd myavd或@myavd选项:

$ emulator -avd myavd

10)编译linux内核映像

a)准备交叉编译工具链

android代码树中有一个prebuilt项目,包含了我们编译内核所需的交叉编译工具。

b)设定环境变量

增加如下两行:

export path=$path:~/android/prebuilt/linux-x86/toolchain/arm-eabi-4.4.0/bin

export arch=arm

保存后,同步变化:

c)获得合适的内核源代码

$ cd ~/android

获得内核源代码仓库

$ git clone git://android.git.kernel.org/kernel/common.git kernel

$ cd kernel

$ git branch

显示

* android-2.6.27

说明你现在在android-2.6.27这个分支上,也是kernel/common.git的默认主分支。

显示所有head分支:

$ git branch -a

remotes/origin/head -> origin/android-2.6.27

remotes/origin/android-2.6.25

remotes/origin/android-2.6.27

remotes/origin/android-2.6.29

remotes/origin/android-goldfish-2.6.27

remotes/origin/android-goldfish-2.6.29

我们选取最新的android-goldfish-2.6.29,其中goldfish是android的模拟器模拟的cpu。

$ git checkout -b android-goldfish-2.6.29 origin/android-goldfish-2.6.29

android-2.6.27

* android-goldfish-2.6.29

我们已经工作在android-goldfish-2.6.29分支上了。

d)设定交叉编译参数

打开kernel目录下的makefile文件,把cross_compile指向刚才下载的prebuilt中的arm-eabi编译器

cross_compile ?= arm-eabi-

ldflags_build_id = $(patsubst -wl$(comma)%,%,\

$(call ld-option, -wl$(comma)–build-id,))

这一行注释掉,并且添加一个空的ldflags_build_id定义,如下:

ldflags_build_id =

e)编译内核映像

$ cd ~/android/kernel

$ make goldfish_defconfig

f)测试生成的内核映像

$ emulator -avd myavd -kernel ~/android/kernel/arch/arm/boot/zimage