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

发布者:DreamyMoon最新更新时间:2015-10-14 来源: eefocus关键字:ARM汇编  linker  script 手机看文章 扫描二维码
随时随地手机看文章
       在写GNU ARM汇编下的linker script之前,还是有必要看一下ldr指令,以及ldr和adr伪指令.

        ldr指令:

        LDR       load word into a register                                         Rd <- mem32[address]

        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.     ldr pc, _main     @a  
  2.     ldr pc, =_main    @b  
  3.     ldr pc,main       @c  
  4.     ldr pc,=main      @d  
  5.     adr pc,_main      @e  
  6.     adr pc,main       @e  
  7.   
  8. _main:  .word main  

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

 

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

 

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

 

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

 

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

 

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

 

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

 

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

 

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

 

 

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

 

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

 

[cpp] view plaincopy
 
  1.  128:   e28fff69    add pc, pc, #420    ; 0x1a4  
  2.   
  3. 0000012c <_main>:  
  4.  12c:   000002d4    .word   0x000002d4  
  5. ......  
  6.   
  7. 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.   
  5.   
  6.   
  7. SECTIONS{  
  8.   
  9.   
  10.  . = 0x30000000;    
  11.     .text ALIGN(4): {*(.text)}    
  12.     .rodata ALIGN(4) : {*(.rodata)}    
  13.     .data ALIGN(4) : {*(.data)}    
  14.     .bss ALIGN(4) : {*(.bss) *(COMMON)}    
  15. }  

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

[page]

 

[cpp] view plaincopy
 
  1. ldr pc,=on_sdram    
  2.    
  3. n_sdram:    
  4.     bl clearsram    

        为了证实后面跑的代码是sdram中的,在跳到sdram后将sram都清除了.

 

 

2.

 

[cpp] view plaincopy
 
  1. OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")  
  2. OUTPUT_ARCH(arm)  
  3. ENTRY(_start)  
  4.   
  5.   
  6.   
  7. SECTIONS{  
  8.   
  9.   
  10.  . = 0x00000000;    
  11.     .text ALIGN(4): {*(.text)}    
  12.     .rodata ALIGN(4) : {*(.rodata)}    
  13.     .data ALIGN(4) : {*(.data)}    
  14.     .bss ALIGN(4) : {*(.bss) *(COMMON)}    
  15. }  
[cpp] view plaincopy
 
  1.   

vma的地址用的是0x00000000,跳转这样写:

 

[cpp] view plaincopy
 
  1. ldr r0,=on_sdram  
  2. add r0,r0,#0x30000000  
  3. mov pc,r0  
  4.   
  5.    
  6. n_sdram:    
  7.     bl clearsram    

        通过两种链接脚本的对比,这个vma lma,绝对跳转的概念那应该就很清楚了.太晚了,明天给出一个有意思的链接脚本.

 

       今天晚上有时间,补充一个有点意思的链接脚本:

 

[cpp] view plaincopy
 
  1. OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")  
  2. OUTPUT_ARCH(arm)  
  3. ENTRY(_start)  
  4.   
  5.   
  6. SECTIONS{  
  7.   .text1  0x00000000 :   
  8.     {  
  9.     start.o(.text)  
  10.   
  11.     }  
  12.   .text2  0x30000000 : AT(2048)   
  13.     {  
  14.     main.o(.text)  
  15.     }  
  16.   
  17. }  

 

        编译出来的二进制文件会大很多,因为.text2段的LMA地址是2048.

        bin档大小为2184

        arm-linux-readelf -a sram2sdram_elf,结果是:

Program Headers:
  Type           Offset   VirtAddr   PhysAddr   FileSiz MemSiz  Flg Align
  LOAD           0x008000 0x00000000 0x00000000 0x003f0 0x003f0 R E 0x8000
  LOAD           0x010000 0x30000000 0x00000800 0x00088 0x00088 R E 0x8000


 Section to Segment mapping:
  Segment Sections...
   00     .text1 
   01     .text2 

第二段.text2的大小是0x88,而.text1的大小是ox3f0,必须要小于2048才行.

        你应该也发现了:2048+0x88 = 2184   这就是最后的bin档大小

 

       可以vi 最后的bin档看一下, 输入%!xxd后,效果如下:

       从0x3f0到0x800都是0000,这一段都是gap.

        既然是这样,我们的拷贝代码也只能拷贝sram中2048到4096这一部分就可以了.

 

        有这三个linker script做对比,对linker script和代码跳转可以说理解的比较透了.

关键字:ARM汇编  linker  script 引用地址:GNU ARM汇编--(十三)GNU ARM汇编下的linker script

上一篇:GNU ARM汇编--(十一)小结一下
下一篇:GNU ARM汇编--(十二)arm汇编指令的B真的那么简单吗?

推荐阅读最新更新时间:2024-03-16 14:35

