编译test_Makefile的方法:
a. gcc -o test a.c b.c
对于a.c: 预处理、编译(C文件转换成汇编)、汇编(汇编转换成机器码)
对于b.c:预处理、编译、汇编
最后链接
优点:命令简单
缺点:如果文件很多,即使你只修改了一个文件,但是所有的文件文件都要重新"预处理、编译、汇编"
效率低

b. 写Makefile
核心:规则

目标:依赖1 依赖2(目标指根据依赖生成文件名字)
  命令

a.o : a.c

  gcc -c -o a.o a.c

%.o:%.c    //这里的%是通配符

  gcc -c -o $@  $<     //$@表示目标  $<表示第一个依赖 %^表示所有的依赖

(gcc还要加上-Wp,-MD,名字.d  用来生成依赖文件,否则.h文件改变后,执行make的时候gcc不会发现文件变化了,这样就不执行gcc指令)

命令执行的条件:
i. "依赖"文件 比 "目标"文件 新
ii.没有"目标"这个文件

一、各级子目录的Makefile:
它最简单,形式如下:
obj-y += file.o
obj-y += subdir/

"obj-y += file.o"表示把当前目录下的file.c编进程序里,
"obj-y += subdir/"表示要进入subdir这个子目录下去寻找文件来编进程序里,是哪些文件由subdir目录下的Makefile决定。

注意: "subdir/"中的斜杠"/"不可省略

二、顶层目录的Makefile:
它除了定义obj-y来指定根目录下要编进程序去的文件、子目录外,主要是定义工具链、编译参数、链接参数──就是文件中用export导出的各变量。导出的目的是给个子目录下makefile使用

export AS LD CC CPP AR NM
export STRIP OBJCOPY OBJDUMP

//链接的时候指定优化选项和目录(目录是告诉编译器去哪里找那些头文件,$(shell pwd)表示执行pwd指令获得当前目录),使用arm-linux-gcc来编译的时候,编译器是去/usr/local/arm/4.3.2/arm-none-linux-guneabi/libc/usr/include目录下找头文件,可以通过-I(i的大小)头文件目录来指定头文件或者通过-L 指定库文件目录

CFLAGS := -Wall -O2 -g  //-Wall表示列出所有警告   -O2表示优化选项  -g表示,等号的前面“:”表示立即变量,没有这个:,赋值过程会被延时,使用的时候财货确定下来,有无冒号的原因具体见gcc文档
CFLAGS += -I $(shell pwd)/include

//LDFLAGS 表示链接参数

LDFLAGS := -lm -lfreetype

三、顶层目录的Makefile.build:(在数码相框的makefile那节的doc目录中有详细介绍,(子目录下的makefile是最先执行))
这是最复杂的部分,它的功能就是把某个目录及它的所有子目录中、需要编进程序去的文件都编译出来,打包为built-in.o
详细的讲解请看视频。

举例说明Makefile.build

PHONY := __build   //PHONY 假象目标,肯定会执行
__build:  //声明最开始的目标

obj-y :=  //先赋空值
subdir-y :=

include Makefile  //因为Makefile中含有obj-y等变量值

# obj-y := a.o b.o c/ d/    //下面用#开始的四条语句是举例怎么从obj-y中取得所有子目录
# $(filter %/, $(obj-y)) : c/ d/  //filter表示把$(obj-y)中不符合%/的移除,保留复合%/的
# __subdir-y : c d
# subdir-y : c d
__subdir-y := $(patsubst %/,%,$(filter %/, $(obj-y)))  //$(patsubst pattern,replacement,text) 在text中用replacement代替pattern
subdir-y += $(__subdir-y)

# c/built-in.o   d/built-in.o   //subdir_objs := c/built-in.o   d/built-in.o
subdir_objs := $(foreach f,$(subdir-y),$(f)/built-in.o)

# a.o b.o//下面六行语句的目前是为了生成依赖文件,这样.h文件改变后在执行make才会让编译器知道.h文件也改变了
cur_objs := $(filter-out %/, $(obj-y))
dep_files := $(foreach f,$(cur_objs),.$(f).d)
dep_files := $(wildcard $(dep_files))

ifneq ($(dep_files),)
  include $(dep_files)
endif

PHONY += $(subdir-y)

__build : $(subdir-y) built-in.o //顶层目录的makefile执行到这里的时候会先去获得subdir-y,其获得规则在下面,同理built-in.o,这里的$(subdir-y) 仅为了生成所有的built-in.o,其后的built-in.o才是最后的依赖文件

$(subdir-y):
  make -C $@ -f $(TOPDIR)/Makefile.build    //$@表示进入各级子目录,使用顶层目录下的Makefile.build来生成各级子目录的built-in.o

built-in.o : $(cur_objs) $(subdir_objs)//依赖的是当前目录下.o文件和各级子目录下built-in.o
  $(LD) -r -o $@ $^

dep_file = .$@.d   //没有冒号,表示延时赋值,在下面使用的时候才赋值,这样$@才有值

%.o : %.c
  $(CC) $(CFLAGS) -Wp,-MD,$(dep_file) -c -o $@ $<

