ARMlinux Makefile分析 根据原子哥

本文详细解析了Makefile的编写,包括定义交叉编译工具、设置头文件路径、查找源文件、编译目标及链接过程。通过Makefile,实现了从.S和.c源文件到最终.bin二进制文件的编译链接流程,适用于嵌入式系统的开发。
摘要由CSDN通过智能技术生成

原子哥的教程中对Makefile有详细的分析,根据他的讲解我又整理了一下。

ARM_GCC ?= arm-linux-gnueabihf-
TARGET  ?= bsp
GCC     := $(ARM_GCC)gcc 
LD      := $(ARM_GCC)ld 
OBJCOPY := $(ARM_GCC)objcopy
OBJDUMP := $(ARM_GCC)objdump 

INCS := imx6ull \
        bsp/clk \
        bsp/led \
        bsp/delay
SRCS := bsp/clk   \
        bsp/led   \
        bsp/delay \
        project
INCLUDES := $(patsubst %, -I %,$(INCS))
#带路径的.S和.c
SFILES := $(foreach dir, $(SRCS), $(wildcard $(dir)/*.S) )
CFILES := $(foreach dir, $(SRCS), $(wildcard $(dir)/*.c) )
#不带路径的.S和.c
SNFILES := $(notdir $(SFILES))
CNFILES := $(notdir $(CFILES))
#将所有的.S.c换成.o
SOFILES := $(patsubst %.S, obj/%.o, $(SNFILES) )
COFILES := $(patsubst %.c, obj/%.o, $(CNFILES) )
OBJS := $(SOFILES) $(COFILES)
VPATH := $(SRCS)

.PHONY : clean
$(TARGET).bin : $(OBJS)
	$(LD) -Timx6ull.lds -o $(TARGET).elf $^     
#$^指所有依赖文件的集合 这里就是OBJS 本行将.o文件链接到相关地址
	$(OBJCOPY) -O binary -S $(TARGET).elf $@    
#$@指所有目标文件的集合 这里就是bin文件 本行将.elf转为.bin
	$(OBJDUMP) -D -m arm $(TARGET).elf > $(TARGET).dis 
#反汇编

$(SOFILES) : obj/%.o : %.S 
	$(GCC) -Wall -nostdlib -c -O2 $(INCLUDES) -o $@ $<
$(COFILES) : obj/%.o : %.c 
	$(GCC) -Wall -nostdlib -c -O2 $(INCLUDES) -o $@ $<

clean:
	 rm -rf $(TARGET).elf $(TARGET).bin $(TARGET).dis $(OBJS) 

Makefile的编写分为几部分:

1.定义交叉编译用到的自动变量

2.

INCS := imx6ull \
        bsp/clk \
        bsp/led \
        bsp/delay
SRCS := bsp/clk   \
        bsp/led   \
        bsp/delay \
        project

将所有包含.c和.h文件的文件夹列出

3.指定头文件路径 Makefile指定头文件路径需要 -I 

INCLUDES := $(patsubst %, -I %,$(INCS))

这句的作用就是把所有的.h文件前加 -I

4.

SFILES := $(foreach dir, $(SRCS), $(wildcard $(dir)/*.S) )
CFILES := $(foreach dir, $(SRCS), $(wildcard $(dir)/*.c) )

文件夹列出了就要找到文件,找到所有.c和.S文件,这里的.S文件就是start.S汇编启动文件

$(foreach <var>,<list>,<text>)

这个函数的功能是把list中的单词逐一取出放到var中,然后执行text的表达式。每次text会返回一个字符串,字符串与字符串之间以空格分开。

之所以用到wildcard是因为var是一个变量,要对变量使用通配符就要使用wildcard。

5.

SNFILES := $(notdir $(SFILES))
CNFILES := $(notdir $(CFILES))

上一步获取的是带路径的文件,现在要获取不带路径的文件,也就是xxx.c或者xxx.S。notdir函数的作用就是去掉路径。

6.

SOFILES := $(patsubst %.S, obj/%.o, $(SNFILES) )
COFILES := $(patsubst %.c, obj/%.o, $(CNFILES) )
OBJS := $(SOFILES) $(COFILES)

将所有去掉路径的.c和.S文件转换为obj文件下的.o文件。

7.

VPATH := $(SRCS)

VPATH的作用是指出下面的代码所在的路径,由于下面都是对.c.S文件编译,路径都在SRC下。VPATH详细的解释可以参考这篇:Makefile 里 VPATH 与 vpath 理解 - 简书

8.

$(TARGET).bin : $(OBJS)
	$(LD) -Timx6ull.lds -o $(TARGET).elf $^     
#$^指所有依赖文件的集合 这里就是OBJS 本行将.o文件链接到相关地址
	$(OBJCOPY) -O binary -S $(TARGET).elf $@    
#$@指所有目标文件的集合 这里就是bin文件 本行将.elf转为.bin
	$(OBJDUMP) -D -m arm $(TARGET).elf > $(TARGET).dis 
#反汇编

链接,转为.elf,.elf转为.bin文件

9.

$(SOFILES) : obj/%.o : %.S 
	$(GCC) -Wall -nostdlib -c -O2 $(INCLUDES) -o $@ $<
$(COFILES) : obj/%.o : %.c 
	$(GCC) -Wall -nostdlib -c -O2 $(INCLUDES) -o $@ $<

编译.o文件

$(SOFILES) : obj/%.o : %.S 

这是静态模式的用法,表示依赖目标换为由x.o变为x.S。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值