ARM汇编-MOV PC,LR
MOV PC,LR BL NEXT ;跳转到子程序 ......... ;NEXT处执行 NEXT .......... MOV PC,LR ;从子程序返回 这里的BL是跳转的意思,LR(R14)保存了返回地址 PC(R15)是当前地址,把LR给PC就是从子程序返回 这里有一下总结 首先 1.SP(R13) LR(R14)PC(R15) 2.lr(r14)的作用问题,这个lr一般来说有两个作用: 1》.当使用bl或者blx跳转到子过程的时候,r14保存了返回地址,可以在调用过程结尾恢复。 2》.异常中断发生时,这个异常模式特定的物理R14被设置成该异常模式将要返回的地址。 另外注意pc,在调试的时候显示的是当前指令地址,而用mov
[单片机]
ARM汇编中的立即数
同事遇到这样一个问题: 在eVC编译出的汇编代码中我看到这样的语句: mov r2, #0xFF, 28 和 orr r2, r2, #0xB 这样得到的结果时 r2=#0xffb , 他试图更直接一点优化成一句:MOV r2,#0xffb 但是这样之后编译就出了问题:error A0092: no immediate rotate operand can be created: 4091 ------------------------------------我是无辜的分割线-------------------------------- 在 mov r2,#0xffb 这句中,不是MOV的用法出错,而是立即
[单片机]
ARM汇编指令集之七——数据交换指令
1、SWP指令 SWP指令的格式为: SWP{条件} 目的寄存器,源寄存器1, SWP指令用于将源寄存器2所指向的存储器中的字数据传送到目的寄存器中,同时将源寄存器1中的字数据传送到源寄存器2所指向的存储器中。显然,当源寄存器1和目的寄存器为同一个寄存器时,指令交换该寄存器和存储器的内容。 指令示例: SWP R0,R1, ;将R2所指向的存储器中的字数据传送到R0,同时将R1中的字数据传送到R2所指向的存储单元。 SWP R0,R0, ;该指令完成将R1所指向的存储器中的字数据与R0中的数据交换。 2、SWPB指令 SWPB指令的格式为: SWP{条件}B 目的寄存器,源寄存器1,
[单片机]
linux下ARM汇编程序的调试
最近在学习 ARM汇编语言 ,但是ARM不像x86,可以很方便的调试。不过还好有虚拟机,而且还有GDB这样万能的调试器。Google了很久,又结合自己的实践,终于成功的调试了ARM的汇编,在此向对ARM 嵌入式开发 感兴趣的同学分享一下。 首先说明需要的工具 1. QEMU 这个模拟器不仅可以模拟x86,还可以模拟ARM和MIPS,可谓强大 2. GDB(源代码) 不用说了,万能调试器,但是需要从源代码编译 3. arm-linux交叉工具链 我用的是自己开发板配套的3.4版本,去随便下载一个就可以(不过貌似uclinux被GFW了 ) 先编译GDB,设置target为ARM平台: $ ./c
[单片机]
ARM汇编条件码
操作码 条件码助记符 标志 含义 0000 EQ Z=1 相等 0001 NE(Not Equal) Z=0 不相等 0010 CS/HS(Carry Set/High or Same) C=1 无符号数大于或等于 0011 CC/LO(Carry Clear/LOwer) C=0 无符号数小于 0100 MI(MInus) N=1 负数 0101 PL(PLus) N=0 正数或零 0110 VS(oVerflow set) V=1 溢出 0111 VC(oVerflow clear) V=0 没有溢出 1000 HI(HIgh) C
[单片机]
ARM汇编指令学习(0) ARM 汇编语言程序格式
ARM 汇编语言程序格式 可执行映象文件的格式: *.axm *.bin *.elf *.hex 代码段示例: 汇编语言源程序的基本结构: AREA Init,CODE,READONLY ENTRY Start LDR R0,=0x3FF50000 LDR R1,0xFF STR R1, LDR R0,=0x3FF5008 LDR R1,0x01 STR R1, END Arm 体系结构3种执行流程: 1 顺序执行 2 跳转执行 3 异常中断执行 Arm 子程序调用使用命令 BL 子程序名称 子程序调用示例: AREA I
[单片机]
arm汇编学习(五)
新增个手写GNU语法arm的方法,以后可以狂逆狂写 hello.S文件 .data msg: .ascii Hello, ARM!\n len = . - msg .text .globl _start _start: /* syscall write(int fd, const void *buf, size_t count) */ mov %r0, $1 /* fd - stdout */ ldr %r1, =msg /* buf - msg */ ldr %r2, =len /* count - len(msg) */ mov %r7, $4 /* write is syscall #4 */ swi $0 /* invo
[单片机]
GNU ARM汇编--(十一)小结一下
对s3c2440的一部分做了学习,更深的了解了汇编以及arm体系,还有中断,调试底层汇编过程的最大感触就是仔细阅读datasheet,因为那是英文文档. 前面写的都是嵌入式底层比较基础的东西,后面要做的内容: 1.先裸机搞定iic,iic包括用控制器实现的和用GPIO模拟的,这样可以加深时序的理解. 2.spi也想看一下,虽然2440有spi,但是没有外设,这个比较麻烦. 3.搞清楚嵌入式系统中的存储系统,主要是内存(MMU)和flash 4.在前面的基础上可以整体上把握bootloader了 5.因为对linux的文件系统和设备模型有一定的了解,在3的基础上可以开始
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved