自实现FLASH读取函数中的隐式类型转换bug踩坑记录

问题现象及背景

在自行封装FLASH自用库的过程中,测试读取 API 时发现了一个隐藏Bug.

cpp 复制代码
#define FLASH_BYTES_PER_PAGE (1024UL)   // 无符号常量
bool Cus_Flash_ReadOutPage( uint32_t PageAddress, uint8_t *pOutBuffer, int32_t Size )
{

.............

   if ( (Size - FLASH_BYTES_PER_PAGE) > 0 )    // 缓冲区给大了,已经读完一页了,剩下空间以0填充.
     {
       memset(&pOutBuffer[FLASH_BYTES_PER_PAGE], 0, (Size - FLASH_BYTES_PER_PAGE));
     }

...............

}

(相关 API 问题部分如上)

cpp 复制代码
 uint8_t RxBuf[128] = { 0 }; 
 Cus_Flash_ReadOutPage(Address, RxBuf, sizeof(RxBuf));

(调用示例)

具体问题为:函数调用时当传入的 Size 小于 FLASH_BYTES_PER_PAGE,在经过判断分支后,被判断为true,进而执行memset后触发硬件HardFault。

原因分析

经过调试分析,当有符号整数(intint32_t)与无符号整数(unsigned intuint32_t)进行算术运算(如减法)时,有符号数会被隐式转换为无符号数。如果运算结果本应为负数,则会变成一个很大的正数(无符号下溢),导致后续逻辑判断错误。

cpp 复制代码
uint32_t a = 1024;
int32_t  b = 64;
// b - a 的执行过程:
// 1. b 被转换为 uint32_t,值为 64(无符号)
// 2. 计算 64 - 1024,在无符号算术中结果为 2^32 - 960 = 4294966336
// 3. 这个值大于 0,所以条件永远为真

例如上述当 Size = 64 时,条件本应为假,但由于隐式转换,无符号下溢,条件为真,导致 memset 越界写入大量字节,进而导致HardFault触发。

解决方法

① 通过强制类型转换,将FLASH_BYTES_PER_PAGE转换为(int32_t)形式.

cpp 复制代码
将 if ( (Size - FLASH_BYTES_PER_PAGE) > 0 )   
改为:

    if ( (Size - (int32_t)FLASH_BYTES_PER_PAGE) > 0 )   

② 先赋值给有符号变量再判断.

cpp 复制代码
    int32_t remaining = Size - FLASH_BYTES_PER_PAGE;  
    if (remaining > 0) 
    {
      memset(&pOutBuffer[FLASH_BYTES_PER_PAGE], 0, remaining);
    }

总结

在 C 语言中,要注意有符号数与无符号数的混合运算,应始终确保操作数类型一致,或使用强制转换明确意图。

相关推荐
野生技术架构师2 分钟前
2026 Java面试宝典(春招/社招/秋招通用):没有前言,只有答案,直接开背
java·开发语言·面试
ychqsq6 分钟前
45.新芽
经验分享·职场和发展
SUNNYSPY00121 分钟前
BSS138-ASEMI中低压通用MOS管BSS138
单片机
国科安芯26 分钟前
国科安芯推出商业航天级抗辐照半双工 RS485 收发器 ASC485S2Y
前端·单片机·嵌入式硬件·架构·安全性测试
人道领域1 小时前
【LeetCode刷题日记】131.分割回文串,动态规划优化
java·开发语言·leetcode
z落落1 小时前
C# 接口 interface (多接口实现、类+接口、成员重名)
java·开发语言
知识的宝藏2 小时前
Xpaht self::div 轴语法
开发语言
keykey6.2 小时前
卷积神经网络(CNN):让AI学会“看“
开发语言·人工智能·深度学习·机器学习
ss2732 小时前
【入门OJ题解】分苹果问题(Python/Java/C 实现)
java·c语言·python
嵌入式ZYXC2 小时前
第1篇:《面试题:画一个STM32最小系统电路,每个元件的作用》
stm32·单片机·嵌入式硬件·面试·职场和发展