1、 Android Make脚本简记 原文地址: 原文作者: Email: zcatt@ Blog 一、 1、Build Layers Build Layers描述的是产品的硬件配置情况,据此make时选择不同的配置和模块。按照从上到下的顺序,Build Layer分成4层。 Layer sample Note Arch arm, x86 处理器的种类 Board - 板子类型的代号 Device - device配置的类型代号 Product - 具体产品的代号 2、添加应用
2、 2.1、一个例子 以calculator为例,app代码可以放到packages/apps/目录下边,一个app对应一个目录,此例,pakcages/apps/Calculator/,创建Android.mk,已去除多余的注释行。 Java代码 1. LOCAL_PATH := $(call my-dir) 2. include $(CLEAR_VARS) 3. LOCAL_MODULE_TAGS := optional 4. LOCAL_STATIC_JAVA_LIBRARIES := libarity 5. LOCAL_SRC_FILES := $(ca
3、ll all-java-files-under, src) 6. LOCAL_SDK_VERSION := current 7. LOCAL_PACKAGE_NAME := Calculator 8. include $(BUILD_PACKAGE) 9. include $(CLEAR_VARS) 10. LOCAL_PREBUILT_STATIC_JAVA_LIBRARIES := libarity:arity-2.1.2.jar 11. include $(BUILD_MULTI_PREBUILT) 12. # Use the folloing
4、include to make our test apk.
13. include $(call all-makefiles-under,$(LOCAL_PATH))
Android.mk中需要赋值的几个LOCAL_XXX变量,
LOCAL_PATH ,调用my-dir(在defination.mk中定义),得到当前路径,即,
5、r,如果要预装,则应定义core.mk; LOCAL_SRC_FILES ,app的所有源码,可以调用all-java-files-under得到,如果是java源码的话; LOCAL_PACKAGE_NAME ,package的名字,这个名字在脚本中将标识这个app或package; $(CLEAR_VARS) 指的是clear_vars.mk,脚本会清空所有LOCAL_xxx的变量,不影响后面这些变量的使用; $(BUILD_PACKAGE) 指的是package.mk 而,最后一句all-makefiles-under将会包含当前目录下所有的.mk脚本文件。 2.2、L
6、OCAL_XXX的列表 LOCAL_PATH,当前路径,必须定义; LOCAL_PACKAGE_NAME,必须定义,package的名字,这个名字在脚本中将标识app或package; LOCAL_MODULE_SUFFIX,不用定义,module的后缀,=.apk。 LOCAL_MODULE,不用定义,=$(LOCAL_PACKAGE_NAME)。 LOCAL_JAVA_RESOURCE_DIRS,不用定义。 LOCAL_JAVA_RESOURCE_FILES,不用定义。 LOCAL_MODULE_CLASS,不用定义。 LOCAL_MODULE_TAGS,可选定义。默认op
7、tional。取值范围user debug eng tests optional samples shell_ash shell_mksh LOCAL_ASSET_DIR,可选定义,推荐不定义。默认$(LOCAL_PATH)/assets LOCAL_RESOURCE_DIR,可选定义,推荐不定义。默认product package和device package相应的res路径和$(LOCAL_PATH)/res。 LOCAL_PROGUARD_ENABLED,可选定义,默认为full,如果是user或userdebug。取值full, disabled, custom。 full_a
8、ndroid_manifest,不用定义,=$(LOCAL_PATH)/AndroidManifest.xml。 LOCAL_EXPORT_PACKAGE_RESOURCES,可选定义,默认null。如果允许app的资源被其它模块使用,则设置true。 LOCAL_CERTIFICATE,可选定义,默认为testkey。最终是:private_key := $(LOCAL_CERTIFICATE).pk8 和 certificate := $(LOCAL_CERTIFICATE).x509.pem; 2.3、mm创建apk时的package.mk中变量分析 以Calculator
9、为例, 由LOCAL_PATH,LOCAL_PACKAGE_NAME导出变量LOCAL_MODULE,all_assets,all_assets,all_resources。 设置LOCAL_MODULE_CLASS=APPS,此值local-intermediates-dir会用到。 设置中间生成目录路径,中间路径将放置R.stamp文件。 package_expected_intermediates_COMMON := $(call local-intermediates-dir,COMMON) 这里COMMON是null,而LOCAL_MODULE_CLASS=APPS,所以
10、 package_expected_intermediates_COMMON=out/target/common/obj/$(LOCAL_MODULE_CLASS)/$(LOCAL_MODULE)_intermediates 即 package_expected_intermediates_COMMON=out/target/common/obj/APPS/Calculator_intermediates 设置 LOCAL_BUILT_MODULE_STEM := package.apk 而 LOCAL_BUILT_MODULE := $(built_module_path)/$(
11、LOCAL_BUILT_MODULE_STEM) @base_rules.mk
built_module_path := $(intermediates) @base_rules.mk
intermediates := $(call local-intermediates-dir) @java.mk
最终
LOCAL_BUILT_MODULE=out/target/product/
12、) 即 LOCAL_BUILT_MODULE=out/target/product/generic/obj/APPS/Calculator_intermediates/package.apk 由LOCAL_CERTIFICATE导出 private_key := $(SRC_TARGET_DIR)/product/security/$(LOCAL_CERTIFICATE).pk8 certificate := $(SRC_TARGET_DIR)/product/security/$(LOCAL_CERTIFICATE).x509.pem LOCAL_CERTIFICATE默认为
13、testkey。 2.4、package.mk中定义的几个PACKAGE.xxx变量 PACKAGES.$(LOCAL_PACKAGE_NAME).PRIVATE_KEY := $(private_key) PACKAGES.$(LOCAL_PACKAGE_NAME).CERTIFICATE := $(certificate) PACKAGES.$(LOCAL_PACKAGE_NAME).OVERRIDES := $(strip $(LOCAL_OVERRIDES_PACKAGES)) PACKAGES.$(LOCAL_PACKAGE_NAME).RESOURCE_FILES
14、 $(all_resources) PACKAGES := $(PACKAGES) $(LOCAL_PACKAGE_NAME) 全编译时,PACKAGES变量将会记录遍历到的packages。 二、 1、java.mk分析 选取APPS场景,以Calculator为例说明。 LOCAL_JAVA_LIBRARIES=true时,Android.mk中不能定义LOCAL_SDK_VERSION。 当LOCAL_SDK_VERSION=current时,LOCAL_JAVA_LIBRARIES=android_stubs_current。 package.mk中定
15、义LOCAL_BUILT_MODULE_STEM=package.apk。
两个中间目录的路径,即对应的obj目录下APPS/
16、java.mk有增添了7个。 LOCAL_INTERMEDIATE_TARGETS += \ $(full_classes_jar) \ $(full_classes_compiled_jar) \ $(full_classes_emma_jar) \ $(full_classes_full_names_jar) \ $(full_classes_stubs_jar) \ $(full_classes_jarjar_jar) \ $(built_dex) 此例中,具体值是 LOCAL_INTERMEDIATE_TARG
17、ETS= out/target/common/obj/APPS/Calculator_intermediates/src/R.stamp @defined in package.mk out/target/common/obj/APPS/Calculator_intermediates/classes.jar @full_classes_jar out/target/common/obj/APPS/Calculator_intermediates/classes-full-debug.jar @full_c
18、lasses_compiled_jar out/target/common/obj/APPS/Calculator_intermediates/emma_out/lib/classes-full-debug.jar @full_classes_emma_jar out/target/common/obj/APPS/Calculator_intermediates/classes-full-names.jar @full_classes_full_names_jar out/target/common/obj/APPS/Calculator_intermedi
19、ates/stubs.jar @full_classes_stubs_jar out/target/common/obj/APPS/Calculator_intermediates/classes-jarjar.jar @full_classes_jarjar_jar out/target/common/obj/APPS/Calculator_intermediates/classes.dex @built_dex java.mk随后include base_rules.mk 后面处理了EMMA,PROGUARD在enable/disable情况下的
20、动作 最后定义的target, $(LOCAL_MODULE)-findbugs因为prebuilt/common下还没有findbugs,目前不可用。 java.mk还定义了几个特别的变量, ALL_MODULES.$(LOCAL_MODULE).PROGUARD_ENABLED:=$(LOCAL_PROGUARD_ENABLED) ALL_MODULES.$(LOCAL_MODULE).CHECKED := $(full_classes_compiled_jar) ALL_MODULES.$(LOCAL_MODULE).STUBS := $(full_classes_st
21、ubs_jar) 2、base_rules.mk的分析 续1(Java.mk)的场景。 提取变量my_prefix:=TARGET_ LOCAL_MODULE_TAGS在Android.mk或package.mk中已经设定,默认是optional。 确认LOCAL_MODULE_PATH,默认$($(my_prefix)OUT$(use_data)_$(LOCAL_MODULE_CLASS)),此例中是out/target/product/generic/system/app 设定module_id := MODULE.$(TARGET).$(LOCAL_MODULE_CLA
22、SS).$(LOCAL_MODULE),此例MODULE.TARGET.APPS.Calculator。 设定中间目录路径intermediates,intermediates.COMMON,参见1. 设定LOCAL_MODULE_STEM=$(LOCAL_MODULE),此例,Calculator。LOCAL_INSTALLED_MODULE_STEM=Calculator.apk。 LOCAL_INTERMEDIATE_TARGETS追加上package.apk,参见1. 处理aidl,转为java,放在intermediates.COMMON下的目录中。 处理l
23、ogtag,转为java,放在intermediates.COMMON下的目录中。 确定java_sources,这包括android.mk中包含的,aidl和logtag生成的。 处理java_resource_files 处理了java lib相关 定义clean-$(LOCAL_MODULE) target, 可以删除app/package的生成文件,包括$(PRIVATE_CLEAN_FILES),$(LOCAL_BUILT_MODULE),$(LOCAL_INSTALLED_MODULE),$(intermediates),$(intermediates.CO
24、MMON) 还定义了$(LOCAL_MODULE) target, 几个变量的值 LOCAL_MODULE=Calculator LOCAL_BUILT_MODULE=out/target/product/generic/obj/APPS/Calculator_intermediates/package.apk LOCAL_INSTALLED_MODULE=out/target/product/generic/system/app/Calculator.apk 最后定义了几个ALL_MODULES变量。 ALL_MODULES.$(LOCAL_MODULE).CLASS
25、 ALL_MODULES.$(LOCAL_MODULE).PATH ALL_MODULES.$(LOCAL_MODULE).TAGS ALL_MODULES.$(LOCAL_MODULE).CHECKED ALL_MODULES.$(LOCAL_MODULE).BUILT ALL_MODULES.$(LOCAL_MODULE).INSTALLED ALL_MODULES.$(LOCAL_MODULE).REQUIRED ALL_MODULES.$(LOCAL_MODULE).EVENT_LOG_TAGS 3、multi_prebuilt.mk的分析 续1的场景。
26、mulit_prebuilt.mk顾名思义就是多次调用prebuilt.mk,对几种明确的prebuilt library完成需要的copy操作。
multi_prebuilt.mk定义了命令auto-prebuilt-boilerplate。入口有6个参数
# $(1): file list, "
27、BUILT_MODULE_STEM 根据这6个参数,命令确定 LOCAL_IS_HOST_MODULE LOCAL_MODULE_CLASS OVERRIDE_BUILT_MODULE_PATH LOCAL_UNINSTALLABLE_MODULE LOCAL_MODULE LOCAL_SRC_FILES LOCAL_BUILT_MODULE_STEM LOCAL_MODULE_SUFFIX 并调用prebuilt.mk multi_prebuilt.mk中分别对下面5中lib调用了auto-prebuilt-boilerplate。 prebuilt_stat
28、ic_libs := $(filter %.a,$(LOCAL_PREBUILT_LIBS)) prebuilt_shared_libs := $(filter-out %.a,$(LOCAL_PREBUILT_LIBS)) prebuilt_executables := $(LOCAL_PREBUILT_EXECUTABLES) prebuilt_java_libraries := $(LOCAL_PREBUILT_JAVA_LIBRARIES) prebuilt_static_java_libraries := $(LOCAL_PREBUILT_STATIC_JAVA_LIBRAR
29、IES) 4、prebuilt.mk的分析 续1的场景。 首先,include base_rules.mk 定义 PACKAGES.$(LOCAL_MODULE).OVERRIDES 第二步,如果是APPS类型,则zipalign,并拷贝到中间路径$(intermediates)。不是APPS,则不做zipalign。 本例是JAVA_LIBRARY类型,目的路径out/target/common/obj/JAVA_LIBRARIES/libarity_intermediates/javalib.jar,注意其中的libarity和javalib.ja
30、r。
最后检查 signed情况。
三、
1.findleaves.py的分析
main.mk中调用了findleaves.py,得到所有子目录下Android.mk文件的路径。
subdir_makefiles := \
$(shell build/tools/findleaves.py --prune=out --prune=.repo --prune=.git $(subdirs) Android.mk)
$(subdirs)一般编译中取值$(TOP)。
使用方法,
Usage: %(progName)s [ 31、list> 32、有子目录查找Android.mk文件,如果有,则加入到返回列表中。
2. pathmap.mk的分析
pathmap.mk中定义了一个列表pathmap_INCL,列表中每项是"短名:路径"对。命令include-path-for使用这个pathmap_INCL列表,输入短名,得到路径。你可以在这个列表中添加自己的对。使用$(call include-path-for, <短名>)就可以得到路径。
另外,定义了FRAMEWORKS_BASE_JAVA_SRC_DIRS,含有frameworks/base目录下含java文件的所有目录。
3. config.mk的分析
首先 33、包含pathmap.mk, 其次,定义了一些变量,例如通用的编译参数,package的后缀名等。
随后包含buildspec.mk。
接着包含envsetup.mk。
然后包含$(board_config_mk)。$(board_config_mk)是位于build/target/board/$(TARGET_DEVICE)/,device/*/$(TARGET_DEVICE)/,或vendor/*/$(TARGET_DEVICE)/目录下的BoardConfig.mk文件。
-------TODO
4. buildspec.mk的分析
buildspec.mk是用户应 34、当配置的脚本文件,模板可以使用buildspec.mk.default,放到$(TOP)下。
在buildspec.mk中,用户应该配置好主要的参数,例如 TARGET_PRODUCT, TARGET_BUILD_VARIANT, CUSTOM_MODULES, TARGET_SIMULATOR, TARGET_BUILD_TYPE, CUSTOM_LOCALES, 和BUILD_ENV_SEQUENCE_NUMBER等。
如果不使用buildspec.mk配置参数,也可以使用环境变量的形式。若不配置参数,那么android会使用默认的参数。
5. envsetup.mk的分析
35、
首先包含进version_defaults.mk,定义好一些版本相关的变量。参见version_defaults.mk。
定义CORRECT_BUILD_ENV_SEQUENCE_NUMBER,这个数字用于buildspec.mk更新时的提醒,应该同buildspec.mk中的或环境变量中的BUILD_ENV_SEQUENCE_NUMBER相等。一般不用关注。
随后检查TARGET_PRODUCT,若为空,则置generic。TARGET_PRODUCT应当在buildspec.mk或环境变量中已经定义好。
再检查TARGET_BUILD_VARIANT,若为空,则置eng 36、TARGET_BUILD_VARIANT应当在buildspec.mk或环境变量中已经定义好。
然后包含进product_config.mk。
接着,检查$(TARGET_BUILD_VARIANT),取值范围是eng user userdebug tests。
随后判定HOST_OS(linux),HOST_ARCH(x86)
接着,确定TARGET_ARCH和TARGET_OS,若没有定义,则取默认值。
TARGET_ARCH := arm
TARGET_OS := linux
接着,确定TARGET_BUILD_TYPE,若没有定义,则取默认值。
T 37、ARGET_BUILD_TYPE := release
接着,确定OUT_DIR。OUT_DIR是存放中间文件和最终结果的地方。若没有定义,则取默认值。
OUT_DIR := $(TOPDIR)out
随后,定义了一些列的路径变量
DEBUG_OUT_DIR,TARGET_OUT_ROOT_release,TARGET_OUT_ROOT_debug,TARGET_OUT_ROOT,BUILD_OUT,PRODUCT_OUT,TARGET_COMMON_OUT_ROOT,等等。
6、version_defaults.mk的分析
version_defaults.mk是检 38、查一些跟版本相关的变量是否定义,如果未定义,则使用默认值。这些变量包括
PLATFORM_VERSION,默认AOSP
PLATFORM_SDK_VERSION,默认8
PLATFORM_VERSION_CODENAME,默认AOSP
DEFAULT_APP_TARGET_SDK,默认AOSP
BUILD_ID,默认UNKNOWN
BUILD_NUMBER,默认eng.$(USER).$(shell date +%Y%m%d.%H%M%S)的形式。
version_defaults.mk首先包含进build_id.mk。用户应当配置build_id.mk,而不应该改动ve 39、rsion_defaults.mk文件。
然后检查上述变量,如未定义则赋值默认值。
7. build_id.mk的分析
用户可以在build_id.mk中定义这样几个参数,
PLATFORM_VERSION
PLATFORM_SDK_VERSION
PLATFORM_VERSION_CODENAME
DEFAULT_APP_TARGET_SDK
BUILD_ID
BUILD_NUMBER
这些参数最终将出现build.prop中。
Froyo的build_id.mk中定义了2个变量,
BUILD_ID,通常用于说明分支branch的,默认的是OPENMA 40、STER,用户应该配置这个参数。
DISPLAY_BUILD_NUMBER,在TARGET_BUILD_VARIANT=user的版本中,build.prop中是ro.build.id是显示成$(BUILD_ID).$(BUILD_NUMBER),还是显示成$(BUILD_ID)形式。设成true,则显示前者。
8. product_config.mk的分析
make PRODUCT- 41、RIANT:= 42、dream make installclean
使用make PRODUCT- 43、ed_goals): $(MAKECMDGOALS)
使用make APP- 44、CT.XXX,和DEVICE.XXX变量。
接着,使用$(call import-products, $(get-all-product-makefiles))遍历Prodcut相关的AndroidProducts.mk文件,读入PRODCUTS.xxx变量。可以去掉文件中下面两句话的注释符,查看。
#$(dump-products)
#$(error done)
随后,使用PRODCUT.xxx和TARGET_PRODUCT,得到INTERNAL_PRODUCT信息,即指定product的路径。
再由INTERNAL_PRODUCT得到TARGET_DEVICE, PR 45、ODUCT_LOCALES, PRODUCT_BRAND, PRODUCT_MODEL, PRODUCT_MANUFACTURER, PRODUCT_DEFAULT_WIFI_CHANNELS, PRODUCT_POLICY, PRODUCT_COPY_FILES, PRODUCT_PROPERTY_OVERRIDES, PRODUCT_PACKAGE_OVERLAYS, DEVICE_PACKAGE_OVERLAYS, PRODUCT_TAGS, PRODUCT_OTA_PUBLIC_KEYS。
由PRODUCT_LOCALES导出PRODUCT_AAPT_CONFIG。
ADDITIO 46、NAL_BUILD_PROPERTIES中追加PRODUCT_PROPERTY_OVERRIDES中的值。
上面所说的这些值,实际上都是在product的mk文件中定义。
9. node_fns.mk的分析
定义了一些命令。这些命令在product.mk,device.mk,和product_config.mk中会使用。这里重点说明import-nodes。
import-nodes需要3个入口参数:
$(1)是一个字串,是输出变量的主干名。例如”PRODUCTS"和”DEVICES“。
$(2)是一个makefile文件列表,这些文件中应该含有对$(3)中变量的定义。 47、
$(3)是一个变量列表。
import-nodes会创建这样形式的变量,以$(1)="PRODUCTS", $(2)中含有"build/target/product/core.mk", $(3)中含有"PRODUCT_NAME", 而且core.mk中定义了PRODUCT_NAME:=core为例,
PRODUCT.build/target/product/core.mk.PRODUCT_NAME:=core
import-nodes中还考虑了inherit的问题,如果某个PRODUCTS.XXX变量的值中有‘@inherit: 48、把那个mk文件中相应的变量的属性添加到PRODUCTS.XXX中。'@inherit: 49、vice/和vendor/, 包括子目录,以及build/target/product/下的AndroidProducts.mk文件列表。
get-all-product-makefiles这个命令会得到所有$(_find-android-products-files)的AndroidProducts.mk文件中PRODUCT_MAKEFILES变量定义的mk文件。
_product_var_list对应的是import-nodes命令的$(3), 定义了会生成那些PRODUCT属性名的变量。这些变量实际也是在product的mk文件中要考虑定义的。
_product_var_list := \
PRODUCT_NAME \
PRODUCT_MODEL \
PRODUCT_LOCALES \
PRODUCT_PACKAGES \
PRODUCT_DEVICE \
PRODUCT_MANUFACTURER \
PRODUCT_BRAND \
PRODUCT_PROPERTY_OVERRIDES \
PRODUCT_






