w25qxx.c 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  1. #include "main.h"
  2. #include "spi.h"
  3. #include "w25qxx.h"
  4. #include "iwdg.h"
  5. /**
  6. * W25Q128 代表128Mb,实际 128/8 = 16 MB
  7. */
  8. /*
  9. 引脚配置:
  10. SPI1_CS PB12
  11. SPI1_SCK PB13
  12. SPI1_MISO PB14
  13. SPI1_MOSI PB15
  14. */
  15. /**
  16. * @brief spi 读一个字节
  17. * @param None
  18. * @retval None
  19. */
  20. uint32_t W25_Tempaddress = W25Q128_TEMP_DATA_BASE_ADDR; // 当前写入地址
  21. volatile uint8_t flash_print_busy = 0;
  22. uint8_t flash_print_buffer[W25Q128_PAGE_SIZE] = {0};;
  23. uint32_t flash_print_page = 0;
  24. uint32_t flash_print_end_page = 0;
  25. uint8_t sector_erased_flag[W25Q128_SECTOR_COUNT] = {0}; // 记录哪些扇区被擦除过
  26. static uint8_t spi_flash_read_byte(void)
  27. {
  28. uint8_t t_data, r_data;
  29. if(HAL_SPI_TransmitReceive(&hspi2, &t_data, &r_data, 1, 0xFFFFFF) != HAL_OK)
  30. {
  31. r_data = 0xff;
  32. }
  33. return r_data;
  34. }
  35. /**
  36. * @brief spi 写一个字节
  37. * @param byte 写入的字节
  38. * @retval 写状态 0成功 1失败
  39. */
  40. static uint8_t spi_flash_send_byte(uint8_t byte)
  41. {
  42. uint8_t r_data;
  43. if(HAL_SPI_TransmitReceive(&hspi2, &byte, &r_data, 1, 0xFFFFFF) != HAL_OK)
  44. {
  45. return 1;
  46. }
  47. return 0;
  48. }
  49. /**
  50. * @brief FLASH 写使能
  51. * @param None
  52. * @retval None
  53. */
  54. static void spi_flash_write_enable(void)
  55. {
  56. SPI_FLASH_CS_LOW();/* 通讯开始:CS低 */
  57. spi_flash_send_byte(W25X_WriteEnable); /* 发送写使能命令*/
  58. SPI_FLASH_CS_HIGH();/*通讯结束:CS高 */
  59. }
  60. /**
  61. * @brief FLASH 等待写结束
  62. * @param None
  63. * @retval None
  64. */
  65. void spi_flash_wait_for_write_end(void)
  66. {
  67. uint8_t state = 0;
  68. SPI_FLASH_CS_LOW();/* 选择 FLASH: CS 低 */
  69. spi_flash_send_byte(W25X_ReadStatusReg); /* 发送 读状态寄存器 命令 */
  70. /* 若FLASH忙碌,则等待 */
  71. do
  72. {/* 读取FLASH芯片的状态寄存器 */
  73. state = spi_flash_read_byte();
  74. }
  75. while((state & 0x01) == SET);/* 正在写入标志 */
  76. SPI_FLASH_CS_HIGH(); /* 停止信号 FLASH: CS 高 */
  77. }
  78. /**
  79. * @brief FLASH 读ID
  80. * @param None
  81. * @retval None
  82. */
  83. uint32_t spi_flash_read_ID(void)
  84. {
  85. uint32_t temp, temp0, temp1, temp2;
  86. SPI_FLASH_CS_LOW();/* 开始通讯:CS低电平 */
  87. spi_flash_send_byte(W25X_JedecDeviceID);/* 发送JEDEC指令,读取ID */
  88. temp0 = spi_flash_read_byte();
  89. temp1 = spi_flash_read_byte();
  90. temp2 = spi_flash_read_byte();
  91. SPI_FLASH_CS_HIGH();/* 停止通讯:CS高电平 */
  92. temp = (temp0 << 16) | (temp1 << 8) | temp2;
  93. return temp;
  94. }
  95. /**
  96. * @brief 读FLASH
  97. * @param addr 读flash的起始地址
  98. * @param pdata 读到的数据存放起始地址
  99. * pdata size 读数据大小
  100. * @retval None
  101. */
  102. void spi_flash_read(uint32_t addr,uint8_t *pdata, uint16_t size)
  103. {
  104. SPI_FLASH_CS_LOW();
  105. spi_flash_send_byte(W25X_ReadData);
  106. spi_flash_send_byte((addr & 0xFF0000) >> 16);
  107. spi_flash_send_byte((addr & 0xFF00) >> 8);
  108. spi_flash_send_byte(addr & 0xFF);
  109. while (size--)
  110. {
  111. *pdata = spi_flash_read_byte();
  112. pdata++;
  113. }
  114. SPI_FLASH_CS_HIGH();
  115. }
  116. /**
  117. * @brief 按页写FLASH
  118. * @param addr 写入flash的起始地址
  119. * @param pdata 写入数据的起始地址
  120. * pdata size 写数据大小
  121. * @retval None
  122. */
  123. void spi_flash_page_write(uint32_t addr, uint8_t *pdata, uint16_t size)
  124. {
  125. uint16_t i;
  126. spi_flash_write_enable();/* 发送FLASH写使能命令 */
  127. SPI_FLASH_CS_LOW();/* 选择FLASH: CS低电平 */
  128. spi_flash_send_byte(W25X_PageProgram);/* 写送写指令*/
  129. spi_flash_send_byte((uint8_t)((addr)>>16));
  130. spi_flash_send_byte((uint8_t)((addr)>>8));
  131. spi_flash_send_byte((uint8_t)addr);
  132. for(i = 0; i < size; i++)
  133. {
  134. spi_flash_send_byte(pdata[i]);
  135. }
  136. SPI_FLASH_CS_HIGH();/* 停止信号 FLASH: CS 高电平 */
  137. spi_flash_wait_for_write_end();/* 等待写入完毕*/
  138. }
  139. /**
  140. * @brief 写FLASH
  141. * @param addr 写入flash的起始地址
  142. * @param pdata 写入数据的起始地址
  143. * pdata size 写数据大小
  144. * @retval None
  145. */
  146. void spi_flash_write(uint32_t addr, uint8_t *pdata, uint32_t size)
  147. {
  148. uint32_t page_remain;
  149. page_remain = 256 - addr%256;
  150. if(size <= page_remain)
  151. {
  152. page_remain = size;
  153. }
  154. while(1)
  155. {
  156. // 确保写入之前发送 Write Enable 命令
  157. spi_flash_write_enable();
  158. spi_flash_page_write(addr, pdata, page_remain);
  159. if(size == page_remain)
  160. break;
  161. else
  162. {
  163. pdata += page_remain;
  164. addr += page_remain;
  165. size -= page_remain;
  166. if(size > 256)
  167. page_remain = 256;
  168. else
  169. page_remain = size;
  170. }
  171. }
  172. }
  173. /**
  174. * @brief 擦除FLASH扇区
  175. * @param sector_addr 扇区的起始地址
  176. * @retval None
  177. */
  178. void spi_flash_sector_erase(uint32_t sector_addr)
  179. {
  180. spi_flash_write_enable();/* 发送FLASH写使能命令 */
  181. spi_flash_wait_for_write_end();
  182. /* 擦除扇区 */
  183. SPI_FLASH_CS_LOW();/* 选择FLASH: CS低电平 */
  184. spi_flash_send_byte(W25X_SectorErase);/* 发送扇区擦除指令*/
  185. spi_flash_send_byte((sector_addr & 0xFF0000) >> 16);/*发送擦除扇区地址的高位*/
  186. spi_flash_send_byte((sector_addr & 0xFF00) >> 8);/* 发送擦除扇区地址的中位 */
  187. spi_flash_send_byte(sector_addr & 0xFF); /* 发送擦除扇区地址的低位 */
  188. /* 停止信号 FLASH: CS 高电平 */
  189. SPI_FLASH_CS_HIGH();
  190. /* 等待擦除完毕*/
  191. spi_flash_wait_for_write_end();
  192. }
  193. /**
  194. * @brief 擦除FLASH块
  195. * @param None
  196. * @retval None
  197. */
  198. void spi_flash_block_erase(void)
  199. {
  200. spi_flash_write_enable();
  201. SPI_FLASH_CS_LOW();
  202. spi_flash_send_byte(W25X_ChipErase);
  203. SPI_FLASH_CS_HIGH();
  204. spi_flash_wait_for_write_end();
  205. }
  206. /*进入深度掉电模式*/
  207. void spi_flash_enter_deep_power_down(void)
  208. {
  209. SPI_FLASH_CS_LOW(); // 选择FLASH: CS低电平
  210. spi_flash_send_byte(W25X_PowerDown); // 发送深度掉电模式指令0xB9
  211. SPI_FLASH_CS_HIGH(); // 停止信号 FLASH: CS 高电平
  212. }
  213. /*退出深度掉电模式*/
  214. void spi_flash_release_from_deep_power_down(void)
  215. {
  216. SPI_FLASH_CS_LOW(); // 选择FLASH: CS低电平
  217. spi_flash_send_byte(W25X_ReleasePowerDown); // 发送退出深度掉电模式指令0XAB
  218. SPI_FLASH_CS_HIGH(); // 停止信号 FLASH: CS 高电平
  219. // 释放后需等待一段时间,确保W25Q16完全唤醒
  220. HAL_Delay(1); // 延时1ms,确保Flash已完全唤醒
  221. }
  222. /*将温度压力数据循环写入flash*/
  223. void flash_write_tempprecoh2_data(uint8_t* data, uint32_t length)
  224. {
  225. // 写满了,从头循环并清除记录
  226. if ((W25_Tempaddress + length) >= W25Q128_TEMP_DATA_MAX_ADDR)
  227. {
  228. W25_Tempaddress = W25Q128_TEMP_DATA_BASE_ADDR;
  229. // 擦除全部相关区域扇区
  230. for (uint32_t sector = 0; sector < W25Q128_SECTOR_COUNT; sector++)
  231. {
  232. spi_flash_sector_erase(sector * W25Q128_SECTOR_SIZE);
  233. spi_flash_wait_for_write_end();
  234. sector_erased_flag[sector] = 1; // 标记已擦除
  235. }
  236. }
  237. // 计算本次写入涉及的扇区
  238. uint32_t start_sector = W25_Tempaddress / W25Q128_SECTOR_SIZE;
  239. uint32_t end_sector = (W25_Tempaddress + length - 1) / W25Q128_SECTOR_SIZE;
  240. // 擦除未擦除的扇区
  241. for (uint32_t sector = start_sector; sector <= end_sector; sector++)
  242. {
  243. if (sector_erased_flag[sector] == 0)
  244. {
  245. spi_flash_sector_erase(sector * W25Q128_SECTOR_SIZE);
  246. spi_flash_wait_for_write_end();
  247. sector_erased_flag[sector] = 1; // 标记为已擦除
  248. }
  249. }
  250. // 写入数据
  251. spi_flash_write(W25_Tempaddress, data, length);
  252. spi_flash_wait_for_write_end();
  253. // 更新写入指针
  254. W25_Tempaddress += length;
  255. }
  256. /*读取flash*/
  257. void flash_dma_print_all_pages_start(void)
  258. {
  259. if (flash_print_busy) return;
  260. flash_print_page = FLASH_PRINT_START_PAGE;
  261. flash_print_end_page = FLASH_PRINT_END_PAGE;
  262. flash_print_busy = 1;
  263. flash_dma_print_next_page();
  264. }
  265. void flash_dma_print_next_page(void)
  266. {
  267. if (flash_print_page > flash_print_end_page)
  268. {
  269. printf("\r\n[PRINT] Pages %lu to %lu printed\r\n",
  270. (unsigned long)FLASH_PRINT_START_PAGE,
  271. (unsigned long)FLASH_PRINT_END_PAGE);
  272. flash_print_busy = 0;
  273. return;
  274. }
  275. uint32_t addr = flash_print_page * W25Q128_PAGE_SIZE;
  276. spi_flash_read(addr, flash_print_buffer, W25Q128_PAGE_SIZE);
  277. // 打印页号
  278. printf("\r\n[Page %lu - 0x%08lX]:\r\n",
  279. (unsigned long)flash_print_page,
  280. (unsigned long)addr);
  281. // 发送当前页数据
  282. // 每16字节打印一行
  283. for (uint32_t i = 0; i < W25Q128_PAGE_SIZE; i++)
  284. {
  285. printf("%02X ", flash_print_buffer[i]);
  286. if ((i + 1) % 16 == 0)
  287. printf("\r\n");
  288. }
  289. flash_print_page++; // 提前准备下一页
  290. flash_dma_print_next_page();
  291. }
  292. /**
  293. * @brief 同步打印 Flash 中某一页到另一页的数据(每页256字节,按16字节一行打印)
  294. *
  295. * @param start 起始页号(包含)
  296. * @param end 结束页号(包含)
  297. */
  298. void print_flash_pages(uint32_t start, uint32_t end)
  299. {
  300. for (uint32_t page = start; page <= end; page++)
  301. {
  302. uint32_t addr = page * W25Q128_PAGE_SIZE;
  303. spi_flash_read(addr, flash_print_buffer, W25Q128_PAGE_SIZE);
  304. printf("\r\n[Page %lu - 0x%08lX]:\r\n", (unsigned long)page, (unsigned long)addr);
  305. for (uint32_t i = 0; i < W25Q128_PAGE_SIZE; i++)
  306. {
  307. printf("%02X ", flash_print_buffer[i]);
  308. if ((i + 1) % 16 == 0)
  309. printf("\r\n");
  310. }
  311. }
  312. printf("\r\n[PRINT] Pages %lu to %lu printed\r\n", (unsigned long)start, (unsigned long)end);
  313. }
  314. /**
  315. * @brief 清空flash中用于保存温度/压力等数据的区域
  316. * @param None
  317. * @retval None
  318. */
  319. void flash_clear_all_data(void)
  320. {
  321. spi_flash_write_enable(); // 发送写使能指令
  322. spi_flash_send_byte(0xC7); // 或者用 0x60,也是 Chip Erase 指令
  323. // 等待擦除完成
  324. spi_flash_wait_for_write_end();
  325. // 擦除完成后,写入指针归零
  326. W25_Tempaddress = 0;
  327. memset(sector_erased_flag, 0, sizeof(sector_erased_flag));
  328. printf("W25FLASH complete emptied\r\n");
  329. }