OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 4975|回复: 7

关于keil的优化等级造成程序异常

[复制链接]

14

主题

33

帖子

0

精华

初级会员

Rank: 2

积分
103
金钱
103
注册时间
2014-6-27
在线时间
24 小时
发表于 2020-12-12 11:15:17 | 显示全部楼层 |阅读模式
10金钱
使用的芯片是stm32f407ve,ucos_iii,keil5

我在程序中这样几个赋值的语句
其中 #define Speed 6400  
  1. Speed1 = Speed;                        
  2. vx1 = Speed1;
复制代码
程序中 Speed1 的数据类型是 uint32_t,vx1的数据类型是float。
现在遇到了这样一个问题,就是赋值完后 vx1的值为0.5;
出现这个现象是因为,我把程序中几个全局变量的类型由 uint16_t 改成了 uint32_t。
后来我尝试强制类型转换 vx1 = (float)Speed1;  没有效果
尝试直接赋值 vx1 = 6400;   vx1 = 6400.0f;       都没有效果


但是下面的几个尝试可以得到正确的值
1、vx1的数据类型改为 uint32_t
2、keil中的优化等级改为不优化
3、赋一个其他的值,6399  ,  800等其他的值都没有问题。

其中上面第三个尝试最让我迷惑,为什么只有6400这个值不行呢?(可能也有别的值不行,但是我没有挨个实验)
想着应该是由于程序空间的问题,被优化的结果,但是不是很明白,有没有哪个前辈给解答一下

非常感谢

最佳答案

查看完整内容[请看2#楼]

这是经常出现的编译器问题。 首先你要明白C语言和汇编是两种语言,考虑到转换后的效率及资源占用, 在最后优化的编译出来的汇编跟原先的C的执行方法是会被改变的。 比如, 全局变量 int AB=56; 局部变量 int C=23; abc = AB+C; C这个局部变量编译后根本就不会出现, 甚至用Watch也不会提供其数据, 优化后的汇编很可能是这样 ldr r0,=ABC ldr r1,[r0] add r1,#23 @
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

3

主题

1907

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
4106
金钱
4106
注册时间
2018-8-14
在线时间
696 小时
发表于 2020-12-12 11:15:18 | 显示全部楼层
这是经常出现的编译器问题。
首先你要明白C语言和汇编是两种语言,考虑到转换后的效率及资源占用, 在最后优化的编译出来的汇编跟原先的C的执行方法是会被改变的。

比如,
全局变量 int AB=56;
局部变量 int C=23;
abc = AB+C;

C这个局部变量编译后根本就不会出现, 甚至用Watch也不会提供其数据, 优化后的汇编很可能是这样

ldr  r0,=ABC
ldr  r1,[r0]
add r1,#23    @<< AB+C
.....

C是变成一个立即数23, 所以C根本就不存在。

要是你一定想看到C的内容来查错, 方法是有的, 把C暂时搬出函数外, 暂时让它当个全局变量, 方便你查错, 记得查完错后还原。
回复

使用道具 举报

51

主题

2166

帖子

2

精华

论坛元老

Rank: 8Rank: 8

积分
10653
金钱
10653
注册时间
2017-4-14
在线时间
2780 小时
发表于 2020-12-12 15:50:38 | 显示全部楼层
speed1加volatile试试
回复

使用道具 举报

14

主题

33

帖子

0

精华

初级会员

Rank: 2

积分
103
金钱
103
注册时间
2014-6-27
在线时间
24 小时
 楼主| 发表于 2020-12-14 09:00:56 | 显示全部楼层
本帖最后由 jinger0311 于 2020-12-14 09:03 编辑
edmund1234 发表于 2020-12-12 11:15
这是经常出现的编译器问题。
首先你要明白C语言和汇编是两种语言,考虑到转换后的效率及资源占用, 在最后 ...

我知道有时候局部变量在调试的时候可能看不到值,
可是我这里面的几个变量都是全局变量,而且最后这个vx1我是有用到的,他是反应的电机的转速,实际我的电机转速的确也是我看到的vx1的值
所以我还是应该去看最终转换为汇编后是什么吧
回复

使用道具 举报

14

主题

33

帖子

0

精华

初级会员

Rank: 2

积分
103
金钱
103
注册时间
2014-6-27
在线时间
24 小时
 楼主| 发表于 2020-12-14 09:05:37 | 显示全部楼层
本帖最后由 jinger0311 于 2020-12-14 09:07 编辑
nashui_sx 发表于 2020-12-12 15:50
speed1加volatile试试

这样加了就没有问题了,所以其实是speed1在优化的时候出错了?
是不是跟浮点数的在内存中的存储方式有关?为什么其他的值没有问题呢?6399也可以正常赋值
回复

使用道具 举报

22

主题

2251

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
4480
金钱
4480
注册时间
2013-4-22
在线时间
337 小时
发表于 2020-12-14 09:21:03 | 显示全部楼层
强制转化编译器是不会优化的,除非你的代码有bug
回复

使用道具 举报

3

主题

1907

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
4106
金钱
4106
注册时间
2018-8-14
在线时间
696 小时
发表于 2020-12-14 09:23:13 | 显示全部楼层
jinger0311 发表于 2020-12-14 09:00
我知道有时候局部变量在调试的时候可能看不到值,
可是我这里面的几个变量都是全局变量,而且最后这个vx ...

贴一下你出问题那段代码吧
回复

使用道具 举报

3

主题

312

帖子

0

精华

高级会员

Rank: 4

积分
907
金钱
907
注册时间
2011-10-19
在线时间
196 小时
发表于 2020-12-14 09:32:27 | 显示全部楼层
工程配置一般要分Debug跟Release两个版本:
Debug专门用于实验调试而没有开优化;
Release版本打开优化选项是发布到生产环节的版本。
回复

使用道具 举报

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

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-13 17:45

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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