摘要:作用是將標(biāo)準(zhǔn)輸入中的所有大寫字母轉(zhuǎn)換為響應(yīng)的小寫字母。的移植過的源代碼是在源代碼目錄下編譯的,所以源代碼目錄等于目標(biāo)文件目錄,所以條件不滿足,將執(zhí)行分支的代碼。
????????一個嵌入式產(chǎn)品的開發(fā)階段,需要不斷地把bootloader下載到存儲器中,如果存儲器使用nand flash,但是第一次里面什么都沒有,所以只能根據(jù)處理器的啟動方式從其他方式啟動如sd卡或nor存儲器啟動,然后在SD卡或nor存儲器啟動的基礎(chǔ)之上使用USB或網(wǎng)絡(luò)接口把u-boot.bin先下載到內(nèi)存中,然后再把內(nèi)存中的內(nèi)容寫到nand中,但是寫前4頁時只能寫每頁的前2KB數(shù)據(jù)(對于OK6410開發(fā)板來說,處理器使用S3C6410處理器,nand使用每頁4KB的存儲器,當(dāng)從nand啟動時,處理器會自動地把nand的前4頁的每一頁的前2KB拷貝片內(nèi)8KB的SRAM中運行,這是處理器硬件所決定,所以這里只能存每一頁的前2KB),前4頁后的所有頁都是全寫。
????????由于u-boot開發(fā)中需要不斷調(diào)試u-boot,而此時nand中已經(jīng)有u-boot,所以可以從nand存儲器啟動,然后根據(jù)開發(fā)的下載模式下的菜單選項,可以重新下載u-boot到nand中。
????????當(dāng)u-boot開發(fā)好后,并且從nand啟動,一上電,處理器硬件會自動把nand flash的前4頁中每頁前2KB拷貝到片內(nèi)SRAM中運行,而在SRAM中運行的代碼又實現(xiàn)了把nand中從0到240KB(這個大小可以變)的代碼拷貝到內(nèi)存,然后跳到內(nèi)存中運行,它在內(nèi)存運行時,會申請更多的空間(除開本身占用的內(nèi)存空間外,會包括12字節(jié)用于abort異常、堆??臻g、malloc內(nèi)存池空間、環(huán)境參數(shù)空間、某些全局變量空間),總共2MB,詳細(xì)查看(三.2.1)u-boot內(nèi)存分布圖。
????????一個嵌入式產(chǎn)品出廠時,在nand 存儲器里面已經(jīng)有了u-boot、內(nèi)核、文件系統(tǒng),并能實現(xiàn)對應(yīng)功能(如果使用nand存儲器,那么編譯好的映像是存儲在nand中的,運行是在內(nèi)存中運行,對于S3C6410,先在8KB片內(nèi)SRAM運行,然后才跳到內(nèi)存DDR中運行)。U-boot運行時會把nand里的內(nèi)核映像拷貝到內(nèi)存,然后運行內(nèi)核。
一、U-Boot-1.1.6頂層Makefile文件分析
(頂層makefile文件內(nèi)容為黑色,其他文件中的內(nèi)容為藍(lán)色,移植時修改過的代碼為紅色)
?????? 根據(jù)uboot根目錄下的Readme文件的說明,可以知道如果想把u-boot使用于開發(fā)板,應(yīng)先配置,即執(zhí)行make orlinx_nand_ram256_config命令進(jìn)行配置(在頂層目錄Makefile中加入forlinx_nand_ram256_config目標(biāo)等選項),然后執(zhí)行make all,就可以生成如下3個文件:U-Boot.bin、U-Boot ELF格式文件、U-Boot.srec。
????????(1)版本說明
VERSION = 1PATCHLEVEL = 1SUBLEVEL = 6EXTRAVERSION =U_BOOT_VERSION = $(VERSION).$(PATCHLEVEL).$(SUBLEVEL)$(EXTRAVERSION)VERSION_FILE = $(obj)include/version_autogenerated.h
(2)定義主機(jī)系統(tǒng)架構(gòu)
HOSTARCH := $(shell uname -m | / sed -e s/i.86/i386/ / -e s/sun4u/sparc64/ / -e s/arm.*/arm/ / -e s/sa110/arm/ / -e s/powerpc/ppc/ / -e s/macppc/ppc/)
????????“sed? –e”表示后面跟的是一串命令腳本,而表達(dá)式“s/abc/def/”表示要從標(biāo)準(zhǔn)輸入中,查找到內(nèi)容為“abc”的,然后替換成“def”。其中“abc”表達(dá)式用可以使用“.”作為通配符。命令“uname –m”將輸出主機(jī) CPU 的體系架構(gòu)類型。如電腦使用 Intel Core2 系列的CPU,那么 “uname? –m”將輸出“i686”。 “i686”可以匹配命令“sed? -e s/i.86/i386/”中的“i.86”,因此在機(jī)器上執(zhí)行 Makefile,HOSTARCH 將被設(shè)置成“i386” 。
(3)定義主機(jī)操作系統(tǒng)類型
HOSTOS := $(shell uname -s | tr "[:upper:]" "[:lower:]" | / sed -e "s//(cygwin/).*/cygwin/")export HOSTARCH HOSTOS# Deal with colliding definitions from tcsh etc.VENDOR=
????????“uname? –s”輸出主機(jī)內(nèi)核名字,開發(fā)主機(jī)使用 Linux 發(fā)行版 fedora-12,因此“uname? –s”結(jié)果是“Linux”?!皌r "[:upper:]" "[:lower:]"”作用是將標(biāo)準(zhǔn)輸入中的所有大寫字母轉(zhuǎn)換為響應(yīng)的小寫字母。因此執(zhí)行結(jié)果是將 HOSTOS 設(shè)置為“l(fā)inux”。
(4)定義執(zhí)行shell腳本的shell(源碼中沒有這部分)
# Set shell to bash if possible, otherwise fall back to shSHELL := $(shell if [ -x "$$BASH" ]; then echo $$BASH; /else if [ -x /bin/bash ]; then echo /bin/bash; /else echo sh; fi; fi)
????????"$$BASH"的作用實質(zhì)上是生成了字符串“$BASH”(前一個$號的作用是指明第二個$是普通的字符)。若執(zhí)行當(dāng)前 Makefile 的 shell 中定義了“$BASH”環(huán)境變量,且文件“$BASH”是可執(zhí)行文件,則 SHELL 的值為“$BASH”。否則,若“/bin/bash”是可執(zhí)行文件,則 SHELL 值為“/bin/bash”。若以上兩條都不成立,則將“sh”賦值給 SHELL 變量。如果機(jī)器安裝了bash? shell,且shell 默認(rèn)環(huán)境變量中定義了“$BASH”,因此 SHELL 被設(shè)置為$BASH 。
(5)設(shè)定編譯輸出目錄
ifdef O ifeq ("$(origin O)", "command line")BUILD_DIR := $(O)endifendif
????????函數(shù)$( origin, variable) 輸出的結(jié)果是一個字符串,輸出結(jié)果由變量 variable 定義的方式?jīng)Q定,若 variable 在命令行中定義過,則origin函數(shù)返回值為"command line"。假若在命令行中執(zhí)行了“export BUILD_DIR=/tmp/build”的命令,則“$(origin O)”值為“command line”,而 BUILD_DIR 被設(shè)置為“/tmp/build”。
????????下面內(nèi)容表示若${BUILD_DIR}表示的目錄沒有定義,則創(chuàng)建該目錄:
ifneq ($(BUILD_DIR),)saved-output := $(BUILD_DIR)# Attempt to create a output directory.$(shell [ -d ${BUILD_DIR} ] || mkdir -p ${BUILD_DIR})
????????下面內(nèi)容表示若$(BUILD_DIR)為空,則將其賦值為當(dāng)前目錄路徑(源代碼目錄)。并檢查$(BUILD_DIR)目錄是否存在:
# Verify if it was successful.BUILD_DIR := $(shell cd $(BUILD_DIR) && /bin/pwd)$(if $(BUILD_DIR),,$(error output directory "$(saved-output)" does not exist))endif # ifneq ($(BUILD_DIR),)
????????在下面內(nèi)容中,CURDIR 變量指示 Make 當(dāng)前的工作目錄,由于當(dāng)前 Make 在 U-Boot 頂層目錄執(zhí)行 Makefile,因此 CURDIR 此時就是 U-Boot 頂層目錄。執(zhí)行完下面的代碼后, SRCTREE,src變量就是U-Boot代碼頂層目錄,而OBJTREE,obj變量就是輸出目錄,若沒有定義BUILD_DIR環(huán)境變量,則SRCTREE, src變量與OBJTREE,obj變量都是U-Boot 源代碼目錄。而MKCONFIG則表示U-Boot根目錄下的mkconfig 腳本。
OBJTREE := $(if $(BUILD_DIR),$(BUILD_DIR),$(CURDIR))SRCTREE := $(CURDIR)TOPDIR := $(SRCTREE)LNDIR := $(OBJTREE)export TOPDIR SRCTREE OBJTREEMKCONFIG := $(SRCTREE)/mkconfigexport MKCONFIGifneq ($(OBJTREE),$(SRCTREE))REMOTE_BUILD := 1export REMOTE_BUILDendif# $(obj) and (src) are defined in config.mk but here in main Makefile# we also need them before config.mk is included which is the case for# some targets like unconfig, clean, clobber, distclean, etc.ifneq ($(OBJTREE),$(SRCTREE))obj := $(OBJTREE)/src := $(SRCTREE)/elseobj :=src :=endifexport obj src
????????在上面內(nèi)容中,MKCONFIG????? := $(SRCTREE)/mkconfig即在根目錄下的mkconfig文件。
(6)執(zhí)行make ?forlinx_nand_ram256_config過程
????????分析這個過程有助于理解移植U-Boot過程中需要修改哪些文件。執(zhí)行這個命令前提是在移植U-Boot時,在根目錄的Makefile中加入了類似如下的內(nèi)容:
forlinx_nand_ram256_config : unconfig @$(MKCONFIG) smdk6410 arm s3c64xx smdk6410 samsung s3c6410 NAND ram256 其中的依賴“unconfig”定義如下(Makefile文件的330-350行左右):unconfig: @rm -f $(obj)include/config.h $(obj)include/config.mk / $(obj)board/*/config.tmp $(obj)board/*/*/config.tmp
????????其中“@”的作用是執(zhí)行該命令時不在 shell 顯示?!皁bj”變量就是編譯輸出的目錄,因此“unconfig”的作用就是清除上次執(zhí)行 make *_config 命令生成的配置文件(如include/config.h,include/config.mk 等)。
????????$(MKCONFIG)在上面(5)指定為“$(SRCTREE)/mkconfig”,即根目錄的mkconfig文件。如果有$(@:_config=)一項,$(@:_config=)為將傳進(jìn)來的所有參數(shù)中的_config 替換為空(其中“@”指規(guī)則的目標(biāo)文件名,在這里就是“forlinx_nand_ram256_config ”。$(text:patternA=patternB),這樣的語法表示把 text 變量每一個元素中結(jié)尾的 patternA 的文本替換為 patternB,然后輸出)。因此$(@:_config=)的作用就是將forlinx_nand_ram256_config中的_config 去掉,得到 forlinx_nand_ram256,而在OK6410移植過的U-BOOT中沒有$(@:_config=)項。
??? 根據(jù)以上分析,執(zhí)行完make forlinx_nand_ram256_config,實際上執(zhí)行如下命令:
./mkconfig ?smdk6410 ?arm ?s3c64xx ?smdk6410 ?samsung ?s3c6410 ?NAND ?ram256
????????所以執(zhí)行make forlinx_nand_ram256_config即將“smdk6410 ?arm ?s3c64xx ?smdk6410 ?samsung ?s3c6410 ?NAND ?ram256”作為參數(shù)傳遞給當(dāng)前目錄下的mkconfig 腳本執(zhí)行。這些參數(shù)實際意義如下:
smdk6410:Target(目標(biāo)板型號)
arm:Architecture (目標(biāo)板的CPU架構(gòu))
s3c64xx:CPU(具體使用的 CPU型號)
smdk6410:Board
samsung:VENDOR(生產(chǎn)廠家名)
s3c6410:SOC
NAND:
ram256:
在mkconfig文件中,將進(jìn)行如下幾點的工作:
APPEND=no # no 表示創(chuàng)建新的配置文件,yes 表示追加到配置文件中BOARD_NAME="" # Name to print in make outputTARGETS=""while [ $# -gt 0 ] ; docase "$1" in--) shift ; break ;;-a) shift ; APPEND=yes ;;-n) shift ; BOARD_NAME="${1%%_config}" ; shift ;;-t) shift ; TARGETS="`echo $1 | sed "s:_: :g"` ${TARGETS}" ; shift ;;*) break ;;esacdone[ "${BOARD_NAME}" ] || BOARD_NAME ="$1"
????????對于命令./mkconfig ?smdk6410 ?arm ?s3c64xx ?smdk6410 ?samsung ?s3c6410 ?NAND ?ram256,其中沒有“--”“-a”“-n”“-t”“*”符號,所以while循環(huán)里面沒有做任何事情。而頭兩行中的兩個值仍然維持原來的值,但執(zhí)行完最后一行后,BOARD_NAME的值等于第1個參數(shù),即smdk6410(傳進(jìn)的幾個參數(shù)在mkconfig文件中以$x表示,$0= mkconfig,$1= smdk6410,$2= arm,$3= s3c64xx,$4= smdk6410,$5= samsung,$6= s3c6410,$7= NAND,$8= ram256)。
注意:
????????在U-Boot1.1.6-for-OK6410源代碼中,BOARD_NAME=""一行后還有一行SETMMU="no",表示在nand啟動是用mmu,只是這里先把值設(shè)置為no。
[ $# -lt 4 ] && exit 1
[ $# -gt 9 ] && exit 1
echo "Configuring for ${BOARD_NAME} board which boot from $7 $8 $9..."
??? 上面代碼的作用是檢查參數(shù)個數(shù)和參數(shù)是否正確,參數(shù)個數(shù)少于 4 個或多于9個都被認(rèn)為是錯誤的。環(huán)境變量$#表示傳遞給腳本的參數(shù)個數(shù),這里的命令有9個參數(shù),因此$#是9。Configuring for ${BOARD_NAME} board which boot from $7 $8 $9...即執(zhí)行完make forlinx_nand_ram256_config命令后串口終端輸出的信息。
# Create link to architecture specific headersif [ "$SRCTREE" != "$OBJTREE" ] ; then mkdir -p ${OBJTREE}/include mkdir -p ${OBJTREE}/include2 cd ${OBJTREE}/include2 rm -f asm ln -s ${SRCTREE}/include/asm-$2 asm LNPREFIX="../../include2/asm/" cd ../include rm -rf asm-$2 rm -f asm mkdir asm-$2 ln -s asm-$2 asmelse cd ./include rm -f asm ln -s asm-$2 asm # 符號連接,即軟鏈接fi
????????第一行代碼判斷源代碼目錄和目標(biāo)文件目錄是否一樣,可以選擇在其他目錄下編譯U-BOOT,這可令源代碼保持干凈,可以同時使用不同的配置進(jìn)行編譯。OK6410的U-BOOT移植過的源代碼是在源代碼目錄下編譯的,所以源代碼目錄等于目標(biāo)文件目錄,所以條件不滿足,將執(zhí)行else分支的代碼。
????????在else分支的代碼中,先進(jìn)入include目錄,刪除asm文件(這是上一次配置時建立的鏈接文件),然后再次建立asm文件,并令它鏈接向asm-$2目錄,即asm-arm目錄。此舉的作用為:在源碼中常調(diào)用頭文件,如#include
1:rm -f asm-$2/arch2:if [ -z "$6" -o "$6" = "NULL" ] ; then3: ln -s ${LNPREFIX}arch-$3 asm-$2/arch4:else5: ln -s ${LNPREFIX}arch-$6 asm-$2/arch6:fi7:if [ "$2" = "arm" ] ; then8: rm -f asm-$2/proc9: ln -s ${LNPREFIX}proc-armv asm-$2/proc10:fi
????????第1行刪除include/asm-arm/arch目錄,對于命令./mkconfig ?smdk6410 ?arm ?s3c64xx ?smdk6410 ?samsung ?s3c6410 ?NAND ?ram256,$6為S3C6410,不為空,也不為NULL,所以第2行條件不滿足,執(zhí)行else分支。第5行中,LNPREFIX為空(在頂層makefile中未定義),所以第5行即執(zhí)行ln ?–s? arch-s3c6410? asm-arm/arch(在U-Boot1.1.6-for-OK6410源代碼中,后面又把include/asm-arm/arch鏈接到了include/asm-arm/arch-s3c64xx)。
?????? 第8-9行表示:若目標(biāo)板是arm架構(gòu),則上面的代碼將建立符號連接 include/asm-arm/proc,使其鏈接到目錄 include/asm-arm/proc-armv 目錄。建立以上的鏈接的好處:編譯 U-Boot 時直接進(jìn)入鏈接文件指向的目錄進(jìn)行編譯,而不必根據(jù)不同開發(fā)板來選擇不同目錄。
注意:
??? 在U-Boot1.1.6-for-OK6410源代碼中,在第6行到7行之間,增加了如下部分代碼for OK6410:
# create link for s3c24xx SoC
if [ "$3" = "s3c24xx" ] ; then
?????? rm -f regs.h
?????? ln -s $6.h regs.h
?????? rm -f asm-$2/arch
?????? ln -s arch-$3 asm-$2/arch
fi
# create link for s3c64xx SoC
if [ "$3" = "s3c64xx" ] ; then
?????? rm -f regs.h
?????? ln -s $6.h regs.h
?????? rm -f asm-$2/arch
?????? ln -s arch-$3 asm-$2/arch
fi
即如果"$3" = "s3c64xx",將刪除include/regs.h,并把regs.h鏈接到include/s3c6410.h,在此頭文件中做了寫S3C6410的寄存器定義。然后刪除include/asm-arm/arch目錄,重新建立并把include/asm-arm/arch目錄鏈接到include/asm-arm/arch-S3C64xx目錄。
??? 在第9行和10行之間,增加了如下部分代碼:
fi
# create link for s3c64xx-mp SoC
if [ "$3" = "s3c64xx-mp" ] ; then
?????? rm -f regs.h
?????? ln -s $6.h regs.h
?????? rm -f asm-$2/arch
?????? ln -s arch-$3 asm-$2/arch
?????? 由于$3=s3c64xx,所以上面代碼中條件不成立,即上面代碼沒有做任何事。
????????創(chuàng)建頂層Makfile包含的文件include/config.mk
# Create include file for Makeecho "ARCH = $2" > config.mk echo "CPU = $3" >> config.mkecho "BOARD = $4" >> config.mk[ "$5" ] && [ "$5" != "NULL" ] && echo "VENDOR = $5" >> config.mk[ "$6" ] && [ "$6" != "NULL" ] && echo "SOC = $6" >> config.mk
????????當(dāng)執(zhí)行“./mkconfig ?smdk6410 ?arm ?s3c64xx ?smdk6410 ?samsung ?s3c6410 ?NAND ?ram256”命令后,上面幾行代碼創(chuàng)建的include/config.mk文件內(nèi)容如下:
ARCH = armCPU = s3c64xxBOARD = smdk6410VENDOR = samsungSOC = s3c6410
(可選,如果待移植的U-Boot源代碼中已經(jīng)有了那些目錄,就不需要下面的代碼,比如三星官方提供的U-Boot源代碼。)
# Assign board directory to BOARDIR variableif [ -z "$5" -o "$5" = "NULL" ] ; thenBOARDDIR=$4elseBOARDDIR=$5/$4fi
????????以上代碼指定 board 目錄下的一個目錄為當(dāng)前開發(fā)板專有代碼的目錄。若$5(VENDOR)為空則BOARDDIR設(shè)置為$4(BOARD),否則設(shè)置為$5/$4(VENDOR/BOARD,即samsung/smdk6410)。在這里由于$5 不為空,即BOARDDIR 被設(shè)置為 samsung/smdk6410 。
if [ "$APPEND" = "yes" ] # Append to existing config file2:then3: echo >> config.h4:else5: > config.h # Create new config file6:fi7:echo "/* Automatically generated - do not edit */" >>config.h8:echo "#include " >>config.h9:exit 0 “>” 和 “>>”為linux命令,> config.h表示重新建立config.h文件,echo "#include " >>config.h表示把#include 添加到config.h文件中。 APPEND維持原值”no”,所以config.h被重新建立,并添加了如下內(nèi)容:/* Automatically generated - do not edit */#include 到這里,include/config.h文件中就有以上的內(nèi)容了。
注意: 在U-Boot1.1.6-for-OK6410源代碼中,在第7行到8行之間,增加了如下部分代碼for OK6410: case $7 inSD) echo "#define FORLINX_BOOT_SD" >> config.h SETMMU="no" ;;NAND) echo "#define FORLINX_BOOT_NAND" >> config.h SETMMU="yes" ;;*) ;;esaccase $8 inram128) echo "#define FORLINX_BOOT_RAM128" >> config.h > ../board/samsung/smdk6410/config.mk # clear file context echo "ifndef TEXT_BASE" >> ../board/samsung/smdk6410/config.mk if [ ${SETMMU} = "yes" ] then echo "TEXT_BASE = 0xC7E00000" >> ../board/samsung/smdk6410/config.mk else echo "TEXT_BASE = 0x57E00000" >> ../board/samsung/smdk6410/config.mk fi echo "endif" >> ../board/samsung/smdk6410/config.mk ;;ram256) echo "#define FORLINX_BOOT_RAM256" >> config.h > ../board/samsung/smdk6410/config.mk # clear file context echo "ifndef TEXT_BASE" >> ../board/samsung/smdk6410/config.mk if [ ${SETMMU} = "yes" ] then echo "TEXT_BASE = 0xCFE00000" >> ../board/samsung/smdk6410/config.mk else echo "TEXT_BASE = 0x5FE00000" >> ../board/samsung/smdk6410/config.mk fi echo "endif" >> ../board/samsung/smdk6410/config.mk ;;*) ;;esacif [ "$9" = "hdmi" ] ; then echo "#define FORLINX_LCDOUT_HDMI" >> config.hfi 對于OK6410,$7=NAND,所以會把#define FORLINX_BOOT_NAND添加到include/config.h文件中,并把SETMMU值設(shè)置為yes(在上面確定開發(fā)板名稱BOARD_NAME部分代碼中把SETMMU值設(shè)為no,如果從nand啟動,需要用到mmu,所以這里設(shè)置為yes)。 對于OK6410,$8=ram256,所以會把#define FORLINX_BOOT_RAM256添加到include/config.h文件中,并且會把include的上層目錄的/board/samsung/smdk6410/config.mk文件(開發(fā)板代碼所在目錄)清空(執(zhí)行“> ../board/samsung/smdk6410/config.mk # clear file context”一行即實現(xiàn)清空), 清空后再往里面添加ifndef TEXT_BASE,由于SETMMU的值為yes,所以再往里面添加TEXT_BASE = 0xCFE00000(此地址為映射過的地址,使用mmu后,這個內(nèi)存地址被映射。如果是128內(nèi)存,此值為0XC7E00000,只要不超過DMC1的最大范圍物理地址0x6FFFFFFF對應(yīng)的虛擬地址即可),最后添加endif。 對于OK6410,$9沒有被傳入,所以沒有把#define FORLINX_LCDOUT_HDMI添加到/include/ config.h文件中。
????????總的來說,執(zhí)行make forlinx_nand_ram256_config命令后,會在mkconfig腳本中進(jìn)行上面第(6)點中的所有動作,由這些動作可知,要在board目錄下新建一個開發(fā)板< board_name >目錄,在include/configs目錄下建立一個
?????? U-Boot還沒有類似Linux一樣的可視化配置界面(如用make menuconfig來配置),需要手動修改配置頭文件/include/configs/smdk6410.h來裁剪、設(shè)置U-Boot。此配置頭文件中有以下兩類宏:
#define CONFIG_S3C6410?? ?????? 1??? ? /* in a SAMSUNG S3C6410 SoC???? ?*/
#define CONFIG_S3C64XX ???? ?1??? ? /* in a SAMSUNG S3C64XX Family? ?*/
#define CONFIG_SMDK6410????? 1??? ? /* on a SAMSUNG SMDK6410 Board? */
#define CFG_MALLOC_LEN? (CFG_ENV_SIZE+128*1024)
#define CFG_PROMPT??
#define CFG_LOAD_ADDR? 0x50000000
從下面的編譯、鏈接過程可知,U-Boot中幾乎每個文件都被編譯和鏈接,但是這樣文件是否包含有效代碼,則由宏開關(guān)來設(shè)置。比如對于網(wǎng)卡驅(qū)動drivers/dm9000x.c,它的格式為:#include #include #include #ifdef CONFIG_DRIVER_DM9000#include "dm9000x.h"/*實際代碼*/…#endif /* CONFIG_DRIVER_DM9000 */
????????如果在/include/configs/smdk6410.h中定義了宏CONFIG_DRIVER_DM9000,則文件中包含有效代碼;否則,文件被注釋為空。
??? 可以這樣認(rèn)為,“CONFIG_”除了設(shè)置一些參數(shù)外,主要用來設(shè)置U-Boot的功能、選擇使用文件中的哪一部分;而“CFG_”用來設(shè)置更細(xì)節(jié)的參數(shù)。
2. U-Boot的編譯、鏈接過程(接著頂層目錄中的Makefile分析)
(1)頂層Config.mk文件、頂層Makefile文件剩余代碼分析
配置完后,執(zhí)行“ make? all ”即可編譯。若沒有執(zhí)行過“make forlinx_nand_ram256_config”命令就直接執(zhí)行“make all”命令則會出現(xiàn)“System not configured - see README”錯誤信息,
然后停止編譯。
?????? 由于執(zhí)行完“make forlinx_nand_ram256_config”命令后,會在include目錄下創(chuàng)建一個config.mk文件,U-Boot就是判斷是否有這個文件而確定用戶是否執(zhí)行過“make forlinx_nand_ram256_config”命令,注意exit 1即返回,相關(guān)代碼如下:
114: ifeq ($(OBJTREE)/include/config.mk,$(wildcard $(OBJTREE)/include/config.mk))…241: all:…249: $(obj)u-boot.bin: $(obj)u-boot250: $(OBJCOPY) ${OBJCFLAGS} -O binary $< $@ …322: else …327: @echo "System not configured - see README" >&2328: @ exit 1329: endif 在頂層Makefile中,繼續(xù)分析如下代碼:# load ARCH, BOARD, and CPU configuration 下面為與ARM相關(guān)部分117: include $(OBJTREE)/include/config.mk118: export ARCH CPU BOARD VENDOR SOC…127: ifeq ($(ARCH),arm)128: CROSS_COMPILE = arm-linux-129: endif…162: export CROSS_COMPILE164: # load other configuration165: include $(TOPDIR)/config.mk
上面代碼中127-129行表示ARCH與目標(biāo)機(jī)器體系架構(gòu)相同,則使用 arm-linux編譯器(ARCH=arm)。
注意:
在U-Boot1.1.6-for-OK6410源代碼中,在162行前面加了如下代碼:
CROSS_COMPILE = /usr/local/arm/4.3.2/bin/arm-linux-即不管上面如何判斷等,交叉編譯器始終使用/usr/local/arm/4.3.2/bin目錄下的arm-linux-編譯器。
?????? 在117和165行中,用于包含config.mk文件,117行將make forlinx_nand_ram256_config命令生成的include/config.mk包含進(jìn)來,即配置過程中制作出來的include/config.mk文件,其中定義了ARCH/CPU/BOARD/VENDOR/SOC的值分別為arm、s3c64xx、smdk6410、samsung、s3c6410。165行將 U-Boot 頂層目錄下的 config.mk 文件包含進(jìn)來,該文件包含了對編譯的一些設(shè)置,它根據(jù)ARCH/CPU/BOARD/VENDOR/SOC變量的值確定了編譯器、編譯選項等。
頂層config.mk文件分析:?????
? 設(shè)置 obj 與 srcifneq ($(OBJTREE),$(SRCTREE))ifeq ($(CURDIR),$(SRCTREE))dir :=elsedir := $(subst $(SRCTREE)/,,$(CURDIR))endifobj := $(if $(dir),$(OBJTREE)/$(dir)/,$(OBJTREE)/)src := $(if $(dir),$(SRCTREE)/$(dir)/,$(SRCTREE)/)$(shell mkdir -p $(obj))elseobj :=src :=endif 由于目標(biāo)輸出到源代碼目錄下,因此執(zhí)行完上面的代碼后,src 和 obj 都是空。? 設(shè)置編譯選項PLATFORM_RELFLAGS =PLATFORM_CPPFLAGS = #編譯選項PLATFORM_ = #連接選項用這 3 個變量表示交叉編譯器的編譯選項,在后面 Make 會檢查交叉編譯器支持的編譯選項,然后將適當(dāng)?shù)倪x項添加到這3個變量中。? 包含與開發(fā)板相關(guān)的配置文件(跳過54到74行的代碼,這些代碼是在NetBSD上使用交叉編譯器時需要的定義)ifdef ARCHsinclude $(TOPDIR)/$(ARCH)_config.mk # include architecture dependend rulesendif$(ARCH)的值是“arm”,因此將頂層目錄下的“arm _ config.mk”包含進(jìn)來,而頂層目錄的arm _ config.mk文件中基本上什么都沒有做,只有一行代碼用來設(shè)置PLATFORM_CPPFLAGS編譯選項(與arm處理器相關(guān))。ifdef CPUsinclude $(TOPDIR)/cpu/$(CPU)/config.mk # include CPU specific rulesendif$(CPU)的值是“s3c64xx”,因此將“cpu/ s3c64xx /config.mk”包含進(jìn)來。這個腳本主要設(shè)定了跟s3c64xx處理器相關(guān)的編譯選項(與arm相關(guān)的PLATFORM_CPPFLAGS編譯選項)。ifdef SOCsinclude $(TOPDIR)/cpu/$(CPU)/$(SOC)/config.mk # include SoC specific rulesendif$(SOC)的值是s3c6410,因此Make程序嘗試將cpu/s3c64xx/s3c6410/config.mk包含進(jìn)來,而這個文件并不存在,但是由于用的是“sinclude”命令,所以并不會報錯。ifdef VENDORBOARDDIR = $(VENDOR)/$(BOARD)elseBOARDDIR = $(BOARD)endif$(BOARD)的值是smdk6410,VENDOR的值是 samsung,因此BOARDDIR的值是 samsung/ smdk6410。BOARDDIR 變量表示開發(fā)板特有的代碼所在的目錄。ifdef BOARDsinclude $(TOPDIR)/board/$(BOARDDIR)/config.mk # include board specific rulesendif頂層目錄下的config.mk文件將“board/samsung/smdk6410/config.mk”包含進(jìn)來。該腳本內(nèi)容如下:ifndef TEXT_BASETEXT_BASE = 0xCFE00000endifU-Boot編譯時將使用TEXT_BASE作為代碼段連接的起始地址(這個地址是經(jīng)過MMU映射過的)。? 其他代碼1 (95行---115行)CONFIG_SHELL := $(shell if [ -x "$$BASH" ]; then echo $$BASH; / else if [ -x /bin/bash ]; then echo /bin/bash; / else echo sh; fi ; fi)ifeq ($(HOSTOS)-$(HOSTARCH),darwin-ppc)HOSTCC = ccelseHOSTCC = gccendifHOSTCFLAGS = -Wall -Wstrict-prototypes -O2 -fomit-frame-pointerHOSTSTRIP = stripcc-option = $(shell if $(CC) $(CFLAGS) $(1) -S -o /dev/null -xc /dev/null / > /dev/null 2>&1; then echo "$(1)"; else echo "$(2)"; fi ;)在上面最后兩行代碼中,變量CC和CFLAGS在后面的代碼定義為延時變量,其中的 CC 即 arm-linux-gcc。函數(shù)cc-option 用于檢查編譯器 CC 是否支持某選項。將 2 個選項作為參數(shù)傳遞給 cc-option 函數(shù),該函數(shù)調(diào)用 CC 編譯器檢查參數(shù) 1 是否支持,若支持則函數(shù)返回參數(shù) 1,否則返回參數(shù) 2 (因此CC 編譯器必須支持參數(shù) 1 或參數(shù) 2,若兩個都不支持則會編譯出錯)。可以像下面這樣調(diào)用cc-option 函數(shù),并將支持的選項添加到 FLAGS 中:FLAGS +=$(call cc-option,option1,option2)? 指定交叉編譯工具AS = $(CROSS_COMPILE)asLD = $(CROSS_COMPILE)ldCC = $(CROSS_COMPILE)gccCPP = $(CC) -EAR = $(CROSS_COMPILE)arNM = $(CROSS_COMPILE)nmSTRIP = $(CROSS_COMPILE)stripOBJCOPY = $(CROSS_COMPILE)objcopyOBJDUMP = $(CROSS_COMPILE)objdumpRANLIB = $(CROSS_COMPILE)RANLIB
????????對于 arm 架構(gòu)處理器,其中的 CROSS_COMPILE在頂層makefile文件中定義:
CROSS_COMPILE = /usr/local/arm/4.3.2/bin/arm-linux-(161行左右)
因此以上代碼指定了使用前綴為“arm-linux-”的編譯工具,即arm-linux-gcc,arm-linux-ld 等等。
????????在這部分代碼中,143行(LDSCRIPT := $(TOPDIR)/board/$(BOARDDIR)/u-boot.lds)給出了LDSCRIPT的值為頂層目錄下/board/Samsung/smdk6410/u-boot.lds(:=表示替換以前的值),而189行代碼如下:
LDFLAGS += -Bstatic -T $(LDSCRIPT) -Ttext $(TEXT_BASE) $(PLATFORM_LDFLAGS)
???? 所以,根據(jù)189行,使LDFLAGS中包含“-Bstatic? -T /board/Samsung/smdk6410/u-boot.lds ”和“-Ttext ?0xCFE00000”的字樣(+=表示添加)。
????????在最后部分代碼中(218-246行),指定隱含的編譯規(guī)則,例如:根據(jù)以上的定義,以“.s”結(jié)尾的目標(biāo)文件將根據(jù)第一條規(guī)則由同名但后綴為“.S”的源文件來生成,若不存在“.S”結(jié)尾的同名文件則根據(jù)最后一條規(guī)則由同名的“.c”文件生成。
上面就是頂層目錄config.mk文件的內(nèi)容,下面為頂層目錄中Makefile剩下的內(nèi)容。
166:#########################################################################167:# U-Boot objects....order is important (i.e. start must be first)168:169:OBJS = cpu/$(CPU)/start.o170:ifeq ($(CPU),i386)171:OBJS += cpu/$(CPU)/start16.o172:OBJS += cpu/$(CPU)/reset.o173:endif174:ifeq ($(CPU),ppc4xx)175:OBJS += cpu/$(CPU)/resetvec.o176:endif177:ifeq ($(CPU),mpc83xx)178:OBJS += cpu/$(CPU)/resetvec.o179:endif180:ifeq ($(CPU),mpc85xx)181:OBJS += cpu/$(CPU)/resetvec.o182:endif183:ifeq ($(CPU),mpc86xx)184:OBJS += cpu/$(CPU)/resetvec.o185:endif186:ifeq ($(CPU),bf533)187:OBJS += cpu/$(CPU)/start1.o cpu/$(CPU)/interrupt.o cpu/$(CPU)/cache.o188:OBJS += cpu/$(CPU)/cplbhdlr.o cpu/$(CPU)/cplbmgr.o cpu/$(CPU)/flush.o189:endif190:191:OBJS := $(addprefix $(obj),$(OBJS))192:193:LIBS = lib_generic/libgeneric.a194:LIBS += board/$(BOARDDIR)/lib$(BOARD).a195:LIBS += cpu/$(CPU)/lib$(CPU).a196:ifdef SOC197:LIBS += cpu/$(CPU)/$(SOC)/lib$(SOC).a198:endif199:LIBS += lib_$(ARCH)/lib$(ARCH).a200:LIBS += fs/cramfs/libcramfs.a fs/fat/libfat.a fs/fdos/libfdos.a fs/jffs2/libjffs2.a /201: fs/reiserfs/libreiserfs.a fs/ext2/libext2fs.a202:LIBS += net/libnet.a203:LIBS += disk/libdisk.a204:LIBS += rtc/librtc.a205:LIBS += dtt/libdtt.a206:LIBS += drivers/libdrivers.a207:LIBS += drivers/nand/libnand.a208:LIBS += drivers/nand_legacy/libnand_legacy.a209:LIBS += drivers/sk98lin/libsk98lin.a210:LIBS += post/libpost.a post/cpu/libcpu.a211:LIBS += common/libcommon.a212:LIBS += $(BOARDLIBS)213:214:LIBS := $(addprefix $(obj),$(LIBS))215:.PHONY : $(LIBS)216:217:# Add GCC lib218:PLATFORM_LIBS += -L $(shell dirname `$(CC) $(CFLAGS) -print-libgcc-file-name`) -lgcc219:220:# The "tools" are needed early, so put this first221:# Don"t include stuff already done in $(LIBS)222:SUBDIRS = tools /223: examples /224: post /225: post/cpu226:.PHONY : $(SUBDIRS)227:228:ifeq ($(CONFIG_NAND_U_BOOT),y)229:NAND_SPL = nand_spl230:U_BOOT_NAND = $(obj)u-boot-nand.bin231:endif232:233:__OBJS := $(subst $(obj),,$(OBJS))234:__LIBS := $(subst $(obj),,$(LIBS))235:236:#########################################################################237:#########################################################################238:239:ALL = $(obj)u-boot.srec $(obj)u-boot.bin $(obj)System.map $(U_BOOT_NAND)240:241:all: $(ALL)242:243:$(obj)u-boot.hex: $(obj)u-boot244: $(OBJCOPY) ${OBJCFLAGS} -O ihex $< $@245:246:$(obj)u-boot.srec: $(obj)u-boot247: $(OBJCOPY) ${OBJCFLAGS} -O srec $< $@248:249:$(obj)u-boot.bin: $(obj)u-boot250: $(OBJCOPY) ${OBJCFLAGS} -O binary $< $@251:252:$(obj)u-boot.img: $(obj)u-boot.bin253: ./tools/mkimage -A $(ARCH) -T firmware -C none /254: -a $(TEXT_BASE) -e 0 /255: -n $(shell sed -n -e "s/.*U_BOOT_VERSION//p" $(VERSION_FILE) | /256: sed -e "s/"[ ]*$$/ for $(BOARD) board"/") /257: -d $< $@258:259:$(obj)u-boot.dis: $(obj)u-boot260: $(OBJDUMP) -d $< > $@261:262:$(obj)u-boot: depend version $(SUBDIRS) $(OBJS) $(LIBS) $(LDSCRIPT)263: UNDEF_SYM=`$(OBJDUMP) -x $(LIBS) |sed -n -e "s/.*/(__u_boot_cmd_.*/)/-u/1/p"|sort|uniq`;/264: cd $(LNDIR) && $(LD) $(LDFLAGS) $$UNDEF_SYM $(__OBJS) /265: --start-group $(__LIBS) --end-group $(PLATFORM_LIBS) /266: -Map u-boot.map -o u-boot267:268:$(OBJS):269: $(MAKE) -C cpu/$(CPU) $(if $(REMOTE_BUILD),$@,$(notdir $@))270:271:$(LIBS):272: $(MAKE) -C $(dir $(subst $(obj),,$@))273:274:$(SUBDIRS):275: $(MAKE) -C $@ all276:277:$(NAND_SPL): version278: $(MAKE) -C nand_spl/board/$(BOARDDIR) all279:280:$(U_BOOT_NAND): $(NAND_SPL) $(obj)u-boot.bin281: cat $(obj)nand_spl/u-boot-spl-16k.bin $(obj)u-boot.bin > $(obj)u-boot-nand.bin282:283:version:284: @echo -n "#define U_BOOT_VERSION /"U-Boot " > $(VERSION_FILE); /285: echo -n "$(U_BOOT_VERSION)" >> $(VERSION_FILE); /286: echo -n $(shell $(CONFIG_SHELL) $(TOPDIR)/tools/setlocalversion /287: $(TOPDIR)) >> $(VERSION_FILE); /288: echo "/"" >> $(VERSION_FILE)289:290:gdbtools:291: $(MAKE) -C tools/gdb all || exit 1292:293:updater:294: $(MAKE) -C tools/updater all || exit 1295:296:env:297: $(MAKE) -C tools/env all || exit 1298:299:depend dep:300: for dir in $(SUBDIRS) ; do $(MAKE) -C $$dir _depend ; done301:302:tags ctags:303: ctags -w -o $(OBJTREE)/ctags `find $(SUBDIRS) include /304: lib_generic board/$(BOARDDIR) cpu/$(CPU) lib_$(ARCH) /305: fs/cramfs fs/fat fs/fdos fs/jffs2 /306: net disk rtc dtt drivers drivers/sk98lin common /307: /( -name CVS -prune /) -o /( -name "*.[ch]" -print /)`308:309:etags:310: etags -a -o $(OBJTREE)/etags `find $(SUBDIRS) include /311: lib_generic board/$(BOARDDIR) cpu/$(CPU) lib_$(ARCH) /312: fs/cramfs fs/fat fs/fdos fs/jffs2 /313: net disk rtc dtt drivers drivers/sk98lin common /314: /( -name CVS -prune /) -o /( -name "*.[ch]" -print /)`315:316:$(obj)System.map: $(obj)u-boot317: @$(NM) $< | /318: grep -v "/(compiled/)/|/(/.o$$/)/|/( [aUw] /)/|/(/./.ng$$/)/|/(LASH[RL]DI/)" | /319: sort > $(obj)System.map320:321:#########################################################################322:else323:all $(obj)u-boot.hex $(obj)u-boot.srec $(obj)u-boot.bin /324:$(obj)u-boot.img $(obj)u-boot.dis $(obj)u-boot /325:$(SUBDIRS) version gdbtools updater env depend /326:dep tags ctags etags $(obj)System.map:327: @echo "System not configured - see README" >&2328: @ exit 1329:endif330:331:.PHONY : CHANGELOG332:CHANGELOG:333: git log --no-merges U-Boot-1_1_5.. | /334: unexpand -a | sed -e "s//s/s*$$//" > $@335:336:#########################################################################337:338:unconfig:339: @rm -f $(obj)include/config.h $(obj)include/config.mk /340: $(obj)board/*/config.tmp $(obj)board/*/*/config.tmp341:
????????在上面代碼中的169行,OBJS的第一個值為“cpu/$(CPU)/start.o”,即“cpu/s3c64xx/start.o”,根據(jù)上一行的注釋,u-boot中OBJS(目標(biāo))非常重要,start.o必須放在第一個,即首先編譯的start.s。在上面代碼的170-191行為其他架構(gòu)CPU的目標(biāo),在這里沒什么用處。
注意:
?????? 175-177行的代碼在U-Boot1.1.6-for-OK6410源代碼中沒有,理論上有沒有無影響。
????????在代碼193行到218行中,LIBS 變量指明了 U-Boot 需要的庫文件,包括平臺/開發(fā)板相關(guān)的目錄、通用目錄下相應(yīng)的庫,都通過相應(yīng)的子目錄編譯得到的,在215行中,定義偽目標(biāo)(.PHONY : $(LIBS)),意在當(dāng)前目錄下如果有LIBS變量對應(yīng)的值(即U-Boot 需要的庫文件,.a文件)時,執(zhí)行make后不會顯示“*.a is up-to-date”,而且會重新編譯生成對應(yīng)的.a文件。
注意:在208-209行之間,U-Boot1.1.6-for-OK6410源代碼中增加了如下代碼:
# add to support onenand. by scsuhLIBS += drivers/onenand/libonenand.aifeq ($(CPU),mpc83xx)LIBS += drivers/qe/qe.aendif 增加對onenand flash的支持。
????????228到231行是些與平臺無關(guān)的代碼,因為對于某些開發(fā)板(包括OK6410),u-boot支持在nand flash啟動,這些開發(fā)板配置文件中可能宏定義了CONFIG_NAND_U_BOOT,這樣在239行依賴U_BOOT_NAND不會為空。239行代碼即在執(zhí)行make all后,將要生成u-boot.srec,u-boot.bin,System.map。其中u-boot.srec 是 Motorola S-Record format 文件,System.map 是 U-Boot 的符號表,u-boot.bin 是最終燒寫到開發(fā)板的二進(jìn)制可執(zhí)行的文件。如果U_BOOT_NAND不為空,還將生成u-boot-nand.bin文件。
????????OBJS、LIBS所代表的.o、.a文件就是U-boot的構(gòu)成,它們通過268-272行命令,由相應(yīng)的源文件(或相應(yīng)子目錄下的文件)編譯得到。第268-269行規(guī)則表示,對于OBJS的每個成員,都將進(jìn)入cpu/$(CPU)目錄(即cpu/s3c64xx)編譯它們,對于smdk6410開發(fā)板,OBJS為cpu/s3c64xx/start.o,它將由cpu/s3c64xx/start.S編譯得到。第271-272行規(guī)則表示,對于LIBS中的每個成員,都將進(jìn)入相應(yīng)的子目錄執(zhí)行“make”命令。這些子目錄的Makefile,結(jié)構(gòu)相似,它們將Makefile中指定的文件編譯、鏈接成一個庫文件。
????????當(dāng)所有的OBJS、LIBS所表示的.o和.a文件都生成后,最后連接,這對應(yīng)上面代碼中的243到267行,先使用262-266行的規(guī)則鏈接得到ELF格式的U-BOOT,最后轉(zhuǎn)換為二進(jìn)制格式的u-boot.bin、S-Record格式的U-Boot.srec等等(即243-261行)。對于ELF格式的U-Boot的依賴,下面分別介紹:
行為依賴目標(biāo)depend的規(guī)則,對于300行中$(SUBDIRS),進(jìn)入該目錄執(zhí)行“make _depend”,生成各個子目錄的.depend 文件,.depend 列出每個目標(biāo)文件的依賴文件。
對于version依賴,即U-Boot的版本,這里不詳細(xì)介紹。274-275行為依賴SUBDIRS的規(guī)則,SUBDIRS的值222-226有定義,所以將執(zhí)行tools、examples、post、post/cpu目錄下的makefile。
這兩個依賴在上面已經(jīng)說明。
此依賴在頂層config.mk文件中有定義,LDSCRIPT的值為頂層目錄下/board/Samsung/smdk6410/u-boot.lds。
??????? U-BOOT規(guī)則命令中,264-266行表示真正連接,LDFLAGS確定了連接方式,LDFLAGS里/board/Samsung/smdk6410/u-boot.lds ”和“-Ttext? 0xCFE00000”的字樣(根據(jù)頂層config.mk文件),這些字樣指定了程序的布局、地址(但是連接起始地址為連接腳本中的偏移地址0+0xCFE00000)。所以,$(LDFLAGS)即使用u-boot.lds鏈接腳本及-Ttext? 0xCFE00000。執(zhí)行連接命令其實就是把 start.o 和各個子目錄 makefile 生成的庫文件按照 LDFLAGS 連接在一起,生成 ELF 文件 u-boot 和連接時內(nèi)存分配圖文件 u-boot.map。
注意:在上面代碼中250到251行之間, U-Boot1.1.6-for-OK6410源代碼中增加了一行代碼:$(OBJDUMP) -d $< > $<.dis。目的是生成U-Boot的反匯編文件。
????????頂層Makefile代碼中341行以后的代碼都是添加CPU架構(gòu)相關(guān)的_config文件(整個 makefile 剩下的內(nèi)容全部是各種不同的開發(fā)板的*_config:目標(biāo)的定義),本開發(fā)板OK6410添加了如下(當(dāng)然還有三星s3c24及64系列的其他處理器):
forlinx_nand_ram256_config :? unconfig
??? @$(MKCONFIG) smdk6410 arm s3c64xx smdk6410 samsung s3c6410 NAND ram256
注意:在頂層目錄makefile文件的末尾部分,是執(zhí)行make clean后的規(guī)則,在U-Boot1.1.6-for-OK6410源代碼中增加了二行代碼:分別在2261行后加了:????????
-o -name "*~" -o -name ".depend*" /
在2263行后增加了:
?rm -f u-boot*
(2)鏈接腳本分析(/board/Samsung/smdk6410/u-boot.lds)
24:OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")25:/*OUTPUT_FORMAT("elf32-arm", "elf32-arm", "elf32-arm")*/26:OUTPUT_ARCH(arm) /* 指定輸出平臺為arm */27:ENTRY(_start)28:SECTIONS29:{30: . = 0x00000000; 31:32: . = ALIGN(4); /* 代碼以4字節(jié)對齊 */33: .text : /* 指定代碼段,.text的基地址由LDFLAGS中-Ttext $(TEXT_BASE)指定*/34: {35: cpu/s3c64xx/start.o (.text) /* 代碼段的第一個代碼部份 */36: cpu/s3c64xx/s3c6410/cpu_init.o (.text)37: cpu/s3c64xx/onenand_cp.o (.text)38: cpu/s3c64xx/nand_cp.o (.text)39: cpu/s3c64xx/movi.o (.text) 40: *(.text) /*其他代碼部分*/41: lib_arm/div0.o42: }43:44: .
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/121955.html
摘要:要想放到樹莓派上運行,需要官方提供的固件四下載直接下載我們所需的在文件夾中,將其中的復(fù)制到卡的分區(qū)中之前分區(qū)時大小為,格式為的那個,再將編譯好的一同復(fù)制進(jìn)去。 2021SC@SDUSC 一、uboot的介紹 1.1、計算機(jī)系統(tǒng)的主要部件 (1)計算機(jī)系統(tǒng)就是以CPU為核心來運行的系統(tǒng)。典型的...
摘要:學(xué)習(xí)樹莓派的的初識學(xué)習(xí)樹莓派的的初識初識初識學(xué)習(xí)書籍正點原子嵌入式驅(qū)動開發(fā)指南章節(jié)第三十章學(xué)習(xí)內(nèi)容書中介紹的獲取可以有三個途徑第一個是的官方代碼。網(wǎng)上的燒寫樹莓派教程很多,但是為了學(xué)習(xí)我選擇了官方的代碼。Linux學(xué)習(xí) - 樹莓派4b的U-Boot的初識初識U-Boot學(xué)習(xí)書籍:《【正點原子】I.MX6U嵌入式Linux驅(qū)動開發(fā)指南V1.5.1》章節(jié):第三十章學(xué)習(xí)內(nèi)容:書中介紹uboot的獲...
摘要:表示,不是用壓縮的。兩者的不同之處在于,解壓縮內(nèi)核到低端內(nèi)存第一個,解壓縮內(nèi)核到高端內(nèi)存以上。如果內(nèi)核比較小,那么采用或都行,如果比較大應(yīng)該用。使得內(nèi)核可以啟動文件的拷貝或指向的鏈接。 ...
摘要:基于開發(fā)的軟件包導(dǎo)師汪禮超學(xué)員崔林威摘要騰訊物聯(lián)網(wǎng)操作系統(tǒng)是騰訊面向物聯(lián)網(wǎng)領(lǐng)域開發(fā)的實時操作系統(tǒng),具有低功耗,低資源占用,模塊化,可裁剪等特性。圖中斷函數(shù)處理進(jìn)行生成工程配置,按如下界面進(jìn)行配置,最后點擊,并點擊。 ...
閱讀 3976·2021-10-09 09:43
閱讀 2883·2021-10-08 10:05
閱讀 2745·2021-09-08 10:44
閱讀 889·2019-08-30 15:52
閱讀 2819·2019-08-26 17:01
閱讀 3026·2019-08-26 13:54
閱讀 1657·2019-08-26 10:48
閱讀 815·2019-08-23 14:41