stmflash.c 3.7 KB

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