一个通用的Makefile (转)
- src/a/a.c
- src/b/b.c
- src/main.c
如何写Makefile使临时文件.o .d(之所以需要生成.d,是因为头文件依赖,后面会提到)等自动生成到
- build/a/a.o 和a.d
- build/b/b.o 和b.d
- build/main.o 和main.d
通常可以做到所有临时文件都生成到build下,如下规则:
- ......
- TARGETMAIN = testmk
- OBJECTDIR = build
- VPATH = $(shell ls -AxR ./src|grep ":"|grep -v "\.svn"|tr -d ':')
- SOURCEDIRS = $(VPATH)
- # search source file in the current dirs
- SOURCES = $(foreach subdir,$(SOURCEDIRS),$(wildcard $(subdir)/*.c))
- SRCOBJS = $(patsubst %.c,%.o,$(SOURCES))
- BASE_FILES = $(notdir $(TMSRCOBJS))
- BUILDOBJS = $(BASE_FILES:%=$(OBJECTDIR)/%)
- all:$(TARGETMAIN)
- $(TARGETMAIN) :$(BUILDOBJS)
- $(CC) $(CFLAGS) -o $@ -c $<
- @$(STRIP) --strip-unneeded $(TARGETMAIN)
- ......
- $(OBJECTDIR)/%.o: %.c $(DEPS_DIR)
- @$(CC) $(CPPFLAGS) $(CFLAGS) -o $@ -c $<
- #声明伪目标,防止Makefile去生成all等
- .PHONY : all install clean
- #定义路径变量,所有.c文件和所有非公开的.h应该放在src下,所有需要的.a文件放在lib
- #下,所有公开的.h(比如生成库文件的时候)或者多个.c公用的.h放在include文件夹下
- #global directory defined
- TOPDIR = $(shell pwd)
- SRCDIR = $(TOPDIR)/src
- LIBDIR = $(TOPDIR)/lib
- OBJECTDIR = $(TOPDIR)/build
- INCLUDEDIR = $(TOPDIR)/include
- #定义交叉编译环境变量,当需要编译arm/mips等平台应用程序/库的时候修改它
- #cross compile tools defined
- CROSS_COMPILE =
- AS = $(CROSS_COMPILE)as
- LD = $(CROSS_COMPILE)ld
- CC = $(CROSS_COMPILE)gcc
- CPP = $(CC) -E
- AR = $(CROSS_COMPILE)ar
- NM = $(CROSS_COMPILE)nm
- STRIP = $(CROSS_COMPILE)strip
- RANLIB = $(CROSS_COMPILE)ranlib
- #本机相关的命令,一般无需修改
- #local host tools defined
- CP := cp
- RM := rm
- MKDIR := mkdir
- SED := sed
- FIND := find
- MKDIR := mkdir
- XARGS := xargs
- #目标名称,这里我们给出了三种常用的目标格式:目标文件,静态库和共享库
- #target name
- TARGETMAIN = testmk
- TARGETLIBS = libmk.a
- TARGETSLIBS = libmk.so
- #所有源码文件的路径被放入SOURCEDIRS,所有.c源码文件(含路径)放入SOURCES
- #.c .o and .d files defined
- VPATH = $(shell ls -AxR $(SRCDIR)|grep ":"|grep -v "\.svn"|tr -d ':')
- SOURCEDIRS = $(VPATH)
- SOURCES = $(foreach subdir,$(SOURCEDIRS),$(wildcard $(subdir)/*.c))
- #所有目标文件.o(含路径)放入BUILDOBJS,注意它们的路径已经是build了。
- SRCOBJS = $(patsubst %.c,%.o,$(SOURCES))
- BUILDOBJS = $(subst $(SRCDIR),$(OBJECTDIR),$(SRCOBJS))
- #所有.d依赖文件放入DEPS
- DEPS = $(patsubst %.o,%.d,$(BUILDOBJS))
- #注意-MD,是为了生成.d文件后,构造对.h的依赖
- #external include file define
- CFLAGS = -O2 -Wall -MD $(foreach dir,$(INCLUDEDIR),-I$(dir))
- ARFLAGS = rc
- #special parameters for app
- CFLAGS +=
- #LDFLAGS指明所有-llibxx,libxx.a应该放到lib下,当然也可以添加.so。Xlinker是为了
- #在提供多个.a时,未知它们之间的依赖顺序时,自动查找依赖顺序
- #c file compile parameters and linked libraries
- CPPFLAGS =
- LDFLAGS =
- XLDFLAGS = -Xlinker "-(" $(LDFLAGS) -Xlinker "-)"
- LDLIBS += -L$(LIBDIR)
- #如果要生成.a或者.so,那么不要将main函数所在的.c放入src。另外添加$(TARGETLIBS)
- #或$(TARGETSLIBS)到all中
- #defaut target:compile the currrent dir file and sub dir
- all: $(TARGETMAIN)
- #for .h header files dependence
- -include $(DEPS)
- $(TARGETMAIN) :$(BUILDOBJS)
- @$(CC) $(subst $(SRCDIR),$(OBJECTDIR),$^) $(CPPFLAGS) $(CFLAGS) $(XLDFLAGS) -o $@ $(LDLIBS)
- @$(STRIP) --strip-unneeded $(TARGETMAIN)
- $(TARGETLIBS) :$(BUILDOBJS)
- @$(AR) $(ARFLAGS) $@ $(BUILDOBJS)
- @$(RANLIB) $@
- $(TARGETSLIBS) :$(BUILDOBJS)
- @$(CC) -shared $(subst $(SRCDIR),$(OBJECTDIR),$^) $(CPPFLAGS) $(CFLAGS) $(XLDFLAGS) -o $@ $(LDLIBS)
- #这里是Makefile的核心,根据%中的内容,查找src路径下对应的.c,注意到$@和$<自动
- #变量的取值,首先查看路径build/xx是否存在,不存在则创建,然后我们尝试将$@中的src
- #替换为build,这样所有的.o和.d都将被创建到对应的build下了。
- $(OBJECTDIR)%.o: $(SRCDIR)%.c
- @[ ! -d $(dir $(subst $(SRCDIR),$(OBJECTDIR),$@)) ] & $(MKDIR) -p $(dir $(subst $(SRCDIR),$(OBJECTDIR),$@))
- @$(CC) $(CPPFLAGS) $(CFLAGS) -o $(subst $(SRCDIR),$(OBJECTDIR),$@) -c $<
- #添加安装的路径
- intall:
- clean:
- @$(FIND) $(OBJECTDIR) -name "*.o" -o -name "*.d" | $(XARGS) $(RM) -f
- @$(RM) -f $(TARGETMAIN) $(TARGETLIBS) $(TARGETSLIBS)
经测试,完全满足了要求,虽然不是完美无缺,但是对于通常的工程项目,是绰绰有余了,欢迎大家指正,以便改善。
另外对于Makefile的理解,个人认为可以分成几个部分:依赖关系,自定义变量和自动变量以及Makefile提供的相关函数。理解了它们,对写出结构良好,通用性强的Makefile会有大的帮助,虽然现在有了automake,但是研究一下Makefile的写法还是有收获的。
附件testmk.rar中提供了一个完整的测试程序,目录如下:
- |-- build
- |-- include
- | `-- hello.h
- |-- lib
- |-- Makefile
- `-- src
- |-- a
- | `-- a.c
- |-- b
- | `-- b.c
- `-- main.c
- 6 directories, 5 files
在linux下解压开后运行命令一下命令,可以根据需要定制Makefile。
- #make // 当前目录下生成testmk
- #make libmk.a // 当前目录下生成libmk.a
- #make libmk.so // 当前目录下生成libmk.so
参考资料:
- 跟我一起写Makefile(通俗易懂,易于理解)
- GNU make中文手册(内容很全,是对GNU makefile手册的完全翻译,做手册参考)
思考:
根据Makefile的依赖原理,似乎应该使用一个树类型的结构,a依赖b,c等,b又依赖其他的文件,但是不能反方向依赖(否则为死循环),但是d可以同时依赖b,c,所以不是简单的树形结构,应该是一个图,另外依赖是有方向性的,所以应该是一个有向图结构的实现,不知道对不对?
# A generic template Makefile
# Author Red_Liu lli_njupt@.com v0.
#
# This file is a(part of) free software; you can redistribute it
# and/or modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version , or
# (at your option) any later version. .PHONY : all install clean #global directory defined
TOPDIR = $(shell pwd)
SRCDIR = $(TOPDIR)/src
LIBDIR = $(TOPDIR)/lib
OBJECTDIR = $(TOPDIR)/build
INCLUDEDIR = $(TOPDIR)/include #cross compile tools defined
CROSS_COMPILE ?=
AS = $(CROSS_COMPILE)as
LD = $(CROSS_COMPILE)ld
CC = $(CROSS_COMPILE)gcc
CPP = $(CC) -E
AR = $(CROSS_COMPILE)ar
NM = $(CROSS_COMPILE)nm
STRIP = $(CROSS_COMPILE)strip
RANLIB = $(CROSS_COMPILE)ranlib #local host tools defined
CP := cp
RM := rm
MKDIR := mkdir
SED := sed
FIND := find
MKDIR := mkdir
XARGS := xargs #target name
TARGETMAIN = testmk
TARGETLIBS = libmk.a
TARGETSLIBS = libmk.so #FILE' INFOMATION COLLECT
VPATH = $(shell ls -AxR $(SRCDIR)|grep ":"|grep -v "\.svn"|tr -d ':')
SOURCEDIRS = $(VPATH) #search source file in the current dir
SOURCES = $(foreach subdir,$(SOURCEDIRS),$(wildcard $(subdir)/*.c))
SRCOBJS = $(patsubst %.c,%.o,$(SOURCES))
BUILDOBJS = $(subst $(SRCDIR),$(OBJECTDIR),$(SRCOBJS))
DEPS = $(patsubst %.o,%.d,$(BUILDOBJS)) #external include file define
CFLAGS = -O2 -Wall -MD $(foreach dir,$(INCLUDEDIR),-I$(dir))
ARFLAGS = rc #special parameters for apps
CFLAGS += #c file compile parameters and linked libraries
CPPFLAGS =
LDFLAGS =
XLDFLAGS = -Xlinker "-(" $(LDFLAGS) -Xlinker "-)"
LDLIBS += -L$(LIBDIR) #defaut target:compile the currrent dir file and sub dir
all: $(TARGETMAIN) #for .h header files dependence
-include $(DEPS) $(TARGETMAIN) :$(BUILDOBJS)
@$(CC) $(subst $(SRCDIR),$(OBJECTDIR),$^) $(CPPFLAGS) $(CFLAGS) $(XLDFLAGS) -o $@ $(LDLIBS)
@$(STRIP) --strip-unneeded $(TARGETMAIN) $(TARGETLIBS) :$(BUILDOBJS)
@$(AR) $(ARFLAGS) $@ $(BUILDOBJS)
@$(RANLIB) $@ $(TARGETSLIBS) :$(BUILDOBJS)
@$(CC) -shared $(subst $(SRCDIR),$(OBJECTDIR),$^) $(CPPFLAGS) $(CFLAGS) $(XLDFLAGS) -o $@ $(LDLIBS) $(OBJECTDIR)%.o: $(SRCDIR)%.c
@[ ! -d $(dir $(subst $(SRCDIR),$(OBJECTDIR),$@)) ] & $(MKDIR) -p $(dir $(subst $(SRCDIR),$(OBJECTDIR),$@))
@$(CC) $(CPPFLAGS) $(CFLAGS) -o $(subst $(SRCDIR),$(OBJECTDIR),$@) -c $< intall: clean:
@$(FIND) $(OBJECTDIR) -name "*.o" -o -name "*.d" | $(XARGS) $(RM) -f
@$(RM) -f $(TARGETMAIN) $(TARGETLIBS) $(TARGETSLIBS)
Linux Makefile 生成 *.d 依赖文件及 gcc -M -MF -MP 等相关选项说明
1. 为什么要使用后缀名为 .d 的依赖文件?
在 Makefile 中, 我们的依赖关系可能需要包含一系列的头文件。
比如 main.c 源文件内容如下:
#include "stdio.h"
#include "defs.h" int main(int argc, char *argv[])
{
printf("Hello, %s!\n", NAME);
return 0;
}
defs.h 头文件如下:
#ifndef _DEFS_H_
#define _DEFS_H_ #define NAME "makefile" #endif _DEFS_H_
那么依赖关系应该如下:
main.o : main.c stdio.h defs.h ...
但如果是一个比较大型的工程,你必需清楚每一个 C 源文件包含了哪些头文件,并且在加入或删除头文件时,也需要小心地修改 Makefile,这是一个很没有维护性的工作。为了避免这种繁重而又容易出错的事情,我们可以使用 C/C++ 编译的一个功能。大多数的 C/C++ 编译器都支持一个 “-M” 的选项,即自动找寻源文件中包含的头文件,并生成一个依赖关系。例如,执行下面的命令:
gcc -M main.c
其输出如下:
main.o : main.c defs.h
由编译器自动生成依赖关系,这样做的好处有以下几点:
- 不必手动书写若干文件的依赖关系,由编译器自动生成
- 不管是 .c 文件还是 .h 文件有更新,目标文件都会重新编译
2. 使用说明:
参数介绍:
-M
生成文件的依赖关系,同时也把一些标准库的头文件也包含了进来。本质是告诉预处理器输出一个适合 make 的规则,用于描述各目标文件的依赖关系。对于每个源文件,预处理器输出 一个 make 规则,该规则的目标项 (target) 是源文件对应的目标文件名,依赖项 (dependency) 是源文件中 ‘#include’ 引用的所有文件,生成的规则可以是单行,但如果太长,就用’\’换行符续成多行。规则 显示在标准输出,不产生预处理过的C程序。
注意:该选项默认打开了 -E 选项, -E 参数的用处是使得编译器在预处理结束时就停止编译
例如: gcc -M main.c
则在终端上输出如下:
main.o: main.c defs.h \
/usr/include/stdio.h \
/usr/include/features.h \
/usr/include/sys/cdefs.h /usr/include/gnu/stubs.h \
/usr/lib/gcc-lib/i486-suse-linux/2.95.3/include/stddef.h \
/usr/include/bits/types.h \
/usr/include/bits/pthreadtypes.h \
/usr/include/_G_config.h /usr/include/wchar.h \
/usr/include/bits/wchar.h /usr/include/gconv.h \
/usr/lib/gcc-lib/i486-suse-linux/2.95.3/include/stdarg.h \
/usr/include/bits/stdio_lim.h
-MM
生成文件的依赖关系,和 -M 类似,但不包含标准库的头文件
例如:gcc -MM main.c
则在终端上输出如下:
main.o: main.c defs.h
-MG
要求把缺失的头文件按存在对待,并且假定他们和源程序文件在同一目录下.必须和 ‘-M’ 选项一起用.
-MF File
当使用了 ‘-M’ 或者 ‘-MM’ 选项时,则把依赖关系写入名为 ‘File’ 的文件中。若同时也使用了 ‘-MD’ 或 ‘-MMD’,’-MF’ 将覆写输出的依赖文件的名称
例如:gcc -M -MF main.d main.c
则 '—M' 输出的内容就存在于 main.d 文件中了
-MD
等同于 ‘-M -MF File’,但是默认关闭了 ‘-E’ 选项. 其输出的文件名是基于 ‘-o’ 选项,若给定了 ‘-o’ 选项,则输出的文件名是 ‘-o’ 指定的文件名,并添加 .d 后缀,若没有给定,则输入的文件名作为输出的文件名,并添加.d后缀,同时继续指定的编译工作
注意:’-MD’ 不会像 ‘-M’ 那样阻止正常的编译任务. 因为它默认关闭了 ‘-E’ 选项, 比如命令中使用了 -c 选项,其结果要生成 .o 文件,若使用了 ‘-M’ 选项,则不会生成 .o 文件,若使用的是 ‘-MD’ 选项,则会生成 .o 文件
例如1:
例如2:gcc -E -o tmp.i -MD main.c
本目录下生成了以下文件:
tmp.d tmp.i
例如3:gcc -c -MD main.c
本目录下生成了以下文件:
main.d main.o
例如4:gcc -c -o tmp.o -MD main.c
本目录下生成了以下文件:
tmp.d tmp.o
例如5: gcc -MD main.c
本目录下生成了以下文件:
a.out main.d
例如6: gcc -M -MD main.c
本目录下生成了以下文件:
main.d //并不会生成a.out可执行文件,因为 '-M' 默认打开了 '-E' 选项,使得编译器在预处理结束后就停止编译
-MMD
类似于 ‘-MD’,但是输出的依赖文件中,不包含标准头文件
-MP
生成的依赖文件里面,依赖规则中的所有.h依赖项都会在该文件中生成一个伪目标,其不依赖任何其他依赖项。该伪规则将避免删除了对应的头文件而没有更新 “Makefile” 去匹配新的依赖关系而导致make出错的情况出现。
(英文描述:This option instructs CPP to add a phony target for each dependency
other than the main file, causing each to depend on nothing. These
dummy rules work around errors ‘make’ gives if you remove header
files without updating the ‘Makefile’ to match.)
例如1: gcc -c -MM -MD main.c
生成的 main.d 文件内容如下:
main.o: main.c defs.h
例如2: gcc -c -MM -MD main.c -MP
生成的 main.d 文件内容如下:
main.o: main.c defs.h
defs.h: //该选项会生成该伪目标,其没有任何依赖项,若不使用 '-MP' 选项,则不会生成该伪目标规则
-MT Target
在生成的依赖文件中,指定依赖规则中的目标
例如: gcc -MF main.d -MG -MM -MP -MT main.d -MT main.o main.c
$ cat main.d #查看生成的依赖文件的内容,输出以下内容
main.d main.o: main.c
注:依赖规则中main.d 和 main.o 目标都是通过'-MT'选项指定的
3. 使用参考:
以上简单介绍了 gcc -M 相关的选项,旨在让 make 自动推导并生成文件的依赖关系.
以下提供一个比较好的 gcc -M 选项的参考示例, 它将自动生成依赖文件,并保存在指定目录下的 ‘.d’ 文件中。
makefile如下所示:
SRCS=$(wildcard *.c)
OBJS=$(SRCS:.c=.o)
DEPS=$(SRCS:.c=.d) .PHONY: all clean all: main -include $(DEPS) #注释:'-'号的作用:加载错误时,会继续执行 make,主要是考虑到首次 make 时,目录中若不存在 '*.d' 文件时,加载便会产生错误而停止 make 的执行 %.o:%.c
gcc -c -g -Wall $< -o $@ -MD -MF $*.d -MP main: $(OBJS)
gcc $^ -o $@ #注释:$^:表示所有的依赖文件 $@:表示目标文件 clean:
rm -f *.d *.o main
仍旧以本篇文章开头的源文件进行 make,将生成如下文件:
main : 可执行文件
main.o : 编译的二进制目标文件
main.d:保存了 main.o 依赖关系的文件
注释: $* 表示目标模式中 '%' 及其之前的部分.如果目标是 'dir/a.foo.b',
并且目标的模式为 'a.%.b',那么 '$*' 的值就是 'dir/a.foo'.
如果目标中没有模式的定义,那么 '$*' 就不能被推导出.
但是,如果目标文件是 make 所识别的,那么 '$*' 就是除了后缀的那一部分, 例如:目标是 'foo.c',因为 '.c' 是 make 所能识别的后缀名,
所以 '$*' 的值就是 'foo'.这个特性是 GNU make 的.
一个通用的Makefile (转)的更多相关文章
- 编写一个通用的Makefile文件
1.1在这之前,我们需要了解程序的编译过程 a.预处理:检查语法错误,展开宏,包含头文件等 b.编译:*.c-->*.S c.汇编:*.S-->*.o d.链接:.o +库文件=*.exe ...
- Linux C编程学习之开发工具3---多文件项目管理、Makefile、一个通用的Makefile
GNU Make简介 大型项目的开发过程中,往往会划分出若干个功能模块,这样可以保证软件的易维护性. 作为项目的组成部分,各个模块不可避免的存在各种联系,如果其中某个模块发生改动,那么其他的模块需要相 ...
- 一步一步写一个简单通用的makefile(四)--写一个通用的makefile编译android可执行文件
通常要把我们自己的的代码编译成在android里面编译的可执行文件,我们通常是建一个文件夹 . ├── Android.mk ├── Application.mk ├── convolve.cl ├─ ...
- 一个通用的Makefile(二)
1.各级子目录的Makefile: obj-y += file.o obj-y += subdir/ “obj-y += file.o” 表示把当前目录下的file.c编进程序里. “obj-y += ...
- 一个通用的makefile(一)
最近在编写Android编译系统时,需要遍历每一个目录下每一个文件夹下的makefile,网上的方法有些繁琐 :就直接贴上自己遍历子目录深度为1:(for temporary)(之后会继续更新) 下 ...
- 一个通用的Makefile框架
先做一个简单的记录,后续有时间再慢慢完善补充细节. 先上一个整体图片: 其中,最重要的文件就是:program_template.mk. 下面是program_template.mk最重要的内容: $ ...
- 14、编写一个通用的Makefile
编译test_Makefile的方法:a. gcc -o test a.c b.c对于a.c: 预处理.编译(C文件转换成汇编).汇编(汇编转换成机器码)对于b.c:预处理.编译.汇编最后链接优点:命 ...
- 一个通用的makefile
# ESDK the makefile setting file - chenwg@20131014 # you can modify "PC = 1" such as " ...
- 我所使用的一个通用的Makefile模板
话不多说,请看: 我的项目有的目录结构有: dirls/ ├── include │ └── apue.h ├── lib │ ├── error.c │ ├── error.o │ ...
随机推荐
- Android LayoutInflater.from(context).inflate
在应用中自定义一个view,需要获取这个view的布局,需要用到 (LinearLayout) LayoutInflater.from(context).inflate(R.layout.conten ...
- PCL—低层次视觉—关键点检测(NARF)
关键点检测本质上来说,并不是一个独立的部分,它往往和特征描述联系在一起,再将特征描述和识别.寻物联系在一起.关键点检测可以说是通往高层次视觉的重要基础.但本章节仅在低层次视觉上讨论点云处理问题,故所有 ...
- openfire+spark+smack实现即时通讯
近公司项目需要用到即时通讯功能,经过调研发现openfire+spark+smack可以实现.在网上找了很久,资料都十分有限,即使有些朋友实现了也说的不清不楚.于是决定自己研究,耗时一周的时间实现了文 ...
- Eclipse - FindBugs Plugin 的安装和使用
Eclipse - FindBugs Plugin 的安装和使用 FindBugs is a static analysis tool that examines the classes in se ...
- awk当中使用外部变量
1.awk命令使用双引号的情况下 此时在awk命令里面使用\"$var\"就可以引用外部环境变量的var的值 $ var="BASH";echo "u ...
- Hibernate+JPA (EntityMange讲解)
近年来ORM(Object-Relational Mapping)对象关系映射,即实体对象和数据库表的映射)技术市场人声音鼎沸,异常热闹, Sun在充分吸收现有的优秀ORM框架设计思想的基础上,制定了 ...
- js风格技巧
1.一个页面的所有js都可以写成这样,比如: var index ={}; index.User = ****; index.Init = function(){ $("$tes ...
- 浏览器检测是否安装flash插件,若没有安装,则弹出安装提示
说白了其实就是在html中前途flash的使用代码 <!-- html嵌入flash,检测浏览器是否安装flash插件,并提示安装.--> <object type=&q ...
- $http POST 转字符串
- 无线端不响应键盘事件(keydown,keypress,keyup)
今天在项目时,在android手机上使用输入法的智能推荐的词的话,不会触发keyup事件,一开始想到在focus时使用一个定时器,每隔100ms检测输入框的值是否发生了改变,如果改变了就作对应的处理, ...