deepsleep.c 2.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667
  1. /* Wake from deep sleep stub
  2. See esp_deepsleep.h esp_wake_deep_sleep() comments for details.
  3. */
  4. #include <stddef.h>
  5. #include <sys/lock.h>
  6. #include "rom/cache.h"
  7. #include "rom/rtc.h"
  8. #include "soc/rtc_cntl_reg.h"
  9. #include "soc/dport_reg.h"
  10. #include "esp_attr.h"
  11. #include "esp_deepsleep.h"
  12. #include "rtc.h"
  13. /* Updating RTC_MEMORY_CRC_REG register via set_rtc_memory_crc()
  14. is not thread-safe. */
  15. static _lock_t lock_rtc_memory_crc;
  16. esp_deep_sleep_wake_stub_fn_t esp_get_deep_sleep_wake_stub(void)
  17. {
  18. _lock_acquire(&lock_rtc_memory_crc);
  19. uint32_t stored_crc = REG_READ(RTC_MEMORY_CRC_REG);
  20. set_rtc_memory_crc();
  21. uint32_t calc_crc = REG_READ(RTC_MEMORY_CRC_REG);
  22. REG_WRITE(RTC_MEMORY_CRC_REG, stored_crc);
  23. _lock_release(&lock_rtc_memory_crc);
  24. if(stored_crc == calc_crc) {
  25. return (esp_deep_sleep_wake_stub_fn_t)REG_READ(RTC_ENTRY_ADDR_REG);
  26. } else {
  27. return NULL;
  28. }
  29. }
  30. void esp_set_deep_sleep_wake_stub(esp_deep_sleep_wake_stub_fn_t new_stub)
  31. {
  32. _lock_acquire(&lock_rtc_memory_crc);
  33. REG_WRITE(RTC_ENTRY_ADDR_REG, (uint32_t)new_stub);
  34. set_rtc_memory_crc();
  35. _lock_release(&lock_rtc_memory_crc);
  36. }
  37. void RTC_IRAM_ATTR esp_default_wake_deep_sleep(void) {
  38. /* Clear MMU for CPU 0 */
  39. REG_SET_BIT(DPORT_PRO_CACHE_CTRL1_REG, DPORT_PRO_CACHE_MMU_IA_CLR);
  40. REG_CLR_BIT(DPORT_PRO_CACHE_CTRL1_REG, DPORT_PRO_CACHE_MMU_IA_CLR);
  41. }
  42. void __attribute__((weak, alias("esp_default_wake_deep_sleep"))) esp_wake_deep_sleep(void);
  43. void esp_deep_sleep(uint64_t time_in_us)
  44. {
  45. rtc_set_cpu_freq(CPU_XTAL);
  46. if (esp_get_deep_sleep_wake_stub() == NULL) {
  47. esp_set_deep_sleep_wake_stub(esp_wake_deep_sleep);
  48. }
  49. uint32_t period = rtc_slowck_cali(CALI_RTC_MUX, 128);
  50. uint32_t cycle_l, cycle_h;
  51. rtc_usec2rtc(time_in_us >> 32, time_in_us, period, &cycle_h, &cycle_l);
  52. rtc_slp_prep_lite(1, 0);
  53. rtc_sleep(cycle_h, cycle_l, TIMER_EXPIRE_EN, 0);
  54. while (1) {
  55. ;
  56. }
  57. }
  58. void system_deep_sleep(uint64_t) __attribute__((alias("esp_deep_sleep")));