<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設計應用 > 基于Nand Flash的VIVI裝載器的分析與改進

          基于Nand Flash的VIVI裝載器的分析與改進

          作者: 時間:2016-12-02 來源:網(wǎng)絡 收藏


          3.2 程序拷貝的改進

          在嵌入式系統(tǒng)中,映像文件都是存儲在Flash存儲器等一些非易失性器件中的,而在運行時,映像文件中的RW段必須重新裝載到可瀆寫的RAM中。這就涉及到映像文件的加載時域和運行時域:加載時域就是指映像文件燒入nash中的狀態(tài),
          也就是映像文件運行之前的地址;運行時域是指映像文件執(zhí)行時的狀態(tài),針對本文提到的Nand Flash啟動方式可以這么理解加載時域與運行時域:加載時域的起始地址從(映射后的內(nèi)部SRAM處)0x00000000開始,運行時域的地址從0x33f00000開始。由于加載時域與運行時域的地址不同,從加載時域到運行時域的轉(zhuǎn)換要由系統(tǒng)引導程序完成,所以VIVI必須進行數(shù)據(jù)和代碼的拷貝及程序跳轉(zhuǎn)工作,以完成從加載時域到運行時域的轉(zhuǎn)換。

          VIVI的拷貝首先要確定拷貝的起始地址和目標地址,還要確定要拷貝多少代碼。在此筆者對所搬運代碼量進行了改進,下面是改進前的代碼:

          ldr r0,=VIVI_RAM_BASE
          <2>mov rl,#Ox0
          <3>mov r2,#0x20000
          <4>bl nand_read_ll

          其中:第<1>行:獲取VIVI在RAM中的基地址VIVI_RAM_BASE,也是運行時域的首地址。第<2>行:獲取VIVI映像在Flash中的起始地址OxO。第<3>行:獲取拷貝的代碼量0x20000。第<4>行:跳轉(zhuǎn)到nand_read_ll函數(shù),它是用C語言寫的拷貝函數(shù)(略)。此時寄存器rO,rl,r2是傳遞給函數(shù)nand_read_ll的三個參數(shù)。

          程序這樣設計的缺點是不論VIVI映像有多大,它都會拷貝128KB的代碼量,這樣造成時間及空間的浪費或者拷貝不完整,為此筆者對上述代碼進行改進:

          [1]ldr r0,=VIVI_RAM_BASE
          [2]ldr r2,=vivi_end
          [3]sub r2,r2,r0
          [4]mov r2,r2,lsr #9
          [5]mov r2,r2,lsr #9
          [6]add r2,r2, #0x200
          [7]mov r1,#OxO
          [8]bl nand_read_ll

          代碼中用到了外部變量vivi_end,它是在鏈接腳本文件中定義的,是VIVI映像運行時域的末地址,在此代碼中使用前要用如下語句進行聲明:

          .extern vivi_end

          其中:第[l]、[7]、[8]行的解釋分別與未改進前的第、<2>、<4>行。第[2]行:獲取VIVI映像運行時域的末地址。第[3]行:獲取拷貝的真實代碼量。第[4],[5],[6]行:上文論述到Nand Flash是按頁進行讀寫的,本文用到的Nand Flash每頁有(512+16)Byte,實際用于存儲映像文件的是每頁的前512Byte。所以需要對上述“真實代碼量”進行調(diào)整,把它調(diào)整為整數(shù)頁大小,它的大小必須是頁對齊的。首先把它的低9位調(diào)整為0,即是把代碼量不足512Byte的部分清零,然后再加1頁(page)大小(Ox200)以保證VlVI數(shù)據(jù)的完整性,這就是第[4],[5],[6]行的作用。對此代碼進行改進后,拷貝的代碼量更接近實際的代碼量,拷貝的冗余代碼量不會達到1頁(page)的大小。對于其它類型的Nand Flash可以根據(jù)頁的大小進行類似的改進。

          3.3 程序的跳轉(zhuǎn)

          針對本文所論及的系統(tǒng),當系統(tǒng)加電或復位后,首先Nand Flash中的前4KB由硬件拷貝到位于0x40000000處的大小為4KB的內(nèi)部SRAM中,然后此SRAM被映射到BankO處(Ox00000000)。PC從0x00000000處取指令執(zhí)行。當遇到B或BL等跳轉(zhuǎn)指令時,它會跳到當前地址加上一個偏移量的位置,它們屬于相對跳轉(zhuǎn),它們的跳轉(zhuǎn)范圍是±32 MB,這使得B或BL指令不依賴于代碼的存儲位置,此時這些地址為加載時域的地址。在嵌入式系統(tǒng)中,還有一種實現(xiàn)長跳轉(zhuǎn)的方式,就是使用ldr指令,它町以實現(xiàn)程序的絕對跳轉(zhuǎn),跳轉(zhuǎn)范圍為4G空間。

          VIVI中實現(xiàn)程序跳轉(zhuǎn)的代碼為:

          @jump to rain
          ldr r1.=on_the_ram
          <2>add pc, r1, #O
          <3>1: blb@ infinite loop
          <4>on_the_ram:

          上文提到了加載時域與運行時域的概念,此時第<1>行獲取的on_the_ram的地址就是運行時域的地址,此地址由上述鏈接腳本文件決定,第<2>行跳轉(zhuǎn)到SDRAM中的on_the_ram處。

          為了進一步深入說明程序的跳轉(zhuǎn),可以利用VIVI的反匯編文件來查看上述代碼的反匯編情況。現(xiàn)分別給出此段代碼下載時域和運行時域的存儲布局。

          下載時域此段代碼在內(nèi)部SRAM中的存儲布局為:

          <1>000000dc: e59f1278 ldr rl,[pc,#278];0x35c
          <2>000000eO: e281f000 add pc,rl,#0
          <3>000000e4: eafffffeb Oxe4
          <4>000000e8
          ...
          <5>00000358:000055aa andeq r5,rO,r10,lsr #ll
          <6>0000035c: 33f000e8 mvnccs rO.#240

          運行時域此段代碼被拷貝到SDRAM中的存儲布局:

          [1]33fOOOdc: e59f1278 ldr r1,[pc,#278]
          [2133f000eO: e281f000 add pc,rl,#0
          [3] 133mooe4: eaffffffeb Oxe4
          [4]33fOOOe8
          ...
          [5]33f00358: 000055aa andeqr5,rO,r10,lsr #11
          [6]33f0035c: 33fOOOe8 mvnccs rO,#240

          系統(tǒng)加電或復位從基地址0x00000000運行到上述代碼的第<1>行時,r1獲得地址0x0000035c處的值,從第<6>行知道此地址處的值為33f000e8,運行到第<2>行處,進行跳轉(zhuǎn),由于此時程序映像已經(jīng)拷貝到SDRAM中,程序就跳到了運行時域此段代碼的第[4]行處斷續(xù)執(zhí)行下面的程序,從而實現(xiàn)了程序從SRAM到SDRAM的跳轉(zhuǎn)。

          4 結(jié)語

          Boot Loader的設計是嵌入式系統(tǒng)中的重要環(huán)節(jié),它為系統(tǒng)的正常啟動完成了一系列的初始化工作,設計一個簡單高效功能強大的Boot Loader是嵌入式系統(tǒng)設計中一項重要工作。

          本文的創(chuàng)新點:深入研究和分析了VIVI如何實現(xiàn)自己拷貝以及拷貝后如何實現(xiàn)程序的跳轉(zhuǎn),并給出了原理性和實驗性的說明;<2>對程序進行了改進,主要體現(xiàn)在拷貝的代碼量上及VIVI的鏈接腳本上,使程序設計更合理。

          上一頁 1 2 下一頁

          關鍵詞: NandFlashVIVI裝載

          評論


          技術專區(qū)

          關閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();