ESP32 в настоящее время является одним из самых популярных микроконтроллеров с поддержкой технологии Wi-Fi и широко применяется в различных проектах, относящихся к тематике интернета вещей (IoT). ESP32 – достаточно мощный микроконтроллер, который поддерживает двухъядерное программирование и имеет встроенную поддержку технологии BLE (Bluetooth Low Energy), которая делает его отличным решением для использования в портативной электронике. Но для устройств, запитываемых от батареек/аккумуляторов, энергопотребление является весьма критичным фактором и одной технологии BLE для снижения потребления энергии часто бывает недостаточно. В этих условиях значительного снижения энергопотребления устройств, построенных на основе модуля ESP32, можно добиться за счет использования спящих режимов в модуле.
В данной статье мы рассмотрим программирование модуля ESP32 для его работы в режиме глубокого сна (deep sleep mode) и сравним его энергопотребление в обычном (активном) режиме и режиме глубокого сна. Ранее на нашем сайте мы рассматривали использование спящих режимов в плате Arduino и в модуле ESP8266 NodeMCU.
Необходимые компоненты
- Модуль ESP32 (купить на AliExpress).
- Светодиод (купить на AliExpress).
- Кнопка (тактильный переключатель) – 2 шт.
- Резистор 4,7 кОм – 2 шт. (купить на AliExpress).
- Резистор 680 Ом (купить на AliExpress).
- Адаптер 5V.
- Кабель micro-USB.
- Компьютер/ноутбук с установленной Arduino IDE.
- Макетная плата.
- Соединительные провода.
Для проекта необходима версия модуля ESP32 на основе Devkit V4.0 (или аналогичная) от компании Espressif, которая содержит преобразователь (мост) из USB в UART. Программировать модуль ESP32 мы будем с помощью Arduino IDE.
Требования для нашего проекта будут следующие:
- Модуль ESP32 должен переходить в режим глубокого сна при нажатии кнопки.
- Модуль ESP32 должен выходить из режима глубокого сна при нажатии другой кнопки.
- Для индикации состояния модуля ESP32 будет использоваться светодиод – в активном состоянии модуля он будет мигать с интервалом в 1 секунду, а в режиме глубокого сна он будет выключен.
Схема проекта
Схема для демонстрации возможностей режима глубокого сна в модуле ESP32 представлена на следующем рисунке.
Как видите, схема достаточно проста и содержит модуль ESP32 и две кнопки – для перехода модуля в режим глубокого сна и выхода из него. Кнопки подключены к контактам 16 и 33 модуля ESP32. При нажатии кнопок на соответствующий контакт модуля будет подаваться уровень low, поэтому для подключения кнопок необходимы дополнительные подтягивающие резисторы. Светодиод, который будет показывать в каком режим (спящем или активном) будет находиться модуль ESP32, подключен к его контакту 4.
Краткий обзор спящих режимов в модуле ESP32
В модуле ESP32 есть несколько режимов энергопотребления:
- активный режим (active mode);
- режим сна модема (modem sleep mode);
- "легкий" спящий режим (light sleep mode);
- режим глубокого сна (deep sleep mode);
- режим пониженного энергопотребления/"спячки" (hibernation mode).
В нормальных условиях эксплуатации модуль ESP32 работает в активном режиме. В этом режиме центральный процессор модуля, аппаратное обеспечение WiFi/BT, память часов реального времени (RTC memory), периферийные устройства реального времени (RTC peripherals) и ULP сопроцессоры активированы и их работа зависит от текущей загрузки модуля. Но в некоторых режимах часть этих устройств может отключаться, как показано в следующей таблице.
Аппаратные средства | Active Mode | Modem-sleep Mode | Light Sleep Mode | Deep-sleep Mode | Hibernation |
CPU | ON | ON | PAUSE | OFF | OFF |
WiFi/BT | ON | OFF | OFF | OFF | OFF |
RTC and RTC Peripherals | ON | ON | ON | ON | OFF |
ULP-Co Processor | ON | ON | ON | ON/OFF | OFF |
Режим глубокого сна (deep sleep mode) в модуле ESP32 часто называют шаблоном мониторинга датчиков (ULP sensor monitored pattern) – в этом режиме включены только память RTC и периферийные устройства RTC, а CPU, WiFi/BT и ULP сопроцессоры – выключены.
Чтобы выйти из режима глубокого сна модуль ESP32 должен получить "пробуждающий" (wake-up) сигнал от какого-нибудь источника. И, поскольку, периферийные устройства RTC в режиме глубокого сна включены, модуль ESP32 может "пробудиться" при помощи контактов, доступных для использования устройствами RTC (реального времени). Но существуют и другие способы "пробудить" модуль, например, при помощи внешнего прерывания на одном из контактов или по таймеру. В данном проекте мы будем использовать пробуждение модуля ESP32 из режима глубокого сна при помощи сигнала внешнего прерывания (ext0), подаваемого на его контакт 33.
Объяснение программы для модуля ESP32
Полный код программы приведен в конце статьи, здесь же мы кратко рассмотрим его основные фрагменты.
В начале программы объявим необходимые переменные и инициализируем используемые контакты.
1 2 3 4 5 6 |
//Create a PushButton variable PushBnt pushBtn = {GPIO_NUM_16, 0, false}; // define Led Pin uint8_t led_pin = GPIO_NUM_4; // define wake up pin uint8_t wakeUp_pin = GPIO_NUM_33; |
В функции void setup() инициализируем последовательную связь со скоростью 115200 бод, зададим режимы работы используемых контактов (на ввод или вывод данных), назначим функцию обработки прерывания и создадим необходимую нам задачу.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
void setup() { // put your setup code here, to run once: // set the serial port at 115200 Serial.begin(115200); delay(1000); // set the pushButton pin as input with internal PullUp pinMode(pushBtn.pin, INPUT_PULLUP); // set the Interrupt handler with the pushButton pin in Falling mode attachInterrupt(pushBtn.pin, isr_handle, FALLING); // set the Led pin as ouput pinMode(led_pin, OUTPUT); //create a task that will be executed in the blinkLed() function, with priority 1 and executed on core 0 xTaskCreate( blinkLed, /* Task function. */ "blinkLed", /* name of task. */ 1024*2, /* Stack size of task */ NULL, /* parameter of the task */ 5, /* priority of the task */ &taskBlinkled); /* Task handle to keep track of created task */ delay(500); //Configure Pin 33 as ext0 wake up source with LOW logic level esp_sleep_enable_ext0_wakeup((gpio_num_t)wakeUp_pin, 0); } |
Прерыванием мы будем считать спад импульса (уровня напряжения) на контакте pushBtn.pin. Для обработки прерывания будет вызываться функция isr_handle. Более подробно о работе с прерываниями в модуле ESP32 вы можете прочитать в этой статье.
1 |
attachInterrupt(pushBtn.pin, isr_handle, FALLING); |
Таким образом, при нажатии кнопки уровень на контакте обработки прерывания будет изменяться от логической 1 (3.3V) до логического 0 (0V). Модуль ESP32 будет обнаруживать это изменение уровня. Далее создадим задачу для мигания светодиодом.
1 2 3 4 5 6 7 8 |
xTaskCreate( blinkLed, /* Task function. */ "blinkLed", /* name of task. */ 1024*2, /* Stack size of task */ NULL, /* parameter of the task */ 5, /* priority of the task */ &taskBlinkled); /* Task handle to keep track of created task */ delay(500); |
Контакт 33 конфигурируется с помощью следующей команды чтобы принимать внешний сигнал пробуждения, обозначенный как ext0.
1 |
esp_sleep_enable_ext0_wakeup((gpio_num_t)wakeUp_pin, 0); |
В функции void loop() мы будем выполнять следующие операции:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 |
void loop() { // put your main code here, to run repeatedly: if (pushBtn.pressed) { Serial.printf("PushButton(%d) Pressed \n", pushBtn.pin); Serial.printf("Suspend the 'blinkLed' Task \n"); // Suspend the blinkLed Task vTaskSuspend( taskBlinkled ); digitalWrite(led_pin, LOW); Serial.printf("Going to sleep..... \n", pushBtn.pin); pushBtn.pressed = false; //Go to sleep now esp_deep_sleep_start(); } esp_sleep_wakeup_cause_t wakeupReason; wakeupReason = esp_sleep_get_wakeup_cause(); switch(wakeupReason) { case ESP_SLEEP_WAKEUP_EXT0 : Serial.println("using external signal ext0 for WakeUp From sleep"); break; case ESP_SLEEP_WAKEUP_EXT1 : Serial.println("using external signal ext1 for WakeUp From sleep"); break; case ESP_SLEEP_WAKEUP_TIMER : Serial.println("using Timer signal for WakeUp From sleep"); break; case ESP_SLEEP_WAKEUP_TOUCHPAD : Serial.println("using TouchPad signal for WakeUp From sleep"); break; case ESP_SLEEP_WAKEUP_ULP : Serial.println("using ULP signal for WakeUp From sleep"); break; default : break; Serial.printf("Resume the 'blinkLed' Task \n"); // restart the blinkLed Task vTaskResume( taskBlinkled ); } } |
В цикле мы будем постоянно проверять нажата ли кнопка перехода в спящий режим или нет. Если нажата, то мы будем останавливать задачу мигания светодиодом и запускать функцию перехода в спящий режим.
1 |
esp_deep_sleep_start(); |
А при нажатии кнопки внешнего прерывания ext0 модуль ESP32 будет немедленно выходить из режима глубокого сна и возобновлять выполнение задачи мигания светодиодом.
Код функции для мигания светодиодом с интервалом 1000 мс выглядит следующим образом:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
void blinkLed(void* param){ while(1){ static uint32_t pin_val = 0; // toggle the pin value pin_val ^= 1; digitalWrite(led_pin, pin_val); Serial.printf("Led ----------------- %s\n" , pin_val? "On" : "Off"); /* Simply toggle the LED every 1000ms or 1sec */ vTaskDelay( 1000 / portTICK_PERIOD_MS ); } taskBlinkled = NULL; vTaskDelete( NULL ); } |
Тестирование работы модуля ESP32 в режиме глубокого сна
После сборки схемы проекта на макетной плате мы использовали мультиметр для измерения тока. В активном режиме ток, потребляемый схемой, составил почти 58 mA.
А в режиме глубокого сна (deep sleep mode) он оказался равен 3.95mA (4.10 mA).
В режиме глубоко сна энергопотребление модуля ESP32 должно составлять примерно 150 мкА. Но измеренное энергопотребление платы ESP32 Devkit в этом режиме оказалось равным 4.10 mA. Это произошло из-за микросхемы CP2102 и линейного регулятора в составе платы модуля, которые оба подключены к шине 5V. Также у нас в схеме присутствует светодиод, который потребляет почти 2mA тока.
Тем не менее, по результатам проведенного теста можно отметить, что модуль ESP32 в режиме глубокого сна потребляет очень небольшое количество электроэнергии, что является его весомым преимуществом для использования в устройствах, питаемых от батареек/аккумуляторов.
Более подробно работу проекта вы можете посмотреть на видео, приведенном в конце статьи.
Исходный код программы (скетча)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 |
/* * File Name: esp32-deep-sleep.ino * Created on: 30-Jun-2020 * Author: Noyel Seth (noyelseth@gmail.com) */ // создаем структуру для работы с кнопкой push_button struct PushBnt{ const uint8_t pin; uint32_t pressCount; bool pressed; }; //переменная для работы с кнопкой PushBnt pushBtn = {GPIO_NUM_16, 0, false}; // define Led Pin (контакт, к которому подключен светодиод) uint8_t led_pin = GPIO_NUM_4; // define wake up pin uint8_t wakeUp_pin = GPIO_NUM_33; // define taskBlinkled TaskHandle_t variable TaskHandle_t taskBlinkled; // define ISR function for PushButtton's Interrupt (функция для обработки прерывания от кнопки) void IRAM_ATTR isr_handle() { pushBtn.pressed = true; pushBtn.pressCount = pushBtn.pressCount + 1; } void setup() { // put your setup code here, to run once: // set the serial port at 115200 Serial.begin(115200); delay(1000); // set the pushButton pin as input with internal PullUp pinMode(pushBtn.pin, INPUT_PULLUP); // set the Interrupt handler with the pushButton pin in Falling mode attachInterrupt(pushBtn.pin, isr_handle, FALLING); // set the Led pin as ouput pinMode(led_pin, OUTPUT); //создаем задачу, которая будет исполняться в функции blinkLed() на ядре 0 с приоритетом 1 xTaskCreate( blinkLed, /* Task function. */ "blinkLed", /* name of task. */ 1024*2, /* Stack size of task */ NULL, /* parameter of the task */ 5, /* priority of the task */ &taskBlinkled); /* Task handle to keep track of created task */ delay(500); //Configure Pin 33 as ext0 wake up source with LOW logic level esp_sleep_enable_ext0_wakeup((gpio_num_t)wakeUp_pin, 0); } void loop() { // put your main code here, to run repeatedly: if (pushBtn.pressed) { Serial.printf("PushButton(%d) Pressed \n", pushBtn.pin); Serial.printf("Suspend the 'blinkLed' Task \n"); // останавливаем задачу мигания светодиодом vTaskSuspend( taskBlinkled ); digitalWrite(led_pin, LOW); Serial.printf("Going to sleep..... \n", pushBtn.pin); pushBtn.pressed = false; //переходим в спящий режим esp_deep_sleep_start(); } esp_sleep_wakeup_cause_t wakeupReason; wakeupReason = esp_sleep_get_wakeup_cause(); switch(wakeupReason) { case ESP_SLEEP_WAKEUP_EXT0 : Serial.println("using external signal ext0 for WakeUp From sleep"); break; case ESP_SLEEP_WAKEUP_EXT1 : Serial.println("using external signal ext1 for WakeUp From sleep"); break; case ESP_SLEEP_WAKEUP_TIMER : Serial.println("using Timer signal for WakeUp From sleep"); break; case ESP_SLEEP_WAKEUP_TOUCHPAD : Serial.println("using TouchPad signal for WakeUp From sleep"); break; case ESP_SLEEP_WAKEUP_ULP : Serial.println("using ULP signal for WakeUp From sleep"); break; default : break; Serial.printf("Resume the 'blinkLed' Task \n"); // возобновляем задачу мигания светодиодом vTaskResume( taskBlinkled ); } } void blinkLed(void* param){ while(1){ static uint32_t pin_val = 0; // переключаем значение контакта pin_val ^= 1; digitalWrite(led_pin, pin_val); Serial.printf("Led ----------------- %s\n" , pin_val? "On" : "Off"); /* переключаем состояние светодиода каждый 1000 мс */ vTaskDelay( 1000 / portTICK_PERIOD_MS ); } taskBlinkled = NULL; vTaskDelete( NULL ); } |
Статья интересная. Но мне кажется, что строки:
Serial.printf("Resume the 'blinkLed' Task \n");
// возобновляем задачу мигания светодиодом
vTaskResume( taskBlinkled );
должны идти ниже закрывающей фигурной скобки.
И что будет происходить в loop(), если кнопка не нажата? Всё равно будем вызывать esp_sleep_get_wakeup_cause(), и vTaskResume( taskBlinkled ) ??? Скорее всего надо добавить else и выходить в нём из функции.
Да, спасибо, насчет этих двух строк скорее всего это опечатка у автора. Насчет else и выхода из функции пока не готов сказать