加入星計(jì)劃,您可以享受以下權(quán)益:

  • 創(chuàng)作內(nèi)容快速變現(xiàn)
  • 行業(yè)影響力擴(kuò)散
  • 作品版權(quán)保護(hù)
  • 300W+ 專(zhuān)業(yè)用戶(hù)
  • 1.5W+ 優(yōu)質(zhì)創(chuàng)作者
  • 5000+ 長(zhǎng)期合作伙伴
立即加入
  • 正文
    • MfgTools工具目錄
    • MfgTool工具工作過(guò)程
    • ucl2.xml文件講解
  • 相關(guān)推薦
  • 電子產(chǎn)業(yè)圖譜
申請(qǐng)入駐 產(chǎn)業(yè)圖譜

飛凌嵌入式ElfBoard ELF 1板卡-mfgtools燒錄流程之燒寫(xiě)原理

2024/12/21
1055
閱讀需 35 分鐘
加入交流群
掃碼加入
獲取工程師必備禮包
參與熱點(diǎn)資訊討論

MfgTools工具目錄

MfgTool工具是一個(gè)NXP提供的OTG燒錄工具,工具主目錄主要由以下文件組成:

Profiles/目錄:

(一)ProfilesLinuxOS Firmwarefiles中,存儲(chǔ)著將要燒寫(xiě)到存儲(chǔ)介質(zhì)(啟動(dòng)介質(zhì))的目標(biāo)鏡像(uboot、內(nèi)核、文件系統(tǒng)、設(shè)備樹(shù)等)

(二)ProfilesLinuxOS Firmwarefirmware中,存儲(chǔ)著引導(dǎo)燒寫(xiě)的鏡像(uboot、內(nèi)核、ramfs文件系統(tǒng)、設(shè)備樹(shù))

(三)ProfilesLinuxOS Firmware中,存儲(chǔ)著ucl2.xml腳本和mksdcard.sh.tar分區(qū)腳本。

mx6ull-4gemmc-512mddr-qt5.6.vbs主要用于啟動(dòng)MfgTool2.exe程序,并向其傳遞配置參數(shù),MfgTool2.exe程序會(huì)根據(jù)這些參數(shù)執(zhí)行ProfilesLinuxOS Firmwareucl2.xml中的命令。

MfgTool工具工作過(guò)程

MfgTool工具工作過(guò)程主要分以下幾個(gè)步驟:

(一)撥碼無(wú)誤,開(kāi)發(fā)板上電之后,首先boot ROM(SOC內(nèi)部code,boot ROM具體作用在以后章節(jié)會(huì)進(jìn)一步講解)開(kāi)始運(yùn)行,對(duì)USB外設(shè)進(jìn)行初始化,并將開(kāi)發(fā)板枚舉成HID設(shè)備。

(二)Mfgtool工具檢測(cè)到此HID設(shè)備,并與之建立連接。然后Mfgtool工具會(huì)解析Firmware目錄下uboot,并使用uboot中的內(nèi)存配置信息對(duì)開(kāi)發(fā)板DDR進(jìn)行初始化配置。

(三)DDR內(nèi)存初始化完成之后,Mfgtool工具就會(huì)按照mfgtoolsProfilesLinuxOS Firmwareucl2.xml腳本中的命令將mfgtoolsProfilesLinuxOS Firmwarefirmware目錄下uboot鏡像、內(nèi)核鏡像、設(shè)備樹(shù)以及ramfs加載到DDR內(nèi)存中指定地址,然后發(fā)送指令給boot ROM,跳轉(zhuǎn)到DDR內(nèi)存的uboot入口地址,執(zhí)行uboot,uboot進(jìn)而啟動(dòng)內(nèi)核,內(nèi)核掛載ramfs文件系統(tǒng)。

(四)系統(tǒng)啟動(dòng)之后,會(huì)枚舉USB作為一個(gè)MSC設(shè)備,此時(shí)SoC可以和主機(jī)進(jìn)行高速數(shù)據(jù)傳輸。

(五)然后,Mfgtool會(huì)逐一執(zhí)行mfgtoolsProfilesLinuxOS Firmwareucl2.xml腳本中的命令,將mfgtoolsProfilesLinuxOS Firmwarefiles目錄下的uboot、內(nèi)核鏡像、設(shè)備樹(shù)文件、文件系統(tǒng)等逐一燒寫(xiě)到存儲(chǔ)介質(zhì)(啟動(dòng)介質(zhì))中。

ucl2.xml文件講解

