最近在使用STM32F103C8T6这款芯片,在读写内部flash的时候,读出数据没有问题,但是当写入数据的时候出现问题
我的代码如下(参考的正点原子例程)
不校验的情况下写入数据的数据
- void STMFLASH_Write_NoCheck(u32 WriteAddr,u16 *pBuffer,u16 NumToWrite)
- {
- u16 i;
- for(i=0;i<NumToWrite;i++)
- {
- FLASH_ProgramHalfWord(WriteAddr,pBuffer[i]);
- WriteAddr+=2;//地址增加2.
- }
- }
复制代码
- u16 STMFLASH_BUF[STM_SECTOR_SIZE/2];//最多是2K字节
- void STMFLASH_Write(u32 WriteAddr,u16 *pBuffer,u16 NumToWrite)
- {
- u32 secpos; //扇区地址
- u16 secoff; //扇区内偏移地址(16位字计算)
- u16 secremain; //扇区内剩余地址(16位字计算)
- u16 i;
- u32 offaddr; //去掉0X08000000后的地址
- if(WriteAddr<STM32_FLASH_BASE||(WriteAddr>=(STM32_FLASH_BASE+1024*STM32_FLASH_SIZE)))return;//非法地址
- FLASH_Unlock(); //解锁
- offaddr=WriteAddr-STM32_FLASH_BASE; //实际偏移地址.
- secpos=offaddr/STM_SECTOR_SIZE; //扇区地址 0~127 for STM32F103RBT6
- secoff=(offaddr%STM_SECTOR_SIZE)/2; //在扇区内的偏移(2个字节为基本单位.)
- secremain=STM_SECTOR_SIZE/2-secoff; //扇区剩余空间大小
- if(NumToWrite<=secremain)
- {
- secremain=NumToWrite;//不大于该扇区范围
- }
- while(1)
- {
- STMFLASH_Read(((secpos*STM_SECTOR_SIZE)+STM32_FLASH_BASE),STMFLASH_BUF,STM_SECTOR_SIZE/2);//读出整个扇区的内容
- for(i=0;i<secremain;i++)//校验数据
- // for(i=0;i<(STM_SECTOR_SIZE/2);i++)//校验数据
- {
- if(STMFLASH_BUF[secoff+i]!=0XFFFF)break;//需要擦除
- // if(STMFLASH_BUF[i]!=0XFFFF)break;//需要擦除
- }
- if(i<secremain)//需要擦除
- // if(i<(STM_SECTOR_SIZE/2))//需要擦除
- {
- FLASH_ClearFlag(FLASH_FLAG_EOP | FLASH_FLAG_PGERR | FLASH_FLAG_WRPRTERR);
- FLASH_ErasePage(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE);//擦除这个扇区
- for(i=0;i<secremain;i++)//复制
- {
- STMFLASH_BUF[i+secoff]=pBuffer[i];
- }
- STMFLASH_Write_NoCheck(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//写入整个扇区
- }else STMFLASH_Write_NoCheck(WriteAddr,pBuffer,secremain);//写已经擦除了的,直接写入扇区剩余区间.
- if(NumToWrite==secremain)break;//写入结束了
- else//写入未结束
- {
- secpos++; //扇区地址增1
- secoff=0; //偏移位置为0
- pBuffer+=secremain; //指针偏移
- WriteAddr+=secremain; //写地址偏移
- NumToWrite-=secremain; //字节(16位)数递减
- if(NumToWrite>(STM_SECTOR_SIZE/2))
- {
- secremain=STM_SECTOR_SIZE/2;//下一个扇区还是写不完
- }
- else
- {
- secremain=NumToWrite;//下一个扇区可以写完了
- }
- }
- }
- FLASH_Lock();//上锁
- }
复制代码
主函数调用
- for(u16 i=0;i<6;i++)
- {
- flash_buff_write[i]=i;
- }
- for(u8 i=0;i<88;i++)
- {
- STMFLASH_Write(((0x08005000)+(i*12)),flash_buff_write,6);
- if(i>80)
- {
- receive_num=1;
- }
- }
复制代码
|
最好是把这个放到库函数中,因为每次写入前都清除一次异常标志,要不然如果写错一个,再继续写的话就全部都是错误的,不管你写的是什么,也不管后面是不是0XFF都会出问题
是半字编程啊,您看我都是每次写16位的数进去,单独写的话没事儿,但是当跨页的时候如果遇到新页是0XFFFF的话是不进行擦除的,这样的话写入会有问题,是不是字节对齐的原因?
而且您说的PM0075搜索不到啊
我用的是标准库,例程参考的正点原子的例子,其中他们缺少的清除异常标志位,我也是添加到代码内的,如果不跨页写的话是没问题的
每次写入6个半字
每次写入6个半字 一页是512个半字,所以只能写入85个半字 剩下两个半字,这一页写入是完全正确的 但是写入剩下4个半字的时候就出错了,第二页是没有经过擦除的,因为第二页起始位置是4个0xFFFF按照手册上是可以写入的,但实际情况是写入后会出现异常,通过memory查看对应地址上的数据,确实没有改变,能解释一下是为什么吗?
不管用哪个库都是可以的。
PM0075是专门介绍STM32F1系列flash编程的用户手册,实在找不到我可以发给你。当然,对于你眼前
问题有没有它或许没有决定性作用,但是了解下基本编程规程为宜。需要的话 可以留下你的邮箱。
你跨页写的话,之前应是被erased过。同时 还要注意地址对齐方面的问题。
确认下地址对不对,另外,有无对齐方面的问题。
您好,PM0075我找到了,里面说的是在写入前确认被擦除过,因为在烧写前,我选择的是全片擦除,所以肯定是擦除了的 ,还是有地址肯定是对的,对齐的话是按照半字写入的,应该不存在没有对齐的情况
仿真,一步一步走,地址是没有问题的
这个情况,以前在STM32F429上遇到过,当时没注意,这次是想搞明白怎么回事儿,读写过程肯定是没有问题的,只是不清楚为啥写不进去的
我还做过一个实验,就是无论什么情况下,都在写入前擦除一次,这样的情况是,每次都能正确的写进去,能解释一下这是为什么吗?
嗯 看来根本原因还是地址出了问题。