自实现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 语言中,要注意有符号数与无符号数的混合运算,应始终确保操作数类型一致,或使用强制转换明确意图。

相关推荐
FreakStudio6 天前
W55MH32L-EVB 上手测评:硬件 TCP/IP 加持的以太网单片机,MicroPython 零门槛开发
python·单片机·嵌入式·大学生·面向对象·并行计算·电子diy·电子计算机
ifndef8 天前
GPIO同时复用(ADC与TIM)
mcu
LDR00611 天前
Type-C 快充全面升级!LDR6601 赋能个人护理便携电机,重塑剃须刀 / 理发器新体验
c语言·开发语言
雪碧聊技术11 天前
Tree.js是什么?一文讲透
开发语言·javascript·ecmascript
码云数智-园园11 天前
C++20 Modules 模块详解
java·开发语言·spring
✎ ﹏梦醒͜ღ҉繁华落℘11 天前
单片机基础知识---stm32单片机的优先级
stm32·单片机·mongodb
swordbob11 天前
NIO的channel中什么是 fd(File Descriptor,文件描述符)
java·开发语言·nio
源分享11 天前
Java线程同步的多种实现方法(非常详细)
java·开发语言·jvm
SM1771521183811 天前
NSK紧凑型FA系列丝杠技术详解
经验分享·规格说明书
Luminous.11 天前
C语言--day30
c语言·开发语言