stmflash.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. #include "stmflash.h"
  2. #include "usart.h"
  3. uint16_t STMFLASH_ReadHalfWord(uint32_t faddr)
  4. {
  5. return *(uint16_t*)faddr;
  6. }
  7. #if STM32_FLASH_WREN //如果使能了写
  8. //不检查的写入
  9. //WriteAddr:起始地址
  10. //pBuffer:数据指针
  11. //NumToWrite:半字(16位)数
  12. void STMFLASH_Write_NoCheck(uint32_t WriteAddr,uint16_t *pBuffer,uint16_t NumToWrite)
  13. {
  14. uint16_t i;
  15. for(i=0;i<NumToWrite;i++)
  16. {
  17. HAL_FLASH_Program(FLASH_TYPEPROGRAM_HALFWORD,WriteAddr,pBuffer[i]);
  18. WriteAddr+=2;//地址增加2.
  19. }
  20. }
  21. //从指定地址开始写入指定长度的数据
  22. //WriteAddr:起始地址(此地址必须为2的倍数!!)
  23. //pBuffer:数据指针
  24. //NumToWrite:半字(16位)数(就是要写入的16位数据的个数.)
  25. #if STM32_FLASH_SIZE<256
  26. #define STM_SECTOR_SIZE 1024 //字节
  27. #else
  28. #define STM_SECTOR_SIZE 2048
  29. #endif
  30. uint16_t STMFLASH_BUF[STM_SECTOR_SIZE/2];//最多是2K字节
  31. void STMFLASH_Write(uint32_t WriteAddr,uint16_t *pBuffer,uint16_t NumToWrite)
  32. {
  33. uint32_t secpos; //扇区地址
  34. uint16_t secoff; //扇区内偏移地址(16位字计算)
  35. uint16_t secremain; //扇区内剩余地址(16位字计算)
  36. uint16_t i;
  37. uint32_t offaddr; //去掉0X08000000后的地址
  38. if(WriteAddr<STM32_FLASH_BASE||(WriteAddr>=(STM32_FLASH_BASE+1024*STM32_FLASH_SIZE)))return;//非法地址
  39. HAL_FLASH_Unlock(); //解锁
  40. offaddr=WriteAddr-STM32_FLASH_BASE; //实际偏移地址.
  41. secpos=offaddr/STM_SECTOR_SIZE; //扇区地址 0~127 for STM32F103RBT6
  42. secoff=(offaddr%STM_SECTOR_SIZE)/2; //在扇区内的偏移(2个字节为基本单位.)
  43. secremain=STM_SECTOR_SIZE/2-secoff; //扇区剩余空间大小
  44. if(NumToWrite<=secremain)secremain=NumToWrite;//不大于该扇区范围
  45. while(1)
  46. {
  47. STMFLASH_Read(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//读出整个扇区的内容
  48. for(i=0;i<secremain;i++) //校验数据
  49. {
  50. if(STMFLASH_BUF[secoff+i]!=0XFFFF)break;//需要擦除
  51. }
  52. if(i<secremain) //需要擦除
  53. {
  54. FLASH_PageErase(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE); //擦除这个扇区
  55. FLASH_WaitForLastOperation(FLASH_WAITETIME); //等待上次操作完成
  56. CLEAR_BIT(FLASH->CR, FLASH_CR_PER); //清除CR寄存器的PER位,此操作应该在FLASH_PageErase()中完成!
  57. //但是HAL库里面并没有做,应该是HAL库bug!
  58. for(i=0;i<secremain;i++)//复制
  59. {
  60. STMFLASH_BUF[i+secoff]=pBuffer[i];
  61. }
  62. STMFLASH_Write_NoCheck(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//写入整个扇区
  63. }else
  64. {
  65. FLASH_WaitForLastOperation(FLASH_WAITETIME); //等待上次操作完成
  66. STMFLASH_Write_NoCheck(WriteAddr,pBuffer,secremain);//写已经擦除了的,直接写入扇区剩余区间.
  67. }
  68. if(NumToWrite==secremain)break;//写入结束了
  69. else//写入未结束
  70. {
  71. secpos++; //扇区地址增1
  72. secoff=0; //偏移位置为0
  73. pBuffer+=secremain; //指针偏移
  74. WriteAddr+=secremain*2; //写地址偏移(16位数据地址,需要*2)
  75. NumToWrite-=secremain; //字节(16位)数递减
  76. if(NumToWrite>(STM_SECTOR_SIZE/2))secremain=STM_SECTOR_SIZE/2;//下一个扇区还是写不完
  77. else secremain=NumToWrite;//下一个扇区可以写完了
  78. }
  79. };
  80. HAL_FLASH_Lock(); //上锁
  81. }
  82. #endif
  83. //从指定地址开始读出指定长度的数据
  84. //ReadAddr:起始地址
  85. //pBuffer:数据指针
  86. //NumToWrite:半字(16位)数
  87. void STMFLASH_Read(uint32_t ReadAddr,uint16_t *pBuffer,uint16_t NumToRead)
  88. {
  89. uint16_t i;
  90. for(i=0;i<NumToRead;i++)
  91. {
  92. pBuffer[i]=STMFLASH_ReadHalfWord(ReadAddr);//读取2个字节.
  93. ReadAddr+=2;//偏移2个字节.
  94. }
  95. }
  96. //////////////////////////////////////////测试用///////////////////////////////////////////
  97. //WriteAddr:起始地址
  98. //WriteData:要写入的数据
  99. void Test_Write(uint32_t WriteAddr,uint16_t WriteData)
  100. {
  101. STMFLASH_Write(WriteAddr,&WriteData,1);//写入一个字
  102. }