一.编译成模块的前置条件
Android编译环境初始化完成后,我们就可以用m/mm/mmm/make命令编译源代码了。lunch命令其实是定义在build/envsetup.sh文件中的函数lunch提供的。与lunch命令一样,m、mm和mmm命令也分别是由定义在build/envsetup.sh文件中的函数m、mm和mmm提供的,而这三个函数又都是通过make命令来对源代码进行编译的。事实上,命令m就是对make命令的简单封装,并且是用来对整个Android源代码进行编译,而命令mm和mmm都是通过make命令来对Android源码中的指定模块进行编译。
二.m、mm和mmm的实现
各命令作用及用法:
mmm命令:编译指定路径下的模块,需要指定模块路径(必须为该模块的根目录,包含Android.mk的目录)。
mm命令:编译当前目录下的模块,需要进到模块所在目录,本质上跟mmm没什么区别。
m命令:编译全部模块。
函数m的实现如下所示:
function m()
{T=$(gettop)if [ "$T" ]; thenmake -C $T $@elseecho "Couldn't locate the top of the tree. Try setting TOP."fi
}
函数m调用函数gettop得到的是Android源代码根目录T。在执行make命令的时候,先通过-C选项指定工作目录为T,即Android源代码根目录,接着又将执行命令m指定的参数$@作为命令make的参数。从这里就可以看出,命令m实际上就是对命令make的简单封装。
函数mm的实现如下所示:
function mm()
{# If we're sitting in the root of the build tree, just do a# normal make.if [ -f build/core/envsetup.mk -a -f Makefile ]; thenmake $@else# Find the closest Android.mk file.T=$(gettop)local M=$(findmakefile)# Remove the path to top as the makefilepath needs to be relativelocal M=`echo $M|sed 's:'$T'/::'`if [ ! "$T" ]; thenecho "Couldn't locate the top of the tree. Try setting TOP."elif [ ! "$M" ]; thenecho "Couldn't locate a makefile from the current directory."elseONE_SHOT_MAKEFILE=$M make -C $T all_modules $@fifi
}
函数mm首先是判断当前目录是否就是Android源码根目录,即当前目录下是否存在一个build/core/envsetup.mk文件和一个Makefile文件。如果是的话,就将命令mm当作是一个普通的make命令来执行。否则的话,就调用函数findmakefile从当前目录开始一直往上寻找是否存在一个Android.mk文件。如果在寻找的过程中,发现了一个Android.mk文件,那么就获得它的绝对路径,并且停止上述寻找过程。
由于接下来执行make命令时,我们需要指定的是要编译的Android.mk文件的相对于Android源码根目录路径,因此函数mm需要将刚才找到的Android.mk绝对文件路径M中与Android源码根目录T相同的那部分路径去掉。这是通过sed命令来实现的,也就是将字符串M前面与字符串T相同的子串删掉。
最后,将找到的Android.mk文件的相对路径设置给环境变量ONE_SHOT_MAKE,表示接下来要对它进行编译。另外,函数mm还将make命令目标设置为all_modules。这是什么意思呢?我们知道,一个Android.mk文件同时可以定义多个模块,因此,all_modules就表示要对前面指定的Android.mk文件中定义的所有模块进行编译。
函数mmm的实现如下所示:
function mmm()
{T=$(gettop)if [ "$T" ]; thenlocal MAKEFILE=local MODULES=local ARGS=local DIR TO_CHOPlocal DASH_ARGS=$(echo "$@" | awk -v RS=" " -v ORS=" " '/^-.*$/')local DIRS=$(echo "$@" | awk -v RS=" " -v ORS=" " '/^[^-].*$/')for DIR in $DIRS ; doMODULES=`echo $DIR | sed -n -e 's/.*:\(.*$\)/\1/p' | sed 's/,/ /'`if [ "$MODULES" = "" ]; thenMODULES=all_modulesfiDIR=`echo $DIR | sed -e 's/:.*//' -e 's:/$::'`if [ -f $DIR/Android.mk ]; thenTO_CHOP=`(cd -P -- $T && pwd -P) | wc -c | tr -d ' '`TO_CHOP=`expr $TO_CHOP + 1`START=`PWD= /bin/pwd`MFILE=`echo $START | cut -c${TO_CHOP}-`if [ "$MFILE" = "" ] ; thenMFILE=$DIR/Android.mkelseMFILE=$MFILE/$DIR/Android.mkfiMAKEFILE="$MAKEFILE $MFILE"elseif [ "$DIR" = snod ]; thenARGS="$ARGS snod"elif [ "$DIR" = showcommands ]; thenARGS="$ARGS showcommands"elif [ "$DIR" = dist ]; thenARGS="$ARGS dist"elif [ "$DIR" = incrementaljavac ]; thenARGS="$ARGS incrementaljavac"elseecho "No Android.mk in $DIR."return 1fifidoneONE_SHOT_MAKEFILE="$MAKEFILE" make -C $T $DASH_ARGS $MODULES $ARGSelseecho "Couldn't locate the top of the tree. Try setting TOP."fi
}
函数mmm的实现就稍微复杂一点:
三.mm编译模块实例
要使用m、mm、mmm命令对Android代码进行模块编译,需要先执行如下命令初始化环境。
source build/envsetup.shlunch
mm编译成功后,在编译log会显示生成的so库。
在根目录find -name " " 搜索编译生成的so,然后push到板子对应的目录下。