×
嵌入式 > 嵌入式开发 > 详情

GNU ARM汇编--(十三)GNU ARM汇编下的linker script

发布时间:2020-09-01 发布时间:
|
在写GNU ARM汇编下的linker script之前,还是有必要看一下ldr指令,以及ldr和adr伪指令.

ldr指令:

LDR load word into a register Rd

ldr伪指令:

LDR Rd, =constant

LDR load constant pseudoinstruction Rd=32-bit constant

adr伪指令:

ADR Rd, label

ADR load address pseudoinstruction Rd=32-bit relative address

我们的程序从nandflash启动,运行在s3c2440的4K大小的SRAM中,linker script的.text放在. = 0x00000000;处.

测试代码如下:

[cpp]view plaincopy
  1. ldrpc,_main@a
  2. ldrpc,=_main@b
  3. ldrpc,main@c
  4. ldrpc,=main@d
  5. adrpc,_main@e
  6. adrpc,main@e
  7. _main:.wordmain

main中放置的一个流水灯.分别测试这六种情况:

a.成功跳转,分析下反汇编:

[cpp]view plaincopy
  1. 128:e51ff004ldrpc,[pc,#-4];12c<_main>
  2. 0000012c<_main>:
  3. 12c:000002d4.word0x000002d4
  4. ......
  5. 000002d4
    :
0x00000128+8-4=0x0000012c ldr指令将地址为0x0000012c的word(0x000002d4)放到pc中,那么就跳转到main了.

b.无法跳转,分析下反汇编:

[cpp]view plaincopy
  1. 128:e59ff244ldrpc,[pc,#580];374
  2. 0000012c<_main>:
  3. 12c:000002d4.word0x000002d4
  4. ......
  5. 000002d4
    :
0x00000128+580+8=0x00000374 ldr指令将地址为0x00000374的word(374:0000012c.word0x0000012c)放入pc中,自然无法正确跳转

c.无法跳转,分析下反汇编:

[cpp]view plaincopy
  1. 128:e59ff1a4ldrpc,[pc,#420];2d4
  2. 0000012c<_main>:
  3. 12c:000002d4.word0x000002d4
  4. ......
  5. 000002d4
    :
0x00000128+420+8=0x000002d4 ldr指令将地址为0x000002d4的word放入pc中,自然也无法跳转

d.成功跳转,分析下反汇编:

[cpp]view plaincopy
  1. 128:e59ff244ldrpc,[pc,#580];374
  2. 0000012c<_main>:
  3. 12c:000002d4.word0x000002d4
  4. ......
  5. 000002d4
    :
  6. ......
  7. 374:000002d4.word0x000002d4
0x00000128+580+8=0x00000374 ldr指令将地址为0x00000374的word(000002d4)放入pc中,成功跳转
e.无法跳转,分析下反汇编:

[cpp]view plaincopy
  1. 128:e24ff004subpc,pc,#4;0x4
  2. 000012c<_main>:
  3. 12c:000002d4.word0x000002d4
pc=pc+8-4=0x0000012c 所以无法跳转

f.成功跳转,分下下反汇编:

[cpp]view plaincopy
  1. 128:e28fff69addpc,pc,#420;0x1a4
  2. 0000012c<_main>:
  3. 12c:000002d4.word0x000002d4
  4. ......
  5. 000002d4
    :

pc=pc+420+8=0x00000128+420+8=0x00002d4 所以成功跳转.

完全理解相对跳转和绝对跳转是为了后面的linker script做准备的,linker script的理论只是可以看下gnu.org的官方文档,下面才开始这次的正题.

给出两种linker script的写法:

1.

[cpp]view plaincopy
  1. OUTPUT_FORMAT("elf32-littlearm","elf32-littlearm","elf32-littlearm")
  2. OUTPUT_ARCH(arm)
  3. ENTRY(_start)
  4. SECTIONS{
  5. .=0x30000000;
  6. .textALIGN(4):{*(.text)}
  7. .rodataALIGN(4):{*(.rodata)}
  8. .dataALIGN(4):{*(.data)}
  9. .bssALIGN(4):{*(.bss)*(COMMON)}
  10. }

vma给的是0x30000000,那么跳转就这么跳:



『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
基于嵌入式系统的CAN总线网络通信流程设计