免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 8952 | 回复: 11
打印 上一主题 下一主题

[转贴]Makefile解读 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2003-02-09 12:27 |只看该作者 |倒序浏览
原文出自:http://www.linuxforum.net
作者:jkl

==========================================
Makefile 初探
==========================================
Linux的内核配置文件有两个,一个是隐含的.config文件,嵌入到主Makefile中;另一个是include/linux/autoconf.h,嵌入到各个c源文件中,它们由make config、make menuconfig、make xconfig这些过程创建。几乎所有的源文件都会通过linux/config.h而嵌入autoconf.h,如果按照通常方法建立文件依赖关系(.depend),只要更新过autoconf.h,就会造成所有源代码的重新编绎。

为了优化make过程,减少不必要的重新编绎,Linux开发了专用的mkdep工具,用它来取代gcc来生成.depend文件。mkdep在处理源文件时,忽略linux/config.h这样的头文件,识别源文件宏指令中具有"CONFIG_"特征的行。例如,如果有"#ifdef CONFIG_SMP"这样的行,它就会在.depend文件中输出$(wildcard /usr/src/linux/include/config/smp.h)。

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就不会重新编绎内核。

如果系统的编绎选项发生了变化,Linux也能进行增量编绎。为了做到这一点,make每编绎一个源文件时生成一个flags文件。例如编绎sched.c时,会在相同的目录下生成隐含的.sched.o.flags文件。它是Makefile的一个片断,当make进入某个子目录编绎时,会搜索其中的flags文件,将它们嵌入到Makefile中。这些flags代码测试当前的编绎选项与原来的是不是相同,如果相同,就将自已对应的目标文件加入FILES_FLAGS_UP_TO_DATE列表,然后,系统从编绎对象表中删除它们,得到FILES_FLAGS_CHANGED列表,最后,将它们设为目标进行更新。

下一步准备逐步深入的剖析Makefile代码。

==========================================
Makefile解读之二: sub-make
==========================================
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以外的所有模块子目录。

Rules.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_"。

一点说明,子目录中的Makefile与Rules.make都没有嵌入.config文件,它是通过主Makefile向下传递MAKEFILES变量完成的。MAKEFILES是make自已识别的一个变量,在执行新的Makefile之前,make会首先加载MAKEFILES所指的文件。在主Makefile中它即指向.config。


==========================================
Makefile解读之三: 模块的版本化处理
==========================================
模块的版本化是内核与模块接口之间进行严格类型匹配的一种方法。当内核配置了CONFIG_MODVERSIONS之后,make dep操作会在include/linux/modules/目录下为各级Makefile中export-objs变量所对应的源文件生成扩展名为.ver的文件。

例如对于kernel/ksyms.c,make用以下命令生成对应的ksyms.ver:

gcc -E -D__KERNEL__ -D__GENKSYMS__ ksyms.c | /sbin/genksyms -k 2.4.1 >; ksyms.ver

-D__GENKSYMS__的作用是使ksyms.c中的EXPORT_SYMBOL宏不进行扩展。genksyms命令识别EXPORT_SYMBOL()中的函数名和对应的原型,再根据其原型计算出该函数的版本号。

例如ksyms.c中有一行:
EXPORT_SYMBOL(kmalloc);
kmalloc原型是:
void *kmalloc(size_t, int);
genksyms程序对应的输出为:
#define __ver_kmalloc 93d4cfe6
#define kmalloc _set_ver(kmalloc)
在内核符号表和模块中,kmalloc将变成kmalloc_R93d4cfe6。

在生成完所有的.ver文件后,make将重建include/linux/modversions.h文件,它包含一系列#include指令行嵌入各个.ver文件。在编绎内核本身export-objs中的文件时,make会增加一个"-DEXPORT_SYMTAB"编绎标志,它使源文件嵌入modversions.h文件,将EXPORT_SYMBOL宏展开中的函数名字符串进行版本名扩展;同时,它也定义_set_ver()宏为一空操作,使代码中的函数名不受其影响。
在编绎模块时,make会增加"-include=linux/modversion.h -DMODVERSIONS"编绎标志,使模块中代码的函数名得到相应版本扩展。

