主题 : S3C2440 搬移代码后跳转的疑惑 复制链接 | 浏览器收藏 | 打印
级别: 新手上路
UID: 50148
精华: 0
发帖: 2
金钱: 10 两
威望: 2 点
贡献值: 0 点
综合积分: 4 分
注册时间: 2011-06-19
最后登录: 2012-05-15
楼主  发表于: 2011-06-28 23:54

 S3C2440 搬移代码后跳转的疑惑

弱弱请教大家一个简单的问题,但是我没有想通哦
在S3C2440的有4kB的stepingstone,就是把放在Nand flash的从0开始的4KB代码,在一开机会搬移到这个stepingstone里执行(起始就是SRAM)。
韦东山的有个SDRAM的程序例程,
是在SRAM里运行的时候,把从0地址开始的4KB代码复制到SDRAM(起始地址0x30000000),然后从SDRAM中执行该代码。
疑问如下:
1.如果把4KB里面的代码搬移到SDRAM里,那跳转的标号在4KB里的SRAM中有一个,在SDRAM 中也有一个,怎么跳的过去呢,比如下面代码的on_sdram??
2.复制之后,假如说在SDRAM执行了,那如果中断来了,还是会跳到中断向量地址那里,岂不是又跑回去SRAM去了。虽然中断向量拷贝到SDRAM(0x30000000)了,但是似乎没有用哦?
完整的代码如下所示,共两个文件,一个head.S,一个是Led.c
----------------------------------head.s-------------------------
;*************************************************************************
; File:head.S
; 功能:设置SDRAM,将程序复制到SDRAM,然后跳到SDRAM继续执行
;************************************************************************      
MEM_CTL_BASE EQU       0x48000000
SDRAM_BASE  EQU       0x30000000
  IMPORT main
  PRESERVE8
  AREA    RESET, CODE, READONLY
  ENTRY  
START
    bl  disable_watch_dog               ; 关闭WATCHDOG,否则CPU会不断重启
    bl  memsetup                        ; 设置存储控制器
    bl  copy_steppingstone_to_sdram     ; 复制代码到SDRAM中
    ldr pc, =on_sdram                   ; 跳到SDRAM中继续执行
on_sdram        ;
    ldr sp, =0x34000000                 ; 设置堆栈
    bl  main
halt_loop
    b   halt_loop
disable_watch_dog
    ; 往WATCHDOG寄存器写0即可
    mov r1,     #0x53000000
    mov r2,     #0x0
    str r2,     [r1]
    bx LR      ;@ 返回 也可以使用mov PC,LR
copy_steppingstone_to_sdram
    ;@ 将Steppingstone的4K数据全部复制到SDRAM中去
    ;@ Steppingstone起始地址为0x00000000,SDRAM中起始地址为0x30000000
    
    mov r1, #0
    ldr r2, =SDRAM_BASE
    mov r3, #4*1024
1
    ldr r4, [r1],#4     ;@ 从Steppingstone读取4字节的数据,并让源地址加4
    str r4, [r2],#4     ;@; 将此4字节的数据复制到SDRAM中,并让目地地址加4
    cmp r1, r3          ;@ 判断是否完成:源地址等于Steppingstone的未地址?
    bne %B1              ;@ 若没有复制完,继续
    bx  lr      ;@ 返回
memsetup
    ;@ 设置存储控制器以便使用SDRAM等外设
    mov r1,     #MEM_CTL_BASE      ; @ 存储控制器的13个寄存器的开始地址
    adrl    r2, mem_cfg_val         ;@ 这13个值的起始存储地址
    add r3,     r1, #52             ;@ 13*4 = 54
1  
    ldr r4,     [r2], #4            ;@ 读取设置值,并让r2加4
    str r4,     [r1], #4            ;@ 将此值写入寄存器,并让r1加4
    cmp r1,     r3                  ;@ 判断是否设置完所有13个寄存器
    bne %B1                          ;@ 若没有写成,继续
    bx  lr                  ;@ 返回
mem_cfg_val
    ;@ 存储控制器13个寄存器的设置值
DCD  0x22011110     ;@ BWSCON
    DCD    0x00000700     ;@ BANKCON0
    DCD    0x00000700     ;@ BANKCON1
    DCD    0x00000700     ;@ BANKCON2
    DCD    0x00000700     ;@ BANKCON3  
    DCD    0x00000700      ;@ BANKCON4
    DCD    0x00000700     ;@ BANKCON5
    DCD    0x00018005      ;@ BANKCON6
    DCD    0x00018005      ;@ BANKCON7
    DCD    0x008C07A3      ;@ REFRESH
    DCD    0x000000B1      ;@ BANKSIZE
    DCD    0x00000030      ;@ MRSRB6
    DCD    0x00000030      ;@ MRSRB7
END
---------------------------------------------------------------------
---------------------------LED.C-----------------------------------

#define GPBCON  (*(volatile unsigned long *)0x56000010)
#define GPBDAT  (*(volatile unsigned long *)0x56000014)
#define GPB5_out (1<<(5*2))
#define GPB6_out (1<<(6*2))
#define GPB7_out (1<<(7*2))
#define GPB8_out (1<<(8*2))
void  wait(unsigned long dly)
{
for(; dly > 0; dly--);
}
int main()
{
unsigned long i = 0;
    unsigned long c = 1;
GPBCON = GPB5_out|GPB6_out|GPB7_out|GPB8_out;  // 将LED1-4对应的GPB5/6/7/8四个引脚设为输出
while(c){
  wait(30000);
  GPBDAT = (~(i<<5));   // 根据i的值,点亮LED1-4
  if(++i == 16)
   i = 0;
}
return 0;
}
--------------------------------------------------------------------
-----------------------------分散加载文件------------------------
; *************************************************************
; *** Scatter-Loading Description File generated by uVision ***
; *************************************************************
LR_IROM1 0x00000000 0x04000000  {    ; load region size_region
  ER_IROM1 0x00000000 0x04000000  {  ; load address = execution address
   head.o(RESET, +First)
   leds.o(+RO)
   ;*(InRoot$$Sections)
   .ANY (+RO)
  }
  RW_RAM1 0x30000000 UNINIT 0x04000000  {  ; RW data
   .ANY (+RW +ZI)
  }
}
--------------------------------------------------------------------------
级别: 新手上路
UID: 52894
精华: 0
发帖: 23
金钱: 115 两
威望: 23 点
贡献值: 0 点
综合积分: 46 分
注册时间: 2011-07-27
最后登录: 2015-05-24
1楼  发表于: 2011-09-16 19:30
同样的问题
级别: 新手上路
UID: 46666
精华: 0
发帖: 8
金钱: 40 两
威望: 8 点
贡献值: 0 点
综合积分: 16 分
注册时间: 2011-05-15
最后登录: 2012-11-14
2楼  发表于: 2011-12-23 20:36
  bl main  到不了吧, main 被链接到0x30000000 去了,bl 去不了那么远,要用ldr ,这是个绝对跳转哦!!
至于跳到哪的那个地址 编译的时候确定的, 那个跳转就是一个运行域的那个地址,eg, main 0x30000000
ldr pc,=main , 直接跳到那去了,而不会是sram 了,