.PHONY : $(PHONY)

14、编写一个通用的Makefile的更多相关文章

  1. 编写一个通用的Makefile文件

    1.1在这之前,我们需要了解程序的编译过程 a.预处理:检查语法错误,展开宏,包含头文件等 b.编译:*.c-->*.S c.汇编:*.S-->*.o d.链接:.o +库文件=*.exe ...

  2. Linux C编程学习之开发工具3---多文件项目管理、Makefile、一个通用的Makefile

    GNU Make简介 大型项目的开发过程中,往往会划分出若干个功能模块,这样可以保证软件的易维护性. 作为项目的组成部分,各个模块不可避免的存在各种联系,如果其中某个模块发生改动,那么其他的模块需要相 ...

  3. 一个通用的Makefile (转)

    据http://bbs.chinaunix.net/thread-2300778-1-1.html的讨论,发现还是有很多人在问通用Makefile的问题,这里做一个总结.也作为以后的参考.       ...

  4. 一个通用的makefile(一)

    最近在编写Android编译系统时,需要遍历每一个目录下每一个文件夹下的makefile,网上的方法有些繁琐 :就直接贴上自己遍历子目录深度为1:(for  temporary)(之后会继续更新) 下 ...

  5. 一步一步写一个简单通用的makefile(四)--写一个通用的makefile编译android可执行文件

    通常要把我们自己的的代码编译成在android里面编译的可执行文件,我们通常是建一个文件夹 . ├── Android.mk ├── Application.mk ├── convolve.cl ├─ ...

  6. 一个通用的Makefile(二)

    1.各级子目录的Makefile: obj-y += file.o obj-y += subdir/ “obj-y += file.o” 表示把当前目录下的file.c编进程序里. “obj-y += ...

  7. 一个通用的Makefile框架

    先做一个简单的记录,后续有时间再慢慢完善补充细节. 先上一个整体图片: 其中,最重要的文件就是:program_template.mk. 下面是program_template.mk最重要的内容: $ ...

  8. 一个通用的makefile

    # ESDK the makefile setting file - chenwg@20131014 # you can modify "PC = 1" such as " ...

  9. 我所使用的一个通用的Makefile模板

    话不多说,请看: 我的项目有的目录结构有: dirls/ ├── include │   └── apue.h ├── lib │   ├── error.c │   ├── error.o │   ...

随机推荐

  1. iOS8 对开发人员来说意味着什么?

    今天凌晨.Apple WWDC2014 iOS8 正式推出. 或许,对于广大iOS用户来说,iOS8的创新并非特别多. 但对于开发人员来说,影响却将会是无比巨大的! 正如Apple官网上的广告:Hug ...

  2. C/C++(基础-运算符详解)

    运算符 任何表达式是有值的 int a = 2; int b = 3; a*=b+4;//a=a*(b+4);"*"*=的优先级层次和=号的层次一样. printf("% ...

  3. vue 使用同一组件,切换时不触发created、mounted钩子

    两个页面参数不同使用同一组件,默认情况下当这两个页面切换时并不会触发created或者mounted钩子. 方法一:通过watch $route的变化来做处理 watch: { $route() { ...

  4. halt---关闭正在运行的Linux操作系统。

    halt命令用来关闭正在运行的Linux操作系统.halt命令会先检测系统的runlevel,若runlevel为0或6,则关闭系统,否则即调用shutdown来关闭系统. 语法 halt(选项) 选 ...

  5. Python-Flask项目开发--为什么需要搭建虚拟环境?

    在使用python开发过程中,需要使用到某些工具包/框架等,需要联网下载.   例如,联网安装Flask框架flask-0.10.1版本:pip install flask==0.10.1   此时, ...

  6. 【2017 Multi-University Training Contest - Team 5】Rikka with Graph

    [Link]:http://acm.hdu.edu.cn/showproblem.php?pid=6090 [Description] 给你n个点; 让你在这n个点上最多连m条无向边; 使得 ∑ni= ...

  7. Android-Volley网络通信框架(二次封装数据请求和图片请求(包含处理请求队列和图片缓存))

    1.回想 上篇 使用 Volley 的 JsonObjectRequest 和 ImageLoader 写了 电影列表的样例 2.重点 (1)封装Volley 内部 请求 类(请求队列,数据请求,图片 ...

  8. Word Ladder II [leetcode]

    本题有几个注意点: 1. 回溯找路径时.依据路径的最大长度控制回溯深度 2. BFS时,在找到end单词后,给当前层做标记find=true,遍历完当前层后结束.不须要遍历下一层了. 3. 能够将字典 ...

  9. 关于编译com工程的一些体会

    作者:朱金灿 来源:http://blog.csdn.net/clever101 今天发现com的编译的一个重要一步是微软提供的midl工具将其中的idl文件生成一个头文件.c文件(即IID文件)和代 ...

  10. 企业网管软件之SOLARWINDS实战-基于浏览器的网络流量监控

    本文出自 "李晨光原创技术博客" 博客,谢绝转载!