深入浅出Android.bp
早期的Android系统都是采用Android.mk的配置来编译源码,从Android 7.0开始引入Android.bp。很明显Android.bp的出现就是为了替换掉Android.mk。
再来说一说跟着Android版本相应的发展演变过程:
- Android 7.0引入ninja和kati
- Android 8.0使用Android.bp来替换Android.mk,引入Soong
- Android 9.0强制使用Android.bp
转换关系图如下:
;
通过Kati将Android.mk转换成ninja格式的文件,通过Blueprint+ Soong将Android.bp转换成ninja格式的文件,通过androidmk将将Android.mk转换成Android.bp,但针对没有分支、循环等流程控制的Android.mk才有效。
这里涉及到Ninja, kati, Soong, bp概念,接下来分别简单介绍一下。
Ninja
ninja是一个编译框架,会根据相应的ninja格式的配置文件进行编译,但是ninja文件一般不会手动修改,而是通过将Android.bp文件转换成ninja格文件来编译。
Android.bp
Android.bp的出现就是为了替换Android.mk文件。bp跟mk文件不同,它是纯粹的配置,没有分支、循环等流程控制,不能做算数逻辑运算。如果需要控制逻辑,那么只能通过Go语言编写。
Soong
Soong类似于之前的Makefile编译系统的核心,负责提供Android.bp语义解析,并将之转换成Ninja文件。Soong还会编译生成一个androidmk命令,用于将Android.mk文件转换为Android.bp文件,不过这个转换功能仅限于没有分支、循环等流程控制的Android.mk才有效。
Blueprint
Blueprint是生成、解析Android.bp的工具,是Soong的一部分。Soong负责Android编译而设计的工具,而Blueprint只是解析文件格式,Soong解析内容的具体含义。Blueprint和Soong都是由Golang写的项目,从Android 7.0,prebuilts/go/目录下新增Golang所需的运行环境,在编译时使用。
Kati
kati是专为Android开发的一个基于Golang和C++的工具,主要功能是把Android中的Android.mk文件转换成Ninja文件。代码路径是build/kati/,编译后的产物是ckati。
Android.bp语法
Android.bp是一种纯粹的配置文件,设计简单,没有条件判断或控制流语句,采用在Go语言编写控制逻辑。
Android.bp文件记录着模块信息,每一个模块以模块类型开始,后面跟着一组模块的属性,以名值对(name: value)表示,每个模块都必须有一个 name属性。基本格式,以frameworks/base/services/Android.bp文件为例
1 java_library { 2 name: "services", 3 4 dex_preopt: { 5 app_image: true, 6 profile: "art-profile", 7 }, 8 9 srcs: [ 10 "java/**/*.java", 11 ], 12 13 static_libs: [ 14 "services.core", 15 "services.accessibility", 16 "services.appwidget", 17 "services.autofill", 18 "services.backup", 19 "services.companion", 20 "services.coverage", 21 "services.devicepolicy", 22 "services.midi", 23 "services.net", 24 "services.print", 25 "services.restrictions", 26 "services.usage", 27 "services.usb", 28 "services.voiceinteraction", 29 "android.hidl.base-V1.0-java", 30 ], 31 32 libs: [ 33 "android.hidl.manager-V1.0-java", 34 "miuisdk", 35 "miuisystemsdk" 36 ], 37 38 } 39 40 cc_library_shared { 41 name: "libandroid_servers", 42 defaults: ["libservices.core-libs"], 43 whole_static_libs: ["libservices.core"], 44 }