简化编译命令

无论是在Android编译系统中,还是在Yocto编译系统中,要编译一个目标,输入命令都有点费事。

Yocto系统:

source setup-environment $FOLDER
bitbake $TARGET

Android系统:

source build/envsetup.sh
launch $MENU
make -j$N

通常的做法是编写一个build脚本来解析输入的命令。用户只需要输入一行命令来编译目标。

build <board> <target> [arg]

比如,要为名为board-a的项目编译kernel, 则只需要输入一行: build board-a kernel。

在Yocto系统中还可以加入额外的参数 build board-a kernel -f -c compile。(-f -c compile将进行强制编译)

build脚本很大程度上简化了编译命令的输入。

build脚本的实现

build脚本的实现也比较简单。下面以Android编译系统为例。

1. 准备一个target.conf文件,放置变化的参数表格。

#board      u-boot-defconfig    kernel-deconfig    launch-menu
board-a a-defconfig a-defconfig launch-menu-a
board-b b-defconfig b-defconfig launch-menu-b

2. 由命令输入获取board、target和arg.

board=$
target=$2
shift;shift;
arg=$@

3. 通过board名字在target.conf获取必要的配置。

configs=( $(awk '$1 == "board-a"' target.conf) )
ubootdefcfg=${configs[]}
kerneldefcfg=${configs[]}
menu=${configs[]}

4. 对不同的target调用不同的对象。

build_kernel()
{
...
} build_uboot()
{
...
} main()
{
...
if [ -n "${build_$target}" ]; then
build_$target
fi
...
} main $*

这样,如果target是kernel, 则会运行build_kernel函数。

为编译命令添加自动完成功能

无论新手老手,都不可能记住很多杂乱的东西。所以,需要查看target.conf以知晓支持哪些machine。并且,对应target和argc,则更是两眼一抹黑。kernel和uboot可能使用得多,记得牢固一些。但是其它的target的?比如说,单独编译Camera的App,target应该输入什么?错一个字都不行。

所以,需要为编译命令添加自动完成功能。

1. 不用查看target.conf,就能知道支持哪些board.

2. 输入一个C,按一下tab键,Camera App的target就能列举出来。

3. 输入一个-,就能列举支持的参数。

自动完成的实现

要实现自动完成功能,只需要使用complete对Bash完成自动完成功能的注册就可以。

1. 为了不用在每次使用前都手工注册,准备completion.sh文件,并且在.bashrc中加入语句:

source .completion.sh

2. .completion.sh的内容如下:

function _bb_completion()
{
if test -x build/bb_completion; then
COMPWORDS=${COMP_WORDS[@]};
export COMPWORDS COMP_CWORD;
COMPREPLY=( $(build/bb_completion) );
fi
} function _build_completion()
{
if test -x build/build_completion; then
COMPWORDS=${COMP_WORDS[@]};
export COMPWORDS COMP_CWORD;
COMPREPLY=( $(build/build_completion) );
fi
} complete -F _bb_completion bb
complete -F _build_completion build

真正会在shell开始运行时执行的是complete -F _bb_completion bb,其它的部分只是载入到环境变量当中。

比如,运行type _bb_completion,得到结果

: type _bb_completion
_bb_completion is a function
_bb_completion ()
{
if test -x build/bb_completion; then
COMPWORDS=${COMP_WORDS[@]};
export COMPWORDS COMP_CWORD;
COMPREPLY=($(build/bb_completion));
fi
}
:

运行complete | grep bb,得到结果complete -F _bb_completion bb,表明这个自动完成项目被注册了。

在bash侦测到用户敲击tab键之后,就会去调用与输入程序名bb匹配的自动完成函数_bb_completion。

3.  _bb_completion把运行权交给当前目录下的build/bb_completion文件。在bb_complete中,返回COMPREPLY所需要的值,bash根据COMPREPLY的值显示自动完成的提示结果。

build/bb_completion文件的内容

bb_completion()
{
COMPWORDS=( $COMPWORDS );
local cur="${COMPWORDS[COMP_CWORD]}"
local pre="${COMPWORDS[COMP_CWORD - 1]}"
local args="" for i in ""; do
case "${COMP_CWORD}" in
) # get board list from target.conf
args="$(awk '$1 ~ /^[^#]/ { print $1 }' target.conf)"
;;
*) # extra arguments
case "${cur}" in
-*)
args="-c -f"
break;;
esac
;;
esac
done if test -n "${args}"; then
echo $(compgen -W "${args}" -- ${cur});
fi } bb_completion

自动完成处理时,对参数位置、当前参数和前一个参数进行判断,给出适当的自动完成内容。

比如,当参数位置为1时,希望输入<board>。<board>从target.conf文件中进行抽取。

4. target的产生。

除了kerenl, u-boot等常用的target之外,通常还会有需求编译其它的target。这些target数量巨多,正是自动完成的用武之地。无论是Android还是Yocto编译系统,都有产生完整的target列表的方法。将target列表保存下来,供自动完成使用,这样,在命令输入的时候,会很方便,也有助于初学者理解编译系统的target资源。

