android 学习笔记(八)building system8.5 shell脚本的学习

虽然接触她很久了,但只能说认识,。说道很熟,熟到称兄道弟,这是我的目标。想要混熟总得经历些事情,那么现在开始吧。最近在学习android,还是从android开始吧!

android源码中的build/envsetup.sh

顾名思义,这个脚本的主要作用是初始化编译环境,如环境变量。提供一些需要用到的函数,可作为指令调用。

我们看一下build/envsetup.sh 都干了什么事情。

1. 下面是剥离脚本中的函数部分:(写该脚本的老大,不明白为什么把执行的脚本和函数混在一块些?)

转自:

http://www.lupaworld.com/home.php?mod=space&uid=131820&do=blog&id=149463

01.build/envsetup.sh 脚本主体部分  
02.  
03.看源代码之前需要了解代码分布和结构,分析Makefile应该是最好的方法。在Blog记录之以备以后查看,也供有兴趣朋友参考。  
04.  
05.在编译donut之前,需要运行  
06.  
07.cd ~/mydroid  
08.. build/envsetup.sh  
09.lunch aosp_dream_us-eng  
10.make  
11.  
12.build/envsetup.sh 是编译android的入口。该shell脚本定义了一些函数,并设定了编译的环境变量。  
13.  
14.lunch aosp_dream_us-eng 配置用户需要编译的目标产品(target product)和变体(variant)。  
15.  
16.aosp_dream_us-eng 字符串包含了target product  和variant,它们用减号-隔开,aosp_dream_us是target product,eng是variant。  
17.  
18.variant 有三种选择user userdebug eng,可以从字面上看出其意义,但具体的差别还不是特别清楚,待后面分析之。  
19.  
20.lunch 的入口是让用户指定或者选择需要编译的target product 和 variant,出口是设定编译相关的环境变量。  
21.  
22.build/envsetup.sh 定义了很多函数,并且执行的脚本很分散,因此下面将所有的函数都删除掉了,剩余下脚本的主体,便于代码分析和走读。  
23.  
24.该主体主要做的事情是:  
25.  
26.1. 首先脚本设定了两个缺省的target product,一个是generic-eng,一个是simulator  
27.  
28.2. 运行所有vendor下的 vendorsetup.sh文件,在google官方的 donut源代码中,为vendor/aosp/vendorsetup.sh,内容如下:  
29.  
30.add_lunch_combo aosp_emulator_us-eng  
31.add_lunch_combo aosp_emulator_eu-eng  
32.add_lunch_combo aosp_dream_us-userdebug  
33.add_lunch_combo aosp_dream_eu-userdebug  
34.add_lunch_combo aosp_dream_us-eng  
35.add_lunch_combo aosp_dream_eu-eng  
36.add_lunch_combo aosp_sapphire_us-userdebug  
37.add_lunch_combo aosp_sapphire_eu-userdebug  
38.add_lunch_combo aosp_sapphire_us-eng  
39.add_lunch_combo aosp_sapphire_eu-eng  
40.  
41.可以看出,又添加了一些 和 aosp 相关的 target product 和 variant。  
42.  
43.下面是脚本主体,用蓝色添加了一下注释,便于阅读:  
44.  
45.# 定义了 variant的字符范围  
46.VARIANT_CHOICES=(user userdebug eng)  
47.  
48.# 如果是Linux环境,定义了几个和simulator相关的函数  
49.case `uname -s` in  
50.    Linux)  
51.        function choosesim()  
52.        {  
53.            echo "Build for the simulator or the device?"  
54.            echo "     1. Device"  
55.            echo "     2. Simulator"  
56.            echo  
57.  
58.            export TARGET_SIMULATOR=  
59.            local ANSWER  
60.            while [ -z $TARGET_SIMULATOR ]  
61.            do  
62.                echo -n "Which would you like? [1] "  
63.                if [ -z "$1" ] ; then  
64.                    read ANSWER  
65.                else  
66.                    echo $1  
67.                    ANSWER=$1  
68.                fi  
69.                case $ANSWER in  
70.                "")  
71.                    export TARGET_SIMULATOR=false  
72.                    ;;  
73.                1)  
74.                    export TARGET_SIMULATOR=false  
75.                    ;;  
76.                Device)  
77.                    export TARGET_SIMULATOR=false  
78.                    ;;  
79.                2)  
80.                    export TARGET_SIMULATOR=true  
81.                    ;;  
82.                Simulator)  
83.                    export TARGET_SIMULATOR=true  
84.                    ;;  
85.                *)  
86.                    echo  
87.                    echo "I didn't understand your response.  Please try again."  
88.                    echo  
89.                    ;;  
90.                esac  
91.                if [ -n "$1" ] ; then  
92.                    break  
93.                fi  
94.            done  
95.  
96.            set_stuff_for_environment  
97.        }  
98.        ;;  
99.    *)  
100.        function choosesim()  
101.        {  
102.            echo "Only device builds are supported for" `uname -s`  
103.            echo "     Forcing TARGET_SIMULATOR=false"  
104.            echo  
105.            if [ -z "$1" ]  
106.            then  
107.                echo -n "Press enter: "  
108.                read  
109.            fi  
110.  
111.            export TARGET_SIMULATOR=false  
112.            set_stuff_for_environment  
113.        }  
114.        ;;  
115.esac  
116.  
117.  
118.# 首先将 LUNCH_MENU_CHOICES 变量消除  
119.# Clear this variable.  It will be built up again when the vendorsetup.sh  
120.# files are included at the end of this file.  
121.unset LUNCH_MENU_CHOICES            
122.  
123.  
124.# 当前 LUNCH_MENU_CHOICES = generic-eng  
125.# add the default one here  
126.add_lunch_combo generic-eng           
127.  
128.  
129.# 当前 LUNCH_MENU_CHOICES = generic-eng simulator  
130.# if we're on linux, add the simulator.  There is a special case  
131.# in lunch to deal with the simulator  
132.if [ "$(uname)" = "Linux" ] ; then      
133.    add_lunch_combo simulator         
134.fi  
135.  
136.  
137.# 这里是MAC OS的环境,不考虑  
138.case `uname -s` in    
139.    Darwin)  
140.        function mgrep()  
141.        {  
142.            find -E . -type f -iregex '.*/(Makefile|Makefile\..*|.*\.make|.*\.mak|.*\.mk)' -print0 | xargs -0 grep --color -n "$@"  
143.        }  
144.  
145.        function treegrep()  
146.        {  
147.            find -E . -type f -iregex '.*\.(c|h|cpp|S|java|xml)' -print0 | xargs -0 grep --color -n -i "$@"  
148.        }  
149.  
150.        ;;  
151.    *)  
152.        function mgrep()  
153.        {  
154.            find . -regextype posix-egrep -iregex '(.*\/Makefile|.*\/Makefile\..*|.*\.make|.*\.mak|.*\.mk)' -type f -print0 | xargs -0 grep --color -n "$@"  
155.        }  
156.  
157.        function treegrep()  
158.        {  
159.            find . -regextype posix-egrep -iregex '.*\.(c|h|cpp|S|java|xml)' -type f -print0 | xargs -0 grep --color -n -i "$@"  
160.        }  
161.  
162.        ;;  
163.esac  
164.  
165.  
166.# 设定当前shell环境的数组下标从0开始还是1开始,并记录在 _arrayoffset  
167.# determine whether arrays are zero-based (bash) or one-based (zsh)  
168._xarray=(a b c)  
169.if [ -z "${_xarray[${#_xarray[@]}]}" ]  
170.then  
171.    _arrayoffset=1  
172.else  
173.    _arrayoffset=0  
174.fi  
175.unset _xarray  
176.  
177.  
178.# 运行所有vendor下的 vendorsetup.sh文件  
179.#存在vendor/aosp/vendorsetup.sh  
180.当前 LUNCH_MENU_CHOICES = generic-eng simulator aosp_emulator_us-eng aosp_emulator_eu-eng aosp_dream_us-userdebug aosp_dream_eu-userdebug aosp_dream_us-eng aosp_dream_eu-eng aosp_sapphire_us-userdebug aosp_sapphire_eu-userdebug aosp_sapphire_us-eng aosp_sapphire_eu-eng  
181.  
182.# Execute the contents of any vendorsetup.sh files we can find.  
183.for f in `/bin/ls vendor/*/vendorsetup.sh vendor/*/build/vendorsetup.sh 2> /dev/null`   
184.do  
185.    echo "including $f"  
186.    . $f  
187.done  
188.unset f  
189.   



