deep_sleep_example_main.c 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. /*
  2. * SPDX-FileCopyrightText: 2017-2023 Espressif Systems (Shanghai) CO LTD
  3. *
  4. * SPDX-License-Identifier: Unlicense OR CC0-1.0
  5. */
  6. #include <stdio.h>
  7. #include <time.h>
  8. #include <sys/time.h>
  9. #include "sdkconfig.h"
  10. #include "soc/soc_caps.h"
  11. #include "freertos/FreeRTOS.h"
  12. #include "freertos/task.h"
  13. #include "esp_sleep.h"
  14. #include "esp_log.h"
  15. #include "driver/rtc_io.h"
  16. #include "nvs_flash.h"
  17. #include "nvs.h"
  18. #include "deep_sleep_example.h"
  19. #if SOC_RTC_FAST_MEM_SUPPORTED
  20. static RTC_DATA_ATTR struct timeval sleep_enter_time;
  21. #else
  22. static struct timeval sleep_enter_time;
  23. #endif
  24. static void deep_sleep_task(void *args)
  25. {
  26. /**
  27. * Prefer to use RTC mem instead of NVS to save the deep sleep enter time, unless the chip
  28. * does not support RTC mem(such as esp32c2). Because the time overhead of NVS will cause
  29. * the recorded deep sleep enter time to be not very accurate.
  30. */
  31. #if !SOC_RTC_FAST_MEM_SUPPORTED
  32. // Initialize NVS
  33. esp_err_t err = nvs_flash_init();
  34. if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) {
  35. // NVS partition was truncated and needs to be erased
  36. // Retry nvs_flash_init
  37. ESP_ERROR_CHECK(nvs_flash_erase());
  38. err = nvs_flash_init();
  39. }
  40. ESP_ERROR_CHECK(err);
  41. nvs_handle_t nvs_handle;
  42. err = nvs_open("storage", NVS_READWRITE, &nvs_handle);
  43. if (err != ESP_OK) {
  44. printf("Error (%s) opening NVS handle!\n", esp_err_to_name(err));
  45. } else {
  46. printf("Open NVS done\n");
  47. }
  48. // Get deep sleep enter time
  49. nvs_get_i32(nvs_handle, "slp_enter_sec", (int32_t *)&sleep_enter_time.tv_sec);
  50. nvs_get_i32(nvs_handle, "slp_enter_usec", (int32_t *)&sleep_enter_time.tv_usec);
  51. #endif
  52. struct timeval now;
  53. gettimeofday(&now, NULL);
  54. int sleep_time_ms = (now.tv_sec - sleep_enter_time.tv_sec) * 1000 + (now.tv_usec - sleep_enter_time.tv_usec) / 1000;
  55. switch (esp_sleep_get_wakeup_cause()) {
  56. case ESP_SLEEP_WAKEUP_TIMER: {
  57. printf("Wake up from timer. Time spent in deep sleep: %dms\n", sleep_time_ms);
  58. break;
  59. }
  60. #if CONFIG_EXAMPLE_GPIO_WAKEUP
  61. case ESP_SLEEP_WAKEUP_GPIO: {
  62. uint64_t wakeup_pin_mask = esp_sleep_get_gpio_wakeup_status();
  63. if (wakeup_pin_mask != 0) {
  64. int pin = __builtin_ffsll(wakeup_pin_mask) - 1;
  65. printf("Wake up from GPIO %d\n", pin);
  66. } else {
  67. printf("Wake up from GPIO\n");
  68. }
  69. break;
  70. }
  71. #endif //CONFIG_EXAMPLE_GPIO_WAKEUP
  72. #if CONFIG_EXAMPLE_EXT0_WAKEUP
  73. case ESP_SLEEP_WAKEUP_EXT0: {
  74. printf("Wake up from ext0\n");
  75. break;
  76. }
  77. #endif // CONFIG_EXAMPLE_EXT0_WAKEUP
  78. #ifdef CONFIG_EXAMPLE_EXT1_WAKEUP
  79. case ESP_SLEEP_WAKEUP_EXT1: {
  80. uint64_t wakeup_pin_mask = esp_sleep_get_ext1_wakeup_status();
  81. if (wakeup_pin_mask != 0) {
  82. int pin = __builtin_ffsll(wakeup_pin_mask) - 1;
  83. printf("Wake up from GPIO %d\n", pin);
  84. } else {
  85. printf("Wake up from GPIO\n");
  86. }
  87. break;
  88. }
  89. #endif // CONFIG_EXAMPLE_EXT1_WAKEUP
  90. #ifdef CONFIG_EXAMPLE_TOUCH_WAKEUP
  91. case ESP_SLEEP_WAKEUP_TOUCHPAD: {
  92. printf("Wake up from touch on pad %d\n", esp_sleep_get_touchpad_wakeup_status());
  93. break;
  94. }
  95. #endif // CONFIG_EXAMPLE_TOUCH_WAKEUP
  96. case ESP_SLEEP_WAKEUP_UNDEFINED:
  97. default:
  98. printf("Not a deep sleep reset\n");
  99. }
  100. vTaskDelay(1000 / portTICK_PERIOD_MS);
  101. #if CONFIG_IDF_TARGET_ESP32
  102. // Isolate GPIO12 pin from external circuits. This is needed for modules
  103. // which have an external pull-up resistor on GPIO12 (such as ESP32-WROVER)
  104. // to minimize current consumption.
  105. rtc_gpio_isolate(GPIO_NUM_12);
  106. #endif
  107. printf("Entering deep sleep\n");
  108. // get deep sleep enter time
  109. gettimeofday(&sleep_enter_time, NULL);
  110. #if !SOC_RTC_FAST_MEM_SUPPORTED
  111. // record deep sleep enter time via nvs
  112. ESP_ERROR_CHECK(nvs_set_i32(nvs_handle, "slp_enter_sec", sleep_enter_time.tv_sec));
  113. ESP_ERROR_CHECK(nvs_set_i32(nvs_handle, "slp_enter_usec", sleep_enter_time.tv_usec));
  114. ESP_ERROR_CHECK(nvs_commit(nvs_handle));
  115. nvs_close(nvs_handle);
  116. #endif
  117. // enter deep sleep
  118. esp_deep_sleep_start();
  119. }
  120. static void example_deep_sleep_register_rtc_timer_wakeup(void)
  121. {
  122. const int wakeup_time_sec = 20;
  123. printf("Enabling timer wakeup, %ds\n", wakeup_time_sec);
  124. ESP_ERROR_CHECK(esp_sleep_enable_timer_wakeup(wakeup_time_sec * 1000000));
  125. }
  126. void app_main(void)
  127. {
  128. /* Enable wakeup from deep sleep by rtc timer */
  129. example_deep_sleep_register_rtc_timer_wakeup();
  130. #if CONFIG_EXAMPLE_GPIO_WAKEUP
  131. /* Enable wakeup from deep sleep by gpio */
  132. example_deep_sleep_register_gpio_wakeup();
  133. #endif
  134. #if CONFIG_EXAMPLE_EXT0_WAKEUP
  135. /* Enable wakeup from deep sleep by ext0 */
  136. example_deep_sleep_register_ext0_wakeup();
  137. #endif
  138. #if CONFIG_EXAMPLE_EXT1_WAKEUP
  139. /* Enable wakeup from deep sleep by ext1 */
  140. example_deep_sleep_register_ext1_wakeup();
  141. #endif
  142. #if CONFIG_EXAMPLE_TOUCH_WAKEUP
  143. /* Enable wakeup from deep sleep by touch */
  144. example_deep_sleep_register_touch_wakeup();
  145. #endif
  146. xTaskCreate(deep_sleep_task, "deep_sleep_task", 4096, NULL, 6, NULL);
  147. }