由于生成.ver文件比较费时,make还为每个.ver创建了一个后缀为.stamp时戳文件。在make dep时,如果其.stamp文件比源文件旧才重新生成.ver文件,否则只是更新.stamp文件时戳。另外,在生成.ver和modversions.h文件时,make都会比较新文件和旧文件的内容,保持它们修改时间为最旧。



==========================================
Makefile解读之四: Rules.make的注释
==========================================

  1. #
  2. # This file contains rules which are shared between multiple Makefiles.
  3. #

  4. #
  5. # False targets.
  6. #
  7. #
  8. .PHONY: dummy

  9. #
  10. # Special variables which should not be exported
  11. #
  12. # 取消这些变量通过环境向make子进程传递。
  13. unexport EXTRA_AFLAGS        # as 的开关
  14. unexport EXTRA_CFLAGS        # cc 的开关
  15. unexport EXTRA_LDFLAGS         # ld 的开关
  16. unexport EXTRA_ARFLAGS        # ar 的开关
  17. unexport SUBDIRS        #
  18. unexport SUB_DIRS        # 编绎内核需进入的子目录,等于subdir-y
  19. unexport ALL_SUB_DIRS        # 所有的子目录
  20. unexport MOD_SUB_DIRS        # 编绎模块需进入的子目录
  21. unexport O_TARGET        # ld合并的输出对象
  22. unexport ALL_MOBJS        # 所有的模块名

  23. unexport obj-y                # 编绎成内核的文件集
  24. unexport obj-m                # 编绎成模块的文件集
  25. unexport obj-n                #
  26. unexport obj-                #
  27. unexport export-objs        # 需进行版本处理的文件集
  28. unexport subdir-y        # 编绎内核所需进入的子目录
  29. unexport subdir-m        # 编绎模块所需进入的子目录
  30. unexport subdir-n
  31. unexport subdir-

  32. #
  33. # Get things started.
  34. #
  35. first_rule: sub_dirs
  36.         $(MAKE) all_targets

  37. # 在内核编绎子目录中过滤出可以作为模块的子目录。
  38. both-m          := $(filter $(mod-subdirs), $(subdir-y))
  39. SUB_DIRS        := $(subdir-y)
  40. # 求出总模块子目录
  41. MOD_SUB_DIRS        := $(sort $(subdir-m) $(both-m))
  42. # 求出总子目录
  43. ALL_SUB_DIRS        := $(sort $(subdir-y) $(subdir-m) $(subdir-n) $(subdir-))


  44. #
  45. # Common rules
  46. #
  47. # 将c文件编绎成汇编文件的规则,$@为目标对象。
  48. %.s: %.c
  49.         $(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -S $< -o $@
  50. # 将c文件生成预处理文件的规则。
  51. %.i: %.c
  52.         $(CPP) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) $< >; $@
  53. # 将c文件编绎成目标文件的规则,$<为第一个所依赖的对象;
  54. #
  55. 在目标文件的目录下生成flags文件,strip删除多余的空格,subst将逗号替换成冒号

  56. %.o: %.c
  57.         $(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -c -o $@ $<
  58.         @ ( \
  59.             echo 'ifeq ($(strip $(subst $(comma),:,$(CFLAGS) $(EXTRA_CFLAGS)
  60. $(CFLAGS_$@))),$$(strip $$(subst $$(comma),:,$$(CFLAGS) $$(EXTRA_CFLAGS)
  61. $$(CFLAGS_$@))))' ; \
  62.             echo 'FILES_FLAGS_UP_TO_DATE += $@' ; \
  63.             echo 'endif' \
  64.         ) >; $(dir $@)/.$(notdir $@).flags
  65. # 汇编文件生成目标文件的规则。
  66. %.o: %.s
  67.         $(AS) $(AFLAGS) $(EXTRA_CFLAGS) -o $@ $<

  68. # Old makefiles define their own rules for compiling .S files,
  69. # but these standard rules are available for any Makefile that
  70. # wants to use them.  Our plan is to incrementally convert all
  71. # the Makefiles to these standard rules.  -- rmk, mec

  72. ifdef USE_STANDARD_AS_RULE
  73. # 汇编文件生成预处理文件的标准规则。
  74. %.s: %.S
  75.         $(CPP) $(AFLAGS) $(EXTRA_AFLAGS) $(AFLAGS_$@) $< >; $@
  76. # 汇编文件生成目标文件的标准规则。
  77. %.o: %.S
  78.         $(CC) $(AFLAGS) $(EXTRA_AFLAGS) $(AFLAGS_$@) -c -o $@ $<

  79. endif
  80. # c文件生成调试列表文件的规则,$*扩展为目标的主文件名。
  81. %.lst: %.c
  82.         $(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -g -c -o $*.o $<
  83.         $(TOPDIR)/scripts/makelst $* $(TOPDIR) $(OBJDUMP)
  84. #
  85. #
  86. #
  87. all_targets: $(O_TARGET) $(L_TARGET)

  88. #
  89. # Rule to compile a set of .o files into one .o file
  90. #
  91. ifdef O_TARGET
  92. $(O_TARGET): $(obj-y)
  93.         rm -f $@
  94. # $^扩展为全部依赖对象,如果obj-y为空就生成一个同名空的库文件。
  95.     ifneq "$(strip $(obj-y))" ""
  96.         $(LD) $(EXTRA_LDFLAGS) -r -o $@ $(filter $(obj-y), $^)
  97.     else
  98.         $(AR) rcs $@
  99.     endif
  100. # 生成flags文件的shell语句。
  101.         @ ( \
  102.             echo 'ifeq ($(strip $(subst $(comma),:,$(EXTRA_LDFLAGS)
  103. $(obj-y))),$$(strip $$(subst $$(comma),:,$$(EXTRA_LDFLAGS) $$(obj-y))))' ;
  104. \
  105.             echo 'FILES_FLAGS_UP_TO_DATE += $@' ; \
  106.             echo 'endif' \
  107.         ) >; $(dir $@)/.$(notdir $@).flags
  108. endif # O_TARGET

  109. #
  110. # Rule to compile a set of .o files into one .a file
  111. #
  112. # 将obj-y组合成库L_TARGET的方法。
  113. ifdef L_TARGET
  114. $(L_TARGET): $(obj-y)
  115.         rm -f $@
  116.         $(AR) $(EXTRA_ARFLAGS) rcs $@ $(obj-y)
  117.         @ ( \
  118.             echo 'ifeq ($(strip $(subst $(comma),:,$(EXTRA_ARFLAGS)
  119. $(obj-y))),$$(strip $$(subst $$(comma),:,$$(EXTRA_ARFLAGS) $$(obj-y))))' ;
  120. \
  121.             echo 'FILES_FLAGS_UP_TO_DATE += $@' ; \
  122.             echo 'endif' \
  123.         ) >; $(dir $@)/.$(notdir $@).flags
  124. endif


  125. #
  126. # This make dependencies quickly
  127. #
  128. # wildcard为查找目录中的文件名的宏。
  129. fastdep: dummy
  130.         $(TOPDIR)/scripts/mkdep $(wildcard *.[chS] local.h.master) >; .depend
  131. ifdef ALL_SUB_DIRS
  132. #
  133. 将ALL_SUB_DIRS中的目录名加上前缀_sfdep_作为目标运行子make,并将ALL_SUB_DIRS
  134. 通过
  135. # 变量_FASTDEP_ALL_SUB_DIRS传递给子make。
  136.         $(MAKE) $(patsubst %,_sfdep_%,$(ALL_SUB_DIRS))
  137. _FASTDEP_ALL_SUB_DIRS="$(ALL_SUB_DIRS)"
  138. endif

  139. ifdef _FASTDEP_ALL_SUB_DIRS
  140. #
  141. 与上一段相对应,定义子目录目标,并将目标名还原为目录名,进入该子目录make。
  142. $(patsubst %,_sfdep_%,$(_FASTDEP_ALL_SUB_DIRS)):
  143.         $(MAKE) -C $(patsubst _sfdep_%,%,$@) fastdep
  144. endif

  145.        
  146. #
  147. # A rule to make subdirectories
  148. #
  149. # 下面2段完成内核编绎子目录中的make。
  150. subdir-list = $(sort $(patsubst %,_subdir_%,$(SUB_DIRS)))
  151. sub_dirs: dummy $(subdir-list)

  152. ifdef SUB_DIRS
  153. $(subdir-list) : dummy
  154.         $(MAKE) -C $(patsubst _subdir_%,%,$@)
  155. endif

  156. #
  157. # A rule to make modules
  158. #
  159. # 求出有效的模块文件表。
  160. ALL_MOBJS = $(filter-out $(obj-y), $(obj-m))
  161. ifneq "$(strip $(ALL_MOBJS))" ""
  162. # 取主目录TOPDIR到当前目录的路径。
  163. PDWN=$(shell $(CONFIG_SHELL) $(TOPDIR)/scripts/pathdown.sh)
  164. endif

  165. unexport MOD_DIRS
  166. MOD_DIRS := $(MOD_SUB_DIRS) $(MOD_IN_SUB_DIRS)
  167. # 编绎模块时,进入模块子目录的方法。
  168. ifneq "$(strip $(MOD_DIRS))" ""
  169. .PHONY: $(patsubst %,_modsubdir_%,$(MOD_DIRS))
  170. $(patsubst %,_modsubdir_%,$(MOD_DIRS)) : dummy
  171.         $(MAKE) -C $(patsubst _modsubdir_%,%,$@) modules
  172. # 安装模块时,进入模块子目录的方法。
  173. .PHONY: $(patsubst %,_modinst_%,$(MOD_DIRS))
  174. $(patsubst %,_modinst_%,$(MOD_DIRS)) : dummy
  175.         $(MAKE) -C $(patsubst _modinst_%,%,$@) modules_install
  176. endif

  177. # make modules 的入口。
  178. .PHONY: modules
  179. modules: $(ALL_MOBJS) dummy \
  180.          $(patsubst %,_modsubdir_%,$(MOD_DIRS))

  181. .PHONY: _modinst__
  182. # 拷贝模块的过程。
  183. _modinst__: dummy
  184. ifneq "$(strip $(ALL_MOBJS))" ""
  185.         mkdir -p $(MODLIB)/kernel/$(PDWN)
  186.         cp $(ALL_MOBJS) $(MODLIB)/kernel/$(PDWN)
  187. endif

  188. # make modules_install 的入口,进入子目录安装。
  189. .PHONY: modules_install
  190. modules_install: _modinst__ \
  191.          $(patsubst %,_modinst_%,$(MOD_DIRS))

  192. #
  193. # A rule to do nothing
  194. #
  195. dummy:

  196. #
  197. # This is useful for testing
  198. #
  199. script:
  200.         $(SCRIPT)

  201. #
  202. # This sets version suffixes on exported symbols
  203. # Separate the object into "normal" objects and "exporting" objects
  204. # Exporting objects are: all objects that define symbol tables
  205. #
  206. ifdef CONFIG_MODULES
  207. # list-multi列出那些由多个文件复合而成的模块;
  208. # 从编绎文件表和模块文件表中过滤出复合模块名。
  209. multi-used        := $(filter $(list-multi), $(obj-y) $(obj-m))
  210. # 取复合模块的构成表。
  211. multi-objs        := $(foreach m, $(multi-used), $($(basename $(m))-objs))
  212. # 求出需进行编译的总模块表。
  213. active-objs        := $(sort $(multi-objs) $(obj-y) $(obj-m))

  214. ifdef CONFIG_MODVERSIONS
  215. ifneq "$(strip $(export-objs))" ""
  216. # 如果有需要进行版本化的文件。
  217. MODINCL = $(TOPDIR)/include/linux/modules

  218. # The -w option (enable warnings) for genksyms will return here in 2.1
  219. # So where has it gone?
  220. #
  221. # Added the SMP separator to stop module accidents between uniprocessor
  222. # and SMP Intel boxes - AC - from bits by Michael Chastain
  223. #

  224. ifdef CONFIG_SMP
  225.         genksyms_smp_prefix := -p smp_
  226. else
  227.         genksyms_smp_prefix :=
  228. endif
  229. # 从源文件计算版本文件的规则。
  230. $(MODINCL)/%.ver: %.c
  231.         @if [ ! -r $(MODINCL)/$*.stamp -o $(MODINCL)/$*.stamp -ot $< ]; then \
  232.                 echo '$(CC) $(CFLAGS) -E -D__GENKSYMS__ $<'; \
  233.                 echo '| $(GENKSYMS) $(genksyms_smp_prefix) -k
  234. $(VERSION).$(PATCHLEVEL).$(SUBLEVEL) >; $@.tmp'; \
  235.                 $(CC) $(CFLAGS) -E -D__GENKSYMS__ $< \
  236.                 | $(GENKSYMS) $(genksyms_smp_prefix) -k
  237. $(VERSION).$(PATCHLEVEL).$(SUBLEVEL) >; $@.tmp; \
  238.                 if [ -r $@ ] && cmp -s $@ $@.tmp; then echo $@ is unchanged; rm -f
  239. $@.tmp; \
  240.                 else echo mv $@.tmp $@; mv -f $@.tmp $@; fi; \
  241.         fi; touch $(MODINCL)/$*.stamp
  242. #
  243. 将版本处理源文件的扩展名改为.ver,并加上完整的路径名,它们依赖于autoconf.h?br>;?br>;$(addprefix $(MODINCL)/,$(export-objs:.o=.ver)):
  244. $(TOPDIR)/include/linux/autoconf.h

  245. # updates .ver files but not modversions.h
  246. # 通过fastdep,逐个生成export-objs对应的版本文件。
  247. fastdep: $(addprefix $(MODINCL)/,$(export-objs:.o=.ver))

  248. # updates .ver files and modversions.h like before (is this needed?)
  249. # make dep过程的入口
  250. dep: fastdep update-modverfile

  251. endif # export-objs

  252. # update modversions.h, but only if it would change
  253. # 刷新版本文件的过程。
  254. update-modverfile:
  255.         @(echo "#ifndef _LINUX_MODVERSIONS_H";\
  256.           echo "#define _LINUX_MODVERSIONS_H"; \
  257.           echo "#include <linux/modsetver.h>;"; \
  258.           cd $(TOPDIR)/include/linux/modules; \
  259.           for f in *.ver; do \
  260.             if [ -f $$f ]; then echo "#include <linux/modules/$${f}>;"; fi; \
  261.           done; \
  262.           echo "#endif"; \
  263.         ) >; $(TOPDIR)/include/linux/modversions.h.tmp
  264.         @if [ -r $(TOPDIR)/include/linux/modversions.h ] && cmp -s
  265. $(TOPDIR)/include/linux/modversions.h
  266. $(TOPDIR)/include/linux/modversions.h.tmp; then \
  267.                 echo $(TOPDIR)/include/linux/modversions.h was not updated; \
  268.                 rm -f $(TOPDIR)/include/linux/modversions.h.tmp; \
  269.         else \
  270.                 echo $(TOPDIR)/include/linux/modversions.h was updated; \
  271.                 mv -f $(TOPDIR)/include/linux/modversions.h.tmp
  272. $(TOPDIR)/include/linux/modversions.h; \
  273.         fi
  274. $(active-objs): $(TOPDIR)/include/linux/modversions.h

  275. else
  276. # 如果没有配置版本化,modversions.h的内容。
  277. $(TOPDIR)/include/linux/modversions.h:
  278.         @echo "#include <linux/modsetver.h>;" >; $@

  279. endif # CONFIG_MODVERSIONS

  280. ifneq "$(strip $(export-objs))" ""
  281. # 版本化目标文件的编绎方法。
  282. $(export-objs): $(export-objs:.o=.c) $(TOPDIR)/include/linux/modversions.h
  283.         $(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -DEXPORT_SYMTAB -c $(@:.o=.c)
  284.         @ ( \
  285.             echo 'ifeq ($(strip $(subst $(comma),:,$(CFLAGS) $(EXTRA_CFLAGS)
  286. $(CFLAGS_$@) -DEXPORT_SYMTAB)),$$(strip $$(subst $$(comma),:,$$(CFLAGS)
  287. $$(EXTRA_CFLAGS) $$(CFLAGS_$@) -DEXPORT_SYMTAB)))' ; \
  288.             echo 'FILES_FLAGS_UP_TO_DATE += $@' ; \
  289.             echo 'endif' \
  290.         ) >; $(dir $@)/.$(notdir $@).flags
  291. endif

  292. endif # CONFIG_MODULES


  293. #
  294. # include dependency files if they exist
  295. #
  296. # 嵌入源文件之间的依赖关系。
  297. ifneq ($(wildcard .depend),)
  298. include .depend
  299. endif
  300. # 嵌入头文件之间的依赖关系。
  301. ifneq ($(wildcard $(TOPDIR)/.hdepend),)
  302. include $(TOPDIR)/.hdepend
  303. endif

  304. #
  305. # Find files whose flags have changed and force recompilation.
  306. # For safety, this works in the converse direction:
  307. #   every file is forced, except those whose flags are positively
  308. up-to-date.
  309. #
  310. # 已经更新过的文件列表。
  311. FILES_FLAGS_UP_TO_DATE :=

  312. # For use in expunging commas from flags, which mung our checking.
  313. comma = ,
  314. # 将当前目录下所有flags文件嵌入。
  315. FILES_FLAGS_EXIST := $(wildcard .*.flags)
  316. ifneq ($(FILES_FLAGS_EXIST),)
  317. include $(FILES_FLAGS_EXIST)
  318. endif
  319. # 将无需更新的文件从总的对象中删除。
  320. FILES_FLAGS_CHANGED := $(strip \
  321.     $(filter-out $(FILES_FLAGS_UP_TO_DATE), \
  322.         $(O_TARGET) $(L_TARGET) $(active-objs) \
  323.         ))

  324. # A kludge: .S files don't get flag dependencies (yet),
  325. #   because that will involve changing a lot of Makefiles.  Also
  326. #   suppress object files explicitly listed in $(IGNORE_FLAGS_OBJS).
  327. #   This allows handling of assembly files that get translated into
  328. #   multiple object files (see arch/ia64/lib/idiv.S, for example).
  329. #
  330. # 将由汇编文件生成的目件文件从FILES_FLAGS_CHANGED删除。
  331. FILES_FLAGS_CHANGED := $(strip \
  332.     $(filter-out $(patsubst %.S, %.o, $(wildcard *.S)
  333. $(IGNORE_FLAGS_OBJS)), \
  334.     $(FILES_FLAGS_CHANGED)))
  335. # 将FILES_FLAGS_CHANGED设为目标。
  336. ifneq ($(FILES_FLAGS_CHANGED),)
  337. $(FILES_FLAGS_CHANGED): dummy
  338. endif
  339. </pre>;


