flash.c 3.2 KB

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