c - 从阵列投射会导致某些 MCU 崩溃,但不会导致其他 MCU 崩溃
问题描述
我有一段看起来像这样的代码:
void update_clock(uint8_t *time_array)
{
time_t time = *((time_t *) &time_array[0]); // <-- hangs
/* ... more code ... */
}
wheretime_array
是一个 4 字节的数组(即uint8_t time_array[4]
)。
我正在使用 arm-none-eabi-gcc 为 STM32L4 处理器编译它。
几个月前编译它时,我没有遇到任何错误,并且代码在我所有的测试 MCU 上运行得非常好。回到这个项目时,我对我的环境(OpenSTM32)进行了一些更新,现在这段代码在一些 MCU 上崩溃,而在其他 MCU 上运行良好。
我仍然有几个月前的二进制文件,并确认此代码路径在我所有的 MCU 上都可以正常工作(我有大约 5 个要测试),但现在它可以在其中两个上工作,同时导致三个他们。
我已经通过重写这样的代码来缓解这个问题:
time_t time = (
((uint32_t) time_array[0]) << 0 |
((uint32_t) time_array[1]) << 8 |
((uint32_t) time_array[2]) << 16 |
((uint32_t) time_array[3]) << 24
);
虽然这目前可行,但我认为旧代码看起来更干净,而且我还担心如果此代码路径挂起,我可能会在其他地方遇到类似的错误。
有谁知道是什么原因造成的?我可以更改设置中的任何内容以使编译器再次以旧方式工作吗?
解决方案
从版本 7-2017-q4-major 开始,arm gcc 附带 newlib 编译时time_t
定义为 64 位 ( long long
) 整数,导致假定它为 32 位的代码出现各种问题。您的代码正在读取源数组的末尾,将存储在那里的任何内容作为时间值的高位,可能导致大爆炸之前或宇宙热寂之后的日期,这可能不是您的代码期望什么。
如果已知源数组包含 32 位数据,则先将其复制到 32 位int32_t
变量中,然后您可以将其分配给 a time_t
,这样无论time_t
.
推荐阅读
- python - Python3修剪关联数组中的所有数据
- firebird - 将 .DBF 文件迁移到 Firebird 2.5:如何使用不同 DBRMS 的 UNION 创建查询
- angular - HighChart中没有Y轴的散点气泡图
- html - 在 IOS 设备上,固定元素在滚动时被下部元素覆盖
- python - 如何更改字符串的替代字母的大小写
- java - Android:在 JDK 10 上为 1.8 目标构建 Java 代码时出现 Gradle 错误(gradle build -> NullPointerException)
- excel - 如何将文本与共同的单词分组?
- java - 使用一个 map-filter lambda 表达式从项目列表中检索对象字段
- jmeter - jmeter中的负载测试
- ruby-on-rails - 包括 RailsAdmin 组中的所有剩余字段