drv_flash_f1.c 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273
  1. /*
  2. * Copyright (c) 2006-2021, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2018-12-5 SummerGift first version
  9. * 2020-03-05 redoc support stm32f103vg
  10. *
  11. */
  12. #include "board.h"
  13. #ifdef BSP_USING_ON_CHIP_FLASH
  14. #include "drv_config.h"
  15. #include "drv_flash.h"
  16. #if defined(RT_USING_FAL)
  17. #include "fal.h"
  18. #endif
  19. //#define DRV_DEBUG
  20. #define LOG_TAG "drv.flash"
  21. #include <drv_log.h>
  22. /**
  23. * @brief Gets the page of a given address
  24. * @param Addr: Address of the FLASH Memory
  25. * @retval The page of a given address
  26. */
  27. static uint32_t GetPage(uint32_t addr)
  28. {
  29. uint32_t page = 0;
  30. page = RT_ALIGN_DOWN(addr, FLASH_PAGE_SIZE);
  31. return page;
  32. }
  33. /**
  34. * Read data from flash.
  35. * @note This operation's units is word.
  36. *
  37. * @param addr flash address
  38. * @param buf buffer to store read data
  39. * @param size read bytes size
  40. *
  41. * @return result
  42. */
  43. int stm32_flash_read(rt_uint32_t addr, rt_uint8_t *buf, size_t size)
  44. {
  45. size_t i;
  46. if ((addr + size) > STM32_FLASH_END_ADDRESS)
  47. {
  48. LOG_E("read outrange flash size! addr is (0x%p)", (void *)(addr + size));
  49. return -RT_EINVAL;
  50. }
  51. for (i = 0; i < size; i++, buf++, addr++)
  52. {
  53. *buf = *(rt_uint8_t *) addr;
  54. }
  55. return size;
  56. }
  57. /**
  58. * Write data to flash.
  59. * @note This operation's units is word.
  60. * @note This operation must after erase. @see flash_erase.
  61. *
  62. * @param addr flash address
  63. * @param buf the write data buffer
  64. * @param size write bytes size
  65. *
  66. * @return result
  67. */
  68. int stm32_flash_write(rt_uint32_t addr, const rt_uint8_t *buf, size_t size)
  69. {
  70. rt_err_t result = RT_EOK;
  71. rt_uint32_t end_addr = addr + size;
  72. if (addr % 4 != 0)
  73. {
  74. LOG_E("write addr must be 4-byte alignment");
  75. return -RT_EINVAL;
  76. }
  77. if ((end_addr) > STM32_FLASH_END_ADDRESS)
  78. {
  79. LOG_E("write outrange flash size! addr is (0x%p)", (void *)(addr + size));
  80. return -RT_EINVAL;
  81. }
  82. HAL_FLASH_Unlock();
  83. while (addr < end_addr)
  84. {
  85. if (HAL_FLASH_Program(FLASH_TYPEPROGRAM_WORD, addr, *((rt_uint32_t *)buf)) == HAL_OK)
  86. {
  87. if (*(rt_uint32_t *)addr != *(rt_uint32_t *)buf)
  88. {
  89. result = -RT_ERROR;
  90. break;
  91. }
  92. addr += 4;
  93. buf += 4;
  94. }
  95. else
  96. {
  97. result = -RT_ERROR;
  98. break;
  99. }
  100. }
  101. HAL_FLASH_Lock();
  102. if (result != RT_EOK)
  103. {
  104. return result;
  105. }
  106. return size;
  107. }
  108. /**
  109. * Erase data on flash with bank.
  110. * @note This operation is irreversible.
  111. * @note This operation's units is different which on many chips.
  112. *
  113. * @param bank flash bank
  114. * @param addr flash address
  115. * @param size erase bytes size
  116. *
  117. * @return result
  118. */
  119. int stm32_flash_erase_bank(uint32_t bank, rt_uint32_t addr, size_t size)
  120. {
  121. rt_err_t result = RT_EOK;
  122. uint32_t PAGEError = 0;
  123. /*Variable used for Erase procedure*/
  124. FLASH_EraseInitTypeDef EraseInitStruct;
  125. if ((addr + size) > STM32_FLASH_END_ADDRESS)
  126. {
  127. LOG_E("ERROR: erase outrange flash size! addr is (0x%p)\n", (void *)(addr + size));
  128. return -RT_EINVAL;
  129. }
  130. HAL_FLASH_Unlock();
  131. /* Fill EraseInit structure*/
  132. EraseInitStruct.TypeErase = FLASH_TYPEERASE_PAGES;
  133. EraseInitStruct.PageAddress = GetPage(addr);
  134. EraseInitStruct.NbPages = (size + FLASH_PAGE_SIZE - 1) / FLASH_PAGE_SIZE;
  135. EraseInitStruct.Banks = bank;
  136. if (HAL_FLASHEx_Erase(&EraseInitStruct, &PAGEError) != HAL_OK)
  137. {
  138. result = -RT_ERROR;
  139. goto __exit;
  140. }
  141. __exit:
  142. HAL_FLASH_Lock();
  143. if (result != RT_EOK)
  144. {
  145. return result;
  146. }
  147. LOG_D("erase done: addr (0x%p), size %d", (void *)addr, size);
  148. return size;
  149. }
  150. /**
  151. * Erase data on flash .
  152. * @note This operation is irreversible.
  153. * @note This operation's units is different which on many chips.
  154. *
  155. * @param addr flash address
  156. * @param size erase bytes size
  157. *
  158. * @return result
  159. */
  160. int stm32_flash_erase(rt_uint32_t addr, size_t size)
  161. {
  162. #if defined(FLASH_BANK2_END)
  163. rt_err_t result = RT_EOK;
  164. rt_uint32_t addr_bank1 = 0;
  165. rt_uint32_t size_bank1 = 0;
  166. rt_uint32_t addr_bank2 = 0;
  167. rt_uint32_t size_bank2 = 0;
  168. if((addr + size) <= FLASH_BANK1_END)
  169. {
  170. addr_bank1 = addr;
  171. size_bank1 = size;
  172. size_bank2 = 0;
  173. }
  174. else if(addr > FLASH_BANK1_END)
  175. {
  176. size_bank1 = 0;
  177. addr_bank2 = addr;
  178. size_bank2 = size;
  179. }
  180. else
  181. {
  182. addr_bank1 = addr;
  183. size_bank1 = FLASH_BANK1_END + 1 - addr_bank1;
  184. addr_bank2 = FLASH_BANK1_END + 1;
  185. size_bank2 = addr + size - (FLASH_BANK1_END + 1);
  186. }
  187. if(size_bank1)
  188. {
  189. LOG_D("bank1: addr (0x%p), size %d", (void *)addr_bank1, size_bank1);
  190. if(size_bank1 != stm32_flash_erase_bank(FLASH_BANK_1, addr_bank1, size_bank1))
  191. {
  192. result = -RT_ERROR;
  193. goto __exit;
  194. }
  195. }
  196. if(size_bank2)
  197. {
  198. LOG_D("bank2: addr (0x%p), size %d", (void *)addr_bank2, size_bank2);
  199. if(size_bank2 != stm32_flash_erase_bank(FLASH_BANK_2, addr_bank2, size_bank2))
  200. {
  201. result = -RT_ERROR;
  202. goto __exit;
  203. }
  204. }
  205. __exit:
  206. if(result != RT_EOK)
  207. {
  208. return result;
  209. }
  210. return size_bank1 + size_bank2;
  211. #else
  212. return stm32_flash_erase_bank(FLASH_BANK_1, addr, size);
  213. #endif
  214. }
  215. #if defined(RT_USING_FAL)
  216. static int fal_flash_read(long offset, rt_uint8_t *buf, size_t size);
  217. static int fal_flash_write(long offset, const rt_uint8_t *buf, size_t size);
  218. static int fal_flash_erase(long offset, size_t size);
  219. const struct fal_flash_dev stm32_onchip_flash = { "onchip_flash", STM32_FLASH_START_ADRESS, STM32_FLASH_SIZE, FLASH_PAGE_SIZE, {NULL, fal_flash_read, fal_flash_write, fal_flash_erase} };
  220. static int fal_flash_read(long offset, rt_uint8_t *buf, size_t size)
  221. {
  222. return stm32_flash_read(stm32_onchip_flash.addr + offset, buf, size);
  223. }
  224. static int fal_flash_write(long offset, const rt_uint8_t *buf, size_t size)
  225. {
  226. return stm32_flash_write(stm32_onchip_flash.addr + offset, buf, size);
  227. }
  228. static int fal_flash_erase(long offset, size_t size)
  229. {
  230. return stm32_flash_erase(stm32_onchip_flash.addr + offset, size);
  231. }
  232. #endif
  233. #endif /* BSP_USING_ON_CHIP_FLASH */