本文基于AOSP-7.1.1-R9源码分析,源码可以参见build/+/android-7.1.1_r9。
简介
在Android系统中,所有的应用都是以apk的形式存在,那这个apk是如何生成的呢?对于刚开始接触系统开发的开发者来说,经常会使用如下命令编译apk或者系统固件。
source build/envsetup.sh;
lunch
make -j8
or
mmm packages/app/Settings
这些命令背后的原理是什么呢?最终为什么会生成一个apk
文件?接下来的文章将会对相关原理进行剖析。
本文使用AOSP frameworks/base/tests/Split/
的源码来进行解析。由于关注点在apk
的生成原理,所以重点分析和apk
相关的编译原理,对于生成系统固件的原理不是本文讨论的重点,当了解了apk
的生成原理之后,其他的编译原理也就好理解了。
apk编译系统
接下来我们将分析Android编译系统里面用来生成apk
的原理。大致会分为如下几个步骤:
step1. 通过source 命令,读入envsetup.sh里面定义的各种命令,比如mm、mma、mmm、godir、croot等,方便我们在当前终端进行相关的命令输入。
step2.lunch
将要编译的产品,生成产品相关的参数配置。本文不做深入讲解。
source build/envsetup.sh;
lunch
step3. 开始执行编译命令,生成目标的依赖关系。
mmm frameworks/base/tests/Split
查看build/envsetup.sh
的源码,可以看到,当我们执行如上命令的时候,实际上执行的是如下的命令:
ONE_SHOT_MAKEFILE= frameworks/base/tests/Split/Android.mk make -C /home/tanfuhai/data/code/opengrok/android_n -f build/core/main.mk all_modules
接下来就直接进入到Android编译系统,可以看到入口makefile
文件是main.mk
,下面对引用的makefile文件一一讲解:
- build/core/main.mk 编译系统的入口。Android所有的生成目标都是从这个
main.mk
开始,Android默认的maketarget
是droid
。droid
依赖于droid_targets
,Makefile里面定义了各种依赖,当执行make的时候,可以编译出我们需要的所有image,droid
的依赖关系如下图:
.PHONY: droid
DEFAULT_GOAL := droid
$(DEFAULT_GOAL): droid_targets
.PHONY: droid_targets
droid_targets:
droid_targets: apps_only
droid_targets: droidcore dist_files
本文主要讲解apk
生成的原理,当我们执行mmm
的时候,此时的target是all_modules
。make的过程中,首先会读取所有的makefile文件。由于我们传入了ONE_SHOT_MAKEFILE= frameworks/base/tests/Split/Android.mk
,所以此时会include这个ONE_SHOT_MAKEFILE
。all_modules
的依赖定义如下:
# phony target that include any targets in $(ALL_MODULES)
.PHONY: all_modules
ifndef BUILD_MODULES_IN_PATHS
all_modules: $(ALL_MODULES)
else