[ruby] view plaincopyprint?
01.build/envsetup.sh 脚本分析(lunch函数)  
02.  
03.lunch函数提供了一个菜单,让开发人员选择需要编译的目标产品(target product)和变体(variant),并做一些检查,设置环境变量,并打印出主要的环境变量。  
04.  
05.直接运行lunch(必须先运行 build/envsetup.sh,让lunch函数驻留到环境变量中)  
06.  
07.ning@ning-desktop:~/donut-compare/mydroid$ lunch  
08.  
09.You're building on Linux  
10.  
11.generic-eng simulator aosp_emulator_us-eng aosp_emulator_eu-eng aosp_dream_us-userdebug aosp_dream_eu-userdebug aosp_dream_us-eng aosp_dream_eu-eng aosp_sapphire_us-userdebug aosp_sapphire_eu-userdebug aosp_sapphire_us-eng aosp_sapphire_eu-eng  
12.Lunch menu... pick a combo:  
13.     1. generic-eng  
14.     2. simulator  
15.     3. aosp_emulator_us-eng  
16.     4. aosp_emulator_eu-eng  
17.     5. aosp_dream_us-userdebug  
18.     6. aosp_dream_eu-userdebug  
19.     7. aosp_dream_us-eng  
20.     8. aosp_dream_eu-eng  
21.     9. aosp_sapphire_us-userdebug  
22.     10. aosp_sapphire_eu-userdebug  
23.     11. aosp_sapphire_us-eng  
24.     12. aosp_sapphire_eu-eng  
25.  
26.Which would you like? [generic-eng] 7  
27.  
28.============================================  
29.PLATFORM_VERSION_CODENAME=REL  
30.PLATFORM_VERSION=1.6  
31.TARGET_PRODUCT=aosp_dream_us  
32.TARGET_BUILD_VARIANT=eng  
33.TARGET_SIMULATOR=false  
34.TARGET_BUILD_TYPE=release  
35.TARGET_ARCH=arm  
36.HOST_ARCH=x86  
37.HOST_OS=linux  
38.HOST_BUILD_TYPE=release  
39.BUILD_ID=Donut  
40.============================================  
41.  
42.用户也可以直接输入参数,不使用菜单  
43.  
44.ning@ning-desktop:~/donut-compare/mydroid$ lunch aosp_dream_us-eng  
45.  
46.============================================  
47.PLATFORM_VERSION_CODENAME=REL  
48.PLATFORM_VERSION=1.6  
49.TARGET_PRODUCT=aosp_dream_us  
50.TARGET_BUILD_VARIANT=eng  
51.TARGET_SIMULATOR=false  
52.TARGET_BUILD_TYPE=release  
53.TARGET_ARCH=arm  
54.HOST_ARCH=x86  
55.HOST_OS=linux  
56.HOST_BUILD_TYPE=release  
57.BUILD_ID=Donut  
58.============================================  
59.  
60.下面是lunch函数源代码,用蓝色添加了一下注释,便于阅读:  
61.  
62.function lunch()  
63.{  
64.    local answer  
65.  
66.    if [ "$1" ] ; then  
67.       # lunch后面直接带参数  
68.        answer=$1  
69.    else  
70.       # lunch后面不带参数,则打印处所有的target product和variant菜单提供用户选择  
71.        print_lunch_menu     
72.        echo -n "Which would you like? [generic-eng] "  
73.        read answer  
74.    fi  
75.  
76.    local selection=  
77.  
78.    if [ -z "$answer" ]  
79.    then  
80.           # 如果用户在菜单中没有选择,直接回车,则为系统缺省的generic-eng  
81.        selection=generic-eng  
82.    elif [ "$answer" = "simulator" ]  
83.    then  
84.        # 如果是模拟器  
85.        selection=simulator  
86.    elif (echo -n $answer | grep -q -e "^[0-9][0-9]*$")  
87.    then  
88.        # 如果answer是选择菜单的数字,则获取该数字对应的字符串  
89.        if [ $answer -le ${#LUNCH_MENU_CHOICES[@]} ]  
90.        then  
91.            selection=${LUNCH_MENU_CHOICES[$(($answer-$_arrayoffset))]}  
92.        fi  
93.        # 如果 answer字符串匹配 *-*模式(*的开头不能为-)  
94.    elif (echo -n $answer | grep -q -e "^[^\-][^\-]*-[^\-][^\-]*$")  
95.    then  
96.        selection=$answer  
97.    fi  
98.  
99.    if [ -z "$selection" ]  
100.    then  
101.        echo  
102.        echo "Invalid lunch combo: $answer"  
103.        return 1  
104.    fi  
105.  
106.    # special case the simulator  
107.    if [ "$selection" = "simulator" ]  
108.    then  
109.        # 模拟器模式  
110.        export TARGET_PRODUCT=sim  
111.        export TARGET_BUILD_VARIANT=eng  
112.        export TARGET_SIMULATOR=true  
113.        export TARGET_BUILD_TYPE=debug  
114.    else  
115.  
116.        # 将 product-variant模式种的product分离出来  
117.        local product=$(echo -n $selection | sed -e "s/-.*$//")  
118.  
119.        # 检查之,调用关系 check_product()->get_build_var()->build/core/config.mk比较罗嗦,不展开了  
120.        check_product $product  
121.        if [ $? -ne 0 ]  
122.        then  
123.            echo  
124.            echo "** Don't have a product spec for: '$product'"  
125.            echo "** Do you have the right repo manifest?"  
126.            product=  
127.        fi  
128.  
129.        # 将 product-variant模式种的variant分离出来  
130.        local variant=$(echo -n $selection | sed -e "s/^[^\-]*-//")  
131.  
132.        # 检查之,看看是否在 (user userdebug eng) 范围内  
133.        check_variant $variant  
134.        if [ $? -ne 0 ]  
135.        then  
136.            echo  
137.            echo "** Invalid variant: '$variant'"  
138.            echo "** Must be one of ${VARIANT_CHOICES[@]}"  
139.            variant=  
140.        fi  
141.  
142.        if [ -z "$product" -o -z "$variant" ]  
143.        then  
144.            echo  
145.            return 1  
146.        fi  
147.  
148.        export TARGET_PRODUCT=$product  
149.        export TARGET_BUILD_VARIANT=$variant  
150.        export TARGET_SIMULATOR=false  
151.        export TARGET_BUILD_TYPE=release  
152.    fi # !simulator  
153.  
154.    echo  
155.  
156.    # 设置到环境变量,比较多,不再一一列出,最 简单的方法 set >env.txt 可获得  
157.    set_stuff_for_environment  
158.    # 打印一些主要的变量, 调用关系 printconfig()->get_build_var()->build/core/config.mk->build/core/envsetup.mk 比较罗嗦,不展开了  
159.    printconfig  
160.}  

2. envsetup.sh 提供的命令

function help()
function get_abs_build_var()
function get_build_var()
function check_product()
function check_variant()
function setpaths()
function printconfig()
function set_stuff_for_environment()
function set_sequence_number()
function settitle()
function addcompletions()
function choosetype()
function chooseproduct()
function choosevariant()
function choosecombo()
function add_lunch_combo()
function print_lunch_menu()
function lunch()
function _lunch()
function tapas()
function m()
function findmakefile()
function mm()
function mmm()
function croot()
function cproj()
function pid()
function systemstack()
function gdbclient()
       
function jgrep()
function cgrep()
function resgrep()
       
function tracedmdump()
function runhat()
function getbugreports()
function startviewserver()
function stopviewserver()
function isviewserverstarted()
function key_home()
function key_back()
function key_menu()
function smoketest()
function runtest()
function godir () {
function set_java_home()

3. 命令详解

help
显示帮助,列出提供的命令
get_abs_build_var
列出make脚本中某变量值作为绝对路径。
使用方法:
get_abs_build_var VAR_NAME
VAR_NAME是需要显示的make脚本中的变量。
例如:
get_abs_build_var TARGET_PRODUCT
返回
<Your Android Root>\<VAR_NAME Value>

get_build_var
列出make脚本中某变量的值。
Usage:
get_build_var VAR_NAME
VAR_NAME是需要显示的make脚本中的变量。
Return:
<VAR_NAME Value>
Example:
get_abs_build_var TARGET_PRODUCT
check_product
检查指定的TARGET_PRODUCT是否支持,默认的有sim和generic。如果不允许,则输出错误信息,允许则无回显。
Usage:
check_product <YourTargetProduct>
Example:
check_product generic
check_variant
检查variant是否支持,支持则返回0,不支持则返回1。允许的variant列表定义在envsetup.sh中的VARIANT_CHOICES中,默认是user,userdebug,eng。定制android时,可以在VARIANT_CHOICES中添加vairant。
Usage:
check_variant <YourVariant>
Example:
check_variant eng
setpaths
奇次执行时,将ANDROID_BUILD_PATHS路径加到PATH中。偶次执行时,将ANDROID_BUILD_PATHS路径从PATH中去除。ANDROID_BUILD_PATHS包括android编译中要使用到的路径,例如ANDROID_EABI_TOOLCHAIN,ANDROID_TOOLCHAIN,ANDROID_QTOOLS,ANDROID_JAVA_TOOLCHAIN,ANDROID_PRODUCT_OUT等等。
Usage:
setpaths

printconfig
输出类似如下形势的配置信息。
============================================
PLATFORM_VERSION_CODENAME=AOSP
PLATFORM_VERSION=AOSP
TARGET_PRODUCT=generic
TARGET_BUILD_VARIANT=eng
TARGET_SIMULATOR=
TARGET_BUILD_TYPE=release
TARGET_BUILD_APPS=
TARGET_ARCH=arm
HOST_ARCH=x86
HOST_OS=linux
HOST_BUILD_TYPE=release
BUILD_ID=OPENMASTER
============================================
set_stuff_for_environment
依次调用settitle, set_java_home,setpaths,set_sequence_number。设置android编译需要的环境变量。
set_sequence_number
设置环境变量BUILD_ENV_SEQUENCE_NUMBER。
settitle
设置shell的prompt提示,PROMPT_COMMAND中加入TARGET_PRODUCT,TARGET_BUILD_VARIANT,和TARGET_BUILD_APPS等信息提示。
choosesim
配置环境变量TARGET_SIMULATOR。linux下会提示用户选择device或simulator。然后调用set_stuff_for_environment设置。
choosetype
配置环境变量TARGET_BUILD_TYPE_SIMULATOR。会提示用户选择release或debug。然后调用set_stuff_for_environment设置。
chooseproduct
配置环境变量TARGET_PRODUCT。会提示用户选择release或debug。然后调用set_stuff_for_environment设置。
choosevariant
配置环境变量TARGET_BUILD_VARIANT。会提示用户选择release或debug。
choosecombo
依次调用choosesim,choosetype,chooseproduct,choosevariant,set_stuff_for_environment配置,然后调用printconfig输出。
add_lunch_combo
向环境变量LUNCH_MENU_CHOICES标识的列表中添加项。envsetup.sh中默认添加了full-eng,full_x86-eng,和simulator。
print_lunch_menu
列出LUNCH_MENU_CHOICES中的所有选项。
lunch
点菜,用户选择/指定product,variant后,lunch命令设置环境变量TARGET_PRODUCT,TARGET_BUILD_VARIANT,TARGET_SIMULATOR,TARGET_BUILD_TYPE,随后调用set_stuff_for_environment设置,并printconfig显示。
Usage:
lunch [<YourProduct>-<YourBuildVariant>]

不给参数时,将提示用户选择。
Example:
lunch
lunch generic-eng
tapas
用户给定variant和一个或多个app name,就是LOCAL_PACKAGE_NAME的名字。tapas设定
export TARGET_PRODUCT=generic
export TARGET_BUILD_VARIANT=$variant
export TARGET_SIMULATOR=false
export TARGET_BUILD_TYPE=release
export TARGET_BUILD_APPS=$apps
Usage:
tapas <YourVariant>? <YourAppName>*
?代表可选,*代表0个,1个或多个。YourVariant 和YourAppName的次序可颠倒。
Example:
tapas user Calculator Calender
gettop
返回当前android代码树的顶层路径。前提是当前路径位于android代码树中。
m
等价于在当前android代码树的顶层路径下执行make命令。
findmakefile
查找当前或最接近自己的祖辈路径上的Android.mk,返回Android.mk的路径,假设当前路径处于android代码树中。
mm
如果当前路径是代码树顶层,则mm相当于make。如果是深层,测mm相当于
ONE_SHOT_MAKEFILE=$M make -C $T files $@

$M是findmakefile发现的Android.mk,$T是代码树顶层路径,files是main.mk中定义的phony goal,就是完成$M对应目录范围内,所有android需编译的modules以及辅助说明txt文件。
mmm
给定package的路径,则mm会make相应的package。
例如,
mmm package/apps/Calculator

croot
改变当前路径到代码树顶层。
cproj
改变当前路径到最近的还有Android.mk文件的祖父辈路径。
pid
使用adb shell ps命令列出手机上指定名字的进程的pid。
Usage:
pid <YourName>
systemstack
使用kill -3system_server将系统进程中的线程信息写入/data/anr/traces.txt。
gdbclient
建立gdb调试环境,包括两步,手机上运行gdbserver,本机上运行arm-eabi-gdb。
Usage:
gdbclient <EXE> <PORT> <AppName>
EXE: AppName的执行名。

PORT:gdbserver的端口,例如, 192.168.2.102:5039
AppName:手机中ps列出的app名字,据此查pid。
sgrep
查找当前目录及子目录中所有.c,.h,.cpp,.S,.java,.mk,.xml,.sh文件,即源码文件中包含特定单词的行,并颜色显示输出。
Usage:
sgrep <YourWord>
Example:
sgrep Calendar
jgrep
同sgrep,但只查.java文件。
cgrep
同sgrep,但只查c相关的文件,即.c,.cc,.cpp,.h文件。
resgrep
同sgrep,但只查res相关的.xml文件。
mgrep
同sgrep,但只查make相关的脚本文件,包括Makefile文件,Makefile目录下的所有文件,.make文件,.mak文件和.mk文件。
treegrep
查找当前目录及子目录中所有.c,.h,.cpp,.S,.java,.xml文件,即源码文件中包含特定单词的行,并颜色显示输出。
getprebuilt
输出prebuilt的路径。
tracedmdump
生成dexlist文件qtrace.dexlit,dmtrace数据文件dmtrace,和调用dmtracedump工具生成的dmtrace解析文件dmtrace.html,将生成文件放到指定路径。
Usage:
tracedmdump <YourDirName>
如果YourDirName中不含’\’,则将放置的路径是$ANDROID_PRODUCT_OUT/traces/YourDirName。
runhat
貌似使用kill -10的方法得到heap dump并取到本地。使用hat以http方式展现出来。hat可能是个lightweight http server,不曾用过。
getbugreports
将手机/sdcard/bugreports目录下的文件下载到本地并压缩打包。
startviewserver
用指定端口启动viewserver。
Usage:
startviewserver <Port>
不指定端口,则默认4939。
stopviewserver
关闭viewserver。
isviewserverstarted
检查viewserver是否可用。
smoketest
编译smoketest并安装手机运行。
runtest
运行development/testrunner/runtest.py $@

godir
给出一个词,godir会输出一个路径列表供用户选择要进入的路径。路径列表包含的路径满足,路径名中包含这个词,或这路径下的文件有文件名含这个词。out/路径下不考虑。
Usage:
godir <YourKey>
Usage:
godir Calculator
set_java_home

设置JAVA_HOME环境变量为/usr/lib/jvm/java-6-sun

 

参考:http://blog.csdn.net/piaozhiye/article/details/7522153

posted @ 2012-05-04 00:11  夏大王  阅读(2783)  评论(0编辑  收藏  举报