最近由于项目的一些变化,需要摒弃掉用IDE编译的方式,改为使用Makefile手动编译。我又拾起了多年不碰的Makefile,虽然很多关键字不记得叫什么名字了,自己的文章也没有记录,好在网上还有很多同学都有记录的习惯,一搜索就能找到,倒也还好,只是有一个问题,折腾了好些时间,就是怎么处理自动依赖。
我对完成这件事情有两个要求
一是生成的.d .o文件必须独立存放,不要放在.c源文件的目录下,以免污染了source目录。
二是.c会include的所有.h文件,如果有更新,此.c也必须要重新编译。
不知道是因为这个平台(xtensillica)工具链的问题,还是我没搞清楚怎么弄。用-MM自动生成的.d文件,始终在我这两个要求之间二选一才能用,让我非常不爽,最后选择了一种比较土鳖的办法,只用这个工具链找出.h列表,剩下的事情我自己来,也就是对.d文件进行二次改造,按照自己的要求去编译,看起来效果还不错,至少能满足我这两个要求。
核心的代码如下,主要是第4/6两行。
这里的DEP指需要生成的所有.d文件列表,含完整路径,在之前的代码中组合好的。OBJDIR,SRCDIR,HPATH,CMACRO,CFLAGS顾名思义,也是这段代码之前准备好的。
$(DEP):$(OBJDIR)%.d:$(SRCDIR)%.c @if NOT EXIST $(subst /,\,$(dir $@)) (mkdir $(subst /,\,$(dir $@))) @echo generate $(notdir $@) @$(CC) -MM $< $(HPATH) $(CMACRO) -MT $(patsubst %.d,%.o,$@) > $@ @echo $(subst ',,' @echo compile $(notdir $<)')>>$@ @echo $(subst ',,' @$(CC) $(CFLAGS) $< -o $(patsubst %.d,%.o,$@)')>>$@ -include $(DEP)
解析:
第4行,预编译,生成依赖文件。
-MM,最基本的生成依赖的命令,列出所有.h文件列表
-MT,自定义目标文件名称,如果不用-MT,默认情况下只有目标文件命令,没有完整路径。我在这里使用-MT的目的是为了给目标.o加上前面的完整路径,以达到.o不要和.c同目录的目的。但是也是由于这个改动,导致默认的生成.d不work,我才会对.d进行二次改造。
第6行,是对.d的二次改造,把编译命令强行写进.d中,替换掉默认的隐含编译。
原文:http://blog.csdn.net/yang_dk/article/details/45166621