Android.mk简介(9)

简介

Android.mk文件是GNU Make的一小部分,它用来对Android程序进行编译。
因为所有的android.mk都在同一个GNU Make执行环境中进行执行,而Android.mk中所有的变量都是全局的。因此,应尽量少声明变量。
一个Android.mk文件可以编译多个模块,模块类型如下:

1. APK程序

一般的Android程序,编译打包生成apk文件

2. JAVA库

java类库,编译打包生成jar文件

3. C\C++应用程序

可执行的C\C++应用程序

4. C\C++静态库

编译生成C\C++静态库,并打包成.a文件

5. C\C++动态库

编译生成动态库,并打包成.so文,有且只有动态库才能被安装/复制到您的应用软件(APK)包中。

注意,NDK的Anroid.mk语法同公开发 布的Android平台开源代码的Anroid.mk语法很接近,然而编译系统实现他们的方式却是不同的,这是故意这样设计的,可以让程序开发人员重用外部库的源代码更容易。

GNU MAKE系统变量

GNU Make变量在你的Android.mk文件解析之前,就由编译系统定义好了。

  1. CLEAR_VARS: 必须在开始一个新模块之前包含这个脚本:include$(CLEAR_VARS),用于重置除LOCAL_PATH变量外的所有LOCAL_XXX系列变量。
  2. BUILD_SHARED_LIBRARY: 根据所有的LOCAL_XXX变量把列出的源代码文件编译成一个共享库。
    注意,必须至少在包含这个文件之前定义 LOCAL_MODULE 和 LOCAL_SRC_FILES。
  3. BUILD_STATIC_LIBRARY:用于编译一个静态库。静态库不会复制到的APK包中,但是能够用于编译共享库。示例:include $(BUILD_STATIC_LIBRARY),这将会生成一个名为 lib$(LOCAL_MODULE).a 的文件
  4. TARGET_ARCH: 目标 CPU平台的名字, 和 android 开放源码中指定的那样。如果是
    arm,表示要生成 ARM 兼容的指令,与 CPU架构的修订版无关。
  5. TARGET_PLATFORM: Android.mk 解析的时候,目标 Android 平台的名字。详情可参考/development/ndk/docs/stable- apis.txt.
    android-3 -> Official Android 1.5 system images
    android-4 -> Official Android 1.6 system images
  6. TARGET_ARCH_ABI: 暂时只支持armeabi和armeabi-v7a。在现在的版本中一般把这两个值简单的定义为 arm, 通过android平台内部对它重定义来获得更好的匹配。其他的 ABI 将在以后的 NDK 版本中介绍,它们会有不同的名字。注意虽然所有基于ARM的ABI都会把 'TARGET_ARCH'定义成‘arm’, 但是会有不同的‘TARGET_ARCH_ABI’。
  7. TARGET_ABI: 目标平台和ABI的组合,它事实上被定义成$(TARGET_PLATFORM)-$(TARGET_ARCH_ABI) ,在想要在真实的设备中针对一个特别的目标系统进行测试时,会有用。在默认的情况下,它会是'android-3-arm'。

模块描述变量

下面的变量用于向系统描述我们自己的模块,它们应该定义在include $(CLEAR_VARS)include $(BUILD_XXX)之间。

LOCAL_PATH:当前文件的路径,必须在 Android.mk 的开头定义
include $(CLEAR_VARS)
LOCAL_SRC_FILES:编译的源文件
LOCAL_MODULE:模块名称
LOCAL_MODULE_CLASS:用于制定LOCAL_MODULE_PATH的路径
LOCAL_MODULE_PATH:模块的生成地址

LOCAL_PACKAGE_NAME:APK应用的名称
LOCAL_DEX_PREOPT:是否开启odex优化
LOCAL_CERTIFICATE:当前应用的证书名称
LOCAL_MODULE_TAGS:模块的标签,eng,user,debug,development,option

LOCAL_STATIC_JAVA_LIBRARIES: 模块依赖的java静态库,最终会打包到apk里面
LOCAL_JAVA_LIBRARIES:模块依赖的java动态库,比如framework.ja,只是编译的时候引用
LOCAL_STATIC_LIBRARIES: 模块依赖的c/c++静态库 .a
LOCAL_SHARED_LIBRARIES:模块依赖的c/c++动态库 .so
LOCAL_C_INCLUDES: 表示头文件的搜索路径,默认的头文件的搜索路径是LOCAL_PATH目录
LOCAL_CPP_EXTENSION: 用来指定C++代码文件的扩展名,默认是'.cpp'

