Chinaunix首页 | 论坛 | 博客
  • 博客访问: 273991
  • 博文数量: 111
  • 博客积分: 1591
  • 博客等级: 上尉
  • 技术积分: 877
  • 用 户 组: 普通用户
  • 注册时间: 2011-10-20 15:14
文章分类

全部博文(111)

文章存档

2014年(1)

2012年(41)

2011年(69)

分类: LINUX

2011-11-23 18:43:24

本文提供一个在GNU/Linux环境下通用的智能Makefile模板,用户可以将其引入到自己的工程当中。即使用户对此
Makefile的工作原理不甚了解,但是只需要修改少数几个关键变量,就可以满足不同的应用需求。


1. 使用环境
GNU make 3.8.1或者以上版本
sed
gcc/binutils
cygwin或者Linux

2. 关键变量
CROSS_COMPILE:交叉编译器前缀
OPTIMIZE:关于优化的编译参数
WARNINGS:关于warning的编译参数
DEFS:    关于宏定义的编译参数
EXTRA_CFLAGS:其它的编译参数
$(OPTIMIZE) $(WARNINGS) $(DEFS) $(EXTRA_CFLAGS)共同构成了传给gcc的编译参数。

INC_DIR:头文件目录
SRC_DIR:源文件目录(当前目录.是被默认包含的)
OBJ_DIR:object文件的输出目录
EXTRA_SRC:源文件列表
EXCLUDE_FILES:exclude文件列表

SUFFIX:源文件的后缀名
TARGET:最终的目标程序名
TARGET_TYPE:目标程序的类型。
可以定义的类型为:ar(static library),so(shared library)和app(application)。

最终Makefile的行为是:
在$(SRC_DIR)定义的每个目录中查找后缀为$(SUFFIX)的文件,并加上$(EXTRA_SRC)中的文件,然后排除掉
$(EXCLUDE_FILES)中的文件,获得本工程定义的源文件列表。对于每一个源文件,编译生成的一个.o文件和一个.d文件
(依赖文件),放在$(OBJ_DIR)目录下。最终生成的目标文件为$(TARGET)。
此Makefile已经充分考虑到文件之间的依赖关系,即,如果某个头文件发生改变,当运行make的时候,所有依赖于它的
源文件都将被重新编译。

附件是一个例子,演示了如何处理不同目录,如何处理不同目标类型,以及如何exclude文件。
  (3.39 KB)
  1. ###############################################################################
  2. #
  3. # A smart Makefile template for GNU/LINUX programming
  4. #
  5. # Author: PRC (ijkxyz AT msn DOT com)
  6. # Date:   2011/06/17
  7. #
  8. # Usage:
  9. #   $ make           Compile and link (or archive)
  10. #   $ make clean     Clean the objectives and target.
  11. ###############################################################################

  12. CROSS_COMPILE =
  13. OPTIMIZE := -O2
  14. WARNINGS := -Wall -Wno-unused -Wno-format
  15. DEFS     :=
  16. EXTRA_CFLAGS := 

  17. INC_DIR   =
  18. SRC_DIR   =
  19. OBJ_DIR   =
  20. EXTRA_SRC =
  21. EXCLUDE_FILES =

  22. SUFFIX       = c cpp cc cxx
  23. TARGET       :=
  24. #TARGET_TYPE  :=
  25. TARGET_TYPE  :=
  26. #TARGET_TYPE  :=


  27. #####################################################################################
  28. #  Do not change any part of them unless you have understood this script very well  #
  29. #  This is a kind remind.                                                           #
  30. #####################################################################################

  31. #FUNC#  Add a new line to the input stream.
  32. define add_newline
  33. $1

  34. endef

  35. #FUNC# set the variable `src-x' according to the input $1
  36. define set_src_x
  37. src-$1 = $(filter-out $4,$(foreach d,$2,$(wildcard $d/*.$1)) $(filter %.$1,$3))

  38. endef

  39. #FUNC# set the variable `obj-x' according to the input $1
  40. define set_obj_x
  41. obj-$1 = $(patsubst %.$1,$3%.o,$(notdir $2))

  42. endef

  43. #VAR# Get the uniform representation of the object directory path name
  44. ifneq ($(OBJ_DIR),)
  45. prefix_objdir  = $(shell echo $(OBJ_DIR)|sed 's:\(\./*\)*::')
  46. prefix_objdir := $(filter-out /,$(prefix_objdir)/)
  47. endif

  48. GCC      := $(CROSS_COMPILE)gcc
  49. G++      := $(CROSS_COMPILE)g++
  50. SRC_DIR := $(sort . $(SRC_DIR))
  51. inc_dir = $(foreach d,$(sort $(INC_DIR) $(SRC_DIR)),-I$d)

  52. #--# Do smart deduction automatically
  53. $(eval $(foreach i,$(SUFFIX),$(call set_src_x,$i,$(SRC_DIR),$(EXTRA_SRC),$(EXCLUDE_FILES))))
  54. $(eval $(foreach i,$(SUFFIX),$(call set_obj_x,$i,$(src-$i),$(prefix_objdir))))
  55. $(eval $(foreach f,$(EXTRA_SRC),$(call add_newline,vpath $(notdir $f) $(dir $f))))
  56. $(eval $(foreach d,$(SRC_DIR),$(foreach i,$(SUFFIX),$(call add_newline,vpath %.$i $d))))

  57. all_objs = $(foreach i,$(SUFFIX),$(obj-$i))
  58. all_srcs = $(foreach i,$(SUFFIX),$(src-$i))

  59. CFLAGS       = $(EXTRA_CFLAGS) $(WARNINGS) $(OPTIMIZE) $(DEFS)
  60. TARGET_TYPE := $(strip $(TARGET_TYPE))

  61. ifeq ($(filter $(TARGET_TYPE),so ar app),)
  62. $(error Unexpected TARGET_TYPE `$(TARGET_TYPE)')
  63. endif

  64. ifeq ($(TARGET_TYPE),so)
  65. CFLAGS  += -fpic -shared
  66. LDFLAGS += -shared
  67. endif

  68. PHONY = all .mkdir clean

  69. all: .mkdir $(TARGET)

  70. define cmd_o
  71. $(obj-$1): $2%.o: %.$1  $(MAKEFILE_LIST)
  72.         $(GCC) $(inc_dir) -Wp,-MT,$@ -Wp,-MMD,$@.d $(CFLAGS) -c -o $@ $<

  73. endef
  74. $(eval $(foreach i,$(SUFFIX),$(call cmd_o,$i,$(prefix_objdir))))

  75. ifeq ($(TARGET_TYPE),ar)
  76. $(TARGET): AR := $(CROSS_COMPILE)ar
  77. $(TARGET): $(all_objs)
  78.         rm -f $@
  79.         $(AR) rcvs $@ $(all_objs)
  80. else
  81. $(TARGET): LD = $(if $(strip $(src-cpp) $(src-cc) $(src-cxx)),$(G++),$(GCC))
  82. $(TARGET): $(all_objs)
  83.         $(LD) $(LDFLAGS) $(all_objs) -o $@
  84. endif

  85. .mkdir:
  86.         @if [ ! -d $(OBJ_DIR) ]; then mkdir -p $(OBJ_DIR); fi

  87. clean:
  88.         rm -f $(prefix_objdir)*.o $(TARGET)

  89. -include $(patsubst %.o,%.o.d,$(all_objs))

  90. .PHONY: $(PHONY)
阅读(2318) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~