drv_flash_f1.c 6.3 KB

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