当前位置: 代码迷 >> 综合 >> ARMlinux Makefile分析 根据原子哥
  详细解决方案

ARMlinux Makefile分析 根据原子哥

热度:22   发布时间:2024-01-16 00:34:41.0

原子哥的教程中对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。

  相关解决方案