程序在nor flash中真的可以运行吗?

     程序在nor flash中可以运行,但是是有限制的,它不能像RAM那样随意的写(尽管它可以随意的读)。在norflash上,不能运行写存储器的指令,不过排除写的地方是RAM类。实验中的三个文件如下所示:
Makefile如下:

led_on.bin : crt0.S leds.c
 arm-linux-gcc -g -c -O2 -o crt0.o crt0.S
 arm-linux-gcc -g -c -O2 -o leds.o leds.c
 arm-linux-ld -Ttext  0x0 crt0.o leds.o  -o led_on_elf
 arm-linux-objcopy -O binary -S led_on_elf led_on.bin
 arm-linux-objdump -D -m arm  led_on_elf > led_on.dis
clean:
 rm -f led_on.dis led_on_elf *.o led_on.bin 


crt0.S如下:

@******************************************************************************
@ File:crt0.S
@ 功能:通过它转入C程序
@*****************************************************************************      
.text
. global _start
_start:
            ldr     r0, = 0x53000000     @ WATCHDOG寄存器地址
            mov     r1, # 0x0                     
            str     r1, [r0]        @ 写入0,禁止WATCHDOG,否则CPU会不断重启
            ldr     sp, = 0x00001000     
            bl      main                @ 调用C程序中的main函数
halt_loop:
            b       halt_loop
leds.c如下所示:

#define GPBCON      (*(volatile unsigned long *)0x56000010)
#define GPBDAT      (*(volatile unsigned long *)0x56000014)
/*
 * LED1,LED2,LED4对应GPB5、GPB6、GPB7、GPB8
 
*/
#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( volatile unsigned  long dly)
{
  for(; dly >  0; dly--);
}
int main( void)
{
 unsigned  long i =  0;
  //  LED1,LED2,LED4对应的4根引脚设为输出
 GPBCON = GPB5_out | GPB6_out | GPB7_out | GPB8_out;
  while( 1){
  wait( 30000);
  GPBDAT = (~(i<< 5));   //  根据i的值,点亮LED1,2,3,4
   if(++i ==  16)
   i =  0;
 }
  return  0;
}


    实验结果:这样的代码编译出来的程序在steppingstone(SRAM)中可以运行,但是下载到norflash中不能运行。
    修改代码:
crt0.S如下:

@******************************************************************************
@ File:crt0.S
@ 功能:通过它转入C程序
@*****************************************************************************      
.text
. global _start
_start:
            ldr     r0, = 0x53000000     @ WATCHDOG寄存器地址
            mov     r1, # 0x0                     
            str     r1, [r0]        @ 写入0,禁止WATCHDOG,否则CPU会不断重启
            ldr     sp, = 0x40001000     
            bl      main                @ 调用C程序中的main函数
halt_loop:
            b       halt_loop


实验结果:这样的代码编译出来的程序在norflash中可以运行。


原因分析:
由于要跳转到main()函数中去执行,即C函数中去,就需要使用堆栈。代码“ldr     sp, =0x00001000”由于使用堆栈的地址位于norflash中,而当跳转到main()函数中去执行时,就会写norflash,而norflash不能像RAM那样随意写,所以不能成功执行。代码“ldr     sp, =0x40001000”由于使用堆栈的地址是0x40001000,使用的是(0x40000000—0x40001000),即是S3C2440的片内SRAM。这样跳转到main()函数时,用到的     就是片内SRAM,所以能够成功执行。


写内存的时机:
程序在运行过程中大多时候是在读内存,例如取指令、加载数据等等,写内存的时机大概这么几种情况:
1.写特殊功能寄存器(其实这个可以排除出去)
2.修改全局变量的值
3.讲数据压入堆栈保存。

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值