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