FLASH.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. #include "main.h"
  2. //读取指定地址的半字(16位数据)
  3. //faddr:读地址(此地址必须为2的倍数!!)
  4. //返回值:对应数据.
  5. u16 STMFLASH_ReadHalfWord(u32 faddr)
  6. {
  7. return *(vu16*)faddr;
  8. }
  9. //不检查的写入
  10. //WriteAddr:起始地址
  11. //pBuffer:数据指针
  12. //NumToWrite:半字(16位)数
  13. void STMFLASH_Write_NoCheck(u32 WriteAddr,u16 *pBuffer,u16 NumToWrite)
  14. {
  15. u16 i;
  16. for(i=0;i<NumToWrite;i++)
  17. {
  18. FLASH_ProgramHalfWord(WriteAddr,pBuffer[i]);
  19. WriteAddr+=2;//地址增加2.
  20. }
  21. }
  22. //从指定地址开始写入指定长度的数据
  23. //WriteAddr:起始地址(此地址必须为2的倍数!!)
  24. //pBuffer:数据指针
  25. //NumToWrite:半字(16位)数(就是要写入的16位数据的个数.)
  26. #if STM32_FLASH_SIZE<256
  27. #define STM_SECTOR_SIZE 1024 //字节
  28. #else
  29. #define STM_SECTOR_SIZE 2048
  30. #endif
  31. u16 STMFLASH_BUF[STM_SECTOR_SIZE/2];//最多是2K字节
  32. void STMFLASH_Write(u32 WriteAddr,u16 *pBuffer,u16 NumToWrite)
  33. {
  34. u32 secpos; //扇区地址
  35. u16 secoff; //扇区内偏移地址(16位字计算)
  36. u16 secremain; //扇区内剩余地址(16位字计算)
  37. u16 i;
  38. u32 offaddr; //去掉0X08000000后的地址
  39. if(WriteAddr<STM32_FLASH_BASE||(WriteAddr>=(STM32_FLASH_BASE+1024*STM32_FLASH_SIZE)))return;//非法地址
  40. FLASH_Unlock(); //解锁
  41. offaddr=WriteAddr-STM32_FLASH_BASE; //实际偏移地址.
  42. secpos=offaddr/STM_SECTOR_SIZE; //扇区地址 0~127 for STM32F103RBT6
  43. secoff=(offaddr%STM_SECTOR_SIZE)/2; //在扇区内的偏移(2个字节为基本单位.)
  44. secremain=STM_SECTOR_SIZE/2-secoff; //扇区剩余空间大小
  45. if(NumToWrite<=secremain)secremain=NumToWrite;//不大于该扇区范围
  46. while(1)
  47. {
  48. STMFLASH_Read(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//读出整个扇区的内容
  49. for(i=0;i<secremain;i++)//校验数据
  50. {
  51. if(STMFLASH_BUF[secoff+i]!=0XFFFF)break;//需要擦除
  52. }
  53. if(i<secremain)//需要擦除
  54. {
  55. FLASH_ErasePage(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE);//擦除这个扇区
  56. for(i=0;i<secremain;i++)//复制
  57. {
  58. STMFLASH_BUF[i+secoff]=pBuffer[i];
  59. }
  60. STMFLASH_Write_NoCheck(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//写入整个扇区
  61. }else STMFLASH_Write_NoCheck(WriteAddr,pBuffer,secremain);//写已经擦除了的,直接写入扇区剩余区间.
  62. if(NumToWrite==secremain)break;//写入结束了
  63. else//写入未结束
  64. {
  65. secpos++; //扇区地址增1
  66. secoff=0; //偏移位置为0
  67. pBuffer+=secremain; //指针偏移
  68. WriteAddr+=secremain+2; //写地址偏移
  69. NumToWrite-=secremain; //字节(16位)数递减
  70. if(NumToWrite>(STM_SECTOR_SIZE/2))secremain=STM_SECTOR_SIZE/2;//下一个扇区还是写不完
  71. else secremain=NumToWrite;//下一个扇区可以写完了
  72. }
  73. };
  74. FLASH_Lock();//上锁
  75. }
  76. //从指定地址开始读出指定长度的数据
  77. //ReadAddr:起始地址
  78. //pBuffer:数据指针
  79. //NumToWrite:半字(16位)数
  80. void STMFLASH_Read(u32 ReadAddr,u16 *pBuffer,u16 NumToRead)
  81. {
  82. u16 i;
  83. for(i=0;i<NumToRead;i++)
  84. {
  85. pBuffer[i]=STMFLASH_ReadHalfWord(ReadAddr);//读取2个字节.
  86. ReadAddr+=2;
  87. }
  88. }
  89. #define LockAddr 0x800FF10
  90. uint16_t LockNum = 0;
  91. // 程序运行锁
  92. // num 允许重启的次数
  93. uint8_t SystemRunLock(uint8_t num)
  94. {
  95. STMFLASH_Read(LockAddr, &LockNum, 1);
  96. DELAYClass.DelayMs(5);
  97. if(LockNum > 255)
  98. {
  99. LockNum = 0;
  100. STMFLASH_Write(LockAddr, &LockNum, 1);
  101. DELAYClass.DelayMs(5);
  102. NVIC_SystemReset();
  103. }
  104. else
  105. {
  106. LockNum++;
  107. STMFLASH_Write(LockAddr, &LockNum, 1);
  108. if((num == 0) || (LockNum <= num)) return 1;
  109. }
  110. return 0;
  111. }