xml文件中會(huì)根據(jù)不同的板卡類(lèi)型,選擇不同的LIST,主要講解EMMC版本的燒錄流程:

(一)將引導(dǎo)鏡像加載到DDR

在文件中查找<LIST name="eMMC" desc="Choose eMMC as media">

<LIST name="eMMC" desc="Choose eMMC as media">

<CMD state="BootStrap" type="boot" body="BootStrap" file ="firmware/u-boot-imx6ul%lite%%6uluboot%_emmc.imx" ifdev="MX6ULL">Loading U-boot</CMD>

<CMD state="BootStrap" type="load" file="firmware/zImage_emmc" address="0x80800000"

loadSection="OTH" setSection="OTH" HasFlashHeader="FALSE" ifdev="MX6SL MX6SX MX7D MX6UL MX6ULL">Loading Kernel.</CMD>

<CMD state="BootStrap" type="load" file="firmware/%initramfs%" address="0x83800000"

loadSection="OTH" setSection="OTH" HasFlashHeader="FALSE" ifdev="MX6SL MX6SX MX7D MX6UL MX6ULL">Loading Initramfs.</CMD>

<CMD state="BootStrap" type="load" file="firmware/zImage-imx6ul%lite%-%6uldtb%-emmc.dtb" address="0x83000000"

loadSection="OTH" setSection="OTH" HasFlashHeader="FALSE" ifdev="MX6ULL">Loading device tree.</CMD>

<CMD state="BootStrap" type="jump" > Jumping to OS image. </CMD>

可以看到先是加載了ProfilesLinuxOS Firmwarefirmware目錄下的u-boot-imx6ul%lite%%6uluboot%_emmc.imx也就是固件下的u-boot-imx6ull14x14evk_emmc.imx文件。

這兩個(gè)名稱(chēng)是如何定義的呢?在mfgtoos的根目錄下的mx6ull-4gemmc-512mddr-qt5.6.vbs和cfg.ini文件中有相關(guān)定義,可以右鍵使用文本格式打開(kāi)這兩個(gè)文件進(jìn)行查看。

cfg.ini文件如下:

[profiles]

chip = Linux




[platform]

board = SabreSD




[LIST]

name = SDCard




[variable]

board = sabresd

mmc = 0

sxuboot=sabresd

sxdtb=sdb

7duboot=sabresd

7ddtb=sdb

6uluboot=14x14ddr3arm2

6uldtb=14x14-ddr3-arm2

6ulldtb=14x14-ddr3-arm2

ldo=

plus=

lite=l

initramfs=fsl-image-mfgtool-initramfs-imx_mfgtools.cpio.gz.u-boot

seek = 1

sxnor=qspi2

7dnor=qspi1

6ulnor=qspi1

nor_part=0

可以看到,lite=l,6uluboot=14x14ddr3arm2,lite的定義沒(méi)問(wèn)題是l,但是6uluboot的定義也不是14x14evk呀,別著急看下一個(gè)文件。

mx6ull-8gemmc-512mddr-qt5.6.vbs:

Set wshShell = CreateObject("WScript.shell")

wshShell.run "mfgtool2.exe -c ""linux"" -l ""eMMC"" -s ""board=sabresd"" -s ""mmc=1"" -s ""6uluboot=14x14evk"" -s ""6uldtb=14x14-evk"""

Set wshShell = Nothing

能看到這里又重新定義了6uluboot的名稱(chēng),在ucl2.xml文件中使用%lite%和%6uluboot%來(lái)引用這兩個(gè)變量。所以u(píng)-boot-imx6ul%lite%%6uluboot%_emmc.imx就是u-boot-imx6ull14x14evk_emmc.imx文件。

這樣芯片就會(huì)自動(dòng)解析uboot然后對(duì)DDR進(jìn)行初始化,引導(dǎo)固件中的uboot就開(kāi)始運(yùn)行了。

接下來(lái)就是將ProfilesLinuxOS Firmwarefirmware目錄下的zImage_emmc加載到DDR的“0x80800000”地址,命令如下:

<CMD state="BootStrap" type="load" file="firmware/zImage_emmc" address="0x80800000"

loadSection="OTH" setSection="OTH" HasFlashHeader="FALSE" ifdev="MX6SL MX6SX MX7D MX6UL MX6ULL">Loading Kernel.</CMD>

