Android核心原理与系统级应用高效开发
上QQ阅读APP看本书,新人免费读10天
设备和账号都新为新人

2.2 各种部件的构建

一般情况下,在使用了envsetup进行环境设置后,可以通过mmm进行编译一个工程,如下所示:

$ mmm {project_path}

2.2.1 Android.mk的语法

Android.mk是Android工程的管理文件,其作用基本等同于Linux环境中的Makefile。在语法上,Android.mk和普通的Makefile略有不同,主要区别是Android.mk包含一些Android编译系统公共的宏。

Android.mk中选项参考以下文件路径:

build/core/config.mk

各个选项的默认值在以下文件中定义:

build/core/base_rules.mk

Android.mk文件只处理从根目录开始找到的第一个Android.mk文件,如果需要递归,需要在当前目录的Android.mk文件中做如下处理:

# Android.mk文件的最后
include $(call all-makefiles-under,$(LOCAL_PATH))

增加以上依然内容后,如果当前目录的子目录中还有Android.mk,也会对其做出处理。

在一个Android.mk中也可以生成多个目标:可执行程序、动态库、静态库或者Android应用程序包。Android.mk文件可以处理多个内容:

include $(CLEAR_VARS)
# 处理第一个内容
include $(CLEAR_VARS)
# 处理第二个内容

2.2.2 各种部件的构建方式

1.可执行程序

可执行程序是Linux标准的ELF格式文件的一种,具有main入口,可以直接作为一个进程执行。在Android.mk中编译一个可执行程序的模板如下所示:

LOCAL_PATH := $(my-dir)
# Test Exe
include $(CLEAR_VARS)
LOCAL_MODULE_TAGS := eng
LOCAL_SRC_FILES:= \
    main.c
LOCAL_MODULE:= test_exe
LOCAL_C_INCLUDES :=
LOCAL_STATIC_LIBRARIES :=
LOCAL_SHARED_LIBRARIES := libc
include $(BUILD_EXECUTABLE)

编译一个可执行程序,需要在LOCAL_SRC_FILES中加入源文件路径(相对于当前的Android.mk目录的路径),在LOCAL_C_INCLUDES中加入所需要包含的头文件路径;在LOCAL_STATIC_LIBRARIES中加入所需要连接的静态库(*.a)的名称;在LOCAL_SHARED_LIBRARIES加入所需要连接的动态库(*.so)的名称。

LOCAL_MODULE表示模块最终的名称。最后使用include $(BUILD_EXECUTABLE)表示以一个可执行程序的方式进行编译。在本例中,LOCAL_MODULE被定义为test_exe,因此最终生成可执行程序的名称是test_exe。

一个可执行程序编译后生成独立的目标目录在out/target/product/中,路径如下所示:

<TARGET_PRODUCT>/obj/EXECUTABLES/< LOCAL_MODULE >

2.静态库

静态库,也称之为归档文件,在Linux中扩展名通常为.a。在Android.mk中编译一个静态库(归档文件)的模板如下所示:

LOCAL_PATH := $(my-dir)
# Test static lib
include $(CLEAR_VARS)
LOCAL_MODULE_TAGS := eng
LOCAL_SRC_FILES:= \
    hello.c
LOCAL_MODULE:= libtest_static
LOCAL_C_INCLUDES :=
LOCAL_STATIC_LIBRARIES :=
LOCAL_SHARED_LIBRARIES := libc
include $(BUILD_STATIC_LIBRARY)

编译一个静态库,基本的内容和编译可执行程序相似,区别在于使用include$(BUILD_STATIC_LIBRARY) 表示编译静态库。在本例中,LOCAL_MODULE被定义为libtest_static,因此最终生成静态库的名称是libtest_static.a。

一个静态库编译后生成的独立的目标目录在out/target/product/中,路径如下所示:

<TARGET_PRODUCT>/obj/STATIC_LIBRARIES/< LOCAL_MODULE >

3.动态库

动态库,也称之为共享库,是Linux标准的ELF格式文件的一种,在Linux中扩展名通常为.so。在Android.mk中编译一个动态库(共享库)的模板如下所示:

LOCAL_PATH := $(my-dir)
# Test shared lib
include $(CLEAR_VARS)
LOCAL_MODULE_TAGS := eng
LOCAL_SRC_FILES:= \
    hello.c
LOCAL_MODULE:= libtest_shared
TARGET_PRELINK_MODULE := false
LOCAL_C_INCLUDES :=
LOCAL_STATIC_LIBRARIES :=
LOCAL_SHARED_LIBRARIES := libc
include $(BUILD_SHARED_LIBRARY)

编译一个动态库,基本的内容和编译可执行程序、静态库相似,区别在于使用include$(BUILD_SHARED_LIBRARY) 表示编译动态库。在本例中,LOCAL_MODULE被定义为libtest_shared,因此最终生成动态库的名称是libtest_shared.so。

