54.makefile 实例

15.makefile_LingLinTianShang的博客-CSDN博客

如何优雅地使用 makefile - 知乎

Linux内核Makefile.txt文件翻译_漫不经心-CSDN博客

六 makefile.build的分析_lgjjeff的博客-CSDN博客_makefile.build

一个例子

Makefile


CROSS_COMPILE = arm-linux-
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
OBJCOPY		= $(CROSS_COMPILE)objcopy
OBJDUMP		= $(CROSS_COMPILE)objdump

export AS LD CC CPP AR NM
export STRIP OBJCOPY OBJDUMP

CFLAGS := -Wall -Werror -O2 -g
CFLAGS += -I $(shell pwd)/include

LDFLAGS := -lm -lfreetype -lts -lpthread -ljpeg

export CFLAGS LDFLAGS

TOPDIR := $(shell pwd)
export TOPDIR

TARGET := digitpic


obj-y += main.o
obj-y += display/
obj-y += encoding/
obj-y += fonts/
obj-y += input/
obj-y += debug/
obj-y += render/
obj-y += page/
obj-y += file/

all : 
	make -C ./ -f $(TOPDIR)/Makefile.build
	$(CC) $(LDFLAGS) -o $(TARGET) built-in.o


clean:
	rm -f $(shell find -name "*.o")
	rm -f $(TARGET)

distclean:
	rm -f $(shell find -name "*.o")
	rm -f $(shell find -name "*.d")
	rm -f $(TARGET)
	

Makefile.build

PHONY := __build
__build:


obj-y :=
subdir-y :=

include Makefile

# obj-y := a.o b.o c/ d/
# $(filter %/, $(obj-y))   : c/ d/
# __subdir-y  : c d
# subdir-y    : c d
__subdir-y	:= $(patsubst %/,%,$(filter %/, $(obj-y)))
subdir-y	+= $(__subdir-y)

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

# a.o b.o
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

$(subdir-y):
	make -C $@ -f $(TOPDIR)/Makefile.build

built-in.o : $(cur_objs) $(subdir_objs)
	$(LD) -r -o $@ $^

dep_file = .$@.d

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

例子解析:

1.

Makefile.build相当于把部分重复操作单独提出来,然后取名为Makefile.build,换成其它名字,比如test.build,对应修改Makefile里Makefile.build的名字,也可以。

所以,仅仅因为它们名字上有类似,就以为有关系、区别,是没意义的,整个工程必须有Makefile,但不一定有Makefile.build。

在多层级目录的项目中,可以通过创建一个Makefile.build来方便对每个子目录进行操作。

执行make,调用顶层Makefile,进而调用执行Makefile.build,Makefile.build里对每个目录都执行Makefile.build,从而得到各个.o文件,最后将所有.o合并成最终二进制文件。

2.

 $表示依赖的关系,=是延时变量,使用时才会赋值,把CROSS_COMPILE带进去,就等价于,构造出编译的环境

CROSS_COMPILE = arm-linux-
AS		= arm-linux-as
LD		= arm-linux-ld
CC		= arm-linux-gcc
CPP		= arm-linux-gcc -E
AR		= arm-linux-ar
NM		= arm-linux-nm

STRIP		= arm-linux-strip
OBJCOPY		= arm-linux-objcopy
OBJDUMP		= arm-linux-objdump

3.

  

MakeFile中export的使用 - Love流浪的猪 - 博客园

export出来的变量arch将被子make进程继承

4.

gcc -I -i -L -l 参数区别 / -l(静态库/动态库)_Do-CSDN博客_-i gcc

-I是指定头文件路径,-L指定库文件路径

 :=是即时变量,CFLAGS在赋值的时候就获得值,+=表示追加,shell pwd是脚本命令,pwd代表makefile的当前路径,比如在/home/book/example文件夹下面

综合起来,等价于 

CFLAGS += -Wall -Werror -O2 -g -I /home/book/example/include

 5.

all : 
	make -C ./ -f $(TOPDIR)/Makefile.build
	$(CC) $(LDFLAGS) -o $(TARGET) built-in.o
这段话的意思是对子目录递归执行Makefile.build,把built-in.o生成$(TARGET)的可执行文件。
built-in.o从Makefile.build里面得到

 6.在makefile.build里面

makefile中.PHONY的作用是什么?_Sunrise的博客-CSDN博客_makefile phony

Linux之Makefile(filter)_嵌入式后台开发-CSDN博客_filter makefile

Makefile之patsubst - Geeking - 博客园

$(filter %/, $(obj-y))表示把末尾不是/的删掉,有/的就是子目录,留下子目录 

 $(patsubst %/,%,子目录),表示把子目录的/替换成空,就是把/删掉,从而获得子目录的路径

 

 Makefile中foreach使用 - 行走的思想 - 博客园

这段话的意思是通过循环,给每个子目录后加上built-in.o   /子目录/built-in.o 

Linux之Makefile(filter-out)_嵌入式后台开发-CSDN博客_makefile中filter-out

Makefile中wildcard的介绍 - haoxing990 - 博客园

这段的含义是获得.main.o.d这个字符串.d文件为配置文件,主要用于获得头文件信息,头文件被修改时,gcc -o 无法知道,就不会自动更新make结果

这段的作用就是判断前面的 .d文件是否存在,如果存在就包含进makefile,不包含的话就需要在后面生成

__build 依赖子目录$(subdir-y) 和built-in.o ,如果存在子目录,就执行

make -C $@ -f $(TOPDIR)/Makefile.build,执行子目录里面的Makefile.build,其实这就是套娃

如果Makefile.build不存在,则忽略,built-in.o不存在则继续执行下面

 $(LD) -r -o $@ $^等价于arm-linux-ld -r -o  目标文件按  所有依赖文件

意思就是把所有的依赖文件打包连接成一个built-in.o的文件

linux中ld的使用过程,arm-linux-ld 的使用_陈谈的博客-CSDN博客

$(cur_objs) $(subdir_objs)里面的.o文件如果不存在,则执行下面的%.o : %.c去生成.o文件

-Wp,-MD,$(dep_file)的作用则是生成需要的依赖文件,主要是头文件.h更改后能自动更新

7.执行完上面之后再次回到makefile

8.子目录的makefle写法

十分简单,子目录里所有文件统一用obj-y+=xxxx.o就可以

 9.总结

整个makefile的流程就是递归执行子目录里的makefile.build,子目录又递归执行子子目录里的makefile.build,编译连接生成各自的build.o,然后再跳回上一层,上一层把下层生成的build.o最终一起生成目标文件

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值