原创 ARM程序由于字节对齐引起的问题深入分析

2007-9-28 17:34 8622 12 13 分类: MCU/ 嵌入式

      首先说说,什么叫对齐。如果一个数据是从偶地址开始的连续存储,那么它就是半字对齐,否则就是非半字对齐;半字对齐的特征是bit0=0,其他位为任意值。字对齐的特征是bit1=0,bit0=1,其他位为任意值。如果一个数据是以能被4 整除的地址开始的连续存储,那么它就是字对齐,否则就是非字对齐。举例说明四字节对齐: 对内存进行操作时,被访问的地址必须为4的倍数。如果分配到的地址的地址不是4的倍数时,CPU实际访问的地址还是按照字对齐的方式来操作。也就是自动屏蔽bit1和bit0.   


      用ADS的ARM C Complier下Optimization Level可能引起问题,其中的一个问题就是字节对齐的问题。下面讲讲问题的现象及实质。


       当时问题的现象是:程序使用一公共变量Buf创建队列,如果ADS编译优化选项采用Minium则软件工作正常;源码不变,如果采用ALL优化,则不正常,数据紊乱且无法工作。为了发现问题,我们分别用Minium和ALL编译,在反汇编条件下单步跟踪程序,观察CPU寄存器和内存变量的变化情况。发现在Minium模式下,编译器把队列内存块Uart0TxBuf分配到的地址是0x400015cc,这个地址是一个4字节对齐的地址,而在ALL模式下,编译器把Buf分配的地址是0x400015c2,这个地址是一个非4字节对齐的地址。正是由于这个非4字节对齐的地址导致了问题的发生。
问题发生在QueueCreate(void *Buf, uint32 SizeOfBuf, uint8 (* ReadEmpty)(), uint8 (* WriteFull)())这个函数里,问题是如何发生的,



        在了解问题发生的机理前,先了解QueueCreate这个函数的工作原理。QueueCreate工作原理是,首先把buf指向的内存初始化为DataQueue格式的结构体。  DataQueue的结构体格式如下:
typedef struct {
    QUEUE_DATA_TYPE     *Out;                   /* 指向数据输出位置         */
    QUEUE_DATA_TYPE     *In;                    /* 指向数据输入位置         */
    QUEUE_DATA_TYPE     *End;                   /* 指向Buf的结束位置        */
    uint16              NData;                  /* 队列中数据个数           */
    uint16              MaxData;                /* 队列中允许存储的数据个数 */
   
    uint8               (* ReadEmpty)();        /* 读空处理函数             */
    uint8               (* WriteFull)();        /* 写满处理函数             */
    QUEUE_DATA_TYPE     *Buf;                 /* 存储数据的空间           */
} DataQueue;
从结构体可以看出,结构体字节类型在内存分配为: 4字节指针变量(*Out)、4字节指针变量(*In)、4字节指针变量(*End)、2字节变量NData、2字节变量MaxData、4字节函数指针变量ReadEmpty()、4字节函数指针变量(WriteFull())
   观察结构体起始地址放在非对齐时会出现什么情况。


          起始地址为0x400015c2时的由编译器分配得到的地址         实际操作地址
*Out       0x400015c2~0x400015c5                                                0x40015c0~0x400015c3
*In         0x400014c6~0x400015c9                                                 x400014c4~0x400015c7
*End      0x400015ca~0x400015cd                                               0x400015c8~0x400015cb
从表中可以看出,实际操作的地址按照4字节对齐格式得到。例如,当执行*Out进行操作时,自动屏蔽bit1和bit0,因此实际发生变化的是0x40015c0~0x400015c3,而不是0x400015c2~0x400015c5,由于实际操作地址和编译器分配地址互相覆盖,当对*In操作时,会导致*Out一起变化,对*End操作时,*In也跟着变化。正是由于非对齐的原因导致创建队列和对列操作完全错误。
     当内存起始地址为4字节对齐地址的情况时,编译器分配地址和实际地址一致,因此不存在上述问题。


结 论:


          在ARM嵌入式系统中,当把一个内存区域初始化为某个结构体时,必须注意字节对齐的情况。如果该内存起始地址为非对齐地址,不仅得不到预期的结果,还可能导致一些很奇怪的让人无法理解表面问题。在C层面上不太容易观察到这些问题的实质,只有深入到汇编一层去分析程序,才可能理解这些现象的深层原因。


 



 

PARTNER CONTENT

文章评论1条评论)

登录后参与讨论

用户232631 2009-12-1 16:52

呵呵,我就是吃了内存对齐的亏啊,谢谢了!!
相关推荐阅读
用户1524708 2009-11-05 09:52
数字湿度传感器
资料来源:http://www.sensirion.com/该公司还有流量气体传感器、压力传感器。介绍器件的目的是为产品涉设计选型时多一种器件的选择考虑。SHT15 - Digital Humidit...
用户1524708 2009-03-31 15:56
一种低电阻的测量方法
请参照本博客文章介绍的一种测量低阻值导线电阻的方法:http://blog.ednchina.com/cllzs/195043/message.aspx...
用户1524708 2009-03-31 15:49
电平电压为12V的I2C总线通讯
I2C的通讯电压一般为5V或3.3V电压,如果特殊的情况下使用12V座作为通讯电压,可以参照本博客文章:http://blog.ednchina.com/cllzs/215265/message.as...
用户1524708 2009-03-31 15:47
双向电平转换电路在汽车解码器通讯电路中的应用
汽车解码器即汽车故障诊断系统,在这个电子设备中,需要面对各种各样的汽车逻辑电平,如KWP2000,ISO9141等通讯协议都是用12V来通讯。如何实现一种简单的电路就能满足各种各样的汽车ECU通讯电平...
用户1524708 2009-03-31 15:06
两种简易的低成本双向的逻辑电平转化电路
(转载请注明www.ednchina.com/blog/cllzs原创)在电子电路设计中,可能需要这样一种电路:单片机输出5V或3.3V信号,但在总线上的信号是12V或24V甚至更高的电压。单片机I/...
用户1524708 2009-01-16 17:35
精度高达0.001欧姆的简易电阻测量法
首先说明:此方法用于来测量导线等一些电阻值非常小导体。在电子实践中,有时需要测量一些电阻值非常小的导体,如导线电阻。导线的电阻一般都是非常的小,在电流不是很大,如几十到几百毫安时,一般不太考虑导线电阻...
我要评论
1
12
关闭 站长推荐上一条 /3 下一条