2015-07-29 00:40:53 +00:00
|
|
|
/* A very simple OTA example
|
|
|
|
*
|
2016-02-08 05:58:31 +00:00
|
|
|
* Binds a TCP socket, reads an image from it over TFTP and then flashes live.
|
2015-07-29 00:40:53 +00:00
|
|
|
*
|
2016-02-08 05:58:31 +00:00
|
|
|
* For more information about esp-open-rtos OTA see https://github.com/SuperHouse/esp-open-rtos/wiki/OTA-Update-Configuration
|
2015-07-29 00:40:53 +00:00
|
|
|
*
|
|
|
|
* NOT SUITABLE TO PUT ON THE INTERNET OR INTO A PRODUCTION ENVIRONMENT!!!!
|
|
|
|
*/
|
2016-03-23 08:44:27 +00:00
|
|
|
#include <string.h>
|
2015-07-29 00:40:53 +00:00
|
|
|
#include "espressif/esp_common.h"
|
2015-10-06 12:04:19 +00:00
|
|
|
#include "esp/uart.h"
|
2015-07-29 00:40:53 +00:00
|
|
|
#include "FreeRTOS.h"
|
|
|
|
#include "task.h"
|
|
|
|
#include "esp8266.h"
|
2015-08-10 04:50:32 +00:00
|
|
|
#include "ssid_config.h"
|
2016-03-23 08:44:27 +00:00
|
|
|
#include "mbedtls/sha256.h"
|
2015-07-29 00:40:53 +00:00
|
|
|
|
2015-08-04 04:50:50 +00:00
|
|
|
#include "ota-tftp.h"
|
2016-04-07 18:29:28 +00:00
|
|
|
#include "rboot.h"
|
|
|
|
#include "rboot-api.h"
|
2015-07-29 00:40:53 +00:00
|
|
|
|
2016-03-23 06:33:08 +00:00
|
|
|
#define TFTP_IMAGE_SERVER "192.168.1.23"
|
|
|
|
#define TFTP_IMAGE_FILENAME1 "firmware1.bin"
|
|
|
|
#define TFTP_IMAGE_FILENAME2 "firmware2.bin"
|
|
|
|
|
2016-03-23 08:44:27 +00:00
|
|
|
/* Output of the command 'sha256sum firmware1.bin' */
|
|
|
|
static const char *FIRMWARE1_SHA256 = "88199daff8b9e76975f685ec7f95bc1df3c61bd942a33a54a40707d2a41e5488";
|
|
|
|
|
2016-03-23 06:33:08 +00:00
|
|
|
void tftp_client_task(void *pvParameters)
|
|
|
|
{
|
|
|
|
printf("TFTP client task starting...\n");
|
|
|
|
rboot_config conf;
|
|
|
|
conf = rboot_get_config();
|
|
|
|
int slot = (conf.current_rom + 1) % conf.count;
|
|
|
|
printf("Image will be saved in OTA slot %d.\n", slot);
|
|
|
|
if(slot == conf.current_rom) {
|
|
|
|
printf("FATAL ERROR: Only one OTA slot is configured!\n");
|
|
|
|
while(1) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Alternate between trying two different filenames. Probalby want to change this if making a practical
|
|
|
|
example!
|
|
|
|
|
|
|
|
Note: example will reboot into FILENAME1 if it is successfully downloaded, but FILENAME2 is ignored.
|
|
|
|
*/
|
|
|
|
while(1) {
|
|
|
|
printf("Downloading %s to slot %d...\n", TFTP_IMAGE_FILENAME1, slot);
|
|
|
|
int res = ota_tftp_download(TFTP_IMAGE_SERVER, TFTP_PORT, TFTP_IMAGE_FILENAME1, 1000, slot);
|
|
|
|
printf("ota_tftp_download %s result %d\n", TFTP_IMAGE_FILENAME1, res);
|
|
|
|
if(res == 0) {
|
2016-03-23 08:44:27 +00:00
|
|
|
printf("Looks valid, calculating SHA256...\n");
|
|
|
|
uint32_t length;
|
|
|
|
bool valid = rboot_verify_image(conf.roms[slot], &length, NULL);
|
|
|
|
static mbedtls_sha256_context ctx;
|
|
|
|
mbedtls_sha256_init(&ctx);
|
|
|
|
mbedtls_sha256_starts(&ctx, 0);
|
|
|
|
valid = valid && rboot_digest_image(conf.roms[slot], length, (rboot_digest_update_fn)mbedtls_sha256_update, &ctx);
|
|
|
|
static uint8_t hash_result[32];
|
|
|
|
mbedtls_sha256_finish(&ctx, hash_result);
|
|
|
|
mbedtls_sha256_free(&ctx);
|
|
|
|
|
|
|
|
if(!valid)
|
|
|
|
{
|
|
|
|
printf("Not valid after all :(\n");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Image SHA256 = ");
|
|
|
|
bool valid = true;
|
|
|
|
for(int i = 0; i < sizeof(hash_result); i++) {
|
|
|
|
char hexbuf[3];
|
|
|
|
snprintf(hexbuf, 3, "%02x", hash_result[i]);
|
|
|
|
printf(hexbuf);
|
|
|
|
if(strncmp(hexbuf, FIRMWARE1_SHA256+i*2, 2))
|
|
|
|
valid = false;
|
|
|
|
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
if (valid) {
|
|
|
|
printf("SHA256 Matches. Rebooting into slot %d...\n", slot);
|
|
|
|
rboot_set_current_rom(slot);
|
|
|
|
sdk_system_restart();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("Downloaded image SHA256 didn't match expected '%s'\n", FIRMWARE1_SHA256);
|
|
|
|
}
|
|
|
|
}
|
2016-03-23 06:33:08 +00:00
|
|
|
}
|
|
|
|
vTaskDelay(5000 / portTICK_RATE_MS);
|
|
|
|
|
|
|
|
printf("Downloading %s to slot %d...\n", TFTP_IMAGE_FILENAME2, slot);
|
|
|
|
res = ota_tftp_download(TFTP_IMAGE_SERVER, TFTP_PORT, TFTP_IMAGE_FILENAME2, 1000, slot);
|
|
|
|
printf("ota_tftp_download %s result %d\n", TFTP_IMAGE_FILENAME2, res);
|
|
|
|
vTaskDelay(5000 / portTICK_RATE_MS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-29 00:40:53 +00:00
|
|
|
void user_init(void)
|
|
|
|
{
|
2015-10-06 12:04:19 +00:00
|
|
|
uart_set_baud(0, 115200);
|
|
|
|
|
2016-04-07 18:29:28 +00:00
|
|
|
rboot_config conf = rboot_get_config();
|
2015-08-04 04:50:50 +00:00
|
|
|
printf("\r\n\r\nOTA Basic demo.\r\nCurrently running on flash slot %d / %d.\r\n\r\n",
|
|
|
|
conf.current_rom, conf.count);
|
|
|
|
|
|
|
|
printf("Image addresses in flash:\r\n");
|
|
|
|
for(int i = 0; i <conf.count; i++) {
|
2015-09-05 02:54:09 +00:00
|
|
|
printf("%c%d: offset 0x%08x\r\n", i == conf.current_rom ? '*':' ', i, conf.roms[i]);
|
2015-08-04 04:50:50 +00:00
|
|
|
}
|
2015-07-29 00:40:53 +00:00
|
|
|
|
|
|
|
struct sdk_station_config config = {
|
|
|
|
.ssid = WIFI_SSID,
|
|
|
|
.password = WIFI_PASS,
|
|
|
|
};
|
|
|
|
sdk_wifi_set_opmode(STATION_MODE);
|
|
|
|
sdk_wifi_station_set_config(&config);
|
|
|
|
|
2015-08-04 04:50:50 +00:00
|
|
|
ota_tftp_init_server(TFTP_PORT);
|
2016-03-23 08:44:27 +00:00
|
|
|
xTaskCreate(&tftp_client_task, (signed char *)"tftp_client", 2048, NULL, 2, NULL);
|
2015-07-29 00:40:53 +00:00
|
|
|
}
|