INflash.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. /**
  2. ******************************************************************************
  3. * 版 本 :V1.0.0
  4. * 作 者 :liuyanbin
  5. * 版权所有,盗版必究。
  6. * Copyright(C) All rights reserved
  7. ******************************************************************************
  8. * 历史记录
  9. * 序号 版本 变更内容 作者 日期
  10. * 1 V0.0.1 实现flash存储 刘艳斌 2024/1/24
  11. *
  12. ******************************************************************************
  13. */
  14. #include "stm32f10x.h"
  15. #include <stdint.h>
  16. #include "INflash.h"
  17. #include "Regist.h"
  18. #include "string.h"
  19. #include "Delay.h"
  20. #include "Log_Module.h"
  21. typedef uint16_t uint16;
  22. typedef uint32_t uint32;
  23. uint16 STMFLASH_BUF[STM32_SECTOR_SIZE/2];//最多是2K字节
  24. void write_to_flash(uint32 WriteAddr,uint16 *pBuffer,uint16 NumToWrite)
  25. {
  26. uint16 i;
  27. for(i=0;i<NumToWrite;i++)
  28. {
  29. FLASH_ProgramHalfWord(WriteAddr,pBuffer[i]);
  30. WriteAddr+=2;//地址增加2.
  31. }
  32. }
  33. //读取指定地址的半字(16位数据)
  34. //faddr:读地址(此地址必须为2的倍数!!)
  35. //返回值:对应数据.
  36. uint16 STMFLASH_ReadHalfWord(uint32 faddr)
  37. {
  38. return *(vu16*)faddr;
  39. }
  40. //从指定地址开始读出指定长度的数据
  41. //ReadAddr:起始地址
  42. //pBuffer:数据指针
  43. //NumToWrite:半字(16位)数
  44. void STMFLASH_Read(uint32 ReadAddr,uint16 *pBuffer,uint16 NumToRead)
  45. {
  46. uint16 i;
  47. for(i=0;i<NumToRead;i++)
  48. {
  49. pBuffer[i]=STMFLASH_ReadHalfWord(ReadAddr);//读取2个字节.
  50. ReadAddr+=2;//偏移2个字节.
  51. }
  52. }
  53. void STMFLASH_Write(uint32 WriteAddr,uint16 *pBuffer,uint16 NumToWrite)
  54. {
  55. uint32 secpos; //扇区地址
  56. uint16 secoff; //扇区内偏移地址(16位字计算)
  57. uint16 secremain; //扇区内剩余地址(16位字计算)
  58. uint16 i;
  59. uint32 offaddr; //去掉0X08000000后的地址
  60. if(WriteAddr<STM32_FLASH_BASE||(WriteAddr>=(STM32_FLASH_BASE+1024*STM32_FLASH_SIZE)))return;//非法地址
  61. FLASH_Unlock();
  62. offaddr=WriteAddr-STM32_FLASH_BASE; //实际偏移地址.
  63. secpos=offaddr/STM32_SECTOR_SIZE; //扇区地址 0~127 for STM32F103RBT6
  64. secoff=(offaddr%STM32_SECTOR_SIZE)/2; //在扇区内的偏移(2个字节为基本单位.)
  65. secremain=STM32_SECTOR_SIZE/2-secoff; //扇区剩余空间大小
  66. if(NumToWrite<=secremain)secremain=NumToWrite;//不大于该扇区范围
  67. while(1)
  68. {
  69. STMFLASH_Read(secpos*STM32_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM32_SECTOR_SIZE/2);//读出整个扇区的内容
  70. for(i=0;i<secremain;i++)//校验数据
  71. {
  72. if(STMFLASH_BUF[secoff+i]!=0XFFFF)break;//需要擦除
  73. }
  74. if(i<secremain)//需要擦除
  75. {
  76. FLASH_ErasePage(secpos*STM32_SECTOR_SIZE+STM32_FLASH_BASE);//擦除这个扇区
  77. for(i=0;i<secremain;i++)//复制
  78. {
  79. STMFLASH_BUF[i+secoff]=pBuffer[i];
  80. }
  81. write_to_flash(secpos*STM32_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM32_SECTOR_SIZE/2);//写入整个扇区
  82. }else write_to_flash(WriteAddr,pBuffer,secremain);//写已经擦除了的,直接写入扇区剩余区间.
  83. if(NumToWrite==secremain)break;//写入结束了
  84. else//写入未结束
  85. {
  86. secpos++; //扇区地址增1
  87. secoff=0; //偏移位置为0
  88. pBuffer+=secremain; //指针偏移
  89. WriteAddr+=secremain; //写地址偏移
  90. NumToWrite-=secremain; //字节(16位)数递减
  91. if(NumToWrite>(STM32_SECTOR_SIZE/2))secremain=STM32_SECTOR_SIZE/2;//下一个扇区还是写不完
  92. else secremain=NumToWrite;//下一个扇区可以写完了
  93. }
  94. };
  95. FLASH_Lock();//上锁
  96. }