最近在做gd32E230c8t6 BootLoader的时候,发现当打开编译优化-o1的时候,代码运行不正常。
代码会在读取内部flash内存时出现问题,进入了hardfault。但是改为-o0后,运行就正常了。非常诡异,后面经过反复测试逐步摸清楚了问题的原因。先看问题的代码
uint8_t temp[PAGE_SIZE];
void run_bootloader(void)
{
…
flash_read(FLASH_SAVE_ADDR_APP,temp,PAGE_SIZE);
…
}
其中PAGE_SIZE=1024.这段代码的意思很简单,就是将FLASH_SAVE_ADDR_APP后的1K字节读取到temp数组中。flash_read函数内容如下。
int flash_read(uint32_t addr,uint8_t *buf,uint32_t len)
{
// printf(“buf addr:%x %x”,(uint8_t*)buf,addr);
for(uint16_t i = 0;i<len;i+=4) {<!–=“” --=“”>
*( uint32_t*)&buf[i] =*(volatile uint32_t*)(addr+i);
}
return 0;
}
</len;i+=4)>
flash读取是4字节读取,一次读取4字节,这里将buf强制类型转换成了uint32,当i=0时,上述一次赋值相当于将buf[0],buf[1],buf[2],buf[3]进行了赋值,提高了代码的效率。从逻辑上这样做一点问题都没有,但是-o1确出现了问题。
问题原因分析