`

Android编译系统详解(一)

阅读更多

++++++++++++++++++++++++++++++++++++++++++

本文系本站原创,欢迎转载! 转载请注明出处:

http://blog.csdn.net/mr_raptor/article/details/7539978

++++++++++++++++++++++++++++++++++++++++++

Android编译系统详解(一):http://blog.csdn.net/mr_raptor/article/details/7539978

Android编译系统详解(二):http://blog.csdn.net/mr_raptor/article/details/7540066

Android编译系统详解(三):http://blog.csdn.net/mr_raptor/article/details/7540730


Android的优势就在于其开源,手机和平板生产商可以根据自己的硬件进行个性定制自己的手机产品,如小米,LePhone,M9等,因此,在我们在对Android的源码进行定制的时候,很有必要了解下,Android的编译过程。

如果你从来没有做过Android代码的编译,那么最官方的编译过程就是查看Android的官方网站:http://source.android.com/source/building.html

但是,这儿只是告诉你了如何去编译一个通用的系统,并没有详细告诉你细节,我们跟着编译过程来了解下。

+--------------------------------------------------------------------------------------------------------------------+

本文使用Android版本为2.1,采用开发板为华清远见研发的FS_S5PC100 A8开发板。

+--------------------------------------------------------------------------------------------------------------------+

按照google给出的编译步骤如下:

1> source build/envsetup.sh:加载命令

2> lunch:选择平台编译选项

3> make:执行编译

我们按照编译步骤来分析编译过程的细节,最终添加自己的平台产品的编译选项。


1. source build/envsetup.sh

这个命令是用来将envsetup.sh里的所有用到的命令加载到环境变量里去,我们来分析下它。

envsetup.sh里的主要命令如下:

function help()                  # 显示帮助信息
function get_abs_build_var()           # 获取绝对变量
function get_build_var()             # 获取绝对变量
function check_product()             # 检查product
function check_variant()             # 检查变量
function setpaths()                # 设置文件路径
function printconfig()              # 打印配置
function set_stuff_for_environment()       # 设置环境变量
function set_sequence_number()         # 设置序号
function settitle()                # 设置标题
function choosetype()               # 设置type
function chooseproduct()             # 设置product
function choosevariant()             # 设置variant
function tapas()                 # 功能同choosecombo
function choosecombo()              # 设置编译参数
function add_lunch_combo()            # 添加lunch项目
function print_lunch_menu()            # 打印lunch列表
function lunch()                 # 配置lunch
function m()                   # make from top
function findmakefile()              # 查找makefile
function mm()                   # make from current directory
function mmm()                  # make the supplied directories
function croot()                 # 回到根目录
function cproj()
function pid()
function systemstack()
function gdbclient()
function jgrep()                 # 查找java文件
function cgrep()                 # 查找c/cpp文件
function resgrep()
function tracedmdump()
function runhat()
function getbugreports()
function startviewserver()
function stopviewserver()
function isviewserverstarted()
function smoketest()
function runtest()
function godir ()                 # 跳到指定目录 405

# add_lunch_combo函数被多次调用,就是它来添加Android编译选项
# Clear this variable. It will be built up again when the vendorsetup.sh
406 # files are included at the end of this file.
# 清空LUNCH_MENU_CHOICES变量,用来保存编译选项
407 unset LUNCH_MENU_CHOICES
408 function add_lunch_combo()
409 {
410 local new_combo=$1# 获得add_lunch_combo被调用时的参数
411 local c
# 依次遍历LUNCH_MENU_CHOICES里的值,其实该函数第一次调用时,该值为空
412 for c in ${LUNCH_MENU_CHOICES[@]} ; do
413 if [ "$new_combo" = "$c" ] ; then# 如果参数里的值已经存在于LUNCH_MENU_CHOICES变量里,则返回
414 return
415 fi
416 done
# 如果参数的值不存在,则添加到LUNCH_MENU_CHOICES变量里
417 LUNCH_MENU_CHOICES=(${LUNCH_MENU_CHOICES[@]} $new_combo)
418 }


# 这是系统自动增加了一个默认的编译项 generic-eng
420 # add the default one here
421 add_lunch_combo generic-eng# 调用上面的add_lunch_combo函数,将generic-eng作为参数传递过去
422
423 # if we're on linux, add the simulator. There is a special case
424 # in lunch to deal with the simulator
425 if [ "$(uname)" = "Linux" ] ; then
426 add_lunch_combo simulator
427 fi

# 下面的代码很重要,它要从vendor目录下查找vendorsetup.sh文件,如果查到了,就加载它
1037 # Execute the contents of any vendorsetup.sh files we can find.
1038 for f in `/bin/ls vendor/*/vendorsetup.sh vendor/*/build/vendorsetup.sh 2> /dev/null`
1039 do
1040 echo "including $f"
1041 . $f # 执行找到的脚本,其实里面就是厂商自己定义的编译选项
1042 done
1043 unset f

 

envsetup.sh其主要作用如下:

  1. 加载了编译时使用到的函数命令,如:help,lunch,m,mm,mmm等
  2. 添加了两个编译选项:generic-eng和simulator,这两个选项是系统默认选项
  3. 查找vendor/<-厂商目录>/和vendor/<厂商目录>/build/目录下的vendorsetup.sh,如果存在的话,加载执行它,添加厂商自己定义产品的编译选项
其实,上述第3条是向编译系统添加了厂商自己定义产品的编译选项,里面的代码就是:add_lunch_combo xxx-xxx。

 

根据上面的内容,可以推测出,如果要想定义自己的平台产品编译项,简单的办法是直接在envsetup.sh最后添加上add_lunch_combo myProduct-eng,当然这么做,不太符合上面代码最后的本意,我们还是老实的在vendor目录下创建自己公司名字,然后在公司目录下创建一个新的vendorsetup.sh,在里面添加上自己的产品编译项

 

  1. #mkdirvendor/farsight/
  2. #touchvendor/farsight/vendorsetup.sh
  3. #echo"add_lunch_combofs100-eng">vendor/farsight/vendorsetup.sh

这样,当我们在执行source build/envsetup.sh命令的时候,可以在shell上看到下面的信息:

 

 

  1. includingvendor/farsight/vendorsetup.sh


2. 按照android官网的步骤,开始执行lunch full-eng

 

当然如果你按上述命令执行,它编译的还是通用的eng版本系统,不是我们个性定制系统,我们可以执行lunch命令,它会打印出一个选择菜单,列出可用的编译选项

如果你按照第一步中添加了vendorsetup.sh那么,你的选项中会出现:

 

  1. You'rebuildingonLinux
  2. generic-engsimulatorfs100-eng
  3. Lunchmenu...pickacombo:
  4. 1.generic-eng
  5. 2.simulator
  6. 3.fs100-eng

其中第3项是我们自己添加的编译项。

 

lunch命令是envsetup.sh里定义的一个命令,用来让用户选择编译项,来定义Product和编译过程中用到的全局变量。

我们一直没有说明前面的fs100-eng是什么意思,现在来说明下,fs100是我定义的产品的名字,eng是产品的编译类型,除了eng外,还有user, userdebug,分别表示:

eng: 工程机,

user:最终用户机

userdebug:调试测试机

tests:测试机

由此可见,除了eng和user外,另外两个一般不能交给最终用户的,记得m8出来的时候,先放出了一部分eng工程机,然后出来了user机之后,可以用工程机换。

那么这四个类型是干什么用的呢?其实,在main.mk里有说明,在Android的源码里,每一个目标(也可以看成工程)目录都有一个Android.mk的makefile,每个目标的Android.mk中有一个类型声明:LOCAL_MODULE_TAGS,这个TAGS就是用来指定,当前的目标编译完了属于哪个分类里。

PS:Android.mk和Linux里的makefile不太一样,它是Android编译系统自己定义的一个makefile来方便编译成:c,c++的动态、静态库或可执行程序,或java库或android的程序,

好了,我们来分析下lunch命令干了什么?

 

function lunch()
{
local answer

# lunch后面直接带参数的情况,则编译项为指定的参数: $1

if [ "$1" ] ; then

answer=$1
else
# lunch后面不带参数的情况,则调用print_lunch_menu函数,打印所有的target product和variant菜单提供用户选择
print_lunch_menu
echo -n "Which would you like? [generic-eng] "
read answer# 读取用户选择的结果,注意,这儿可以是数字,也可以是字符串的选项内容
fi

local selection=

# 如果用户在菜单中没有输入任何内容(直接回车),则为系统缺省的generic-eng编译项

if [ -z "$answer" ]
then
selection=generic-eng
elif [ "$answer" = "simulator" ]
then
# 如果是用户输入的是模拟器:simulator
selection=simulator

# 如果answer是菜单中的数字,则获取该数字
elif (echo -n $answer | grep -q -e "^[0-9][0-9]*$")
then
if [ $answer -le ${#LUNCH_MENU_CHOICES[@]} ] # 如果用户输入的菜单数字不合法(超过全部选项数组的元素个数)

# ${#LUNCH_MENU_CHOICES[@]}是指,取得LUNCH_MENU_CHOICES这个数组的元素个数,LUNCH_MENU_CHOICES[@]指引用数组里的全部元素,以列表返回
then
selection=${LUNCH_MENU_CHOICES[$(($answer-$_arrayoffset))]} #这儿通过$answer - $_arrayoffset来引用用户输入的数字对应的数组LUNCH_MENU_CHOICES中的编译项,其中_arrayoffset这个变量表示,是否是数组下标从0开始,这儿其值为:1
fi
# 如果 answer字符串匹配 *-*模式(开头结尾不能为-)
elif (echo -n $answer | grep -q -e "^[^\-][^\-]*-[^\-][^\-]*$")
then
selection=$answer
fi

#如果selection为空,出错退出

if [ -z "$selection" ]
then
echo
echo "Invalid lunch combo: $answer"
return 1
fi

# special case the simulator
if [ "$selection" = "simulator" ]#如果用户选项为使用模拟器
then
# 导出4个环境变量,这4个环境变量将指定编译系统编译为模拟器
export TARGET_PRODUCT=sim#产品变量
export TARGET_BUILD_VARIANT=eng#版本型号变量
export TARGET_SIMULATOR=true#编译在模拟器中
export TARGET_BUILD_TYPE=debug#类型变量
else

# 将 product-variant模式中的product分离出来,sed命令是将-后面的字符串替换为空串,也就是只保留-前面的内容
local product=$(echo -n $selection | sed -e "s/-.*$//")

# 检查之,调用关系 check_product()->get_build_var()->build/core/config.mk比较罗嗦,不展开了
check_product $product
if [ $? -ne 0 ]
then
echo
echo "** Don't have a product spec for: '$product'"
echo "** Do you have the right repo manifest?"
product=
fi

# 将 product-variant模式中的variant分离出来,sed命令将-前面的内容替换为空串
local variant=$(echo -n $selection | sed -e "s/^[^\-]*-//")

# 检查之,看看是否在 (user userdebug eng) 范围内
check_variant $variant
if [ $? -ne 0 ]
then
echo
echo "** Invalid variant: '$variant'"
echo "** Must be one of ${VARIANT_CHOICES[@]}"
variant=
fi

if [ -z "$product" -o -z "$variant" ]#再次检查两个变量是否为空
then
echo
return 1
fi
# 导出4个环境变量(和上面模拟器的对比着看),这里很重要,因为后面的编译系统都是依赖于这里定义的几个变量的
export TARGET_PRODUCT=$product
export TARGET_BUILD_VARIANT=$variant
export TARGET_SIMULATOR=false
export TARGET_BUILD_TYPE=release
fi # !simulator

echo

# 设置到环境变量,比较多,不再一一列出,最简单的方法 set >env.txt 可获得
set_stuff_for_environment
# 打印一些主要的变量, 调用关系 printconfig()->get_build_var()->build/core/config.mk->build/core/envsetup.mk 比较罗嗦,不展开了
printconfig
}

 

 

由上面分析可知,lunch命令可以带参数和不带参数,最终导出一些重要的环境变量,从而影响编译系统的编译结果。导出的变量如下(以实际运行情况为例)

 

 

  1. TARGET_PRODUCT=fs100
  2. TARGET_BUILD_VARIANT=eng
  3. TARGET_SIMULATOR=false
  4. TARGET_BUILD_TYPE=release

 

执行完上述两个步骤,就该执行:make命令了,当然如果你按照上述两个步骤做完,执行make之后肯定会出问题,我们还要做一些其它的操作,下篇来分析。

分享到:
评论

相关推荐

    android编译系统详解

    总结来说,Android编译系统是一个复杂的工程化编译方法,它依赖于Makefile的编写和Shell脚本的应用。了解和熟练掌握Make工具的使用、Makefile的规则和结构以及Shell脚本编程,对于理解Android编译构建系统至关重要。...

    android编译过程详解

    我们将重点解析`source build/envsetup.sh`命令及其内部函数,深入理解Android编译过程中的关键步骤。 #### 二、`source build/envsetup.sh`命令详解 `source build/envsetup.sh`命令是Android构建环境设置的重要...

    Android编译过程详解

    在深入探讨Android编译过程之前,首先需要理解的是Android系统的开放性使得开发者和制造商能够根据特定硬件需求定制操作系统。例如,小米、LePhone和M9等品牌就利用这一特性为自家设备打造独特的用户体验。要对...

    Android的编译系统.pdf

    通过上述内容,我们可以看到Android编译系统的复杂性和灵活性,它能够根据不同的需求定制化构建出适合特定设备和应用场景的操作系统。对于初学者来说,掌握这些基础概念和流程是理解和开发Android系统的关键。

    android编译系统框架分析及main.mk分析

    ### Android编译系统框架分析及main.mk解析 #### 一、引言 自从Android源代码公开以来,它在嵌入式行业中引发了巨大的关注与兴趣。众多开发者被Android的强大功能所吸引,纷纷尝试下载源代码进行编译和移植。尽管...

    Android系统定制之Android.mk和Android.bp语法详解(精通版).mp4

    1.Android系统源码定制和编译方法(Android7.0至Android12) 2.Android.mk和Android.bp语法大全及使用方法详解 3.Android.mk或Android.bp中引用Android的第三方(jar、aar、so)、Androidx的类库等的使用方法 4.Android....

    Android 编译系统分析.doc

    本文将深入解析Android编译系统的结构和工作原理,重点关注四种类型的Make文件及其作用。 首先,我们来看第一类Make文件,即For config文件。这类文件主要负责配置Android系统的各个方面,包括产品(product)、...

    android编译系统

    ### Android编译系统详解 #### 一、Android编译系统简介 Android的编译系统是其核心组件之一,为开发者提供了高效且稳定的编译环境。该系统不仅支持Android平台本身的编译,还能够编译主机平台上的工具以及各种SDK...

    Android Android系统开发编译环境配置

    除了Android编译环境的配置外,文档还简述了在Ubuntu系统下使用Samba服务实现局域网内文件共享的方法,便于跨平台文件访问与协作。 #### 安装Samba服务器 通过`sudo apt-get install samba`命令安装Samba服务器,...

    PCL for Android 编译过程遇到的问题及办法

    1. **CMake安装**:CMake是一种开源的、跨平台的自动化构建系统,用于管理软件构建过程。它可以通过编写描述性脚本来控制编译过程。在编译PCL的过程中,CMake起着核心作用。 2. **Android Studio**:虽然文档提到...

    Android编译独立源代码

    ### Android编译独立源代码知识点解析 #### 一、概览 在深入探讨如何编译Android独立源代码之前,我们先来了解下整个过程的基本框架。Android作为一个开源项目,其源代码包含了操作系统的核心部分以及一系列应用...

    Android编译流程makefile流程[参考].pdf

    Android 编译流程是 Android 系统开发的关键步骤之一,Makefile 是 Android 编译流程的核心组件。本文将详细介绍 Android 编译流程 Makefile 流程,并对其进行详细解释。 一、Makefile 概述 Makefile 是一个用于...

    android编译系统makefile(Android.mk)写法扫描.pdf

    本篇将深入探讨`Android.mk`的写法及其在Android编译系统中的作用。 1. **LOCAL_PATH变量**: 在`Android.mk`文件中,`LOCAL_PATH`变量是用来指示当前Makefile所在的目录,通常在文件开头设置。通过`$(call my-dir...

    Android_makefile编译系统_Android.mk_文件语法规范

    ### Android.mk 编译系统与文件语法规范 #### 引言 `Android.mk`作为Android NDK中的核心配置文件之一,主要用于向构建系统描述项目的C/C++源代码组织方式及构建逻辑。对于深入理解Android NDK项目构建流程、优化...

    android反编译工具

    10. **Xposed Framework**:Xposed不直接进行反编译,但它提供了一个平台,允许开发者编写模块来修改系统和应用行为。通过Xposed,可以对APK进行非侵入式的修改,而无需反编译和重新打包。 以上工具各有其特点和...

    android P android9源码编译.ipr.iml文件

    源码编译对于开发者来说是一项基础且至关重要的工作,它涉及到理解系统内部运作,自定义修改以及构建自己的Android ROM。`.ipr`和`.iml`文件是IntelliJ IDEA(Android Studio是其专为Android开发的分支)项目配置...

    windows protobuf android 编译.doc

    ### Windows 下 Android 编译 Protobuf 的步骤及详解 #### 一、下载 Protobuf 源代码 在开始编译之前,首先需要从官方仓库获取最新版本的 Protobuf 源代码。通常可以通过 Git 克隆或者直接下载压缩包的方式进行...

Global site tag (gtag.js) - Google Analytics