硬汉嵌入式论坛

 找回密码
 立即注册
查看: 959|回复: 2
收起左侧

[IAR] 浅谈MDK, IAR,CLANG和GCC的局部变量字节对齐处理差异(2023-10-13)

[复制链接]

1万

主题

6万

回帖

10万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
106678
QQ
发表于 2023-10-13 00:31:27 | 显示全部楼层 |阅读模式
视频:

https://www.bilibili.com/video/BV1CB4y1Z7kA



问题由来:

早期这个帖子里面的局部变量对齐仅测试了MDK AC5,但项目中使用AC6发现了新问题,看来AAPCS规约研究的还是不够细:

https://www.armbbs.cn/forum.php?mod=viewthread&tid=109400

当时对局部变量的描述如下:局部变量使用的是栈空间(除了静态局部变量和编译器优化不使用栈,直接用寄存器做变量空间),也就是大家使用在xxxx.S启动文件开辟的stack空间。

在M内核里面,局部变量的对齐问题如果研究起来是最烧脑的,这个涉及到AAPCS规约(Procedure Call Standard for the Arm Architecture,  Arm架构的程序调用标准)。



上面这个贴图最重要,仅需理解上面这两条就可以,意思是说,栈地址是全程至少保持4字节对齐的,因为M内核的硬件长做了处理,SP最低两个bit,bit0和bit1直接固定为0了。

但是在程序调用入口处必须满足8字节对齐,对于C语言,不需要用户去管,编译器都帮我们处理好了,先来个简单的示例压压惊:


而汇编文件是需要用户去处理的。以xxx.S启动文件为例,通过伪指令PRESERVE8来保证



那么问题来了,我们搞个4对齐是不是会出问题,一般情况下也没问题的,但特殊情况下不行,特别调用C库的sprintf和printf函数,直接给你输出个不知所以然的结果来。比如我在H7上做如下测试:




输出结果:




总结:

MDK AC5和IAR的用法差不多,MDK AC6和GCC的用法差不多(视频里面做了详细对比测试)。

对于GCC和MDK AC6,大家可以手动添加__ALIGNED()设置对齐,这个原定义在CMSIS软件的头文件里面。
GCC的定义如下:
[C] 纯文本查看 复制代码
#define __ALIGNED(x)                           __attribute__((aligned(x)))

AC6定义如下:
[C] 纯文本查看 复制代码
#define __ALIGNED(x)                           __attribute__((aligned(x)))



回复

使用道具 举报

1万

主题

6万

回帖

10万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
106678
QQ
 楼主| 发表于 2023-10-13 00:33:22 | 显示全部楼层
V7-3001_Base Template.7z (2.53 MB, 下载次数: 8)
回复

使用道具 举报

210

主题

1042

回帖

1682

积分

至尊会员

More we do, more we can do.

积分
1682
发表于 2023-10-13 13:41:34 | 显示全部楼层
本帖最后由 emwin 于 2023-10-13 13:50 编辑

IAR是这么描述的"The AEABI standard specifies that the stack pointer must be 8-byte aligned at function entry.",原文:


截图_2023-10-13_13-50-36.png

回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|小黑屋|Archiver|手机版|硬汉嵌入式论坛

GMT+8, 2024-4-29 09:31 , Processed in 0.175180 second(s), 28 queries .

Powered by Discuz! X3.4 Licensed

Copyright © 2001-2023, Tencent Cloud.

快速回复 返回顶部 返回列表