Yocto系统,bitbake machine -c listtasks 将列出所有的target.

Android系统,下面的target将罗列出所有的App package.

show-packages:
@$(foreach m, $(PRODUCTS.$(INTERNAL_PRODUCT).PRODUCT_PACKAGES), echo '$(m)';)

Yocto和Android编译命令的简化和自动完成的实现的更多相关文章

  1. Android 编译命令 make j8 2>&1 | tee build.log 解释

    在编译Android的时候,经常看到这样的命令 make  -j8 2>&1 | tee build.log  其中 make 是编译命令, -j8 这里的 8 指的是线程数量,就是你要 ...

  2. 【Android】MTK Android 编译命令

    命令格式:./maketek [option] [project] [action] [modules] Option: -t ,-tee :输出log信息到当前终端 -o , -opt=-- : 编 ...

  3. MTK Android 编译命令

    一.Target 编译命令 usage: (makeMtk|mk) [options] project actions [moudles] options:       -t,-tcc         ...

  4. Mtk Android编译命令

    一.输入命令: cbk@YCS:~/work/k6/alps$ ./mk help Usage: (makeMtk|mk) [options] project actions [modules] Op ...

  5. 理解Android编译命令(转)

    一.引言 关于Android Build系统,这个话题很早就打算整理下,迟迟没有下笔,决定跟大家分享下.先看下面几条指令,相信编译过Android源码的人都再熟悉不过的. source setenv. ...

  6. Android编译命令

    目录 说在前面 编译流程 编译指令 代码编译 代码检索 其他指令 说在前面 从最开始接触Android系统开始,每次进行代码编译都需要网上搜索编译指令.后来大致熟悉了Android的编译体系,加深了对 ...

  7. mtk Android 编译命令自定义--添加版本号

    1. alps\build\core\Makefile文件:(参照CUSTOM_BUILD_VERNO) ifeq "" "$(SURPLUS_BUILD_VERNO)& ...

  8. 【转】Android源代码编译命令m/mm/mmm/make分析--不错

    原文网址:http://blog.csdn.net/luoshengyang/article/details/19023609 在前文中,我们分析了Android编译环境的初始化过程.Android编 ...

  9. android的m、mm、mmm编译命令

    android的m.mm.mmm编译命令的使用 android源码目录下的build/envsetup.sh文件,描述编译的命令 - m:       Makes from the top of th ...

随机推荐

  1. Linux IO Scheduler(Linux IO 调度器)【转】

    每个块设备或者块设备的分区,都对应有自身的请求队列(request_queue),而每个请求队列都可以选择一个I/O调度器来协调所递交的request.I/O调度器的基本目的是将请求按照它们对应在块设 ...

  2. $(document).ready()和onload() html渲染时的区别

    不谈调用次数,加载先后问题,只看渲染时区别 1.都在数据绑定完加载. 2.ready可以有多个,且都执行,onload虽可以写多个,但是只执行最后一个. 3. $.ready = function ( ...

  3. 入坑C++

    c++中的++来自c语言中的递增运算符,该运算符将变量加1,c++起初也叫c with class ,通过通过名称表面,C++是对c的扩展,因此C++是c语言的超集,这以为这任何有效的c程序都是有效的 ...

  4. celery 大量消息的分布式系统 定时任务

    Celery 1.什么是Celery Celery是一个简单.灵活且可靠的,处理大量消息的分布式系统 专注于实时处理的异步任务队列 同时也支持任务调度 Celery架构 https://www.jia ...

  5. java控台输入

    import java.util.Scanner;//访问util包的Scanner(控台输入) public class HelloWorld {public static void main(St ...

  6. 七.HTTP协议原理介绍

    01. 当用户访问一个网站时,都发生了事情? ①. 利用DNS服务,将输入的域名解析为相应的IP地址   a --本地主机输入域名后,会查询本地缓存信息和本地hosts文件 如果有就进行解析,如果没有 ...

  7. 4.9cf自训9..

    cf401D 状态压缩dp好题,每次把新加入集合的数字放在最后即可 /* 它可以通过重新排列数字n, 它没有任何前导零, x除以m后的余数等于0. 每次把新加的数放在最后 dp[i][j]表示状态i下 ...

  8. mysql 的crud操作(增删改查)

    1.mysql添加记录 --添加记录的语法(可添加单条记录或者多条记录),INTO是可以省略的,字段名也可以省略的,但是如果省略的话,后面对应的value的值就要全部填写 INSERT [INTO] ...

  9. SVN 版本控制安装

    客户端 一路点击next即可 注意事项: 这个一定要选,否则后面使用会出现问题. 选择下拉菜单的第一项

  10. centos7安装elasticsearch

    [root@aaron tools]# wget https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-5.5.1.zi ...