Makefile 初探
上一篇 / 下一篇 2006-06-23 13:14:31 / 天气: 晴朗 / 心情: 高兴 / 个人分类:Backup
Makefile 初探EDA中国门户网站SueK-_"S
5e,KQ a YY:E.{\'e0==========================================
7q4pST'Z KyT0EDA中国门户网站$p%E}H2[
为了优化make过程,减少不必要的重新编绎,linux开发了专用的mkdep工具,用它来取代gcc来生成.depend文件。mkdep在处理源文件时,忽略linux/config.h这样的头文件,识别源文件宏指令中具有"CONFIG_"特征的行。例如,如果有"#ifdef CONFIG_SMP"这样的行,它就会在.depend文件中输出$(wildcard /usr/src/linux/include/config/smp.h)。
7T![ MnCVUeJ nz0
K+X*d q*uYg0EDA中国门户网站L;F|c;p(y^
include/config/下的文件是另一个工具split-include从autoconf.h中生成,它利用autoconf.h中的CONFIG_标记,生成与mkdep相对应的文件。例如,如果autoconf.h中有"#undef CONFIG_SMP"这一行,它就生成include/config/smp.h文件,内容为"#undef CONFIG_SMP"。这些文件名只在.depend文件中出现,内核源文件是不会嵌入它们的。每配置一次内核,运行split-include一次。split-include会检查旧的子文件的内容,确定是不是要更新它们。这样,不管autoconf.h修改日期如何,只要其配置不变,make就不会重新编绎内核。
;E h!\g6X!^3s7r0EDA中国门户网站Z,N~7I1z)|[y&[mvQ;]
0~ Lh;o#T0如果系统的编绎选项发生了变化,linux也能进行增量编绎。为了做到这一点,make每编绎一个源文件时生成一个flags文件。例如编绎sched.c时,会在相同的目录下生成隐含的.sched.o.flags文件。它是Makefile的一个片断,当make进入某个子目录编绎时,会搜索其中的flags文件,将它们嵌入到Makefile中。这些flags代码测试当前的编绎选项与原来的是不是相同,如果相同,就将自已对应的目标文件加入FILES_FLAGS_UP_TO_DATE列表,然后,系统从编绎对象表中删除它们,得到FILES_FLAGS_CHANGED列表,最后,将它们设为目标进行更新。
w1aAI3^bz]O0d0
SC3[0m Fwe0EDA中国门户网站BFV a'{9Yxf3a H QR
下一步准备逐步深入的剖析Makefile代码。EDA中国门户网站*w;M+Z4^H-?
EDA中国门户网站/I4{#Wo0p,C']@
EDA中国门户网站%g*n Pr~%B,p7VE"Z
==========================================EDA中国门户网站GT)[v2hr#c~V$W i
L+aH|kzq0Makefile解读之二: sub-make
_Q4rlWR5@0
U:lqJ [-^+@ na0CAU0==========================================EDA中国门户网站[:mH{'v:h1Ate~v$h0A
EDA中国门户网站(fG'j4S;k'G9_,zb"?j
linux各级内核源代码的子目录下都有Makefile,大多数Makefile要嵌入主目录下的Rule.make,Rule.make将识别各个Makefile中所定义的一些变量。变量obj-y表示需要编绎到内核中的目标文件名集合,定义O_TARGET表示将obj-y连接为一个O_TARGET名称的目标文件,定义L_TARGET表示将obj-y合并为一个L_TARGET名称的库文件。同样obj-m表示需要编绎成模块的目标文件名集合。如果还需进行子目录make,则需要定义subdir-y和subdir-m。在Makefile中,用"obj-$(CONFIG_BINFMT_ELF) = binfmt_elf.o"和"subdir-$(CONFIG_EXT2_FS) = ext2"这种形式自动为obj-y、obj-m、subdir-y、subdir-m添加文件名。有时,情况没有这么单纯,还需要使用条件语句个别对待。Makefile中还有其它一些变量,如mod-subdirs定义了subdir-m以外的所有模块子目录。EDA中国门户网站"`eh~.H-\/WQ'f
EDA中国门户网站-ECS8|&B1WKloS(n
Q{.K T#~eq)?0Rules.make是如何使make进入子目录的呢? 先来看subdir-y是如何处理的,在Rules.make中,先对subdir-y中的每一个文件名加上前缀"_subdir_"再进行排序生成subdir-list集合,再以它作为目标集,对其中每一个目标产生一个子make,同时将目标名的前缀去掉得到子目录名,作为子make的起始目录参数。subdir-m与subdir-y类似,但情况稍微复杂一些。由于subdir-y中可能有模块定义,因此利用mod-subdirs变量将subdir-y中模块目录提取出来,再与subdir-m合成一个大的MOD_SUB_DIRS集合。subdir-m的目标所用的前缀是"_modsubdir_"。EDA中国门户网站9B[ M^H7NH/d wb(O
EDA中国门户网站"ah2CLo;N+|
s(Km8}v;O!k K'MEs0一点说明,子目录中的Makefile与Rules.make都没有嵌入.config文件,它是通过主Makefile向下传递MAKEFILES变量完成的。MAKEFILES是make自已识别的一个变量,在执行新的Makefile之前,make会首先加载MAKEFILES所指的文件。在主Makefile中它即指向.config。EDA中国门户网站w.vNY#V6\,w%gn
EDA中国门户网站&e6P G(p!G1j _
*Fh^ R(V-d1^-H0EDA中国门户网站:[2j}C!C2xO
==========================================
J5ZD"L$OR&n0EDA中国门户网站7jJ3UN#JS
Makefile解读之三: 模块的版本化处理
T5B6N0S*SS3n3v0EDA中国门户网站5^UBig t
==========================================
[{q-p*`a3@)I0
,iC[u%P2]-De0模块的版本化是内核与模块接口之间进行严格类型匹配的一种方法。当内核配置了CONFIG_MODVERSIONS之后,make dep操作会在include/linux/modules/目录下为各级Makefile中export-objs变量所对应的源文件生成扩展名为.ver的文件。
KHQ0z]J`\0
9GC-d p&l0
$K7dr0c,e5H9H H Ix${0例如对于kernel/ksyms.c,make用以下命令生成对应的ksyms.ver:
Rv5tf%q0
*a.T h k V A0EDA中国门户网站f}/B ^j
gcc -E -D__KERNEL__ -D__GENKSYMS__ ksyms.c | /sbin/genksyms -k 2.4.1 > ksyms.ver
bIO!cD&z^uCF0
C$M&Mp+^VvQmN3{0EDA中国门户网站.BmL.p ~"]&rS
-D__GENKSYMS__的作用是使ksyms.c中的EXPORT_SYMBOL宏不进行扩展。genksyms命令识别EXPORT_SYMBOL()中的函数名和对应的原型,再根据其原型计算出该函数的版本号。
`ngT-p_2`n M0EDA中国门户网站6mh4`+p)r{WwL&p
Ua"Y&eN,pX t0例如ksyms.c中有一行:
k7vaL/v0rRt0
5k5H(MeO9sI0EXPORT_SYMBOL(kmalloc);
h$V"~ha7^'z8|0
l.G.i9OEO0kmalloc原型是:
"v4UaH!K;Bi[ WPlD0EDA中国门户网站y$Gly$W6b8Q
void *kmalloc(size_t, int);
'H7RzpTbo0