当前位置:首页 > 单片机 > 单片机
[导读] 在编译下面一段代码时:STACK_TOPEQU0x20002000AREAReset,CODE,READONLYDCD0x20002000DCDStartENTRY;CODE16Startldrr2,=TestLDRDr0,r1,[r2,#4]LDRDr0,r1,[r2]LDRDr0,r1,[r2];movsr0,r0;NOP;align4TestDCD0x12345678

在编译下面一段代码时:

STACK_TOPEQU0x20002000

AREAReset,CODE,READONLY

DCD0x20002000

DCDStart

ENTRY

;CODE16

Start

ldrr2,=Test

LDRDr0,r1,[r2,#4]

LDRDr0,r1,[r2]

LDRDr0,r1,[r2]

;movsr0,r0

;NOP

;align4

Test

DCD0x12345678

END

我发现,如果加上NOP或align4,程序就不会跑飞,否则程序就跑飞了。

经调试发现: 如果不加NOP或align 4的话产生的Test的标号地址就会产生错误,而LDRD 指令操作的地址必须是4字节对节的,如果使用的地址不是四字节对齐,那么程序就会产生异常,所以程序就跑飞了。 那么为什么不加NOP或align 4的话Test标号地址就会产生错误呢? 来看一段手册上的话:
也就是说DCD是需要标号地址按字对齐的,如果你没有对齐就可以看到如下的编译警告:test.asm(18): warning: A1581W: Added 2 bytes of padding at address 0x1a
这说明编译器会自动添加两个字节来帮你对齐,数据分布情况和下面很相似:

这说明编译器会自动添加两个字节来帮你对齐,数据分布情况和下面很相似:

STACK_TOPEQU0x20002000

AREAReset,CODE,READONLY

DCD0x20002000

DCDStart

ENTRY

;CODE16

Start

ldrr2,=Test

LDRDr0,r1,[r2,#4]

LDRDr0,r1,[r2]

LDRDr0,r1,[r2]

;movsr0,r0

;NOP

;align4

Test

dcb00;编译器自动添加

dcb00;编译器自动添加,而movsr0,r0的机器码就是0x0000,会被

;编译器翻译成movsr0,r0,不是当作数据0x0000

DCDU0x12345678

END

也许看来这样就完美了,但是程序依然会跑飞。原因有两点:
1.即使加了两个字节那么Test的标号地址依然不是四字节对齐。

2.这两个字节的零会被编译器当作指令来处理的,这也就是说Test标号会被编译器来当作代码标号来处理,看到了吧,我们的数据编译器一插手就变成代码了,实在无奈的很。再来看一段手册上的讲解:



也就是实际上LDR r2,=Test执行后,r2=Test+1这也解释了为什么不加NOP或align 4的话r2=0x8000017而加了NOP或Align 4就r2=0x8000018。那么来看一下,加nop或align 4后的效果:

STACK_TOPEQU0x20002000

AREAReset,CODE,READONLY

DCD0x20002000

DCDStart

ENTRY

;CODE16

Start

ldrr2,=Test

LDRDr0,r1,[r2,#4]

LDRDr0,r1,[r2]

LDRDr0,r1,[r2]

;movsr0,r0

;如果是align4会被加两个节字的movsr0,r0(机器码为0x0000)

;如果是nop则会被加上nop的机器码(0xBF00)

;NOP

align4

Test

DCDU0x12345678

END

需要说明的是,我总是把align 和nop 放在一块说,并不是说nop也具有对齐作用。是因为加上nop后刚好可以使Test标号地址放在4字节对齐的其他地方。在其他地方,nop也许并无此作用。



本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

近日,东芝宣布推出新品——“M3H族”微控制器,该产品基于全球标准的Arm Cortex-M3内核,专为电机控制而设计,可满足消费设备和工业设备的各种需求。新产品的出现扩大了 “TXZ?系列”微控制器的产品阵容。

关键字: ARM cortex-m3 内核 微控制器 电源新品

已经是很久没有写文章了,因为没有时间,人家都说大四不考研,天天像过年,可我依旧没能有那份闲暇的时间。现在几乎人人的手里都是一款安卓智能机这是我非常羡慕的,而我手

关键字: adxl345 cortex-m3 单片机 电源技术解析 重力感应

1. 摘要 Cortex-M内核实现了一个高效异常处理模块,可以捕获非法内存访问和数个程序错误条件。本应用笔记从程序员角度描述Cortex-M Fault异常,并且讲述在软件开发周期中的Fault用法。 2. 简介...

关键字: cortex-m3 cortex-m4 fault异常

1.1 启动代码内容1) 硬件初始化:最起码的是要初始化堆栈指针。2) C语言环境 :在main函数调用之前要完成对一些变量的初始化。3) 应用初始化: 这主要取决于你的应用。比如设置系统的晶振、时钟。1.2 图解M3启...

关键字: cortex-m3 启动代码

有時候切換了編譯方式如從ARMCC轉變為GCC編譯器,編譯下載時會出現Error:FlashDownloadfailed:-"Cortex-M3"這個錯誤。目前,自己測試解決的方式是:刪除相同目錄下的...

关键字: cortex-m3 error keil

在项目处于调试期间,Fault处理程序可能只是一个断点指令,调试器遇到这个指令后停止程序的运行。默认情况下,由于非硬Fault被禁能,所有发生的非Fault都会上访成硬Fault,因此只要在硬Fault处理程序中设...

关键字: cortex-m3 cortex-m4 fault fault处理函数

1 Cortex-M3/4的Fault简介 Cortex-M3/4的Fault异常是由于非法的存储器访问(比如访问0地址、写只读存储位置等)和非法的程序行为(比如除以0等)等造成的。常见的4种异常及产生异常的情况如...

关键字: 4 cortex-m3 调试方法 hard fault

STM32(Cortex-M3)中有两个优先级的概念——抢占式优先级和响应优先级,有人把响应优先级称作'亚优先级'或'副优先级',每个中断源都需要被指定这两...

关键字: cortex-m3 STM32 中断优先级

TM32笔记之八:来跟PC打个招呼,基本串口通讯a) 目的:在基础实验成功的基础上,对串口的调试方法进行实践。硬件代码顺利完成之后,对日后调试需要用到的printf重定义进行调试,固定在自己的库函数中。b) 初始化函数

关键字: cortex-m3 usb的

电动车一般采用锂电池供电,由多个单体电池 串联成电池组作为动力电源。但由于各个串联单体电池特性不能保证完全一致,因此相同的电流下充电放电速度也会不同,如果不进行均衡干预,电池寿命会大大缩 短,实践证明EMS可以有效延长电...

关键字: cortex-m3 STM32 电池管理系统 触摸屏 设计教程
关闭
关闭