Ubuntu 下编译AOSP源码
前言
AOSP是谷歌维护的一个开源项目,它包含了Android系统的源码和功能。
狭义的Android系统是谷歌的一个商标,除了包括asop的源码部分,还包括手机处理器、摄像头等硬件所需的固件等内核设备驱动,以及要谷歌移动服务(GSM、如Chrome、Google Play等等。其他第三方厂商在AOSP的基础上构建出自己的RoM,获得Android商标的授权后也可以标注Android的字样。
编译完AOSP后,可以导入Android studio进行阅读学习、修改和调试。谷歌提供了的nexus和pixel手机的硬件驱动代码,它们是使用AOSP作为原生系统,因此我们如果有nexus和pixel手机也可以对AOSP进行定制并刷到设备上。
编译配置要求
- 16 GB 以上的内存,越大越好,这里是指构建过程的可用内存, Google 建议提供 64 GB。
- 400G以上可用磁盘空间。代码占150-200GB,构建产物占150多G,重复构建占用更多
- Ubuntu 18.04 以上,64位。官方现已不再支持在 Windows 或 MacOS 上进行构建。
以下以获取android-13.0.0_r44源码为例:
1 搭建编译环境
1.1 检查JDK环境
在Ubuntu请使用OpenJDK,如果是AOSP的master分支会自带OpenJDK,其他版本需要自行安装
1 | # 检查jdk版本,如果提示找不到命令 “java”字样说明未安装 |
1.2 检查python环境
Ubuntu 自 16.04 版本已经默认预装python3,如果变异过程中报错`找不到命令 “python”,只需要建立一个python的链接即可。
1 | # 通过软链接将python命令默认指向python3 |
1.3 安装基础软件包
这条命令一次性安装了常用软件包例如git、curl、zip、unzip以及其他开发工具和库。
1 | sudo apt-get install git-core gnupg flex bison build-essential zip curl zlib1g-dev libc6-dev-i386 libncurses5 lib32ncurses5-dev x11proto-core-dev libx11-dev lib32z1-dev libgl1-mesa-dev libxml2-utils xsltproc unzip fontconfig |
其中Git安装完还需配置一下用户信息,否则在使用repo拉取代码时会被拒绝。
1 | # 配置Git的用户名 |
1.4 安装Repo
Repo是Google专门用来管理Android大型源码库的一个工具,比如AOSP项目。
简单说就是一个封装了Git命令的python脚本,用以简化跨多个Git仓库的代码管理和同步。通过 Repo,你可以在一个命令行中对所有仓库进行同步、检出特定的分支、以及执行其他 Git 操作。
1 | # 安装repo |
2 下载AOSP源码
Android 源代码树位于由 Google 托管的 Git 代码库中,因此我们可以通过Git(Repo)下载源码。
2.1 官方同步方式
这种方式对网络有一定要求,更推荐使用镜像站提供的初始化包的方式。
(1)初始化Repo并指定代码库地址
使用mkdir <dirName>
命令或右键创建文件夹,作为AOSP的根目录。在该目录下打开终端,输入以下命令进行仓库的初始化,指定源和AOSP版本。
1 | # 从Google下载Android13源码 |
命令说明:repo init -u <url> -b <branch>
会获取最新版本的 Repo 及其最新的 bug 修复。执行命令后创建一个 .repo/ 目录,其中包含存放 Repo 源代码和标准 Android 清单文件的 Git 代码库。其中可以通过-b 来指定相应分支,可以从谷歌使用文档-关于-源代码标记和build分类中,根据设备型号和Android版本号来选择标记(分支)。
如果谷歌服务器被墙,可以设置科学上网,或者从清华大学镜像站、中科大镜像等镜像源中获取。
部分AOSP的标记和 build源码版本标记:
Build ID | 标记 | 版本 | 更新日期 |
---|---|---|---|
TQ2A.230505.002.A1 | android-13.0.0_r44 | Android13 | 2023-05-05 |
9575282 | android-security-13.0.0_r3 | Android13 | 2022-08-05 |
SP1A.210812.016.A1 | android-12.0.0_r3 | Android12 | 2021-10-05 |
RQ1A.201205.008.A1 | android-11.0.0_r21 | Android11 P | 2020-12-05 |
6780337 | android-10.0.0_r47 | Android10 | 2019-09-05 |
6780336 | android-9.0.0_r61 | Pie | 2018-08-05 |
OPR5.170623.014 | android-8.0.0_r36 | Oreo | 2017-12-01 |
NRD91D | android-7.0.0_r7 | Nougat | 2016-10-05 |
MRA59B | android-6.0.0_r7 | Marshmallow | 2015-12-01 |
LMY48I | android-5.1.1_r9 | Lollipop | / |
(2)同步源码到工作目录
1 | # 默认命令 |
命令说明:repo sync
会将 Android 源代码树从默认清单中指定的代码库下载到工作目录。如过从未进行过这步操作,这相当于git clone
。
可选参数-c:只拉取当前分支,–no-tags 不拉取tags,-jthreadcount:指定同步任务数量,这些参数有助于加快同步速度。
Android源码非常大,需要同步很长的时间,如果同步过程中有出现某个地方卡顿了很久,可以使用Ctrl+Z来中断同步,重新执行 repo sync进行同步。
2.2 使用镜像站提供的初始化包(推荐)
首次同步的数据包特别大,从Google官方同步方式耗时较长且可能由于网络原因造成失败,可以使用一些镜像站提供的打包好的 AOSP压缩文件。例如清华大学镜像源提供的每月更新初始化包。
1. 下载初始包
使用浏览器或者curl命令进行下载压缩包。文件大约60G,如果使用curl下载下来只有几k就结束了,大概是请求被禁止了,可以使用浏览器直接下载。
1 | # 下载初始化包,-O:下载远程文件到本地,-C:断点续传 |
2. 提取文件
解包后得到的 AOSP 工程目录,里面有一个隐藏的 .repo 目录
1 | # 提取文件,-x:解包,-v:显示过程,-f:指定文件 |
3. 进入aosp目录再指定分支并同步
1 | # 进入aosp目录 |
当终端输出类似下面的信息表示同步完成。
1 | Checking out: 100% (1317/1317), done in 47m8.048s |
下载完成后的部分文件夹:
1 | aosp |
3 开始编译
3.1 设置环境
执行以下命令进行初始化环境。
1 | # 执行该命令,envsetup.sh 脚本初始化环境 |
envsetup.sh
脚本位于aosp根目录/build/envsetup.sh,作用是设置环境变量和封装一些用于构建Android源码的函数。每次开始构建前请先执行一下这个脚本以便加载lunch等命令。
3.2 选择构建目标
使用lunch
命令选择要构建的目标(选择你的午餐)。
1 | # 默认,从后续列表中选择构建目标, |
命令说明:lunch product_name-build_variant
,参数的product_name
表示需要构建的设备型号,build_variant
表示构建类型,共3种。 可以从从谷歌使用文档-构建-刷写设备-选择设备build分类下选择目标机型对应的build参数。
构建类型:
- user: 用户版,权限受限;适用于生产环境。
- userdebug:用户调试版,与“user”类似,但具有 root 权限和调试功能;是进行调试时的首选编译类型。
- eng:工程版,具有额外调试工具的开发配置,适用于深度开发和调试
如果在没有参数的情况下运行,则 lunch 会提示您从菜单中选择目标,但是注意菜单中并未包含所有可能的选项。如果你打算跑在Android模拟器上可以选择带x86_64/x86标识的选项。
1 | # 选择具体目标,例如这里选择x86_64 架构模拟器和用户调试版为构建目标 |
3.3 构建代码
1. 全量编译
使用 make
命令从源代码树的顶部开始编译 Android 源代码。
1 | m -j10 |
-jN可选参数可以设置处理并行任务的数量。如果您没有提供-j参数,构建系统会自动选择您认为最适合您系统的并行任务计数,Google推荐执行源码编译的线程数=CPU核心数x2+2。
构建完成后会生成各种库文件、输出产物默认在out/target/product/<device>
下,例如:
- 各种库文件:libhardware.so、libssl.so 等
- Java 类文件:framework.jar、core.jar 等
- 系统映像文件:boot.img、system.img、recovery.img等,用于刷机将系统烧录到 Android 设备上。
1 | /out/target/product/<device>/ |
2. 编译指定模块
如果只想编译某个模块,可以使用命令mmm <directory>
,该命令只编译指定目录中的模块,而不是整个源代码树。
1 | # 指定编译frameworks/base |
3.4 其他命令
清除文件重新构建。
1 | # 清除已生成的构建输出文件 |
3.5 编译完成
编译过程耗时比较久,对CPU线程数和主频要求比较高,这里用E5-2678 v3(12核24线程)/64G内存大概跑了一个多小时。
当终端输出类似下面的日志说明构建完成。
1 | #### make completed successfully (01:24:41 (hh:mm:ss)) #### |
编译完成后,如果我们选择的目标是x86,可以使用模拟器来运行,由于已经使用 lunch 选择了构建目标,因此只能运行在我们选定的目标上。
1 | # 构建流程会自动将模拟器添加到您的路径中,如需运行模拟器,请输入以下命令 |
此外我们还可以将源码导入Android studio进行阅读或者调试等操作。
4 阅读源码
如果需要将AOSP源码导入Android studio主要有以下3种方式:
4.1 平台版 Android Studio
Android Studio for Platform (ASfP)是 Android Studio针对Framework开发做的版本,添加了对AOSP项目相关的一些支持。不过目前仅支持 Ubuntu 系统。
可以从ASfP官网下载,目前官网简中网页打不开,可以使用英文或者繁中
1 | # 1.从官网网页或执行以下命令下载安装包 |
AIDEGen
AIDEGen是 Android 10 开始引入的为Android studio 等idea工具生成系统源码的project,它能自动配置Android Studio或IntelliJ项目文件,并解析相关模块依赖关系,不过也只支持Ubuntu。
AIDEGen需要运行在选定目标之后,即执行了source build/envsetup.sh && lunch
之后。
1 | # Settings:模块名称 -i:需要运行的ides:Android studio |
idegen
idegen 是 Android 源代码中的一个模块,专门用来为idea工具生成系统源码的project。这是最早提供的一种方式,不推荐。如果需要在Windows 下导入源码或者只关注AOSP的一部分源码,可以使用这种方式。
默认情况下AOSP编译并不会生成该文件,另外编译idegen模块不需要先进行全量编译,直接执行以下命令:
1. 生成ipr和iml文件
1 | # 执行该命令后,将在out目录生成idegen.jar |
其中ipr文件和iml文件是主要的IEDA工程配置文件:
- android.ipr:Android Studio 项目文件的主配置文件,包含了项目的基本信息、模块、依赖项、构建设置等
- android.iml:模块的配置信息,例如源代码目录、依赖项等
- android.iws:存储工作空间(workspace)相关设置的文件,可省略。
2. 导入到Android Studio
如果只想关注framework层的源码,可以删除其他模块,只保留framework文件夹和packages文件夹以及idegen配置文件,这样可以大大减少源码体积。
将项目文件复制到Windows后,点击Android Studio->File->Open,选择刚才生成的android.ipr文件导入项目。
3. 设置源码正确跳转
将源码只关联本地,删除其他依赖:依次打开Android Studio的Project Structure->Modules->dependecies,删除其他依赖模块,只保留android api xx platform和
排除模块
如果不想导入某些模块,可以修改android.iml文件中
1 | <excludeFloder url="file://$MODULE_DIR$"/模块名> |
5.刷机
如果我们谷歌系列的手机,可以将编译出来的AOSP镜像刷写到设备上。
5.1 下载驱动
AOSP只含有纯源代码部分,不包含行与硬件相关的其他专有库(驱动),如果编译AOSP是为了刷机并且拥有Pixel 系列的手机,则还需要下载 Google 的驱动程序。
- 下载驱动文件:在Google官网Nexus 和 Pixel 设备的驱动程序页面可以找设备对应的驱动压缩文件,注意驱动要和aosp build号对应。
- 解压文件:将驱动压缩包文件解压得到.sh文件并放到AOSP源码根目录,
1
2
3
4
5
6# 解压
tar -zxvf FileName.tgz
# 解压后会生成以下两个文件
extract-google_devices-sailfish.sh
extract-qcom-sailfish.sh - 使用bash运行这两个文件,浏览协议并同意后,执行后会创建vendor文件夹,其下包含驱动。
1
2bash extract-google_devices-sailfish.sh
bash extract-qcom-sailfish.sh
5.2使用 fastboot 刷机
- 解锁BL和下载配置fastboot工具,可以在aosp目录下通过make fastboot命令编译出来,也可以直接从网上下载
- 设备确保解锁BL、打开开发者模式、连接上adb,进入BootLoader模式
- 进入编译后产生的镜像的目录…./aosp/out/target/product/
,执行以下命令。刷完会自动重启,系统刷入完成。 1
2
3# fastboot device 电脑能识别到设备
# -w 代表清除/data分区,删除所有用户数据
fastboot flashall -w
6 优化编译环境(可选)
1. 设置 ccache
视需要指示编译过程使用ccache编译工具,ccache是适用于C和C++的编译器缓存,有助于提高编译速度。这对于编译服务器和其他高容量生产环境来说尤其有用,如果您是在执行增量编译(例如个人开发者而非编译服务器),ccache 可能会让您为缓存未命中埋单,从而减慢您的编译速度。
设置步骤:
1.打开bashrc配置文件
1 | gedit ~/.bashrc |
2.在文件末尾添加以下内容:
其中
1 | export USE_CCACHE=1 |
3.使 .bashrc 更改立即生效
1 | source ~/.bashrc |
后续可以通过以下命令查看 ccache 的使用情况:
1 | watch -n1 -d prebuilts/misc/linux-x86/ccache/ccache -s |
2. 自定义输出目录
默认情况下,每次编译的输出都会存储在相应源代码树的out/子目录下,自定义目录如果构建多个目标(不同手机型号等)时能够更清晰地区分各个目标的输出结果
OUT_DIR:如果需要构建多个目标(不同的手机型号或构建类型),修改这个路径可以为每个目标输出到独立的目录。
OUT_DIR_COMMON_BASE:多个构建目标的共享的基础路径,如果有多个物理磁盘,如果将源文件和输出存储在单独的物理磁盘中,构建速度会更快(虚拟机一般用不到)。
操作步骤,修改.bashrc或者每次编译时直接在终端输入以下命令:
1 | # 修改存储输出目录的基础位置 |
7 报错
1.下载源码404无法连接
科学上网或者使用镜像地址
1 | # 如果repo如果更新失败报错404,将repo的git源替换成清华的景象 |
2.python 编码格式错误
1 | File "/home/zhg/桌面/aosp/out/host/linux-x86/bin/protoc-gen-nanopb/nanopb_gene |
解决方法:
1 | vim ~/.bashrc |
3.ca证书错误
1 | server certificate verification failed. CAfile: none CRLfile: none |
解决方法:添加证书或者忽略证书
1 | export GIT_SSL_NO_VERIFY=1 |
4.Python版本不兼容
1 | File "device/generic/goldfish/tools/mk_combined_img.py", line 48 |
或者
1 | SyntaxError: Missing parentheses in call to 'print'. Did you mean print(...)? |
解决方法:编译低版本的aosp可能需要Python2而不是Python3
1 | # 安装Python2 |
5. 执行idegen.sh 脚本报错
1 | Error: A JNI error has occurred, please check your installation and try again |
解决方法: jdk版本太低,idegen.jar的编译java版本是 version 53.0即java 9,需要升级openjdk版本