然后是加載虛擬文件系統(tǒng),將ProfilesLinuxOS Firmwarefirmware目錄下的fsl-image-mfgtool-initramfs-imx_mfgtools.cpio.gz.u-boot加載到DDR的“0x83800000”,%initramfs%的定義在上文的cfg.ini文件中可以找到。這個(gè)虛擬文件系統(tǒng)又叫rmadisk,是將內(nèi)存的一塊地址虛擬成硬盤(pán)來(lái)使用,斷電后是不會(huì)保存的。

<CMD state="BootStrap" type="load" file="firmware/%initramfs%" address="0x83800000"

loadSection="OTH" setSection="OTH" HasFlashHeader="FALSE" ifdev="MX6SL MX6SX MX7D MX6UL MX6ULL">Loading Initramfs.</CMD>

然后加載設(shè)備樹(shù)文件,將ProfilesLinuxOS Firmwarefirmware目錄下的zImage-imx6ull-14x14-evk-emmc.dtb加載到DDR的“0x83000000”,最后jump命令跳轉(zhuǎn)到虛擬文件系統(tǒng)中。

<CMD state="BootStrap" type="load" file="firmware/zImage-imx6ul%lite%-%6uldtb%-emmc.dtb" address="0x83000000"

loadSection="OTH" setSection="OTH" HasFlashHeader="FALSE" ifdev="MX6ULL">Loading device tree.</CMD>

<CMD state="BootStrap" type="jump" > Jumping to OS image. </CMD>

(二)EMMC分區(qū),進(jìn)入到虛擬文件系統(tǒng)后就可以對(duì)EMMC進(jìn)行分區(qū)了。

先是將ProfilesLinuxOS Firmware下的mksdcard.sh.tar包發(fā)送到了虛擬文件系統(tǒng)中。

<!-- create partition -->

<CMD state="Updater" type="push" body="send" file="mksdcard.sh.tar">Sending partition shell</CMD>

然后使用tar命令將壓縮包解壓:

<CMD state="Updater" type="push" body="$ tar xf $FILE "> Partitioning...</CMD>

最后執(zhí)行mksdcard.sh對(duì)mmcblk1進(jìn)行格式化和分區(qū)操作。%mmc%在上文中的mx6ull-8gemmc-512mddr-qt5.6.vbs文件中有定義。

<CMD state="Updater" type="push" body="$ sh mksdcard.sh /dev/mmcblk%mmc%"> Partitioning...</CMD>

接下來(lái)看一下mksdcard.sh中是如何進(jìn)行格式化和分區(qū)的,在windows上解壓mksdcard.sh.tar并打開(kāi)mksdcard.sh腳本:

#!/bin/sh




# partition size in MB

BOOT_ROM_SIZE=10




# wait for the SD/MMC device node ready

while [ ! -e $1 ]

do

sleep 1

echo “wait for $1 appear”

done




# call sfdisk to create partition table

# destroy the partition table

node=$1

dd if=/dev/zero of=${node} bs=1024 count=1




sfdisk --force ${node} << EOF

${BOOT_ROM_SIZE}M,500M,0c

600M,,83

EOF

先是循環(huán)檢測(cè)虛擬文件系統(tǒng)中/dev/mmcblk1節(jié)點(diǎn)是否注冊(cè)成功,$1就是傳進(jìn)來(lái)的第一個(gè)參數(shù)/dev/mmcblk1。

# wait for the SD/MMC device node ready

while [ ! -e $1 ]

do

sleep 1

echo “wait for $1 appear”

done

如果識(shí)別到了mmcblk1設(shè)備,就使用dd命令將EMMC的前1KB的空間清0,EMMC的前1KB空間保存的是它的分區(qū)表信息,所以在正常使用情況下千萬(wàn)不能執(zhí)行如下操作。

# call sfdisk to create partition table

# destroy the partition table

node=$1

dd if=/dev/zero of=${node} bs=1024 count=1

接下來(lái)就是使用sfdisk命令對(duì)EMMC進(jìn)行分區(qū)操作。

sfdisk --force ${node} << EOF

${BOOT_ROM_SIZE}M,500M,0c

600M,,83

EOF

0c為fat32格式,83為ext3格式,其中10-500M為fat32格式,使用來(lái)存儲(chǔ)鏡像的分區(qū)。600M之后的空間都為ext3格式的分區(qū),是給文件系統(tǒng)來(lái)用的。在開(kāi)發(fā)板的命令行執(zhí)行df -h命令即可看到這些分區(qū),如下:

root@ELF1:~# df -h

Filesystem      Size     Used    Avail   Use%  Mounted on

/dev/root        6.5G    746M    5.5G   12%    /