一个动态库编译后生成的独立的目标目录在out/target/product/中,路径如下所示:

<TARGET_PRODUCT>/obj/STATIC_LIBRARIES/< LOCAL_MODULE >

4.Android应用程序包

Android应用程序包是一种特殊的文件,通常以apk为扩展名。在Android.mk中编译一个应用程序包的模板如下所示:

LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE_TAGS := eng
LOCAL_SRC_FILES := $(call all-subdir-java-files)
LOCAL_PACKAGE_NAME := TestApplication
include $(BUILD_PACKAGE)

这里使用BUILD_PACKAGE宏表示编译apk,而LOCAL_SRC_FILES使用自动查找的方法,将找到所有的Java文件进行编译。

在源代码环境中编译和在SDK中编译应用程序包略有不同,涉及的目录主要有以下两个。

● out/target/common/obj/APPS:通用Java字节码目录。

● out/target/product/<TARGET_PRODUCT>/obj/APPS:Android应用包目录。

每个包在这两个目录中均具有名为{LOCAL_PACKAGE_NAME}_intermediates/的独立目录。

例如,对于SkeletonApp包的编译,公共目录的生成结构如下所示:

out/target/common/obj/APPS/SkeletonApp_intermediates/
|-- classes-full-debug.jar
|-- classes.jar
|-- classes-jarjar.jar
|-- emma_out
|   `-- lib
|      `-- classes-jarjar.jar
|-- noproguard.classes.dex
|-- noproguard.classes.jar
|-- noproguard.classes-with-local.dex
|-- proguard_options
|-- public_resources.xml
`-- src                              [自动生成的Java文件目录]
  |-- com
  |   `-- example
  |      `-- android
  |          `-- skeletonapp
  |             `-- R.java
  `-- R.stamp

一般都会根据资源目录res自动生成R.java文件,在具有aidl时还会生成由它自动生成的java源代码文件。

SkeletonApp的目标目录out/target/product/<TARGET_PRODUCT>/obj/APPS中为SkeletonApp_intermediates。其中包含3个apk包,如下所示。

● package.apk.unsigned:第一步生成的没有签名的apk包。

● package.apk.unaligned:第二步生成具有签名,但是没有对齐的apk包。

● package.apk:第三步生成最终的apk包。

在编译应用程序包的时候,有一个额外的宏可以控制编译的行为,如下所示:

WITH_DEXPREOPT := true

如果WITH_DEXPREOPT被定义为true,一个应用程序包将由两个部分组成:一个是不包含Java字节码(classes.dex)的apk文件,一个是名称为classes.odex的字节码文件。两个文件同时预置在系统中依然可以构成能运行的应用程序。但是,这种生成的结果就不能再进行动态的安装了。

提示:WITH_DEXPREOPT可以在每个应用程序包的Android.mk文件中定义,也可以全局定义。Android 4.x之前的版本此宏的全局默认没有打开,Android 4.x中则将此宏全局默认定义为true。

5.各部分内容的生成路径和安装

可执行程序、静态库(*.a)、动态库(*.so)和Android应用程序包(*.apk)生成的编译结果分别放在以下的目录中。

● out/target/product/<TARGET_PRODUCT>/obj/EXECUTABLES/。

● out/target/product/<TARGET_PRODUCT>/obj/STATIC_LIBRARIES/。

● out/target/product/<TARGET_PRODUCT>/obj/SHARED_LIBRARIES/。

● out/target/product/<TARGET_PRODUCT>/obj/APPS/。

几种模块的目标的目录分别为如下所示。

● 可执行程序:<LOCAL_MODULE>__intermediates。

● 静态库:<LOCAL_MODULE>_static_intermediates。

● 动态库:<LOCAL_MODULE>_shared_intermediates。

● Android应用程序包:<LOCAL_PACKAGE_NAME>_intermediates。

其中<LOCAL_MODULE>和<LOCAL_PACKAGE_NAME>将决定每一个工程私有目录的名称。对于可执行程序和动态库,一般在LINK子目录中是带有符号的库(没有经过符号剥离)。

编写Android.mk文件的过程中,有以下两个注意点。

(1)Android的编译过程根据扩展名自动识别文件类型,统一加入LOCAL_SRC_FILES中即可。

(2)在编译过程中LOCAL_SRC_FILES尽量不要使用../source.cpp类型的文件路径,因为各个源代码的编译结果也会放置到相对路径中,使用这种路径的文件将会被放置在编译目录的上级目录中。

在编译过程中,可以编译目标机的内容,也可以编译主机的内容。以上的例子是编译目标机的内容,可执行程序、静态库、动态库和Android应用程序包使用的编译宏分别如下所示:

