這也是《FPGA實(shí)現(xiàn)串口升級(jí)及MultiBoot》系列中的一篇文章,作為一個(gè)專(zhuān)題單獨(dú)出來(lái)說(shuō)明。
本篇文章分為三個(gè)主題:固化、啟動(dòng)和MultiBoot實(shí)現(xiàn)。
固化分為SPI和BPI FLASH兩種情況;啟動(dòng)分為SREC解析及加快啟動(dòng)模式的ELF直讀;最后就是MultiBoot實(shí)現(xiàn)的時(shí)候應(yīng)該注意什么。
固化
軟核的固化和外部FLASH及應(yīng)用程序大小有很大關(guān)系。小應(yīng)用程序,使用BRAM即可運(yùn)行,固化的時(shí)候和邏輯一起固化即可運(yùn)行,這時(shí)候不管外部是什么類(lèi)型FLASH,和邏輯固化一樣。大應(yīng)用程序,需要用到外部DDR,就需要兩個(gè)啟動(dòng)程序,一個(gè)小的Bootloader,和邏輯一起啟動(dòng),起來(lái)后從外部FLASH讀取相應(yīng)的應(yīng)用程序放到DDR里運(yùn)行,這時(shí)候就要根據(jù)不同的FLASH,設(shè)計(jì)相應(yīng)的讀取邏輯及應(yīng)用。下面我們根據(jù)不同的FLASH類(lèi)型進(jìn)行相應(yīng)的說(shuō)明。
SPI FLASH
小應(yīng)用程序
小應(yīng)用程序,比如串口通信,IIC、SPI等配置通信,不占用多少STACK_SIZE和HEAP_SIZE,用內(nèi)部BRAM就可運(yùn)行,這時(shí)候?qū)?yīng)用程序的LD設(shè)置為運(yùn)行到BRAM即可,生成elf后就可以將其和邏輯的BIT融合在一起生成新的BIT,正常生成MCS即可燒寫(xiě)。
elf和bit融合成新的BIT,有兩種方式,下面詳細(xì)介紹:
調(diào)試階段
調(diào)試階段通過(guò)Vitis或者SDK一步即可生成新的BIT:
即在vitis工作目錄-->應(yīng)用程序目錄-->_ide-->bitstream-->download.bit
該bit就是融合后的BIT,接下來(lái)可以使用Vitis/SDK或者Vivado進(jìn)行下載。
穩(wěn)定階段
應(yīng)用程序一般比較簡(jiǎn)單,所以調(diào)試穩(wěn)定后,就可以將elf交給邏輯端,讓他們繼續(xù)折騰。
Vivado中在生成bit前,通過(guò)下面步驟將elf和軟核關(guān)聯(lián)起來(lái),后面Vivado生成bit后,就是包含了軟核程序的BIT,繼續(xù)后續(xù)的固化工作即可。
大應(yīng)用程序
大應(yīng)用程序就會(huì)復(fù)雜很多,主要是建立軟核讀取外部FLASH的通道。
大的應(yīng)用程序需要運(yùn)行到SDRAM中,節(jié)省內(nèi)部BRAM,這里的“大”并不是我們說(shuō)的elf文件大,而是需要的運(yùn)行空間大。
其實(shí)可以完全不需要內(nèi)部BRAM也可以運(yùn)行大應(yīng)用程序程序,不過(guò)制作過(guò)程比較復(fù)雜,我們這次還是需要一點(diǎn)內(nèi)部BRAM啟動(dòng)bootloader。
整個(gè)流程框圖如下:
首先這種方式需要三個(gè)文件:1、FPGA的bit文件(需要包含F(xiàn)LASH的讀寫(xiě)控制器);2、bootloader 生成的elf文件;3、大應(yīng)用應(yīng)用程序的ELF文件。
這里要用到比較重要的IP-axi_quad_spi(可以自己按照需求寫(xiě)IP),將IP按照下面設(shè)置進(jìn)行設(shè)置:
主要是使能STARTUP原語(yǔ)。
PS:7系列的在IP內(nèi)部使能STARTUP即可,對(duì)于U+系列由于FLASH的IO位于BANK0,需要在頂層再使用STARTUP3原語(yǔ)進(jìn)行IO引出(IP內(nèi)部雖然也使用了原語(yǔ),但是我調(diào)試的時(shí)候只有在頂層添加STARTUP3原語(yǔ)才能進(jìn)行通信,不清楚是BUG還是沒(méi)設(shè)置好,按照官方說(shuō)明使用方式也是要在頂層再添加原語(yǔ))。
邏輯端添加上訴IP即可,接下來(lái)是應(yīng)用程序設(shè)計(jì)。
在SDK或者Vitis中添加下面BootLoader程序:
接下來(lái)要修改第一個(gè)地方:
就是這個(gè)起始地址,其中0x44A00000就是Vivado中IP的基地址,0x80000是從FLASH讀取的地址,即應(yīng)用存放在FLASH中的地址,該地址要和生成MCS時(shí)候地址相匹配。
第二個(gè)需要注意的地方就是BootLoader要放到BRAM里運(yùn)行,查看以下LD:
這樣就可以和上面小應(yīng)用程序一樣,隨著邏輯端一起啟動(dòng)。
接下來(lái)就是固化,固化也是分為兩個(gè)階段。
調(diào)試階段
調(diào)試階段首先將大應(yīng)用程序放到FLASH相應(yīng)的偏移地址上,通過(guò)以下工具可以直接燒寫(xiě):
這里先選擇大應(yīng)用程序生成的elf,然后選擇燒寫(xiě)的偏移地址(按照BootLoader里設(shè)置的偏移地址進(jìn)行設(shè)置),然后選擇對(duì)應(yīng)的FLASH,最后一定要勾選Convert ELF to bootloadable SREC format and program,這是目前官方驅(qū)動(dòng)程序能讀取的二進(jìn)制結(jié)構(gòu)-SREC。
關(guān)于怎么減少啟動(dòng)時(shí)間我們后面會(huì)單獨(dú)討論。
注意點(diǎn)1:修改驅(qū)動(dòng)
如果不能讀取FLASH,那么需要修改驅(qū)動(dòng)文件。Board support Package setting,修改xilisf,設(shè)置serial_flash_family(根據(jù)FLASH廠(chǎng)家修改)serial_flash_interface根據(jù)需求修改。
注意點(diǎn)2:讀取超時(shí)
增加等待flash初始化時(shí)間:
原位置:
修改
/* *InitializetheSerialFlashLibrary. */ volatileintwait; do { Status=XIsf_Initialize(&Isf,&Spi,ISF_SPI_SELECT,IsfWriteBuffer); if(Status!=XST_SUCCESS){ xil_printf("XIsf_initializefailed!Status=%d ",Status); for(wait=0;wait100000;?wait++); ???for?(wait=0;wait?100000;?wait++); ??} ?}while(Status?!=?XST_SUCCESS?);
注意點(diǎn)3:讀取速度
某些國(guó)產(chǎn)FLASH的速度遠(yuǎn)低于進(jìn)口FALSH,所以注意看下數(shù)據(jù)手冊(cè),控制好ext_spi_clk,同時(shí)下圖還有一個(gè)分頻系數(shù):
穩(wěn)定階段
如果BootLoader能夠引到啟動(dòng)后,可以將BootLoader的ELF添加到Vivado工程中,這樣在大應(yīng)用程序需要重建后無(wú)需每次都建立BootLoader工程進(jìn)行調(diào)試。同時(shí)axi_quad_spi不更改(基地址不變)情況下,BootLoader都不需要修改。
BPI FLASH
小應(yīng)用程序
小應(yīng)用程序和SPI的一樣,不隨著外圍FLASH不同而改變,就不贅述了。
大應(yīng)用程序
大應(yīng)用程序就更復(fù)雜一些,也是需要建立軟核讀取外部FLASH的通道。
接下來(lái)我們按照另一個(gè)思路講解BPI FLASH大應(yīng)用程序的固化。上面我們說(shuō)過(guò)大應(yīng)用程序的固化需要三個(gè)文件:1、FPGA的bit文件(需要包含F(xiàn)LASH的讀寫(xiě)控制器);2、bootloader 生成的elf文件;3、大應(yīng)用程序的ELF文件。我們按照順序講解每個(gè)文件生成需要的必要條件:
FPGA的bit文件
FPGA設(shè)計(jì)中需要增加對(duì)FLASH控制,對(duì)于BPI FLASH,官方推薦AXI-EMC IP,通過(guò)AXI-MEM映射外部FLASH,這個(gè)IP可以控制常見(jiàn)的BPI FLASH(可配置參數(shù)很低),對(duì)于不能控制的FLASH,需要自己寫(xiě)控制器+控制驅(qū)動(dòng)。
AXI_EMC IP 概述
AXI_EMC是FPGA的一個(gè)ip core,axi 外部存儲(chǔ)控制器,支持sram,nor flash ,psram,cellularRAM,IP核使用AXI4接口,支持32bit和64bit的數(shù)據(jù)位寬,支持以下memory type,即:
異步SRAM
同步SRAM
串行flash or并行nor flash
偽靜態(tài)隨機(jī)存儲(chǔ)器
每個(gè)emc控制器支持掛接4個(gè)存儲(chǔ)設(shè)備
來(lái)源:pg100-figure1-1
信號(hào)連接
我們這次使用的BPI FLASH為S29GL01GSXXXX,可以通過(guò)AXI-EMC 控制,具體在FPGA中的連線(xiàn)如下:
主要將這個(gè)IP連到MB上,外接引腳比較重要,因?yàn)镮P要兼容的東西比較多,所以引出來(lái)的引腳比較多,對(duì)于不同的外設(shè)需要連接不同的引腳,對(duì)于本次設(shè)計(jì)的引腳連接如下表所示:
圖中紅線(xiàn)部分需根據(jù)表進(jìn)行連接
IP界面時(shí)序參數(shù)配置
(1)第一頁(yè)
默認(rèn)配置即可,如果使用比較大的內(nèi)存則可選總線(xiàn)位寬為64位,我們控制FLASH,默認(rèn)32位即可:
這里注意以下,如果使用U+系列,這個(gè)IP會(huì)在下面位置有個(gè)使能STARTUP3原語(yǔ)的選項(xiàng):
使能后和SPI FLASH一樣,需要在頂層添加STARTUP原語(yǔ)引出BPI的DQ0~DQ3。這里說(shuō)下原因,因?yàn)?系列FPGA的DQ0~DQ3是在BANK14,而U+是在BANK0上,同時(shí)官方IP可能有BUG,導(dǎo)致需要2次使用原語(yǔ)。
(2)第二頁(yè)
這一頁(yè)的設(shè)置是核心,需要根據(jù)FLASH數(shù)據(jù)手冊(cè)進(jìn)行配置:
Memory Type
支持Sync SRAM, Async SRAM, Linear Flash, Page Mode Flash, PSRAM, or Micron Flash
Data Width
Memory 數(shù)據(jù)位寬,也就是接的存儲(chǔ)設(shè)備的數(shù)據(jù)位寬,支持8,16,32,64位位寬
Parity
可以設(shè)置為No parity,Odd Parity,or Even Parity,只有Memory Type為Sync SRAM時(shí)才能設(shè)置
Delay Mode
可以設(shè)置為Flow-Through model or Pipeline Model,只有Memory Type為Sync SRAM時(shí)才能設(shè)置
Read CE Low to Data Valid Period
根據(jù)描述,該參數(shù)在不同的memory type下含義不同,如果是flash,則和tELQV的值相等,這里以Page Mode Flash為例說(shuō)明,基本上也就是片選拉低的時(shí)間,其他類(lèi)型的存儲(chǔ)設(shè)備沒(méi)有驗(yàn)證,暫不清楚,以flash型號(hào)為S29GL01GSXXX(容量大小:128Mbyte)為例,根據(jù)芯片手冊(cè)中的描述:
該芯片讀時(shí)序如下:
對(duì)該芯片來(lái)說(shuō),該參數(shù)值就是tCE,即100ns = 100000ps
Read Address Valid to Data Valid Period
這里的意思為讀地址在數(shù)據(jù)有效前的保持時(shí)間,不是片選保持的時(shí)間,對(duì)于S29GL01GSXX來(lái)說(shuō),就是tAVQV的值,和描述的一樣,即為100ns = 100000ps,如上圖所示。
Page Access Period
根據(jù)描述的含義,對(duì)于Page Mode Flash來(lái)說(shuō)就是訪(fǎng)問(wèn)一頁(yè)需要的時(shí)間,以S29GL01GSXX為例,也就是tPACC,為25ns,如下:
Read CE High to Data Bus HZ Period
該參數(shù)的含義是指片選(CE)從有效變?yōu)闊o(wú)效后,至少要保持多長(zhǎng)時(shí)間,即從低變高后,要保持多長(zhǎng)時(shí)間的高電平,以S29GL01GSXX為例,就是圖中的tDF也就是tEHQZ,為20ns
Read OE High to Data Bus HZ Period
和上邊參數(shù)一樣,指OE高電平保持的時(shí)間,也就是上圖中的tDF,為20ns
AXI Read Timing
根據(jù)AXI EMC IP核手冊(cè)的page47頁(yè)得知以上讀時(shí)序圖,結(jié)合“IP核參數(shù)默認(rèn)值”查看
AXI Write Timing
根據(jù)AXI EMC IP核手冊(cè)的page47頁(yè)得知以上寫(xiě)時(shí)序圖,結(jié)合“IP核參數(shù)默認(rèn)值”查看
IP核參數(shù)默認(rèn)值:
Write Cycle Period
根據(jù)描述,AXI EMC core根據(jù)該參數(shù)去保持CE為低的時(shí)間,但不是指CE保持的時(shí)間,以S29GL01GSXX為例,值等于tWC,即60ns,如下
Write Enable Minimum Pulse Width
該參數(shù)指片選(WE)保持的最小時(shí)間,以S29GL01GSXX為例,值等于tWP,即最小25ns
Write Phase Period
該參數(shù)表示兩個(gè)WE之間的最小間隔,以S29GL01GSXX為例,值等于tWPH,即最小20ns
Write WE High to Data Bus LZ Period
該參數(shù)表示在一個(gè)寫(xiě)周期里,寫(xiě)使能(WE)到數(shù)據(jù)總線(xiàn)低阻的時(shí)間,或者寫(xiě)到讀的恢復(fù)時(shí)間(可能理解的不正確,關(guān)于此參數(shù)配置時(shí)請(qǐng)謹(jǐn)慎),以S29GL01GSXX為例,使用的默認(rèn)值,即0ps。
Write Recovery Period for Flash Memory
這個(gè)參數(shù)應(yīng)該指的是數(shù)據(jù)寫(xiě)完后,WE還需要保持多少個(gè)時(shí)鐘周期的時(shí)間,這里使用默認(rèn)值,根據(jù)下圖,此參數(shù)小于等于twrr,具體多少不知道,個(gè)人理解大于等于tWPH應(yīng)該就可以了。
IP其他界面默認(rèn)即可,地址分配界面將IP尋址空間大小設(shè)置成實(shí)際FLASH大小,如下:
編譯后導(dǎo)出bit到SDK或者Vitis,就可以制作BootLoader.
以上參數(shù)解釋?zhuān)瑓⒖枷旅娴牟┪模?/p>
https://blog.csdn.net/qq_33166886/article/details/112490916
這里要用到比較重要的IP-axi_quad_spi(可以自己按照需求寫(xiě)IP),將IP按照下面設(shè)置進(jìn)行設(shè)置:
BootLoader制作
打開(kāi)sdk或者Vitis,新建一個(gè)SREC的工程。這個(gè)就是傳說(shuō)中啟動(dòng)文件。
其中1是用來(lái)制作BPI FALSH的啟動(dòng)文件,2是用來(lái)制作SPI FLASH啟動(dòng)文件的。
修改FLASH的鏡像地址。地址不是亂寫(xiě)的。是根據(jù)你的mcs文件大小,文件大小在runs/imp/ 文件夾下的 prm 文件上有告知。后面接上的。然后生成文件。注意這個(gè)地址也是你后面真正運(yùn)行的elf地址寫(xiě)入。
也可以按照prm文件后續(xù)接上elf文件:
bootloader工程,注意選擇ld看鏈接表設(shè)定。不要把代碼和數(shù)據(jù)放在DDR上面。
然后編譯工程。生成elf文件。然后按照之前的方式將BootLoader elf和vivado生成的bit合成一個(gè)新的bit。
剩下的步驟和之前SPI FLASH一樣了,就不贅述了。
啟動(dòng)
固化完成了,接下來(lái)就是啟動(dòng)了。小應(yīng)用程序和FPGA啟動(dòng)一起,所以就不說(shuō)明了,重點(diǎn)介紹大應(yīng)用程序。
大應(yīng)用程序如果使用官方的BootLoader,在讀取大應(yīng)用到DDR中時(shí),是讀取的SREC,那么為什么要讀取SREC,SREC文件有什么特點(diǎn)?我們接下來(lái)開(kāi)始揭開(kāi)面紗。
SREC文件
SREC文件是帶有程序的地址信息和數(shù)據(jù)校驗(yàn)功能,所以在讀取SREC文件時(shí)是連讀連校驗(yàn)并且要轉(zhuǎn)譯,所以在讀取SREC時(shí)候會(huì)很慢,好處就是可以避免很多因?yàn)槲募e(cuò)誤導(dǎo)致功能異常。結(jié)構(gòu)如下:
Record type | Byte count | Address | Data | Checksum |
---|
SREC 格式文件由一系列ASCII文本記錄組成。這些記錄從左到右具有以下結(jié)構(gòu):
a.Record type——2個(gè)字節(jié)ASCII字符,第一個(gè)字符為‘S’(ASCII 0x53),第二個(gè)字符為ASCII數(shù)字的‘0’~‘9’(ASCII 0x30 到 0x39)
b.Byte count——兩個(gè)十六進(jìn)制數(shù)字(“00”至“FF”),表示記錄其余部分(地址 + 數(shù)據(jù) + 校驗(yàn)和)后面的字節(jié)數(shù)(十六進(jìn)制數(shù)字對(duì))。此字段的最小值為 3(16 位地址字段加 1 個(gè)校驗(yàn)和字節(jié)時(shí)為 2),最大值為 255(0xFF)。“00”/“01”/“02”為非法值。
c.Address——大端地址——4/6/8個(gè)16進(jìn)制的ASCII數(shù)字,取決于Record type的類(lèi)型
d.Data——數(shù)據(jù) ——2*n個(gè)16進(jìn)制的ASCII數(shù)字(n字節(jié)數(shù)據(jù))
e.Checksum ——2個(gè)16進(jìn)制的ASCII數(shù)字,即字節(jié)數(shù)、地址和數(shù)據(jù)字段的兩個(gè)十六進(jìn)制數(shù)字對(duì)所表示的值之和的補(bǔ)碼的最低有效字節(jié)。在C 編程語(yǔ)言中,總和通過(guò)以下方式轉(zhuǎn)換為校驗(yàn)和:0xFF - (sum & 0xFF)
圖片來(lái)源:https://en.wikipedia.org/wiki/SREC_(file_format)
ELF和SREC對(duì)比:
ELF轉(zhuǎn)換成SREC
如果使用官方的BootLoader,那么SREC格式是必不可少的,而SDK或者Vitis只能生成ELF文件,下面介紹幾種方式將ELF轉(zhuǎn)換成SREC。
方式一:通過(guò)XSCT Console
將elf轉(zhuǎn)換成SREC format,打開(kāi)Xilinx—> XSCT Console。
通過(guò)cd 命令進(jìn)入elf所在的目錄:
輸入:mb-objcopy -O srec app.elf app.srec
就會(huì)在ELF所在目錄生成SREC文件。
方式二:自動(dòng)轉(zhuǎn)換
在應(yīng)用上右擊,屬性,打開(kāi)屬性窗口。在圖中位置添加命令:
mb-objcopy -O srec {ProjName}.srec
這樣每次在APP編譯完成后就會(huì)將ELF自動(dòng)轉(zhuǎn)換成SREC(在ELF目錄里)。
方式二:使用命令手動(dòng)設(shè)置
使用下圖中兩個(gè)腳本,cd到elf文件位置后,使用mb-object命令完成轉(zhuǎn)換:
最后這種方式和方式一類(lèi)似,好處就是可以使用Win下的腳本調(diào)用,然后生成一鍵腳本。
加快啟動(dòng)
從上面的分析可知,在讀取SREC時(shí)候會(huì)進(jìn)行很多無(wú)關(guān)操作,導(dǎo)致啟動(dòng)時(shí)間大大增加,尤其是大應(yīng)用程序生成的ELF比較大(SREC也會(huì)比較大)的時(shí)候,這個(gè)時(shí)間肯定不能忍受的。
方式一:減少打印
在進(jìn)行調(diào)試的時(shí)候,BootLoader會(huì)有讀取進(jìn)程通過(guò)串口打印,這一操作方便調(diào)試的時(shí)候快速進(jìn)行問(wèn)題定位,完成調(diào)試的時(shí)候,可以將打印去掉,注釋掉下圖位置語(yǔ)句即可:
更改讀取方式
我們可以減少讀取過(guò)程中的轉(zhuǎn)譯和校驗(yàn)等過(guò)程,直接引導(dǎo)搬運(yùn)FLASH的elf文件至DDR,減少“中間商賺差價(jià)”的時(shí)間,可大大提高啟動(dòng)速度,相關(guān)的工程可以參考下面的鏈接。
https://github.com/henrikbrixandersen/elf-bootloader?_ga=2.259888963.835186866.1691390458-2003547133.1691118205
在eb-config.h文件中更改實(shí)際用到的FLASH地址以及匹配FLASH信息,主要就是修改對(duì)應(yīng)FLASH的opration指令以及dummy cycle,可以根據(jù)你使用的FLASH類(lèi)型查閱數(shù)據(jù)手冊(cè)。
MultiBoot實(shí)現(xiàn)注意點(diǎn)
在小應(yīng)用程序時(shí)候其實(shí)和純FPGA應(yīng)用一樣,沒(méi)什么大的區(qū)別,沒(méi)什么注意的。主要在大應(yīng)用程序:
大應(yīng)用程序主要包含上面幾個(gè)文件組成(根據(jù)自己需求可能有所不同)。考慮到升級(jí)過(guò)程中傳輸完成擦除FLASH后及突然斷電等特殊情況,所以需要考慮CRC error(突然斷電)、IDCODE error(升級(jí)包制作錯(cuò)誤)、Watchdog timer time-out error(擦除FLASH后未升級(jí)、突然斷電)三種情況。
Golden區(qū)無(wú)需變動(dòng),主要在M區(qū)的程序,在升級(jí)APP時(shí)候面對(duì)上面的情況怎么操作?因?yàn)槟壳暗腗ultiBoot機(jī)制只能在FPGA邏輯層進(jìn)行操作,如果APP程序錯(cuò)誤可能觸發(fā)不了回退機(jī)制。
這里提供一個(gè)思路,就是對(duì)M區(qū)APP增加CRC校驗(yàn),BootLoader在讀取APP的ELF或者SREC時(shí)如果CRC校驗(yàn)錯(cuò)誤,那么就破壞M區(qū)的FPGA程序(FLASH),讓FPGA程序啟動(dòng)時(shí)候觸發(fā)四種錯(cuò)誤的任何一種,就可以回退到Golden區(qū)再進(jìn)行升級(jí)。
總結(jié)
總結(jié)就下面的一張圖了:
關(guān)于這部分內(nèi)容,大家有什么建議或者經(jīng)驗(yàn),歡迎大家評(píng)論區(qū)留言討論~
-
應(yīng)用程序
+關(guān)注
關(guān)注
38文章
3287瀏覽量
57814 -
軟核
+關(guān)注
關(guān)注
0文章
14瀏覽量
15869 -
固化
+關(guān)注
關(guān)注
0文章
167瀏覽量
10512
原文標(biāo)題:一篇文章搞懂軟核(MicroBlaze)的固化和啟動(dòng)
文章出處:【微信號(hào):HXSLH1010101010,微信公眾號(hào):FPGA技術(shù)江湖】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論