【补充】常用Makefile格式分析
扫描二维码
随时随地手机看文章
一、Makefile三大组成:目标、依赖、命令
led.bin: led.o
#(目标文件的生成依赖于命令行对依赖文件的处理。要将所有能生成的.o文件都写到依赖里面。)
#.bin为目标文件(启动文件),可直接烧录到芯片中运行
#.o为依赖文件,其生成源于.s(汇编文件)或.c(C文件),%表示名字,编译时名字要一致。
arm-linux-ld -Ttext 0x0 -o led.elf $^
#ld 表示连接。-T用于指定链接脚本(详情请参考韦东山应用开发完全手册-第三章),本例中0x0为编译时链接地址。 (为什么0x0可以作为链接地址,有两种说法一种是地址映射,第二种是位置无关吗。笔者倾向第二种)
#-o 左边为输入,右边为输出。
#.elf文件为系统可执行文件格式,相当于WINDOWS里的EXE格式.
#$^ 代表所有的依赖文件。 $@--目标文件,$<--第一个依赖文件。
arm-linux-objcopy -O binary led.elf led.bin
#objcopy用于从一个文件拷贝二进制目标代码到另一个文件,并在这一过程中执行各种转换。binary:二进制的
arm-linux-objdump -D led.elf > led_elf.dis
#objdump表示反汇编,'>' 表示将这个程序的反汇编程序写入到 .dis文件中,在终端中不显示出来.
gcc mkv210_image.c -o mkx210
./mkx210 led.bin 210.bin
#mkv210_image.c的主要作用是通过编译此文件,使得由usb启动时使用的led.bin制作得到由sd卡启动的镜像210.bin gcc(gnu collect compiler)是一组编译工具的总称。 它主要完成的工作任务是“预处理”和“编译”。
%.o : %.S
arm-linux-gcc -o $@ $< -c
#将目标.S文件编程.o文件
#-c 代表只进行预处理、编译和汇编操作,不进行链接。链接是在最后一步进行的,在生成.o文件的过程中一定要加-c进行说明。
%.o : %.c
arm-linux-gcc -o $@ $< -c
#同理分析
clean:
rm *.o *.elf *.bin *.dis mkx210 -f
#*万能匹配符,-f强制执行。删除所有.o、.elf、.bin、.dis和mkx210文件
补充:
= make会将整个makefile展开后,再决定变量的值。也就是说,变量的值将会是整个makefile中最后被指定的值
+= 是添加等号后面的值
:= 表示变量的值决定于它在makefile中的位置,而不是整个makefile展开后的最终值。
?= 是如果没有被赋值过就赋予等号后面的值
ex:
注:
x := foo y := $(x) bar x := xyz 则 y = foo bar
CC= arm-linux-gcc
#将arm-linux-gcc等价于CC
LD = arm-linux-ld
#将arm-linux-ld等价于LD
OBJCOPY= arm-linux-objcopy
# 将arm-linux-objcopy等价于OBJCOPY
OBJDUMP= arm-linux-objdump
#将arm-linux-objdump 等价于OBJDUMP
AR= arm-linux-ar
#将 arm-linux-ar等价于AR
#把多个.o文件合并成一个.o文件或静态库文件(.a文件)
INCDIR:= $(shell pwd)
#打印本文件夹路径赋值给 INCDIR
CPPFLAGS:= -nostdlib -nostdinc -I$(INCDIR)/include
# C预处理器的flag,flag就是编译器可选的选项
#-nostdlib:不连接系统标准启动文件和标准库文件
#-nostdinc:不在标准系统目录中搜索头文件,只在-I指定的目录中搜索
#-I为添加标准库文件夹
# $(INCDIR)/include:当前目录下的include文件
CFLAGS:= -Wall -O2 -fno-builtin
# C编译器的flag
#-Wall: 选项可以打印出编译时所有的错误或者警告信息
#- O2:表示编译时使用二级优化
#-fno-builtin: 不使用内建函数(如putchar。。。)
export CC LD OBJCOPY OBJDUMP AR CPPFLAGS CFLAGS
#导出这些变量到全局,其实就是给子文件夹下面的Makefile使用
objs := start.o sdram_init.o led.o uart.o main.o
#objs += clock.o
objs += lib/libc.a
#lib/libc.a 在子makefile中生成的目标
uart.bin: $(objs)
$(LD) -Tlink.lds -o uart.elf $^
$(OBJCOPY) -O binary uart.elf uart.bin
$(OBJDUMP) -D uart.elf > uart_elf.dis
gcc mkv210_image.c -o mkx210
./mkx210 uart.bin 210.bin
lib/libc.a:
cd lib;make;cd ..
#切换到lib文件夹,执行make指令,然后返回到上一级目录。
%.o : %.S
$(CC) $(CPPFLAGS) $(CFLAGS) -o $@ $< -c
%.o : %.c
$(CC) $(CPPFLAGS) $(CFLAGS) -o $@ $< -c
clean:
rm *.o *.elf *.bin *.dis mkx210 -f
cd lib; make clean; cd ..