devtmpfs        89M     4.0K    89M    1%     /dev

tmpfs           40K      0       40K    0%     /mnt/ .psplash

tmpfs            249M   140K  249M   1%   /run

tmpfs            249M   140K  249M   1%   /var/volatile

/dev/mmcblk1p1  500M   12M   488M   3%   /run/media/mmcblk1p1

(三)格式化和燒錄

uboot分區(qū)格式化和燒錄,先是清除了mmcblk1偏移量768之后的8k字節(jié),此位置存放的是uboot的環(huán)境變量。

<!-- burn uboot -->

<CMD state="Updater" type="push" body="$ dd if=/dev/zero of=/dev/mmcblk%mmc% bs=1k seek=768 conv=fsync count=8">clear u-boot arg</CMD>

然后使能mmcblk1boot0的寫(xiě)操作,將ProfilesLinuxOS Firmwarefiles目錄下的u-boot.imx燒寫(xiě)到mmcblk1boot0分區(qū)。

<!-- access boot partition -->

<CMD state="Updater" type="push" body="$ echo 0 > /sys/block/mmcblk%mmc%boot0/force_ro">access boot partition 1</CMD>

<CMD state="Updater" type="push" body="send" file="files/ u-boot.imx" ifdev="MX6ULL">Sending u-boot.bin</CMD>

<CMD state="Updater" type="push" body="$ dd if=$FILE of=/dev/mmcblk%mmc%boot0 bs=512 seek=2">write U-Boot to sd card</CMD>

<CMD state="Updater" type="push" body="$ echo 1 > /sys/block/mmcblk%mmc%boot0/force_ro"> re-enable read-only access </CMD>

<CMD state="Updater" type="push" body="$ mmc bootpart enable 1 1 /dev/mmcblk%mmc%">enable boot partion 1 to boot</CMD>

接下來(lái)是格式化FAT32分區(qū),并燒寫(xiě)zImage、logo圖片和dtb文件,同樣的先是檢測(cè)/dev下有沒(méi)有mmcblk1p1節(jié)點(diǎn),使用mkfs.vfat命令將分區(qū)格式化為fat格式,再使用mdir命令創(chuàng)建掛載目錄/mnt/mmcblk1p1,最后使用mount命令/dev下的mmcblk1p1掛載到新建的目錄上。

<!-- create fat partition -->

<CMD state="Updater" type="push" body="$ while [ ! -e /dev/mmcblk%mmc%p1 ]; do sleep 1; echo "waiting..."; done ">Waiting for the partition ready</CMD>

<CMD state="Updater" type="push" body="$ mkfs.vfat /dev/mmcblk%mmc%p1">Formatting rootfs partition</CMD>

<CMD state="Updater" type="push" body="$ mkdir -p /mnt/mmcblk%mmc%p1"/>

<CMD state="Updater" type="push" body="$ mount -t vfat /dev/mmcblk%mmc%p1 /mnt/mmcblk%mmc%p1"/>

有了掛載目錄,就可以將ProfilesLinuxOS Firmwarefiles目錄下的鏡像燒錄到EMMC了,其實(shí)就是一個(gè)拷貝的操作。

先是拷貝ProfilesLinuxOS Firmwarefiles目錄的zImage文件到EMMC。

<!-- burn zImage -->

<CMD state="Updater" type="push" body="send" file="files/zImage">Sending kernel zImage</CMD>

<CMD state="Updater" type="push" body="$ cp $FILE /mnt/mmcblk%mmc%p1/zImage">write kernel image to sd card</CMD>

<CMD state="Updater" type="push" body="$ cp $FILE /mnt/mmcblk%mmc%p1/zImagebak">write kernel image to sd card</CMD>

然后是logo圖片,

<!-- burn logo -->

<CMD state="Updater" type="push" body="send" file="files/logo.bmp">Sending logo bmp</CMD>

<CMD state="Updater" type="push" body="$ cp $FILE /mnt/mmcblk%mmc%p1/logo.bmp">write logo bmp to sd card</CMD>

然后是dtb文件,再使用umount命令將掛載目錄卸載掉。

<!-- burn dtb -->

<CMD state="Updater" type="push" body="send" <CMD state="Updater" type="push" body="send" file="files/imx6ull-elf1-emmc.dtb" ifdev="MX6ULL">Sending Device Tree file</CMD>

<CMD state="Updater" type="push" body="$ cp $FILE /mnt/mmcblk%mmc%p1/imx6ull-elf1-emmc.dtb" ifdev="MX6ULL">write device tree to sd card</CMD>

