【補充】常用Makefile格式分析
一、Makefile三大組成:目標、依賴、命令
led.bin: led.o
#(目標文件的生成依賴于命令行對依賴文件的處理。要將所有能生成的.o文件都寫到依賴里面。)
#.bin為目標文件(啟動文件),可直接燒錄到芯片中運行
#.o為依賴文件,其生成源于.s(匯編文件)或.c(C文件),%表示名字,編譯時名字要一致。
arm-linux-ld -Ttext 0x0 -o led.elf $^
#ld 表示連接。-T用于指定鏈接腳本(詳情請參考韋東山應(yīng)用開發(fā)完全手冊-第三章),本例中0x0為編譯時鏈接地址。 (為什么0x0可以作為鏈接地址,有兩種說法一種是地址映射,第二種是位置無關(guān)嗎。筆者傾向第二種)
#-o 左邊為輸入,右邊為輸出。
#.elf文件為系統(tǒng)可執(zhí)行文件格式,相當于WINDOWS里的EXE格式.
#$^ 代表所有的依賴文件。 $@--目標文件,$<--第一個依賴文件。
arm-linux-objcopy -O binary led.elf led.bin
#objcopy用于從一個文件拷貝二進制目標代碼到另一個文件,并在這一過程中執(zhí)行各種轉(zhuǎn)換。binary:二進制的
arm-linux-objdump -D led.elf > led_elf.dis
#objdump表示反匯編,'>' 表示將這個程序的反匯編程序?qū)懭氲?.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)是一組編譯工具的總稱。 它主要完成的工作任務(wù)是“預(yù)處理”和“編譯”。
%.o : %.S
arm-linux-gcc -o $@ $< -c
#將目標.S文件編程.o文件
#-c 代表只進行預(yù)處理、編譯和匯編操作,不進行鏈接。鏈接是在最后一步進行的,在生成.o文件的過程中一定要加-c進行說明。
%.o : %.c
arm-linux-gcc -o $@ $< -c
#同理分析
clean:
rm *.o *.elf *.bin *.dis mkx210 -f
#*萬能匹配符,-f強制執(zhí)行。刪除所有.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文件或靜態(tài)庫文件(.a文件)
INCDIR:= $(shell pwd)
#打印本文件夾路徑賦值給 INCDIR
CPPFLAGS:= -nostdlib -nostdinc -I$(INCDIR)/include
# C預(yù)處理器的flag,flag就是編譯器可選的選項
#-nostdlib:不連接系統(tǒng)標準啟動文件和標準庫文件
#-nostdinc:不在標準系統(tǒng)目錄中搜索頭文件,只在-I指定的目錄中搜索
#-I為添加標準庫文件夾
# $(INCDIR)/include:當前目錄下的include文件
CFLAGS:= -Wall -O2 -fno-builtin
# C編譯器的flag
#-Wall: 選項可以打印出編譯時所有的錯誤或者警告信息
#- O2:表示編譯時使用二級優(yōu)化
#-fno-builtin: 不使用內(nèi)建函數(shù)(如putchar。。。)
export CC LD OBJCOPY OBJDUMP AR CPPFLAGS CFLAGS
#導(dǎo)出這些變量到全局,其實就是給子文件夾下面的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文件夾,執(zhí)行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 ..