Общие сведения
В этом проекте мы сделаем устройство на базе Микроконтроллер Piranha Set ESP32 которое будет загружать выполняемый код OTA (Over-The-Air - по воздуху) из сети Интернет.
Видео
Редактируется...
Нам понадобится
Аппаратная часть
- Микроконтроллер Piranha Set ESP32
- Set Expander
- Дисплей 1602 и I2C конвертер для связи с ним
- Две TREMA-кнопки
- Два TREMA-светодиода
- Соединительные провода «папа-мама»
Программная часть
- Arduino IDE 1.8.19
- ESP32 core (устанавливается в Arduino IDE, пункт меню Инструменты->Плата->Менеджер плат)
- Библиотека Button2
Про установку библиотек в Arduino IDE можно узнать по ссылке: https://wiki.iarduino.ru/page/Installing_libraries/
Подключение
Установите Set Expander на Piranha Set ESP32
Установите Кнопки и Светодиоды на Piranha Set ESP32
Подключите дисплей к колодке I2C на Set Expander
Принцип работы и таблица разделов ESP32
После загрузки скетча в микроконтроллер левой кнопкой можно выбрать
новый скетч для загрузки, правой начать загрузку. После загрузки
устройство начнёт мигать соответствующим прошивке светодиодом. При
использовании собственного хостинга необходимо три раза скомпилировать и
экспортировать двоичный код, каждый раз с разным определением
FW_LED
и правильно указать названия двоичных файлов.
Для загрузки скетча в ESP32 через интернет необходимо где-то хранить скомпилированный код скетча. Для этого будем использовать бесплатный хостинг Github Pages. Пример как создать такой сайт находится после раздела "Скетч проекта"
Для корректной работы необходимо выбрать таблицу разделов, поддерживающую загрузку OTA в меню "Инструменты"->"Partition Scheme:".
В примере используется схема разделов "default":
Название | Сдвиг адреса | Описание |
---|---|---|
nvs | 0x9000 | Non Volatile Storage - энергонезавизимая память, здесь храниться калибровочные данные WiFi, а так же другие данные |
otadata | 0xe000 | Раздел, в котором хранится информация о том, какой раздел нужно загружать (app0 или app1 в данном случае) |
app0 | 0x10000 | Раздел в котором находиться скомпилированный код скетча |
app1 | 0x150000 | Второй раздел для скетча (может быть дубликатом или предыдущей версией скетча, до загрузки OTA) |
spiffs | 0x290000 | Файловая система (SPI Flash File System) |
coredump | 0x3F0000 | Бектрейс стека в случае неисправимой ошибки |
При загрузке новой прошивки OTA будет использоваться текущий неактивный раздел app0 или app1. В случае, если соединение прервётся во время загрузки, устройство останеться работать на текущей прошивке и будет загружаться с текущего раздела.
Скетч проекта
#include <WiFi.h> #include <HTTPClient.h> #include <Update.h> #include <SPIFFS.h> #include <LiquidCrystal_I2C.h> // 1.1.4 #include <Button2.h> // 2.2.4 // данные для подключения const char* ssid = "Имя Вашей точки доступа WiFi"; const char* password = "Пароль Вашей точки доступа WiFi"; // адрес сервера // const char* server = "адрес сервера для хранения файлов (слеш в конце строки обязателен)/"; // для проверки можно использовать наш репозиторий с уже загруженными прошивками: const char* server = "https://tremaru-file.github.io/OTA_update/"; // выводы светодиодов (разкомментировать только один для каждой версии прошивки) #define FW_LED LED_BUILTIN // BLUE //#define FW_LED 18 // GREEN //#define FW_LED 32 // RED // макрос возможных прошивок (ИНДЕКС, СТРОКА ДЛЯ ЭКРАНА, НАЗВАНИЕ ФАЙЛА НА СЕРВЕРЕ) #define MENU\ X(FW1, "FW 1", "led_red.bin")\ X(FW2, "FW 2", "led_green.bin")\ X(FW3, "FW 3", "led_blue.bin")\ // вывод левой кнопки constexpr unsigned BUTTON_LEFT = 13; // вывод правой кнопки constexpr unsigned BUTTON_RIGHT = 33; // состояние меню (cм. макрос MENU) typedef enum { #define X(INDEX, STR, FILENAME) INDEX, MENU #undef X NMENUSTATES } state_t; // переопределение оператора приращивания для смены текущего состояния меню state_t& operator++(state_t& s) { switch (s) { #define X(INDEX, STR, FILENAME) \ case INDEX: s = static_cast<state_t>(INDEX + 1); s == NMENUSTATES ? s = FW1: 0; return s; MENU #undef X } } state_t menu_state = FW1; // строки для вывода на дисплей (см. макрос MENU) const char* disp_strings[] { #define X(INDEX, STR, FILENAME) STR, MENU #undef X }; // строки названия файлов (см. макрос MENU) const char* filename_strings[] { #define X(INDEX, STR, FILENAME) FILENAME, MENU #undef X }; // название файла настроек const char* settings_file = "/wifi.cfg"; // объект файла настроек fs::File g_file; // объект дисплея LiquidCrystal_I2C disp(0x27, 20, 4); // объекты кнопок Button2 leftButton; Button2 rightButton; void setup() { initWiFi(); initFlashMemory(); saveWiFiCred(); initDisplay(); initButtons(); initLED(); } void loop() { handleInput(); handleDisplay(); handleMenu(); handleLED(); } void handleDisplay() { disp.setCursor(0,0); disp.print(disp_strings[menu_state]); disp.setCursor(0,1); disp.print(filename_strings[menu_state]); } void handleMenu() { } void handleLED() { static unsigned long blink_millis = 0; static constexpr unsigned long blink_interval = 1000; if (millis() - blink_millis < blink_interval) return; blink_millis = millis(); static bool blink = false; digitalWrite(FW_LED, blink); blink = !blink; } void initWiFi() { WiFi.begin(ssid, password); while (WiFi.status() != WL_CONNECTED) { delay(500); disp.print("."); } } void initLED() { pinMode(FW_LED, OUTPUT); } // инициализация дисплея void initDisplay() { disp.init(); disp.backlight(); disp.noBlink(); disp.print("CTAPTyEM...."); delay(500); disp.clear(); } // инициализация кнопок void initButtons() { leftButton.begin(BUTTON_LEFT, INPUT, false); rightButton.begin(BUTTON_RIGHT, INPUT, false); leftButton.setReleasedHandler(released); rightButton.setReleasedHandler(released); } // обработка ввода void handleInput() { leftButton.loop(); rightButton.loop(); } // инициализация памяти void initFlashMemory() { if (!SPIFFS.begin()) { Serial.println("Форматируем память, нужно подождать..."); disp.print("F "); SPIFFS.format(); } if (!SPIFFS.begin()) { Serial.println("Не получилось отформатировать, остановка."); disp.print("E 01"); while(true) ; } } // сохранение настроек WiFi (необходимо, если используется наш репозиторий) bool saveWiFiCred() { try { g_file = SPIFFS.open(settings_file, "w"); if (!g_file) { Serial.println("Ошибка открытия файла"); return false; } g_file.print(ssid); g_file.print('\n'); g_file.print(password); g_file.print('\n'); g_file.flush(); g_file.close(); Serial.println("Настройки сохранены"); return true; } catch (...) { Serial.println("Исключение при сохранении файла"); return false; } } void released(Button2& btn) { switch (btn.getPin()) { default: break; case BUTTON_RIGHT: ++menu_state; disp.clear(); break; case BUTTON_LEFT: update(); break; } } // обновление прошивки void update() { // запрос на сервер HTTPClient client; client.begin((String)server + filename_strings[menu_state]); // если запрос удачный - обновляем раздел новым кодом int httpcode = client.GET(); if (httpcode == HTTP_CODE_OK) { disp.clear(); disp.print("FW UPDATE.."); updateFirmware(client); } else { disp.clear(); disp.printf("HTTP: %d", httpcode); Serial.printf("HTTP: %d", httpcode); return; } client.end(); // сбрасываем конфигурацию вывода pinMode(FW_LED, INPUT); // перезагрузка ESP.restart(); } int g_full_length = 0; int g_curr_length = 0; // обновление раздела OTA void updateFirmware(HTTPClient& client) { // хранение части полученных данных uint8_t buff[128]{}; // получение информации о размере файла g_full_length = client.getSize(); // размер для декрементации в цикле int len = g_full_length; // начинаем обновление Update.begin(UPDATE_SIZE_UNKNOWN); // получаем указатель на поток данных WiFiClient* stream = client.getStreamPtr(); while (client.connected() && (len > 0 || len == -1)) { // пока данные есть - записываем поэтапно в Flash память size_t size = stream->available(); if (size) { int bytes_read = stream->readBytes(buff, ((size > sizeof(buff))?sizeof(buff):size)); updateFlash(buff, bytes_read); if (len > 0) len -= bytes_read; } delay(1); } } // запись из RAM в Flash (параметры: указатель на массив в RAM, длинна массива) void updateFlash(uint8_t* data, size_t len) { // Запись массива в Flash Update.write(data, len); // Обновление счётчика записсаного размера g_curr_length += len; // Вывод на дисплей точек на дисплей static int count = 0; count++; if (count % 1024 == 0) disp.print("."); // ранний выход из функции, если записаны не все данные if (g_curr_length != g_full_length) return; // завершение обновления Update.end(true); }
Экспорт скомпилированного скетча
В Arduino IDE выберите пункт "Скетч"->"Экспортировать скомпилированный бинарный файл"
Выберите пункт "Скетч"->"Показать папку скетча"