<CMD state="Updater" type="push" body="$ umount /mnt/mmcblk%mmc%p1">Unmounting vfat partition</CMD>

最后是格式化ext3分區(qū)并燒錄文件系統(tǒng),與fat分區(qū)是同樣的操作,先是使用mkfs.ext3命令將分區(qū)格式化為ext3格式,然后使用mkdir命令創(chuàng)建掛載分區(qū)目錄/mnt/mmcblk1p2,并將分區(qū)掛載到目錄上。

<!-- burn rootfs -->

<CMD state="Updater" type="push" body="$ mkfs.ext3 -F -E nodiscard /dev/mmcblk%mmc%p2">Formatting rootfs partition</CMD>

<CMD state="Updater" type="push" body="$ mkdir -p /mnt/mmcblk%mmc%p2"/>

<CMD state="Updater" type="push" body="$ mount -t ext3 /dev/mmcblk%mmc%p2 /mnt/mmcblk%mmc%p2"/>

有了掛載目錄,就可以將ProfilesLinuxOS Firmwarefiles下的文件系統(tǒng)壓縮包rootfs.tar.bz2和模塊驅(qū)動(dòng)壓縮包modules.tar.bz2解壓到掛載目錄中了,解壓之后使用umount命令卸載掛載目錄。

<CMD state="Updater" type="push" body="pipe tar -jxv -C /mnt/mmcblk%mmc%p2" file="files/rootfs.tar.bz2" ifdev="MX6UL MX7D MX6ULL">Sending and writting rootfs</CMD>

<CMD state="Updater" type="push" body="frf">Finishing rootfs write</CMD>

<CMD state="Updater" type="push" body="pipe tar -jxv -C /mnt/mmcblk%mmc%p2" file="files/modules.tar.bz2" ifdev="MX6UL MX7D MX6ULL">Sending and writting modules</CMD>

<CMD state="Updater" type="push" body="frf">Finishing modules software write</CMD>

<CMD state="Updater" type="push" body="$ umount /mnt/mmcblk%mmc%p2">Unmounting rootfs partition</CMD>

<CMD state="Updater" type="push" body="$ echo Update Complete!">Done</CMD>

(四)燒錄技巧

前面了解了整個(gè)燒錄的流程,我們就可以對(duì)ucl2.xml文件進(jìn)行修改來(lái)方便燒寫(xiě),其中燒錄文件系統(tǒng)是最耗時(shí)的,因?yàn)槲募到y(tǒng)的壓縮包比較大,需要完全把壓縮包解壓到EMMC中,如果我們想要加快燒錄時(shí)間,不希望燒錄文件系統(tǒng),我們就可以將ucl2.xml文件中燒錄文件系統(tǒng)的部分屏蔽掉,屏蔽語(yǔ)法為<!-- -->方法如下:

<!-- burn rootfs -->

<!--

<CMD state="Updater" type="push" body="$ mkfs.ext3 -F -E nodiscard /dev/mmcblk%mmc%p2">Formatting rootfs partition</CMD>

<CMD state="Updater" type="push" body="$ mkdir -p /mnt/mmcblk%mmc%p2"/>

<CMD state="Updater" type="push" body="$ mount -t ext3 /dev/mmcblk%mmc%p2 /mnt/mmcblk%mmc%p2"/>

<CMD state="Updater" type="push" body="pipe tar -jxv -C /mnt/mmcblk%mmc%p2" file="files/rootfs-qt.tar.bz2" ifdev="MX6UL MX7D MX6ULL">Sending and writting rootfs</CMD>

<CMD state="Updater" type="push" body="frf">Finishing rootfs write</CMD>

<CMD state="Updater" type="push" body="pipe tar -jxv -C /mnt/mmcblk%mmc%p2" file="files/modules.tar.bz2" ifdev="MX6UL MX7D MX6ULL">Sending and writting modules</CMD>

<CMD state="Updater" type="push" body="frf">Finishing modules software write</CMD>

<CMD state="Updater" type="push" body="$ umount /mnt/mmcblk%mmc%p2">Unmounting rootfs partition</CMD>

<CMD state="Updater" type="push" body="$ echo Update Complete!">Done</CMD>

-->

通過(guò)上面的學(xué)習(xí),我們對(duì)Mfgtools工具的燒錄原理已經(jīng)有了充分的理解,那么我們就可以將這個(gè)工具進(jìn)行修改變成自己的工具啦

相關(guān)推薦

電子產(chǎn)業(yè)圖譜