mirror of
https://github.com/cesanta/mongoose.git
synced 2025-01-19 16:13:07 +08:00
Implement firmware update in ESP32 device-dashboard example using espressif OTA APIs
This commit is contained in:
parent
7581229bbf
commit
c20cc4ae09
@ -8,3 +8,5 @@ component_compile_options(-DMG_ENABLE_PACKED_FS)
|
||||
component_compile_options(-DHTTP_URL="http://0.0.0.0:80")
|
||||
component_compile_options(-DHTTPS_URL="https://0.0.0.0:443")
|
||||
component_compile_options(-DMG_TLS=MG_TLS_NONE) # change to 'MG_TLS_MBED' to enable TLS
|
||||
component_compile_options(-DMG_OTA=MG_OTA_CUSTOM)
|
||||
component_compile_options(-DMG_DEVICE=MG_DEVICE_CUSTOM)
|
||||
|
@ -1,22 +1,13 @@
|
||||
// Copyright (c) 2020 Cesanta Software Limited
|
||||
// All rights reserved
|
||||
|
||||
#include "esp_spiffs.h"
|
||||
#include "mongoose.h"
|
||||
#include "net.h"
|
||||
|
||||
#define WIFI_SSID "YOUR_WIFI_NETWORK_NAME" // SET THIS!
|
||||
#define WIFI_PASS "YOUR_WIFI_PASSWORD" // SET THIS!
|
||||
#define FS_ROOT "/spiffs"
|
||||
|
||||
void app_main(void) {
|
||||
// Mount filesystem
|
||||
esp_vfs_spiffs_conf_t conf = {
|
||||
.base_path = FS_ROOT, .max_files = 20, .format_if_mount_failed = true};
|
||||
int res = esp_vfs_spiffs_register(&conf);
|
||||
MG_INFO(("FS %s, %d", conf.base_path, res));
|
||||
mg_file_printf(&mg_fs_posix, FS_ROOT "/hello.txt", "%s", "hello from ESP");
|
||||
|
||||
// Setup wifi. This function is implemented in wifi.c
|
||||
// It blocks until connected to the configured WiFi network
|
||||
void wifi_init(const char *ssid, const char *pass);
|
||||
@ -35,3 +26,240 @@ void app_main(void) {
|
||||
web_init(&mgr);
|
||||
for (;;) mg_mgr_poll(&mgr, 1000); // Infinite event loop
|
||||
}
|
||||
|
||||
#if MG_DEVICE == MG_DEVICE_CUSTOM
|
||||
void *mg_flash_start(void) {
|
||||
return NULL;
|
||||
}
|
||||
size_t mg_flash_size(void) {
|
||||
return 0;
|
||||
}
|
||||
size_t mg_flash_sector_size(void) {
|
||||
return 0;
|
||||
}
|
||||
size_t mg_flash_write_align(void) {
|
||||
return 0;
|
||||
}
|
||||
int mg_flash_bank(void) {
|
||||
return 0;
|
||||
}
|
||||
bool mg_flash_erase(void *location) {
|
||||
(void) location;
|
||||
return false;
|
||||
}
|
||||
bool mg_flash_swap_bank(void) {
|
||||
return true;
|
||||
}
|
||||
bool mg_flash_write(void *addr, const void *buf, size_t len) {
|
||||
(void) addr, (void) buf, (void) len;
|
||||
return false;
|
||||
}
|
||||
void mg_device_reset(void) {
|
||||
esp_restart();
|
||||
}
|
||||
#endif
|
||||
|
||||
#if MG_OTA == MG_OTA_CUSTOM
|
||||
#include "esp_app_format.h"
|
||||
#include "esp_ota_ops.h"
|
||||
|
||||
static size_t s_size = 0; // Firmware size to flash. In-progress indicator
|
||||
static bool rx_checked = false; // Whether firmware being received has been checked as valid
|
||||
const esp_partition_t * update_partition = NULL; // The partition the update is being applied to
|
||||
esp_ota_handle_t update_handle = 0; // Handle of the current update process
|
||||
|
||||
const esp_partition_t * get_partition_from_fw(int fw) {
|
||||
if (MG_FIRMWARE_CURRENT == fw) {
|
||||
return esp_ota_get_running_partition();
|
||||
}
|
||||
const esp_partition_t * p = esp_ota_get_last_invalid_partition();
|
||||
if (NULL == p) {
|
||||
p = esp_ota_get_next_update_partition(NULL);
|
||||
}
|
||||
return p;
|
||||
}
|
||||
|
||||
// Returns true if the data buffer contains the header of a valid firmware
|
||||
const size_t fw_size_required = sizeof(esp_image_header_t) + sizeof(esp_image_segment_header_t) + sizeof(esp_app_desc_t);
|
||||
bool check_fw_header(const void* buf, size_t len) {
|
||||
// Ensure we have received enough data to do the check
|
||||
if (len < fw_size_required) {
|
||||
MG_ERROR(("Insufficient data to check firmware header - %d bytes, require %d bytes", len, fw_size_required));
|
||||
return false;
|
||||
}
|
||||
|
||||
// Check the magic word of the received firmare
|
||||
esp_app_desc_t* rx_app_info = (esp_app_desc_t*)(((char*)buf) + sizeof(esp_image_header_t) + sizeof(esp_image_segment_header_t));
|
||||
if (ESP_APP_DESC_MAGIC_WORD != rx_app_info->magic_word) {
|
||||
MG_ERROR(("Invalid app - invalid magic word"));
|
||||
return false;
|
||||
}
|
||||
MG_INFO(("Received firmware: %s %s", rx_app_info->project_name, rx_app_info->version));
|
||||
|
||||
// Get the name/version of the running firmware
|
||||
const esp_partition_t *current = esp_ota_get_running_partition();
|
||||
esp_app_desc_t curr_app_info;
|
||||
esp_err_t res = esp_ota_get_partition_description(current, &curr_app_info);
|
||||
if (ESP_OK != res) {
|
||||
MG_ERROR(("Unable to get name/version of running firmware"));
|
||||
return false;
|
||||
}
|
||||
MG_INFO(("Running firmware: %s %s", curr_app_info.project_name, curr_app_info.version));
|
||||
|
||||
// Check the name of the new firmware is the same as the old firmware
|
||||
if (memcmp(curr_app_info.project_name, rx_app_info->project_name, sizeof(curr_app_info.project_name)) != 0) {
|
||||
MG_ERROR(("Firmware name is different. Wrong firmware uploaded! Current %s Uploaded %s", curr_app_info.project_name, rx_app_info->project_name));
|
||||
return false;
|
||||
}
|
||||
|
||||
// Check the version of the new firmware is different from the old firmware
|
||||
if (memcmp(curr_app_info.version, rx_app_info->version, sizeof(curr_app_info.version)) == 0) {
|
||||
MG_ERROR(("Firmware version is the same. Wrong firmware uploaded! Current %s, Uploaded %s", curr_app_info.version, rx_app_info->version));
|
||||
//return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool mg_ota_begin(size_t new_firmware_size) {
|
||||
rx_checked = false;
|
||||
if (s_size) {
|
||||
MG_ERROR(("OTA already in progress. Call mg_ota_end()"));
|
||||
return false;
|
||||
}
|
||||
int partition_size = mg_ota_size(MG_FIRMWARE_PREVIOUS);
|
||||
if (new_firmware_size > partition_size) {
|
||||
MG_ERROR(("Firmware %lu bytes, max %lu", new_firmware_size, partition_size));
|
||||
return false;
|
||||
}
|
||||
update_partition = esp_ota_get_next_update_partition(NULL);
|
||||
if (NULL == update_partition) {
|
||||
MG_ERROR(("esp_ota_get_next_update_partition returned NULL"));
|
||||
return false;
|
||||
}
|
||||
esp_err_t res = esp_ota_begin(update_partition, OTA_WITH_SEQUENTIAL_WRITES, &update_handle);
|
||||
if (res != ESP_OK) {
|
||||
esp_ota_abort(update_handle);
|
||||
MG_ERROR(("esp_ota_begin failed (%s)", esp_err_to_name(res)));
|
||||
return false;
|
||||
}
|
||||
s_size = new_firmware_size;
|
||||
MG_INFO(("Starting OTA, firmware size %lu", s_size));
|
||||
return true;
|
||||
}
|
||||
|
||||
bool mg_ota_write(const void *buf, size_t len) {
|
||||
if (s_size == 0) {
|
||||
MG_ERROR(("OTA is not started, call mg_ota_begin()"));
|
||||
return false;
|
||||
}
|
||||
if (!rx_checked) {
|
||||
if (!check_fw_header(buf, len)) {
|
||||
return false;
|
||||
}
|
||||
rx_checked = true;
|
||||
}
|
||||
esp_err_t res = esp_ota_write(update_handle, buf, len);
|
||||
if (res != ESP_OK) {
|
||||
esp_ota_abort(update_handle);
|
||||
MG_ERROR(("esp_ota_write FAILED (%s)", esp_err_to_name(res)));
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
bool mg_ota_end(void) {
|
||||
if (0 == s_size) {
|
||||
MG_INFO(("Finishing OTA: fail"));
|
||||
return false;
|
||||
}
|
||||
s_size = 0;
|
||||
esp_err_t err = esp_ota_end(update_handle);
|
||||
if (err != ESP_OK) {
|
||||
MG_ERROR(("esp_ota_end failed (%s)!", esp_err_to_name(err)));
|
||||
return false;
|
||||
}
|
||||
err = esp_ota_set_boot_partition(update_partition);
|
||||
if (err != ESP_OK) {
|
||||
MG_ERROR(("esp_ota_set_boot_partition failed (%s)!", esp_err_to_name(err)));
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
bool mg_ota_commit(void) {
|
||||
esp_ota_mark_app_valid_cancel_rollback();
|
||||
return true;
|
||||
}
|
||||
|
||||
bool mg_ota_rollback(void) {
|
||||
if(esp_ota_check_rollback_is_possible()) {
|
||||
esp_ota_mark_app_invalid_rollback_and_reboot();
|
||||
return true;
|
||||
}
|
||||
MG_ERROR(("Rollback NOT possible"));
|
||||
return false;
|
||||
}
|
||||
|
||||
int mg_ota_status(int fw) {
|
||||
const esp_partition_t *p = get_partition_from_fw(fw);
|
||||
if (NULL == p) {
|
||||
return 0;
|
||||
}
|
||||
int status = MG_OTA_UNAVAILABLE;
|
||||
|
||||
esp_ota_img_states_t img_state = ESP_OTA_IMG_UNDEFINED;
|
||||
esp_err_t res = esp_ota_get_state_partition(p, &img_state);
|
||||
if (ESP_OK == res) {
|
||||
if (ESP_OTA_IMG_VALID == img_state) {
|
||||
status = MG_OTA_COMMITTED;
|
||||
} else if (ESP_OTA_IMG_UNDEFINED == img_state) {
|
||||
status = MG_OTA_UNCOMMITTED;
|
||||
} else if (ESP_OTA_IMG_INVALID == img_state) {
|
||||
status = MG_OTA_UNAVAILABLE;
|
||||
} else if (ESP_OTA_IMG_ABORTED == img_state) {
|
||||
status = MG_OTA_UNAVAILABLE;
|
||||
} else if (ESP_OTA_IMG_NEW == img_state) {
|
||||
status = MG_OTA_FIRST_BOOT;
|
||||
} else if (ESP_OTA_IMG_PENDING_VERIFY == img_state) {
|
||||
status = MG_OTA_UNCOMMITTED;
|
||||
}
|
||||
}
|
||||
return status;
|
||||
}
|
||||
|
||||
uint32_t mg_ota_crc32(int fw) {
|
||||
(void) fw;
|
||||
return 0;
|
||||
}
|
||||
|
||||
uint32_t mg_ota_timestamp(int fw) {
|
||||
const esp_partition_t *p = get_partition_from_fw(fw);
|
||||
|
||||
esp_app_desc_t app;
|
||||
esp_err_t res = esp_ota_get_partition_description(p, &app);
|
||||
if (ESP_OK != res)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct tm datetime = {};
|
||||
if (NULL == strptime(app.date, "%b %d %Y", &datetime))
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
if (NULL == strptime(app.time, "%H:%M:%S", &datetime))
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
return mktime(&datetime);
|
||||
}
|
||||
|
||||
size_t mg_ota_size(int fw) {
|
||||
const esp_partition_t *p = get_partition_from_fw(fw);
|
||||
if (NULL == p) {
|
||||
return 0;
|
||||
}
|
||||
return p->size;
|
||||
}
|
||||
#endif
|
||||
|
@ -32,7 +32,7 @@ static void event_handler(void *arg, esp_event_base_t event_base,
|
||||
esp_wifi_connect();
|
||||
} else if (event_base == WIFI_EVENT &&
|
||||
event_id == WIFI_EVENT_STA_DISCONNECTED) {
|
||||
if (s_retry_num < 3) {
|
||||
if (s_retry_num < 10) {
|
||||
esp_wifi_connect();
|
||||
s_retry_num++;
|
||||
MG_INFO(("retry to connect to the AP"));
|
||||
|
@ -1,5 +1,6 @@
|
||||
# Name, Type, SubType, Offset, Size, Flags
|
||||
nvs, data, nvs, 0x9000, 0x6000,
|
||||
phy_init, data, phy, 0xf000, 0x1000,
|
||||
storage, data, spiffs, 0x10000, 0x10000,
|
||||
factory, app, factory, 0x100000, 1M,
|
||||
nvs, data, nvs, , 0x6000
|
||||
otadata, data, ota, , 0x2000,
|
||||
phy_init, data, phy, , 0x1000,
|
||||
ota_0, app, ota_0, , 948K,
|
||||
ota_1, app, ota_1, , 948K,
|
||||
|
|
@ -1,4 +1,5 @@
|
||||
CONFIG_PARTITION_TABLE_CUSTOM=y
|
||||
CONFIG_PARTITION_TABLE_CUSTOM_FILENAME="partitions.csv"
|
||||
CONFIG_PARTITION_TABLE_FILENAME="partitions.csv"
|
||||
CONFIG_BOOTLOADER_APP_ROLLBACK_ENABLE=y
|
||||
#CONFIG_ESP_MAIN_TASK_STACK_SIZE=8192
|
||||
|
Loading…
Reference in New Issue
Block a user