LOCAL_CFLAGS: 提供给c/c++编译器额外的编译参数。指定一个附加的包含路径,宏定义,或者编译选项
LOCAL_CXXFLAGS: 与 LOCAL_CFLAGS同理,针对 C++源文件
LOCAL_CPPFLAGS: 与 LOCAL_CFLAGS同理,但是对 C 和 C++ source files都适用
LOCAL_LDLIBS: 编译模块时要使用的附加的链接器选项

include $(BUILD_XXX)

自定义变量

可以定义其他变量为自己使用,但是不能使用NDK编译系统保留的变量名:

  1. 以 LOCAL_开头的名字(例如 LOCAL_MODULE)
  2. 以 PRIVATE_, NDK_ 或 APP_开头的名字(内部使用)
  3. 小写名字(内部使用,例如‘my-dir’)

如果为了方便在 Android.mk 中定义自己的变量,建议使用 MY_前缀,一个小例子:

MY_SOURCES := foo.c
LOCAL_SRC_FILES += $(MY_SOURCES)

注意:

  1. = 是最基本的赋值,make会将整个mk文件展开后,再决定变量的值。也就是说,变量的值将会是整个mk文件最后被指定的值
  2. := 表示变量的值决定于它在mk文件中的位置,而不是整个mk展开后的最终值。
  3. ?= 是如果没有被赋值过就赋予等号后面的值
  4. += 是添加等号后面的值

宏函数

$(call command) 执行宏函数
my-dir:返回当前Android.mk文件所在的目录路径
all-subdir-makefiles:返回一个当前my-dir路径子目录中所有Android.mk列表
all-subdir-java-files:这个目录下的所有.java文件
this-makefile:返回当前makefile的路径,即这个函数调用的地方
parent-makefile:返回调用树中父makefile的路径,即包含当前makefile的makefile的路径

重点语法详解

include $(CLEAR_VARS)
CLEAR_VARS 由编译系统提供(可以在AOSP的build/core/config.mk 文件看到其定义,为 CLEAR_VARS:=$(BUILD_SYSTEM)/clear_vars.mk),其功能是清除除LOCAL_PATH外的其他LOCAL_XXX 变量 ( 如 LOCAL_MODULE , LOCAL_SRC_FILES 等)。因为所有的编译文件都在同一个 GNU Make 执行环境中,所有的变量都是全局的,所以我们需要先清空这些变量(LOCAL_PATH除外)。

LOCAL_MODULE_PATH :=$(TARGET_ROOT_OUT)
指定最后生成的模块的目标地址。
TARGET_ROOT_OUT:根文件系统,路径为out/target/product/generic/root
TARGET_OUT:system文件系统,路径为out/target/product/generic/system
TARGET_OUT_DATA:data文件系统,路径为out/target/product/generic/data

LOCAL_MODULE_TAGS :=user
user: 指该模块只在user版本下才编译
eng: 指该模块只在eng版本下才编译
tests: 指该模块只在tests版本下才编译
optional:指该模块在所有版本下都编译

include $(BUILD_SHARED_LIBRARY)
BUILD_SHARED_LIBRARY:位于build\core\shared_library.mk,负责收集自从上次调用 include $(CLEAR_VARS) 后的所有LOCAL_XXX信息并决定编译为动态库。
BUILD_STATIC_LIBRARY:编译为静态库。
BUILD_EXECUTABLE:编译为Native C可执行程序
BUILD_PREBUILT:该模块预先编译
BUILD_STATIC_JAVA_LIBRARY:静态jar包,使用.class文件打包,可以运行在任何Java虚拟机平台
BUILD_JAVA_LIBRARY:动态jar包,在静态jar包基础上使用.dex打包而成的jar文件。
BUILD_PACKAGE:编译APK

LOCAL_MODULE_CLASS := $(EXECUTABLES)
用于制定LOCAL_MODULE_PATH的路径
在编译为library,exe时,会默认就指定当前的LOCAL_MODULE_CLASS的值,比如EXECUTABLES、SHARED_LIBRARIES等。但当遇到include $(BUILD_PREBUILT)的预编译选项时不会默认指定LOCAL_MODULE_CLASS的值,需要在Android.mk中明确指定值,从而帮助系统确定LOCAL_MODULE_PATH的路径。

