之前一直没有整理过makefile的使用,现在整理一下便于以后查找。
0、makefile中的常用符号
$@ 表示目标
$^ 表示所有的依赖
$< 表示第一个依赖
$? --代表当前目标所依赖的文件列表中比当前目标文件还要新的文件。
$* --不包括后缀名的当前依赖文件的名字
:= 即时赋值
?= 延时赋值
+= 附加
AR 函数库打包程序,可创建静态库.a文档。默认是"ar"。
AS 汇编程序。默认是"as"。
CC C编译程序。默认是"cc"。
CXX C++编译程序。默认是"g++"。
CPP C/C++预处理器。默认是"$(CC) -E"。
FC Fortran编译器。默认是"f77"。
PC Pascal语言编译器。默认是"pc"。
YACC Yacc文法分析器。默认是"yacc"。
ARFLAGS 函数库打包程序的命令行参数。默认值是"rv"。
ASFLAGS 汇编程序的命令行参数。
CFLAGS C编译程序的命令行参数。
CXXFLAGS C++编译程序的命令行参数。
CPPFLAGS C/C++预处理器的命令行参数。
FFLAGS Fortran编译器的命令行参数。
PFLAGS Pascal编译器的命令行参数。
YFLAGS Yacc文法分析器的命令行参数。
LDFLAGS 链接器的命令行参数。
LIBS:告诉链接器要链接哪些库文件,如LIBS = -lpthread -liconv
1、模板1
ROOTDIR = .
OBJS1 = $(patsubst %.c,%.o,$(wildcard *.c))
OBJS2 = $(patsubst %.cpp,%.o,$(wildcard *.cpp))
DEPE1 = ${OBJS1:%.o=%.d}
DEPE2 = ${OBJS2:%.o=%.d}
CC = arm-linux-gnueabihf-gcc
CXX = arm-linux-gnueabihf-g++
INC =
LIBS =
TARGET = test
#INC、LIBS、TARGET根据实际情况填写
#INC为工程头文件目录,如不需额外指定,保持空值即可
#LIBS为库文件目录,如不需额外指定,保持空值即可
#TARGET为生成的目标文件名,必须指定
$(TARGET) : $(OBJS1) $(OBJS2)
$(CXX) $^ -o $@ $(LIBS) -pthread
$(OBJS1):%.o:%.c
$(CXX) -c $< -o $@ $(INC)
$(OBJS2):%.o:%.cpp
$(CXX) -c $< -o $@ $(INC)
$(DEPE1):%.d:%.c
@set -e;rm -f $@;$(CXX) -MM $< $(INC) > $@.$$$$; \
sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \
rm -f $@.$$$$
$(DEPE2):%.d:%.cpp
@set -e;rm -f $@;$(CXX) -MM $< $(INC) > $@.$$$$; \
sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \
rm -f $@.$$$$
-include $(DEPE1) $(DEPE2)
.PHONY: clean
clean :
rm -f $(OBJS1) $(OBJS2) $(DEPE1) $(DEPE2) $(TARGET)
2、模板2
CC=gcc
SRCS=$(wildcard *.c */*.c)
OBJS=$(patsubst %.c, %.o, $(SRCS))
FLAG=-g
NAME=$(wildcard *.c)
TARGET=$(patsubst %.c, %, $(NAME))
$(TARGET):$(OBJS)
$(CC) -o $@ $^ $(FLAG)
%.o:%.c
$(CC) -o $@ -c $< -g
.PHONY: clean
clean:
rm -rf $(TARGET) $(OBJS)
解释一下:
(1)第1行指定编译器为 gcc,可以根据需要修改为 g++ 或者 arm-linux-gcc 等交叉编译工具链。
(2)第2行是为了获取匹配模式的文件名,*.c
表示当前工程目录的 c 文件,*/*.c
表示所有下一级目录的 .c 文件。
(3)第3行是将 $(SRCS)
中的 .c 文件替换成 .o 文件。
(4)第4行用于指定编译选项,根据需要添加,比如 -g、-ml、-Wall、-O2 等等。
(5)第5、6行,同样使用了 wildcard 和 patsubst 函数来处理,目的是为了指定最终生成的可执行文件名。
(6)第8行 $(TARGET):$(OBJS)
表示由 .o 文件链接成可执行文件。
(7)注意第9行前面是一个 <tab>
键,而 $@
表示目标,也就是 $(TARGET)
,$^
表示依赖列表,也就是 $(OBJS)
。
(8)第11行 %.o:%.c
表示由 .c 文件生成 .o 文件。
(9)第12行中的 $<
表示搜索到的第一个匹配的文件。
(10)第14行的 clean 是伪目标(.PHONY),它不需要依赖,执行 make
命令时是不会执行的,当执行 make clean
时才会执行。
另外,Makefile 的赋值除了使用等号(=),还可以使用 :=
、+=
、?=
符号。具体含义如下:
:=
表示简单赋值(注意右边的 $ 值只会向上寻找)+=
表示追加赋值?=
表示判断赋值(判断左边是否已经定义过,如果是则忽略本次赋值)
3、模板3
CC = g++
SRCS = $(wildcard *.cpp)
OBJS = $(patsubst %cpp, %o, $(SRCS))
LDFLAGS :=-lsqlite3 -lTR3A
CFLAGS := -Wall -O2 -g -L/home/usr
INCLUDE = -I ./com/inc \
-I ./com/lib/inc
CFLAGS += $(INCLUDE)
TARGET = TEST
all : $(TARGET)
$(TARGET):$(OBJS)
$(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS)
%.o:%.c
$(CC) $(CFLAGS) -o $@ -c $< $(LDFLAGS)
解释(更新中):
$(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS)
(1)-o $@ 与 $^各为一个整体,语句不存在-c,就决定了这行命令是编译,链接过程,
-o后面的$@代表链接后生成的可执行文件的取名,你可以TARGET = main.o,但是main.o代
表的是可执行文件,只是后缀名为.o的可执行文件,而不是编译生成的.o文件,你也可以去掉
-o $@,则链接器自动生成名为a.out的可执行文件。
$(CC) $(CFLAGS) -o $@ -c $< $(LDFLAGS)
(2)-o $@ 与 -c $<各为一个整体,语句中有-c存在,就决定了这行命令只编译不链接,
-o $@代表指定编译生成的.o按照$@取名,而$@本身代表编译后生成的.o,所以可以省略-o $@。
疑问:关于-o -c $(CFLAGS)在命令中的位置关系
如下三种情况等价
(1)$(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS)
(2)$(CC) $(CFLAGS) $^ -o $@ $(LDFLAGS)
(3)$(CC) $^ -o $@ $(LDFLAGS) $(CFLAGS)
$(CC) $(CFLAGS) -o $@ -c $< $(LDFLAGS) 等价于 $(CC) $(CFLAGS) -o $@ -c $<
4、生成可执行文件的makefile
######################################
#
######################################
#source file
#源文件,自动找所有.c和.cpp文件,并将目标定义为同名.o文件
SOURCE := $(wildcard *.c) $(wildcard *.cpp)
OBJS := $(patsubst %.c,%.o,$(patsubst %.cpp,%.o,$(SOURCE)))
#target you can change test to what you want
#目标文件名,输入任意你想要的执行文件名
TARGET := test
#compile and lib parameter
#编译参数
CC := gcc
LIBS :=
LDFLAGS :=
DEFINES :=
INCLUDE := -I.
CFLAGS := -g -Wall -O3 $(DEFINES) $(INCLUDE)
CXXFLAGS:= $(CFLAGS) -DHAVE_CONFIG_H
#i think you should do anything here
#下面的基本上不需要做任何改动了
.PHONY : everything objs clean veryclean rebuild
everything : $(TARGET)
all : $(TARGET)
objs : $(OBJS)
rebuild: veryclean everything
clean :
rm -fr *.so
rm -fr *.o
veryclean : clean
rm -fr $(TARGET)
$(TARGET) : $(OBJS)
$(CC) $(CXXFLAGS) -o $@ $(OBJS) $(LDFLAGS) $(LIBS)
5、生成静态链接库的makefile
######################################
#
#
######################################
#target you can change test to what you want
#共享库文件名,lib*.a
TARGET := libtest.a
#compile and lib parameter
#编译参数
CC := gcc
AR = ar
RANLIB = ranlib
LIBS :=
LDFLAGS :=
DEFINES :=
INCLUDE := -I.
CFLAGS := -g -Wall -O3 $(DEFINES) $(INCLUDE)
CXXFLAGS:= $(CFLAGS) -DHAVE_CONFIG_H
#i think you should do anything here
#下面的基本上不需要做任何改动了
#source file
#源文件,自动找所有.c和.cpp文件,并将目标定义为同名.o文件
SOURCE := $(wildcard *.c) $(wildcard *.cpp)
OBJS := $(patsubst %.c,%.o,$(patsubst %.cpp,%.o,$(SOURCE)))
.PHONY : everything objs clean veryclean rebuild
everything : $(TARGET)
all : $(TARGET)
objs : $(OBJS)
rebuild: veryclean everything
clean :
rm -fr *.o
veryclean : clean
rm -fr $(TARGET)
$(TARGET) : $(OBJS)
$(AR) cru $(TARGET) $(OBJS)
$(RANLIB) $(TARGET)
6、生成动态链接库的makefile
######################################
#
#
######################################
#target you can change test to what you want
#共享库文件名,lib*.so
TARGET := libtest.so
#compile and lib parameter
#编译参数
CC := gcc
LIBS :=
LDFLAGS :=
DEFINES :=
INCLUDE := -I.
CFLAGS := -g -Wall -O3 $(DEFINES) $(INCLUDE)
CXXFLAGS:= $(CFLAGS) -DHAVE_CONFIG_H
SHARE := -fPIC -shared -o
#i think you should do anything here
#下面的基本上不需要做任何改动了
#source file
#源文件,自动找所有.c和.cpp文件,并将目标定义为同名.o文件
SOURCE := $(wildcard *.c) $(wildcard *.cpp)
OBJS := $(patsubst %.c,%.o,$(patsubst %.cpp,%.o,$(SOURCE)))
.PHONY : everything objs clean veryclean rebuild
everything : $(TARGET)
all : $(TARGET)
objs : $(OBJS)
rebuild: veryclean everything
clean :
rm -fr *.o
veryclean : clean
rm -fr $(TARGET)
$(TARGET) : $(OBJS)
$(CC) $(CXXFLAGS) $(SHARE) $@ $(OBJS) $(LDFLAGS) $(LIBS)
7、makefile中添加安装与卸载
install:
mkdir -p $(OUTPUT)/include/algorithm
cp $(TGT_LIB_H) $(OUTPUT)/include/algorithm/
cp $(TGT_LIB_A) $(OUTPUT)/lib/
cp $(TGT_LIB_SO) $(OUTPUT)/lib/
uninstall:
rm -rf $(OUTPUT)/include/algorithm
rm -f $(OUTPUT)/lib/$(TGT_LIB_A)
rm -f $(OUTPUT)/lib/$(TGT_LIB_SO)
8、makefile万能模板
######################################
# Generic makefile
######################################
# 用户设定
# 如果需要,调整下面的东西。 EXECUTABLE 是目标的可执行文件名, LIBS
# 是一个需要连接的程序包列表(例如 alleg, stdcx, iostr 等等)。当然你
# 可以在 make 的命令行覆盖它们,你愿意就没问题。
EXECUTABLE :=../../obj/xxxxx.so
LIBS=../lib/centos-5.4-64/libx1.a ../lib/libx2.a ../lib/libx3.a ../lib/libx4.a ../lib/libx5.a ../lib/libx6.a ../lib/libx7.a ../lib/libiconv.a ../lib/libjasper.a
# 现在来改变任何你想改动的隐含规则中的变量,例如
#CFLAGS := -g -Wall -O -I /usr/lib/jvm/jdk1.8.0_05/include/linux/ -I /usr/lib/jvm/jdk1.8.0_05/include/
CFLAGS := -O2 -I /usr/lib/jdk1.7.0_79/include/linux/ -I /usr/lib/jdk1.7.0_79/include/ -fPIC -m64
CXXFLAGS := $(CFLAGS)
# 下面先检查你的 djgpp 命令目录下有没有 rm 命令,如果没有,我们使用
# del 命令来代替,但有可能给我们 \'File not found\' 这个错误信息,这没
# 什么大碍。如果你不是用 DOS ,把它设定成一个删文件而不废话的命令。
# (其实这一步在 UNIX 类的系统上是多余的,只是方便 DOS 用户。 UNIX
# 用户可以删除这5行命令。)
#ifneq ($(wildcard $(DJDIR)/bin/rm.exe),)
#RM-F := rm -f
#else
#RM-F := del
#endif
# 从这里开始,你应该不需要改动任何东西。
SOURCE := $(wildcard *.c) $(wildcard *.cpp)
OBJS := $(patsubst %.c,%.o,$(patsubst %.cpp,%.o,$(SOURCE)))
DEPS := $(patsubst %.o,%.d,$(OBJS))
MISSING_DEPS := $(filter-out $(wildcard $(DEPS)),$(DEPS))
MISSING_DEPS_SOURCES := $(wildcard $(patsubst %.d,%.c,$(MISSING_DEPS)) \
$(patsubst %.d,%.cpp,$(MISSING_DEPS)))
CPPFLAGS +=-MD
.PHONY : everything deps objs clean veryclean rebuild dllLib
everything : $(EXECUTABLE)
deps : $(DEPS)
objs : $(OBJS)
clean :
@$(RM) *.o
@$(RM) *.d
veryclean: clean
@$(RM) $(EXECUTABLE)
cd ../libjasper/libjasper && $(MAKE) -f libjasperMakefile64 veryclean
cd ../libiconv-1.14/libiconv && $(MAKE) -f libiconvMakefile64 veryclean
cd ../x2 && $(MAKE) -f x2Makefile64 veryclean
cd ../x3 && $(MAKE) -f x3Makefile64 veryclean
cd ../x4 && $(MAKE) -f x4 veryclean
cd ../x1 && $(MAKE) -f x1MakefileASmart64 veryclean
rebuild: veryclean everything
ifneq ($(MISSING_DEPS),)
$(MISSING_DEPS) :
@$(RM) $(patsubst %.d,%.o,$@)
endif
-include $(DEPS)
$(EXECUTABLE) : $(OBJS) $(LIBS)
g++ -shared -o $(EXECUTABLE) $(OBJS) $(LIBS) -l pthread
$(LIBS):dllLib
cd ../libjasper/libjasper && $(MAKE) -f libjasperMakefile64
cd ../libiconv-1.14/libiconv && $(MAKE) -f libiconvMakefile64
cd ../x2 && $(MAKE) -f x2Makefile64
cd ../x3 && $(MAKE) -f x3Makefile64
cd ../x4 && $(MAKE) -f x4ImageMakefile64
cd ../x1 && $(MAKE) -f x1MakefileASmart64
dllLib:;