include $(BUILD_EXECUTABLE)             # 可执行程序
include $(BUILD_STATIC_LIBRARY)         # 静态库
include $(BUILD_SHARED_LIBRARY)         # 动态库
include $(BUILD_PACKAGE)                # Android应用程序包

编译主机的内容时,可执行程序、静态库、动态库使用宏分别如下所示:

include $(BUILD_HOST_EXECUTABLE)        # 主机可执行程序
include $(BUILD_HOST_STATIC_LIBRARY)    # 主机静态库
include $(BUILD_HOST_SHARED_LIBRARY)    # 主机动态库

几种模块默认的安装路径如下所示。

● Android应用程序包:TARGET_OUT/app。

● 动态库:TARGET_OUT/lib。

● 可执行程序:TARGET_OUT/bin。

● 静态库不需要安装,因此在目标文件系统中没有静态库。

在Android.mk文件中,可以指定最后的目标安装路径,使用LOCAL_MODULE_PATH宏来指定模块的安装路径。

增加可以安装到不同文件系统路径的方式如下所示:

LOCAL_MODULE_PATH := $(TARGET_ROOT_OUT)
LOCAL_UNSTRIPPED_PATH := $(TARGET_ROOT_OUT_UNSTRIPPED)

不同的文件系统路径使用以下的宏进行选择。

TARGET_ROOT_OUT表示根文件系统,路径为

out/target/product/<TARGET_PRODUCT>/root

TARGET_OUT表示system文件系统,路径为

out/target/product/<TARGET_PRODUCT>/system

TARGET_OUT_DATA表示data文件系统,路径为

out/target/product/<TARGET_PRODUCT>/data

如果需要进一步指定子目录,可以使用这3个宏加上其中子目录的方式。如果不指定LOCAL_MODULE_PATH,安装到默认的目录中。

2.2.3 预编译内容的安装

在各个部分的构建过程中,除了编译各种内容外,有时还需要向目标文件系统复制一些文件,例如运行时配置脚本、资源文件、预制的程序和库等,也有时需要在目标文件系统中创建子目录。预编译内容的安装有两种方法,一种是使用命令向文件系统中复制,另一种是使用Android的预编译模板。

1.使用命令安装

使用命令的安装方式,实际上就是在Android.mk中调用主机命令行的程序,执行向文件系统中复制等工作。

在Android.mk中,使用命令行进行目录创建和安装的方法如下所示:

LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)
copy_from := \
A.txt \
B.txt
copy_to := $(addprefix $(TARGET_OUT)/txt/,$(copy_from))
$(copy_to) : PRIVATE_MODULE := txt
$(copy_to) : $(TARGET_OUT)/txt/% : $(LOCAL_PATH)/% | $(ACP)
$(transform-prebuilt-to-target)
ALL_PREBUILT = $(copy_to)
DIRS := $(addprefix $(TARGET_OUT)/, \
txt \
$(DIRS):
@echo Directory: $@
@mkdir -p $@

其中执行的echo和mkdir都是标准的Shell命令,$(ACP)是Android中对cp命令的一个封装(Android cp),执行的也是复制工作。此时,Android.mk将会进行如下的工作。

(1)在system文件系统(TARGET_OUT)中创建目录txt。

(2)将当前路径下的A.txt和B.txt文件复制到system/txt目录中。

2.使用预编译模板的安装

Android具有一个特殊的预编译模板:BUILD_PREBUILT,使用这个模板可以将内容复制到目标系统中,也可以自动建立子目录。

在Android.mk中,使用预编译模板进行目录创建和安装的方法如下所示:

LOCAL_PATH := $(my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := target.txt
LOCAL_MODULE_CLASS := TEXT
LOCAL_MODULE_TAGS := eng
LOCAL_MODULE_PATH := $(TARGET_OUT)/text
LOCAL_SRC_FILES := source.txt
include $(BUILD_PREBUILT)

此时,LOCAL_SRC_FILES和LOCAL_MODULE两个变量被赋予了不同的含义,前者表示需要复制的源,后者表示要复制的目标命令。LOCAL_MODULE_PATH依然用于指定目标的路径。LOCAL_MODULE_CLASS表示模块的类型,其中可以使用自定义的字符串。

此时,Android.mk将会进行如下的动作:

(1)在system文件系统(TARGET_OUT)中创建目录txt。

(2)将当前路径下的source.txt复制为system/txt目录中的target.txt。

在进行处理的过程中,也会生成模块的中间目录,路径为:

out/target/product/<TARGET_PRODUCT>/obj/text/target.txt_intermediates/

其中将具有target.txt文件,text来自于LOCAL_MODULE_CLASS的定义,实际上表示的是一个在obj目录中的新目标的类型。