复制代码

论坛徽章:
0
2 [报告]
发表于 2003-02-19 23:08 |只看该作者

[转贴]Makefile解读

UP!

论坛徽章:
0
3 [报告]
发表于 2003-06-19 11:21 |只看该作者

[转贴]Makefile解读

mark

论坛徽章:
0
4 [报告]
发表于 2003-09-08 21:49 |只看该作者

[转贴]Makefile解读

我也顶!
这样的帖子,对像我这样的Unix和make的初学者来说,特别有用,谢谢!!

论坛徽章:
0
5 [报告]
发表于 2006-01-17 16:47 |只看该作者
to: Makefile解读之四: Rules.make的注释

最后,这个文件,根本就不能work 啊.

论坛徽章:
0
6 [报告]
发表于 2006-01-17 16:52 |只看该作者
jkl大侠这篇文档是5年前的, 自从他01年归隐之后, 似乎没人续写过
P.S. 翻遍linuxforum旧贴,从未见jkl解答不出来的问题……

论坛徽章:
0
7 [报告]
发表于 2006-01-18 11:10 |只看该作者
像这样的修行,是要有个10年20年的...

论坛徽章:
0
8 [报告]
发表于 2006-06-17 01:28 |只看该作者
谢谢. 膜拜...

论坛徽章:
0
9 [报告]
发表于 2006-06-17 11:48 |只看该作者

回复 6楼 albcamus 的帖子

albcamus, 我是 linuxforum.net 里的小小潜水家了。
linuxforum 里的猛男太多了,包括你,呵呵。

可是最近很长一段时间那里都登不上了。觉得生活少了些什么,:<. 郁闷啊!

现在我跑到这里来,希望你多多照顾啊,呵呵。能不能顺便也介绍下这个论坛的历史等情况啊?呵呵。

论坛徽章:
0
10 [报告]
发表于 2006-06-19 20:58 |只看该作者
厉害阿
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP