色婷婷AⅤ一区二区三区|亚洲精品第一国产综合亚AV|久久精品官方网视频|日本28视频香蕉

          新聞中心

          EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > arm 匯編中 .word偽指令的用法

          arm 匯編中 .word偽指令的用法

          作者: 時(shí)間:2016-11-20 來(lái)源:網(wǎng)絡(luò) 收藏
          PS:在u-boot源碼時(shí)遇到_armboot_start、_bss_start等這些變量,不知道指向什么地址,于是查了一下,弄清了ARM匯編中“.word”這個(gè)偽指令是什么意思了,感覺(jué)自己很菜。借鑒一下網(wǎng)友帖子的內(nèi)容,關(guān)鍵在帖子最后的總結(jié):

          匯編和C引用變量的不同:匯編是“絕對(duì)”引用,即沒(méi)有指針的概念,引用得到的就是值;c語(yǔ)言是“間接”引用,相當(dāng)于指針的概念,引用地址變量,得到的就是該變量所指的內(nèi)容值。

          本文引用地址:http://cafeforensic.com/article/201611/318796.htm

          感謝原作者,以下為原帖:

          aaronwong: u-boot中代碼的疑問(wèn)(_armboot_start與_start)?
          ---------------------------
          我使用的是u-boot-1.3.0-rc2。在cpu/pxa/start.S中,有如下的標(biāo)號(hào)定義:
          _TEXT_BASE:
          .word TEXT_BASE/*uboot映像在SDRAM中的重定位地址,我設(shè)置為0xa170 0000 */

          .globl _armboot_start
          _armboot_start:
          .word _start/*_start是程序入口,鏈接完畢它的值應(yīng)該是0xa170 0000=TEXT_BASE*/
          /*這句話的意思應(yīng)該是在_armboot_start標(biāo)號(hào)處,保存了_start的值,也就是說(shuō),_armboot_start是存放_(tái)start的地址,該地址對(duì)應(yīng)的存儲(chǔ)單元內(nèi)容是0xa1700000*/
          /*
          * These are defined in the board-specific linker script.下面的定義與上面應(yīng)該是一個(gè)意思。
          */
          .globl _bss_start
          _bss_start:
          .word __bss_start
          ======================
          按照上面的理解,__bss_start是uboot的bss段起始地址,那么uboot映像的大小就是__bss_start- _start;在relocate代碼段中計(jì)算uboot的大小時(shí),也體現(xiàn)了這一點(diǎn)。
          實(shí)際上,_armboot_start并沒(méi)有實(shí)際意義,它只是在"ldrr2, _armboot_start"中用來(lái)尋址_start的值而已,_bss_start也是一樣的道理,真正有意義的應(yīng)該是_start和__bss_start本身。
          但是,令我不解的是,在C入口函數(shù)start_armboot()中(對(duì)應(yīng)文件為lib_arm/board.c),有如下代碼:
          void start_armboot (void)
          {
          .........
          gd = (gd_t*)(_armboot_start - CFG_MALLOC_LEN - sizeof(gd_t));//第一句話
          ..........
          monitor_flash_len = _bss_start - _armboot_start;//第二句話
          ...............
          mem_malloc_init (_armboot_start - CFG_MALLOC_LEN);//第三句話
          ..........
          }
          ==============================================
          按照上面的理解,_armboot_start與_bss_start都是沒(méi)有實(shí)際意義的,它們只是一個(gè)地址,有實(shí)際意義的是地址中的內(nèi)容_start和__bss_start(雖然也還是地址)。象第一句話,其“意圖”很明顯,是把gd作為全局?jǐn)?shù)據(jù)結(jié)構(gòu)體的指針,并初始化為“SDRAM中的uboot起始地址(即TEXT_BASE)-CFG_MALLOC_LEN-全局?jǐn)?shù)據(jù)結(jié)構(gòu)體大小”。
          要實(shí)現(xiàn)這個(gè)“意圖”,應(yīng)該是寫成:gd= (gd_t*)(_start - CFG_MALLOC_LEN - sizeof(gd_t));或者gd= (gd_t*)(TEXT_BASE- CFG_MALLOC_LEN - sizeof(gd_t));才對(duì)阿?用_armboot_start來(lái)作運(yùn)算應(yīng)該是沒(méi)有任何意義才對(duì)?。?br />第二句話也是一樣的道理,它的意圖是要計(jì)算u-boot映像的大小,應(yīng)該寫成__bss_start- _start才對(duì)阿?
          我使用readelf工具查看編譯所得到的uboot映像文件得到信息如下:
          [aaronwong@localhost build]$ readelf -s u-boot|grep _start
          1018: a1700048 0 NOTYPE GLOBAL DEFAULT 1 _bss_start
          1083: a1700044 0 NOTYPE GLOBAL DEFAULT 1 _armboot_start
          1142: a1700000 0 NOTYPE GLOBAL DEFAULT 1 _start
          1197: a171b070 0 NOTYPE GLOBAL DEFAULT ABS __bss_start
          上面我刪除了與該討論無(wú)關(guān)的包含“_start""t的標(biāo)號(hào)信息。
          顯然,我前面的理解應(yīng)該是正確的(_start=TEXT_BASE=0xa1700000)。那么u-boot源代碼中的monitor_flash_len=_bss_start -_armboot_start=0xa1700048 - 0xa1700044 = 4,有什么意義??
          迷茫中,期盼大蝦指點(diǎn)迷津,謝謝~!??!


          eltshan: [Re: aaronwong]
          -----------------
          1018: a1700048 0 NOTYPE GLOBAL DEFAULT 1 _bss_start
          1083: a1700044 0 NOTYPE GLOBAL DEFAULT 1 _armboot_start
          1142: a1700000 0 NOTYPE GLOBAL DEFAULT 1 _start
          1197: a171b070 0 NOTYPE GLOBAL DEFAULT ABS __bss_start

          我想:
          _start所在的地址是a1700000,
          _armboot_start 所在的地址是a1700044,
          那么 根據(jù)這句:
          _armboot_start: .word _start
          所以_armboot_start的值應(yīng)該是a1700000

          所以
          monitor_flash_len = _bss_start - _armboot_start = a171b070 - a1700000 = 1b070
          而不是你說(shuō)的 = 4

          以上個(gè)人意見(jiàn).


          aaronwong: [Re: eltshan]
          -------------------
          謝謝,eltshan!你的理解是正確的,不過(guò)我看了之后還是沒(méi)能想得很明白,因?yàn)槲以谙?,按你所說(shuō),那么_start的值應(yīng)該是多少呢?難道是“breset”這條指令的機(jī)器碼?所以我對(duì)ELF格式的u-boot映像文件作了反匯編,分析之后終于找到了癥結(jié)所在。以下是部分分析過(guò)程,首先是反匯編:
          arm-iwmmxt-linux-gnueabi-objectdump -D u-boot > u-boot.s
          并提取了monitor_flash_len = _bss_start - _armboot_start;這條語(yǔ)句相關(guān)的反匯編代碼如下:
          ==============================
          a1700044 <_armboot_start>:
          a1700044: a1700000 .word 0xa1700000

          a1700048 <_bss_start>:
          a1700048: a171b070 .word 0xa171b070

          a171b070 :
          a171b070: 00000000 .word 0x00000000

          .....
          a1700f40: e59f41d0 ldr r4, [pc, #464] ; a1701118
          //r4=[a1701118]=a1700044
          .....
          a1700f7c: e59f3198ldr r3, [pc, #408] ; a170111c
          //r3=[a1700044]=a1700048
          a1700f80: e5942000 ldrr2, [r4]
          //r2=[a1700044]=a1700000
          a1700f84:e59f4194 ldr r4, [pc,#404] ; a1701120
          //r4=[a1701120]=a1719d24
          a1700f88: e5933000 ldrr3, [r3]
          //r3=[a1700048]=a171b070
          a1700f8c: e0623003 rsb r3, r2, r3
          //r3= r3-r2 = a171b070-a1700000 = 1b070;
          a1700f90: e59f218cldr r2, [pc, #396] ; a1701124
          //r2=[a1701124]=a171b070
          a1700f94: e5823000 strr3, [r2]
          //monitor_flash_len=[r2]=r3=1b070
          ......

          a1701118: a1700044 .word 0xa1700044
          a170111c: a1700048.word 0xa1700048
          a1701120: a1719d24 .word 0xa1719d24
          a1701124: a171b070 .word 0xa171b070
          ========================================
          上面//是我自己的注釋。這表明,你的理解的確是正確的。
          經(jīng)過(guò)這個(gè)過(guò)程之后,我終于認(rèn)識(shí)到自己的誤解在哪里了。原來(lái),我是把"匯編語(yǔ)言中LDR偽指令對(duì)符號(hào)的引用"與"C語(yǔ)言中對(duì)匯編程序中符號(hào)/常量/變量的引用"搞混淆了。我想說(shuō)明以下幾點(diǎn):

          (1) readelf以及u-boot.map和System.map所給出的符號(hào)表中符號(hào)的值,實(shí)際上是表示符號(hào)所在的地址,而不是指符號(hào)本身的值。

          (2) 匯編語(yǔ)言中沒(méi)有指針的概念,因此對(duì)符號(hào)的引用是"赤裸裸"的。例如:
          ==========
          .globl _armboot_start
          _armboot_start: .word _start
          ldr r2, _armboot_start
          ==========
          實(shí)際上反匯編以后是:
          ============
          a1700044 <_armboot_start>:
          a1700044: a1700000 .word 0xa1700000
          a1700074: e51f2038ldr r2, [pc, #-56] ; a1700044 <_armboot_start>
          ============
          也就是說(shuō),_armboot_start是一個(gè)地址0xa1700044,其中的內(nèi)容是0xa1700000,上面對(duì)_armboot_start的引用是直接將其替換為其表示的地址0xa1700044,而非其中的內(nèi)容0xa1700000。這就是"赤裸裸"的引用。

          (3) C語(yǔ)言則不同,對(duì)變量/符號(hào)/常量的引用必須要通過(guò)地址來(lái)尋址,不管是全局變量還是局部變量,不同的是局部變量在生命期結(jié)束后,所占的地址空間會(huì)被釋放而已。即使是函數(shù)調(diào)用時(shí)的參數(shù)傳遞,雖然是將實(shí)參的值"拷貝"給形參,但"拷貝"的過(guò)程也是通過(guò)實(shí)參和形參的地址來(lái)對(duì)兩者進(jìn)行訪問(wèn)的。
          所以,在C語(yǔ)言中的 "monitor_flash_len =_bss_start - _armboot_start" 這句話中對(duì)_armboot_star的引用,實(shí)際上是把它用作了指針,把它作為訪問(wèn)對(duì)象的地址來(lái)使用,通過(guò)這個(gè)地址即a1700044 來(lái)訪問(wèn)對(duì)應(yīng)存儲(chǔ)空間所存放的內(nèi)容亦即0xa1700000,_bss_start也是同樣的道理。所以這句話實(shí)際上是monitor_flash_len=[a1700048]-[0xa1700044]=a171b070-a1700000 = 1b070,這樣就得到了正確的結(jié)果。

          現(xiàn)在,我們?cè)倩卮鹱钋懊娴膯?wèn)題:_start的值是什么?_start表示地址0xa1700000,在匯編語(yǔ)言中,對(duì)_start的"絕對(duì)引用"(這里是與用相對(duì)尋址進(jìn)行跳轉(zhuǎn)進(jìn)行區(qū)別)就是將其替換為0xa1700000,但其中存放的內(nèi)容的的確確就是"b reset"這條指令的機(jī)器碼,所以如果在C語(yǔ)言中引用_start,得到的結(jié)果反而就是這個(gè)指令的機(jī)器碼了。其實(shí)這個(gè)問(wèn)題很簡(jiǎn)單,只是和C語(yǔ)言的引用攪在一起,一些概念被偷換了而已。


          關(guān)鍵詞: arm匯編中word偽指

          評(píng)論


          技術(shù)專區(qū)

          關(guān)閉