【ARM裸机s5pv210 】重定位操作

makefile

final.bin: start.o led.o
    arm-linux-ld -Tlink.lds  -o start.elf $^     //link.lds为链接脚本,指定链接地址
    arm-linux-objcopy -O binary start.elf final.bin
    arm-linux-objdump -D start.elf > start_elf.dis
    
%.o : %.S
    arm-linux-gcc -o $@ $< -c -nostdlib

%.o : %.c
    arm-linux-gcc -o $@ $< -c -nostdlib

clean:
    rm *.o *.elf *.bin   -f

link.lds

SECTIONS
{
    . = 0xd0024000;   // . 表示程序自以为的地址,本行指定了链接地址
    
    .text : {
        start.o
        * (.text)    //表示start.o中的.text段放在前面,然后再把所有的文件的.text段放进来
    }
            
    .data : {
        * (.data)    //把所有的文件的.data段放进来
    }
    
    bss_start = .;    //把现在的地址赋给一个变量,供清bss段时使用
    .bss : {
        * (.bss)    //把所有的文件的.bss段放进来
    }
    
    bss_end  = .;    //把现在的地址赋给一个变量,供清bss段时使用
}

start.S

#define WTCON        0xE2700000
#define SVC_STACK    0xd0037d80

.global _start
_start:

    // 第1步:关看门狗(向WTCON的bit5写入0即可)
    ldr r0, =WTCON
    ldr r1, =0x0
    str r1, [r0]
    
    // 第2步:设置SVC栈    满减栈,满的意思是入栈先移动指针再填入数据,减的意思是栈从高到低用
    ldr sp, =SVC_STACK
    
    // 第3步:开/关icache
    mrc p15,0,r0,c1,c0,0;            // 读出cp15的c1到r0中  mrc和mcr是协处理器的相关汇编指令
    //bic r0, r0, #(1<<12)            // bit12 置0  关icache
    orr r0, r0, #(1<<12)            // bit12 置1  开icache
    mcr p15,0,r0,c1,c0,0;
    
    
    // 第4步:重定位
    adr r0, _start          // 将运行地址加载到r0    0xd0020010
    ldr r1, =_start         // 将链接地址加载到r0    0xd0024000
    
    ldr r2, =bss_start    // bss_start在链接脚本中定义
    cmp r0, r1            // 比较_start的运行时地址和链接地址是否相等
    beq clean_bss        // 如果相等说明不需要重定位,所以跳过copy_loop,直接到clean_bss
                        // 如果不相等说明需要重定位,那么直接执行下面的copy_loop进行重定位
                        // 重定位完成后继续执行clean_bss。

// 将运行地址开始所有的数据复制到链接地址开始的地方(即复制.text段和.data段),链接地址此时也在增加,当增加到bss段该开始的地方停止,并且开始清bss段
copy_loop:
    ldr r3, [r0], #4    // 源
    str r3, [r1], #4    // 目的   这两句代码就完成了4个字节内容的拷贝
    cmp r1, r2            // r1和r2都是用ldr加载的,都是链接地址,所以r1不断+4总能等于r2
    bne copy_loop

// 清bss段,其实就是在链接地址处把bss段全部清零
clean_bss:
    ldr r0, =bss_start                    
    ldr r1, =bss_end
    cmp r0, r1                // 如果r0等于r1,说明bss段为空,直接下去
    beq run_on_dram            // 清除bss完之后的地址
    mov r2, #0
clear_loop:
    str r2, [r0], #4        // 先将r2中的值放入r0所指向的内存地址(r0中的值作为内存地址),
    cmp r0, r1                // 然后r0 = r0 + 4
    bne clear_loop

run_on_dram:    
    // 长跳转到led_blink开始第二阶段
    ldr pc, =led_blink                // ldr指令实现长跳转,跳转到完全复制过的链接地址处开始执行

    b .

led.c

#define GPJ0CON        0xE0200240
#define GPJ0DAT        0xE0200244
void delay(void)
{
    unsigned int cnt = 900000;
    while(cnt)
    {
        cnt--;
    }
    
}

void led_blink(void)
{
    unsigned int *p1 = (unsigned int *)GPJ0CON;
    unsigned int *p2 = (unsigned int *)GPJ0DAT;
    *p1 = 0x11111111;
    
    while(1)
    {
        *p2 = ~(1<<3);
        delay();
        *p2 = ~(1<<4);
        delay();
        *p2 = ~(1<<5);
        delay();
        
        
    }
    
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值