编译文件

#编译C/C++静态库
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := lib_hello_static #编译后的名称
LOCAL_SRC_FILES := hellos.c #编译的源文件
include $(BUILD_STATIC_LIBRARY) #指明要编译成静态库
    
#编译C/C++动态库
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := lib_hello_shared
LOCAL_SRC_FILES := hellod.c
include $(BUILD_SHARED_LIBRARY)

#编译C/C++应用程序
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := exe_hello
LOCAL_SRC_FILES := hellod.c
include $(BUILD_EXECUTABLE)    

#编译Java静态库
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_SRC_FILES := $(call all-subdir-java-files)
LOCAL_JAVA_LIBRARIES := android.test.runner #依赖的动态jar
LOCAL_MODULE := sample
include $(BUILD_STATIC_JAVA_LIBRARY)

Settings中的Android.mk文件,编译APK

以Settings作为分析对象,来分析一下Android.mk文件的内容:


LOCAL_PATH:= $(call my-dir) #LOCAL_PATH表示当前目录的地址,一般位于include $(CLEAR_VARS)之前
include $(CLEAR_VARS)#清除除了LOCAL_PATH以外的所有LOCAL_打头的变量

LOCAL_JAVA_LIBRARIES := bouncycastle conscrypt telephony-common ims-common#依赖动态jar  不会打包到APK,使用时从系统获取
LOCAL_STATIC_JAVA_LIBRARIES := android-support-v4 android-support-v13 jsr305#依赖静态jar  需要打包到APK中
LOCAL_MODULE_TAGS := optional#模块tag为optional,表示不管是选择了什么模式都会编译该模块

LOCAL_SRC_FILES := \  #本地代码文件
        $(call all-java-files-under, src) \
        src/com/android/settings/EventLogTags.logtags

LOCAL_RESOURCE_DIR := $(LOCAL_PATH)/res #本地资源文件

LOCAL_PACKAGE_NAME := Settings#模块名

LOCAL_CERTIFICATE := platform#模块证书签名

LOCAL_PRIVILEGED_MODULE := true#是否是特权文件

LOCAL_PROGUARD_FLAG_FILES := proguard.flags#使用代码混淆

ifneq ($(INCREMENTAL_BUILDS),)#判断是否进行增量编译
    LOCAL_PROGUARD_ENABLED := disabled
    LOCAL_JACK_ENABLED := incremental
endif
#include三个makefile文件,进项相关变量赋值
include frameworks/opt/setupwizard/navigationbar/common.mk
include frameworks/opt/setupwizard/library/common.mk
include frameworks/base/packages/SettingsLib/common.mk
#开始编译Settings模块,对应package.mk文件。感兴趣的可以进一步研究apk是怎么被编译出来的,里面还是很复杂的
include $(BUILD_PACKAGE)
# 如果使用的是mm或mmm命令单编Settings模块的话会额外include test目录下的Android.mk,用于编译测试模块。
# Use the following include to make our test apk.
ifeq (,$(ONE_SHOT_MAKEFILE))
include $(call all-makefiles-under,$(LOCAL_PATH))
endif
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,099评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,828评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,540评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,848评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,971评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,132评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,193评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,934评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,376评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,687评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,846评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,537评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,175评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,887评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,134评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,674评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,741评论 2 351

推荐阅读更多精彩内容

  • 隔着人群我老远就听到了一个小女孩在叽叽喳喳,声音如铜铃一般清脆,在安静的车厢里特别悦耳。她像鸟儿一样地和她的妈咪在...
    纽约蓝蓝阅读 535评论 0 4
  • 今天是个特殊的日子,一年前的今天,第一次见到波波,如今已经一年过去了,很多故事还在回旋。昨天晚上突然收到波波给我发...
    廷生阅读 585评论 0 2
  • 这两天晚上回到家,月月天天和姥姥出去卖灯笼了,舅舅说我们有毅力,我说没有什么毅力,月月觉得好玩,感觉有趣,所以才每...
    刘慢慢_9744阅读 502评论 0 1
  • 《遇》 就像千万年里,赏遍万水千山的景,喝过四海八荒的酒,一朝醉醒,一梦芳华。只为遇见你,从此,地老天荒,繁华永复...
    走过岁月的朝夕阅读 272评论 0 0