[2018年最新整理]2Android源代码编译命令m和mm和mmm以及make分析
老罗的新浪微博:/shengyangluo,欢迎关注!
在前文中,我们分析了Android编译环境的初始化过程。Android编译环境初始化完成后,我们就可以用m/mm/mmm/make命令编译源代码了。当然,这要求每一个模块都有一个Android.mk文件。Android.mk实际上是一个Makefile脚本,用来描述模块编译信息。Android编译系统通过整合Android.mk文件完成编译过程。本文就对Android源代码的编译过程进行详细分析。
??????? 从前面Android编译系统环境初始化过程分析这篇文章可以知道,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的实现,然后进一步分析它们是如何通过make命令来编译代码的。
??????? 函数m的实现如下所示:
function?m()??
{??
????T=$(gettop)??
????if?[?"$T"?];?then??
????????make?-C?$T?$@??
????else??
????????echo?"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?];?then??
????????make?$@??
????else??
????????#?Find?the?closest?Android.mk?file.??
????????T=$(gettop)??
????????local?M=$(findmakefile)??
????????#?Remove?the?path?to?top?as?the?makefilepath?needs?to?be?relative??
????????local?M=`echo?$M|sed?'s:'$T'/::'`??
????????if?[?!?"$T"?];?then??
????????????echo?"Couldn't?locate?the?top?of?the?tree.??Try?setting?TOP."??
????????elif?[?!?"$M"?];?then??
????????????echo?"Couldn't?locate?a?makefile?from?the?current?directory."??
????????else??
????????????ONE_SHOT_MAKEFILE=$M?make?-C?$T?all_modules?$@??
????????fi??
????fi??
}??
??????? 函数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命令