spi_flash_hal_iram.c 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. /*
  2. * SPDX-FileCopyrightText: 2015-2023 Espressif Systems (Shanghai) CO LTD
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include "sdkconfig.h"
  7. #include "hal/spi_flash_hal.h"
  8. #if SOC_SPI_MEM_SUPPORT_AUTO_SUSPEND
  9. void spi_flash_hal_setup_auto_suspend_mode(spi_flash_host_inst_t *host);
  10. void spi_flash_hal_disable_auto_resume_mode(spi_flash_host_inst_t *host);
  11. void spi_flash_hal_disable_auto_suspend_mode(spi_flash_host_inst_t *host);
  12. void spi_flash_hal_setup_auto_resume_mode(spi_flash_host_inst_t *host);
  13. #define SPI_FLASH_TSHSL2_SAFE_VAL_NS (30)
  14. #endif //SOC_SPI_MEM_SUPPORT_AUTO_SUSPEND
  15. #ifndef CONFIG_SPI_FLASH_ROM_IMPL
  16. #include "spi_flash_hal_common.inc"
  17. // HAL for
  18. // - MEMSPI
  19. // - SPI1~3 on ESP32/S2/S3/C3/H4/C2
  20. // The common part is in spi_flash_hal_common.inc
  21. void spi_flash_hal_erase_chip(spi_flash_host_inst_t *host)
  22. {
  23. spi_dev_t *dev = get_spi_dev(host);
  24. spi_flash_ll_erase_chip(dev);
  25. #if SOC_SPI_MEM_SUPPORT_AUTO_WAIT_IDLE
  26. if((((spi_flash_hal_context_t*)host)->flags & SPI_FLASH_HOST_CONTEXT_FLAG_AUTO_SUSPEND) == 0) {
  27. host->driver->poll_cmd_done(host);
  28. }
  29. #else
  30. host->driver->poll_cmd_done(host);
  31. #endif
  32. }
  33. // Only support 24bit address
  34. void spi_flash_hal_erase_sector(spi_flash_host_inst_t *host, uint32_t start_address)
  35. {
  36. spi_dev_t *dev = get_spi_dev(host);
  37. spi_flash_ll_set_addr_bitlen(dev, 24);
  38. spi_flash_ll_set_address(dev, start_address & ADDRESS_MASK_24BIT);
  39. spi_flash_ll_erase_sector(dev);
  40. #if SOC_SPI_MEM_SUPPORT_AUTO_WAIT_IDLE
  41. if((((spi_flash_hal_context_t*)host)->flags & SPI_FLASH_HOST_CONTEXT_FLAG_AUTO_SUSPEND) == 0) {
  42. host->driver->poll_cmd_done(host);
  43. }
  44. #else
  45. host->driver->poll_cmd_done(host);
  46. #endif
  47. }
  48. // Only support 24bit address
  49. void spi_flash_hal_erase_block(spi_flash_host_inst_t *host, uint32_t start_address)
  50. {
  51. spi_dev_t *dev = get_spi_dev(host);
  52. spi_flash_ll_set_addr_bitlen(dev, 24);
  53. spi_flash_ll_set_address(dev, start_address & ADDRESS_MASK_24BIT);
  54. spi_flash_ll_erase_block(dev);
  55. #if SOC_SPI_MEM_SUPPORT_AUTO_WAIT_IDLE
  56. if((((spi_flash_hal_context_t*)host)->flags & SPI_FLASH_HOST_CONTEXT_FLAG_AUTO_SUSPEND) == 0) {
  57. host->driver->poll_cmd_done(host);
  58. }
  59. #else
  60. host->driver->poll_cmd_done(host);
  61. #endif
  62. }
  63. // Only support 24bit address
  64. void spi_flash_hal_program_page(spi_flash_host_inst_t *host, const void *buffer, uint32_t address, uint32_t length)
  65. {
  66. spi_dev_t *dev = get_spi_dev(host);
  67. spi_flash_ll_set_addr_bitlen(dev, 24);
  68. spi_flash_ll_set_address(dev, (address & ADDRESS_MASK_24BIT) | (length << 24));
  69. spi_flash_ll_program_page(dev, buffer, length);
  70. host->driver->poll_cmd_done(host);
  71. }
  72. esp_err_t spi_flash_hal_set_write_protect(spi_flash_host_inst_t *host, bool wp)
  73. {
  74. spi_dev_t *dev = get_spi_dev(host);
  75. spi_flash_ll_set_write_protect(dev, wp);
  76. host->driver->poll_cmd_done(host);
  77. return ESP_OK;
  78. }
  79. #else // defined CONFIG_SPI_FLASH_ROM_IMPL
  80. static inline spi_dev_t *get_spi_dev(spi_flash_host_inst_t *host)
  81. {
  82. return ((spi_flash_hal_context_t*)host)->spi;
  83. }
  84. static inline int get_host_id(spi_flash_host_inst_t* host)
  85. {
  86. spi_dev_t *dev = get_spi_dev(host);
  87. return spi_flash_ll_hw_get_id(dev);
  88. }
  89. #endif // !CONFIG_SPI_FLASH_ROM_IMPL
  90. uint32_t spi_flash_hal_check_status(spi_flash_host_inst_t *host)
  91. {
  92. spi_dev_t *dev = get_spi_dev(host);
  93. uint32_t status = spi_flash_ll_host_idle(dev);
  94. #if SOC_SPI_MEM_SUPPORT_AUTO_WAIT_IDLE
  95. uint32_t sus_status = spimem_flash_ll_sus_status((spi_mem_dev_t*)dev) << 1;
  96. #else
  97. uint32_t sus_status = 0;
  98. #endif
  99. // Not clear if this is necessary, or only necessary if
  100. // chip->spi == SPI1. But probably doesn't hurt...
  101. if ((void*) dev == spi_flash_ll_get_hw(SPI1_HOST)) {
  102. #if CONFIG_IDF_TARGET_ESP32
  103. status &= spi_flash_ll_host_idle(&SPI0);
  104. #endif
  105. }
  106. //status and sus_status should be mutual exclusion
  107. return (status | sus_status);
  108. }
  109. esp_err_t spi_flash_hal_setup_read_suspend(spi_flash_host_inst_t *host, const spi_flash_sus_cmd_conf *sus_conf)
  110. {
  111. #if SOC_SPI_MEM_SUPPORT_AUTO_SUSPEND
  112. spi_mem_dev_t *dev = (spi_mem_dev_t *)spi_flash_ll_get_hw(SPI1_HOST);
  113. spi_flash_hal_context_t* ctx = (spi_flash_hal_context_t*)host;
  114. memcpy(&(ctx->sus_cfg), sus_conf, sizeof(spi_flash_sus_cmd_conf));
  115. spimem_flash_ll_suspend_cmd_setup(dev, sus_conf->sus_cmd);
  116. spimem_flash_ll_resume_cmd_setup(dev, sus_conf->res_cmd);
  117. #if SOC_SPI_MEM_SUPPORT_CHECK_SUS
  118. spimem_flash_ll_set_read_sus_status(dev, sus_conf->sus_mask);
  119. spimem_flash_ll_rd_sus_cmd_setup(dev, sus_conf->cmd_rdsr);
  120. #endif // SOC_SPI_MEM_SUPPORT_CHECK_SUS
  121. #endif // SOC_SPI_MEM_SUPPORT_AUTO_SUSPEND
  122. return ESP_OK;
  123. }
  124. #if SOC_SPI_MEM_SUPPORT_AUTO_SUSPEND
  125. void spi_flash_hal_setup_auto_suspend_mode(spi_flash_host_inst_t *host)
  126. {
  127. spi_mem_dev_t *dev = (spi_mem_dev_t*)spi_flash_ll_get_hw(SPI1_HOST);
  128. spi_flash_hal_context_t* ctx = (spi_flash_hal_context_t*)host;
  129. spimem_flash_ll_auto_wait_idle_init(dev, true);
  130. if (ctx->freq_mhz == 120) {
  131. spimem_flash_ll_set_wait_idle_dummy_phase(dev, ctx->extra_dummy);
  132. }
  133. spimem_flash_ll_auto_suspend_init(dev, true);
  134. // tsus = ceil(ctx->tsus_val * ctx->freq_mhz / spimem_flash_ll_get_tsus_unit_in_cycles);
  135. uint32_t tsus = (ctx->tsus_val * ctx->freq_mhz / spimem_flash_ll_get_tsus_unit_in_cycles(dev)) + ((ctx->tsus_val * ctx->freq_mhz) % spimem_flash_ll_get_tsus_unit_in_cycles(dev) != 0);
  136. spimem_flash_ll_set_sus_delay(dev, tsus);
  137. // tshsl2 = ceil(SPI_FLASH_TSHSL2_SAFE_VAL_NS * spimem_flash_ll_get_source_freq_mhz() * 0.001);
  138. uint32_t tshsl2 = (SPI_FLASH_TSHSL2_SAFE_VAL_NS * spimem_flash_ll_get_source_freq_mhz() / 1000) + ((SPI_FLASH_TSHSL2_SAFE_VAL_NS * spimem_flash_ll_get_source_freq_mhz()) % 1000 != 0);
  139. spimem_flash_set_cs_hold_delay(dev, tshsl2);
  140. spimem_flash_ll_sus_set_spi0_lock_trans(dev, SPIMEM_FLASH_LL_SPI0_MAX_LOCK_VAL_MSPI_TICKS);
  141. #if SOC_SPI_MEM_SUPPORT_CHECK_SUS
  142. spimem_flash_ll_sus_check_sus_setup(dev, true);
  143. #endif
  144. }
  145. void spi_flash_hal_setup_auto_resume_mode(spi_flash_host_inst_t *host)
  146. {
  147. spi_mem_dev_t *dev = (spi_mem_dev_t*)spi_flash_ll_get_hw(SPI1_HOST);
  148. spimem_flash_ll_auto_resume_init(dev, true);
  149. #if SOC_SPI_MEM_SUPPORT_CHECK_SUS
  150. spimem_flash_ll_res_check_sus_setup(dev, true);
  151. #endif
  152. }
  153. void spi_flash_hal_disable_auto_suspend_mode(spi_flash_host_inst_t *host)
  154. {
  155. spi_mem_dev_t *dev = (spi_mem_dev_t *)spi_flash_ll_get_hw(SPI1_HOST);
  156. spimem_flash_ll_auto_wait_idle_init(dev, false);
  157. spimem_flash_ll_auto_suspend_init(dev, false);
  158. #if SOC_SPI_MEM_SUPPORT_CHECK_SUS
  159. spimem_flash_ll_sus_check_sus_setup(dev, false);
  160. #endif
  161. }
  162. void spi_flash_hal_disable_auto_resume_mode(spi_flash_host_inst_t *host)
  163. {
  164. spi_mem_dev_t *dev = (spi_mem_dev_t*)spi_flash_ll_get_hw(SPI1_HOST);
  165. spimem_flash_ll_auto_resume_init(dev, false);
  166. #if SOC_SPI_MEM_SUPPORT_CHECK_SUS
  167. spimem_flash_ll_res_check_sus_setup(dev, false);
  168. #endif
  169. }
  170. #endif // SOC_SPI_MEM_SUPPORT_AUTO_SUSPEND
  171. void spi_flash_hal_resume(spi_flash_host_inst_t *host)
  172. {
  173. #if SOC_SPI_MEM_SUPPORT_SW_SUSPEND
  174. spimem_flash_ll_resume((spi_mem_dev_t*)(((spi_flash_hal_context_t *)host)->spi));
  175. #else
  176. abort();
  177. #endif
  178. }
  179. void spi_flash_hal_suspend(spi_flash_host_inst_t *host)
  180. {
  181. #if SOC_SPI_MEM_SUPPORT_SW_SUSPEND
  182. spimem_flash_ll_suspend((spi_mem_dev_t *)(((spi_flash_hal_context_t *)host)->spi));
  183. #else
  184. abort();
  185. #endif
  186. }