2020-07-25 17:25:06 +01:00
|
|
|
|
/*
|
2020-07-27 14:43:24 +01:00
|
|
|
|
xdrv_23_zigbee_9a_upload.ino - zigbee: serial xmodem upload to MCU
|
2020-07-25 17:25:06 +01:00
|
|
|
|
|
2021-01-01 12:44:04 +00:00
|
|
|
|
Copyright (C) 2021 Theo Arends and Stephan Hadinger
|
2020-07-25 17:25:06 +01:00
|
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#ifdef USE_ZIGBEE
|
|
|
|
|
|
|
|
|
|
#ifdef USE_ZIGBEE_EZSP
|
|
|
|
|
/*********************************************************************************************\
|
|
|
|
|
* MCU EFR32 firmware upload using xmodem
|
|
|
|
|
*
|
|
|
|
|
* Step 1 - Upload MCU firmware in ESP8266 flash free space (current size is about 200k)
|
|
|
|
|
* Step 2 - Upload MCU firmware from ESP8266 flash to MCU EFR32 using XMODEM protocol
|
|
|
|
|
* Step 3 - Restart
|
|
|
|
|
\*********************************************************************************************/
|
|
|
|
|
|
2020-07-27 17:31:14 +01:00
|
|
|
|
//#define ZIGBEE_BOOTLOADER_SOFTWARE_RESET_FIRST
|
|
|
|
|
|
2020-07-26 17:55:31 +01:00
|
|
|
|
#define XM_SOH 0x01
|
|
|
|
|
#define XM_EOT 0x04
|
|
|
|
|
#define XM_ACK 0x06
|
|
|
|
|
#define XM_CR 0x0d
|
|
|
|
|
#define XM_NAK 0x15
|
|
|
|
|
#define XM_CAN 0x18
|
2020-07-27 14:43:24 +01:00
|
|
|
|
#define XM_SUB 0x1a
|
2020-07-26 17:55:31 +01:00
|
|
|
|
|
2020-07-27 17:31:14 +01:00
|
|
|
|
enum ZbUploadSteps { ZBU_IDLE, ZBU_INIT,
|
|
|
|
|
ZBU_SOFTWARE_RESET, ZBU_SOFTWARE_SEND, ZBU_HARDWARE_RESET, ZBU_PROMPT,
|
2020-09-01 12:05:46 +01:00
|
|
|
|
ZBU_SYNC, ZBU_UPLOAD, ZBU_EOT, ZBU_COMPLETE, ZBU_DONE, ZBU_ERROR, ZBU_FINISH };
|
2020-07-26 17:55:31 +01:00
|
|
|
|
|
2020-07-25 17:25:06 +01:00
|
|
|
|
const uint8_t PIN_ZIGBEE_BOOTLOADER = 5;
|
|
|
|
|
|
|
|
|
|
struct ZBUPLOAD {
|
|
|
|
|
uint32_t ota_size = 0;
|
|
|
|
|
uint32_t sector_counter = 0;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
uint32_t byte_counter = 0;
|
|
|
|
|
char *buffer;
|
|
|
|
|
uint8_t ota_step = ZBU_IDLE;
|
2020-07-27 17:31:14 +01:00
|
|
|
|
uint8_t bootloader = 0;
|
2020-09-01 12:05:46 +01:00
|
|
|
|
uint8_t state = ZBU_IDLE;
|
2020-07-25 17:25:06 +01:00
|
|
|
|
} ZbUpload;
|
|
|
|
|
|
2020-07-26 17:55:31 +01:00
|
|
|
|
/*********************************************************************************************\
|
2020-11-19 14:43:47 +00:00
|
|
|
|
* Flash from ESP8266 to EZSP
|
2020-07-26 17:55:31 +01:00
|
|
|
|
\*********************************************************************************************/
|
2020-07-25 17:25:06 +01:00
|
|
|
|
|
2020-07-26 17:55:31 +01:00
|
|
|
|
uint32_t ZigbeeUploadAvailable(void) {
|
|
|
|
|
int available = ZbUpload.ota_size - ZbUpload.byte_counter;
|
|
|
|
|
if (available < 0) { available = 0; }
|
|
|
|
|
return available;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char ZigbeeUploadFlashRead(void) {
|
|
|
|
|
if (0 == ZbUpload.byte_counter) {
|
|
|
|
|
if (!(ZbUpload.buffer = (char *)malloc(SPI_FLASH_SEC_SIZE))) {
|
|
|
|
|
return (-1); // Not enough (memory) space
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint32_t index = ZbUpload.byte_counter % SPI_FLASH_SEC_SIZE;
|
|
|
|
|
if (0 == index) {
|
|
|
|
|
ESP.flashRead(ZbUpload.sector_counter * SPI_FLASH_SEC_SIZE, (uint32_t*)ZbUpload.buffer, SPI_FLASH_SEC_SIZE);
|
|
|
|
|
ZbUpload.sector_counter++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char data = ZbUpload.buffer[index];
|
|
|
|
|
ZbUpload.byte_counter++;
|
|
|
|
|
|
|
|
|
|
if (ZbUpload.byte_counter > ZbUpload.ota_size) {
|
2021-01-19 10:54:49 +00:00
|
|
|
|
|
|
|
|
|
// static bool padding = true;
|
|
|
|
|
// if (padding) {
|
|
|
|
|
// AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Start padding from %d"), ZbUpload.byte_counter);
|
|
|
|
|
// padding = false;
|
|
|
|
|
// }
|
|
|
|
|
|
2020-07-27 14:43:24 +01:00
|
|
|
|
// When the source device reaches the last XModem data block, it should be padded to 128 bytes
|
|
|
|
|
// of data using SUB (ASCII 0x1A) characters.
|
|
|
|
|
data = XM_SUB;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
// if (ZbUpload.buffer) { free(ZbUpload.buffer); } // Don't in case of retries
|
|
|
|
|
}
|
|
|
|
|
return data;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*********************************************************************************************\
|
|
|
|
|
* XModem protocol
|
|
|
|
|
\*********************************************************************************************/
|
|
|
|
|
|
2020-10-01 11:58:42 +01:00
|
|
|
|
// Number of milliseconds to wait before prompt is received
|
|
|
|
|
const uint32_t XMODEM_FLUSH_DELAY = 1000;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
// Number of seconds until giving up hope of receiving sync packets from host.
|
|
|
|
|
const uint8_t XMODEM_SYNC_TIMEOUT = 30;
|
|
|
|
|
// Number of times we try to send a packet to the host until we give up sending..
|
|
|
|
|
const uint8_t XMODEM_MAX_RETRY = 30;
|
|
|
|
|
// Packet size
|
|
|
|
|
const uint8_t XMODEM_PACKET_SIZE = 128;
|
|
|
|
|
|
|
|
|
|
struct XMODEM {
|
|
|
|
|
uint32_t timeout = 0;
|
2020-07-27 10:01:20 +01:00
|
|
|
|
uint32_t delay = 0;
|
2020-10-01 11:58:42 +01:00
|
|
|
|
uint32_t flush_delay = 0xFFFFFFFF;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
uint32_t filepos = 0;
|
2021-01-19 10:54:49 +00:00
|
|
|
|
uint32_t packet_no = 1;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
int crcBuf = 0;
|
|
|
|
|
uint8_t checksumBuf = 0;
|
|
|
|
|
bool oldChecksum;
|
|
|
|
|
} XModem;
|
|
|
|
|
|
|
|
|
|
// Send out a byte of payload data, includes checksumming
|
|
|
|
|
void XModemOutputByte(uint8_t out_char) {
|
|
|
|
|
XModem.checksumBuf += out_char;
|
|
|
|
|
|
|
|
|
|
XModem.crcBuf = XModem.crcBuf ^ (int) out_char << 8;
|
|
|
|
|
for (uint32_t i = 0; i < 8; i++) {
|
|
|
|
|
if (XModem.crcBuf & 0x8000) {
|
|
|
|
|
XModem.crcBuf = XModem.crcBuf << 1 ^ 0x1021;
|
|
|
|
|
} else {
|
|
|
|
|
XModem.crcBuf = XModem.crcBuf << 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ZigbeeSerial->write(out_char);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Wait for the remote to acknowledge or cancel.
|
|
|
|
|
// Returns the received char if no timeout occured or a CAN was received. In this cases, it returns -1.
|
|
|
|
|
char XModemWaitACK(void)
|
|
|
|
|
{
|
|
|
|
|
char in_char;
|
|
|
|
|
do {
|
|
|
|
|
uint8_t i = 0;
|
|
|
|
|
while (!ZigbeeSerial->available()) {
|
|
|
|
|
delayMicroseconds(100);
|
|
|
|
|
i++;
|
|
|
|
|
if (i > 200) { return -1; }
|
|
|
|
|
}
|
|
|
|
|
in_char = ZigbeeSerial->read();
|
2021-01-19 10:54:49 +00:00
|
|
|
|
|
|
|
|
|
// if (in_char != XM_ACK) {
|
|
|
|
|
// AddLog_P(LOG_LEVEL_DEBUG_MORE, PSTR("XMD: Rcvd3 0x%02X"), in_char);
|
|
|
|
|
// }
|
|
|
|
|
|
2020-07-27 14:43:24 +01:00
|
|
|
|
if (XM_CAN == in_char) { return XM_CAN; }
|
2020-07-26 17:55:31 +01:00
|
|
|
|
} while ((in_char != XM_NAK) && (in_char != XM_ACK) && (in_char != 'C'));
|
|
|
|
|
return in_char;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool XModemSendPacket(uint32_t packet_no) {
|
|
|
|
|
XModem.filepos = ZbUpload.byte_counter;
|
|
|
|
|
|
|
|
|
|
// Sending a packet will be retried
|
|
|
|
|
uint32_t retries = 0;
|
|
|
|
|
char in_char;
|
|
|
|
|
do {
|
|
|
|
|
// Seek to start of current data block,
|
|
|
|
|
// will advance through the file as block will be acked..
|
|
|
|
|
ZbUpload.byte_counter = XModem.filepos;
|
|
|
|
|
|
|
|
|
|
// Reset checksum stuff
|
|
|
|
|
XModem.checksumBuf = 0x00;
|
|
|
|
|
XModem.crcBuf = 0x00;
|
|
|
|
|
|
2021-01-19 10:54:49 +00:00
|
|
|
|
uint8_t packet_num = packet_no;
|
|
|
|
|
|
2020-07-26 17:55:31 +01:00
|
|
|
|
// Try to send packet, so header first
|
|
|
|
|
ZigbeeSerial->write(XM_SOH);
|
2021-01-19 10:54:49 +00:00
|
|
|
|
ZigbeeSerial->write(packet_num);
|
|
|
|
|
ZigbeeSerial->write(~packet_num);
|
2020-07-26 17:55:31 +01:00
|
|
|
|
for (uint32_t i = 0; i < XMODEM_PACKET_SIZE; i++) {
|
|
|
|
|
in_char = ZigbeeUploadFlashRead();
|
|
|
|
|
XModemOutputByte(in_char);
|
|
|
|
|
}
|
|
|
|
|
// Send out checksum, either CRC-16 CCITT or classical inverse of sum of bytes.
|
|
|
|
|
// Depending on how the received introduced himself
|
|
|
|
|
if (XModem.oldChecksum) {
|
|
|
|
|
ZigbeeSerial->write((char)XModem.checksumBuf);
|
|
|
|
|
} else {
|
|
|
|
|
ZigbeeSerial->write((char)(XModem.crcBuf >> 8));
|
|
|
|
|
ZigbeeSerial->write((char)(XModem.crcBuf & 0xFF));
|
|
|
|
|
}
|
|
|
|
|
in_char = XModemWaitACK();
|
|
|
|
|
if (XM_CAN == in_char) { return false; }
|
|
|
|
|
retries++;
|
|
|
|
|
if (retries > XMODEM_MAX_RETRY) { return false; }
|
|
|
|
|
} while (in_char != XM_ACK);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*********************************************************************************************\
|
|
|
|
|
* Step 2 - Upload MCU firmware from ESP8266 flash to MCU EFR32 using XMODEM protocol
|
|
|
|
|
*
|
|
|
|
|
* https://www.silabs.com/documents/public/application-notes/an760-using-legacy-standalone-bootloader.pdf
|
|
|
|
|
\*********************************************************************************************/
|
|
|
|
|
|
2020-07-27 17:31:14 +01:00
|
|
|
|
void ZigbeeUploadSetSoftwareBootloader() {
|
|
|
|
|
// https://github.com/arendst/Tasmota/issues/8583#issuecomment-663967883
|
|
|
|
|
SBuffer buf(4);
|
|
|
|
|
buf.add16(EZSP_launchStandaloneBootloader);
|
|
|
|
|
buf.add8(0x01);
|
|
|
|
|
ZigbeeEZSPSendCmd(buf.getBuffer(), buf.len()); // Send software bootloader init
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-26 17:55:31 +01:00
|
|
|
|
void ZigbeeUploadSetBootloader(uint8_t state) {
|
|
|
|
|
pinMode(PIN_ZIGBEE_BOOTLOADER, OUTPUT);
|
|
|
|
|
digitalWrite(PIN_ZIGBEE_BOOTLOADER, state); // Toggle Gecko bootloader
|
|
|
|
|
digitalWrite(Pin(GPIO_ZIGBEE_RST), 0);
|
|
|
|
|
delay(100); // Need to experiment to find a value as low as possible
|
|
|
|
|
digitalWrite(Pin(GPIO_ZIGBEE_RST), 1); // Reboot MCU EFR32
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ZigbeeUploadBootloaderPrompt(void) {
|
|
|
|
|
// Scripts that interact with the bootloader should use only the “BL >” prompt to determine
|
|
|
|
|
// when the bootloader is ready for input. While current menu options should remain functionally
|
|
|
|
|
// unchanged, the menu title and options text is liable to change, and new options might be added.
|
2020-10-01 11:58:42 +01:00
|
|
|
|
|
|
|
|
|
uint8_t serial_buffer[255];
|
|
|
|
|
uint32_t buf_len = 0;
|
|
|
|
|
|
2020-07-26 17:55:31 +01:00
|
|
|
|
while (ZigbeeSerial->available()) {
|
|
|
|
|
yield();
|
|
|
|
|
char bootloader_byte = ZigbeeSerial->read();
|
2020-10-01 11:58:42 +01:00
|
|
|
|
|
2021-01-19 10:54:49 +00:00
|
|
|
|
// [cr][lf]
|
|
|
|
|
// Gecko Bootloader v1.A.3 or Gecko Bootloader v1.9.1.04[cr][lf]
|
|
|
|
|
// 1. upload gbl[cr][lf]
|
|
|
|
|
// 2. run[cr][lf]
|
|
|
|
|
// 3. ebl info[cr][lf]
|
|
|
|
|
// BL >
|
|
|
|
|
|
2020-10-01 11:58:42 +01:00
|
|
|
|
if (((uint8_t)bootloader_byte >=0) && (buf_len < sizeof(serial_buffer) -2)) {
|
|
|
|
|
serial_buffer[buf_len++] = bootloader_byte;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
}
|
2020-10-01 11:58:42 +01:00
|
|
|
|
|
|
|
|
|
if (ZbUpload.byte_counter != 4) {
|
|
|
|
|
switch (ZbUpload.byte_counter) {
|
|
|
|
|
case 0:
|
|
|
|
|
if ('B' == bootloader_byte) { ZbUpload.byte_counter++; } break;
|
|
|
|
|
case 1:
|
|
|
|
|
if ('L' == bootloader_byte) { ZbUpload.byte_counter++; } break;
|
|
|
|
|
case 2:
|
|
|
|
|
if (' ' == bootloader_byte) { ZbUpload.byte_counter++; } break;
|
|
|
|
|
case 3:
|
|
|
|
|
if ('>' == bootloader_byte) {
|
|
|
|
|
ZbUpload.byte_counter++;
|
|
|
|
|
XModem.flush_delay = millis() + XMODEM_FLUSH_DELAY;
|
|
|
|
|
XModem.delay = XModem.flush_delay + XMODEM_FLUSH_DELAY;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (buf_len) {
|
|
|
|
|
char hex_char[256];
|
|
|
|
|
ToHex_P(serial_buffer, buf_len, hex_char, 256);
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG_MORE, PSTR("XMD: Rcvd %s"), hex_char);
|
2020-07-26 17:55:31 +01:00
|
|
|
|
}
|
2020-10-01 11:58:42 +01:00
|
|
|
|
|
|
|
|
|
return ((4 == ZbUpload.byte_counter) && (millis() > XModem.flush_delay));
|
2020-07-26 17:55:31 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ZigbeeUploadXmodem(void) {
|
|
|
|
|
switch (ZbUpload.ota_step) {
|
2020-07-27 14:43:24 +01:00
|
|
|
|
case ZBU_IDLE: { // *** Upload disabled
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2020-07-27 17:31:14 +01:00
|
|
|
|
#ifdef ZIGBEE_BOOTLOADER_SOFTWARE_RESET_FIRST
|
|
|
|
|
case ZBU_INIT: { // *** Init ESF32 bootloader
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Init bootloader"));
|
2020-07-27 17:31:14 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_SOFTWARE_RESET;
|
|
|
|
|
return false; // Keep Zigbee serial active
|
|
|
|
|
}
|
|
|
|
|
case ZBU_SOFTWARE_RESET: {
|
|
|
|
|
SBuffer buf(4);
|
|
|
|
|
buf.add16(EZSP_launchStandaloneBootloader);
|
|
|
|
|
buf.add8(0x01);
|
|
|
|
|
ZigbeeEZSPSendCmd(buf.getBuffer(), buf.len()); // Send software bootloader init
|
|
|
|
|
XModem.timeout = millis() + (10 * 1000); // Allow 10 seconds to send Zigbee command
|
|
|
|
|
ZbUpload.ota_step = ZBU_SOFTWARE_SEND;
|
|
|
|
|
return false; // Keep Zigbee serial active
|
|
|
|
|
}
|
|
|
|
|
case ZBU_SOFTWARE_SEND: {
|
|
|
|
|
if (millis() > XModem.timeout) {
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Bootloader software reset send timeout"));
|
2020-07-27 17:31:14 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_HARDWARE_RESET;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if (EZSP_Serial.to_send == EZSP_Serial.to_end) {
|
|
|
|
|
ZbUpload.bootloader = ZBU_SOFTWARE_RESET;
|
|
|
|
|
XModem.timeout = millis() + (10 * 1000); // Allow 10 seconds to receive EBL prompt
|
2020-10-01 11:58:42 +01:00
|
|
|
|
XModem.delay = millis() + (2 * XMODEM_FLUSH_DELAY);
|
2020-07-27 17:31:14 +01:00
|
|
|
|
ZbUpload.byte_counter = 0;
|
|
|
|
|
ZbUpload.ota_step = ZBU_PROMPT;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case ZBU_HARDWARE_RESET: {
|
|
|
|
|
ZbUpload.bootloader = ZBU_HARDWARE_RESET;
|
|
|
|
|
ZigbeeUploadSetBootloader(0); // Reboot MCU EFR32 which returns below text
|
|
|
|
|
XModem.timeout = millis() + (30 * 1000); // Allow 30 seconds to receive EBL prompt
|
2020-10-01 11:58:42 +01:00
|
|
|
|
XModem.delay = millis() + (2 * XMODEM_FLUSH_DELAY);
|
2020-07-27 17:31:14 +01:00
|
|
|
|
ZbUpload.byte_counter = 0;
|
|
|
|
|
ZbUpload.ota_step = ZBU_PROMPT;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case ZBU_PROMPT: { // *** Wait for prompt and select option upload ebl
|
|
|
|
|
if (millis() > XModem.timeout) {
|
|
|
|
|
if (ZBU_SOFTWARE_RESET == ZbUpload.bootloader) {
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Bootloader software reset timeout"));
|
2020-07-27 17:31:14 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_HARDWARE_RESET;
|
|
|
|
|
} else {
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Bootloader hardware reset timeout"));
|
2020-09-01 12:05:46 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_ERROR;
|
2020-07-27 17:31:14 +01:00
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
#else // No ZIGBEE_BOOTLOADER_SOFTWARE_RESET_FIRST
|
2020-07-27 14:43:24 +01:00
|
|
|
|
case ZBU_INIT: { // *** Init ESF32 bootloader
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Init bootloader"));
|
2020-07-27 14:43:24 +01:00
|
|
|
|
ZigbeeUploadSetBootloader(0); // Reboot MCU EFR32 which returns below text
|
2020-07-27 10:01:20 +01:00
|
|
|
|
XModem.timeout = millis() + (30 * 1000); // Allow 30 seconds to receive EBL prompt
|
2020-10-01 11:58:42 +01:00
|
|
|
|
XModem.delay = millis() + (2 * XMODEM_FLUSH_DELAY);
|
2020-07-27 14:43:24 +01:00
|
|
|
|
ZbUpload.byte_counter = 0;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_PROMPT;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-07-27 14:43:24 +01:00
|
|
|
|
case ZBU_PROMPT: { // *** Wait for prompt and select option upload ebl
|
2020-07-27 10:01:20 +01:00
|
|
|
|
if (millis() > XModem.timeout) {
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Bootloader timeout"));
|
2020-09-01 12:05:46 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_ERROR;
|
2020-07-27 10:01:20 +01:00
|
|
|
|
return true;
|
|
|
|
|
}
|
2020-07-27 17:31:14 +01:00
|
|
|
|
#endif // ZIGBEE_BOOTLOADER_SOFTWARE_RESET_FIRST
|
2020-10-01 11:58:42 +01:00
|
|
|
|
else if (!ZigbeeSerial->available() && (millis() < XModem.flush_delay)) {
|
2020-07-27 14:43:24 +01:00
|
|
|
|
// The target device’s bootloader sends output over its serial port after it receives a
|
|
|
|
|
// carriage return from the source device
|
2020-07-27 10:01:20 +01:00
|
|
|
|
if (millis() > XModem.delay) {
|
|
|
|
|
ZigbeeSerial->write(XM_CR);
|
2020-10-01 11:58:42 +01:00
|
|
|
|
XModem.delay = millis() + (2 * XMODEM_FLUSH_DELAY);
|
2020-07-27 10:01:20 +01:00
|
|
|
|
}
|
2020-07-26 17:55:31 +01:00
|
|
|
|
} else {
|
|
|
|
|
// After the bootloader receives a carriage return from the target device, it displays a menu
|
2021-01-19 10:54:49 +00:00
|
|
|
|
// [cr][lf]
|
|
|
|
|
// Gecko Bootloader v1.A.3 or Gecko Bootloader v1.9.1.04[cr][lf]
|
|
|
|
|
// 1. upload gbl[cr][lf]
|
|
|
|
|
// 2. run[cr][lf]
|
|
|
|
|
// 3. ebl info[cr][lf]
|
2020-07-26 17:55:31 +01:00
|
|
|
|
// BL >
|
|
|
|
|
if (ZigbeeUploadBootloaderPrompt()) {
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Init sync"));
|
2020-07-26 17:55:31 +01:00
|
|
|
|
ZigbeeSerial->flush();
|
2020-07-27 14:43:24 +01:00
|
|
|
|
ZigbeeSerial->write('1'); // upload ebl
|
2020-10-29 15:16:34 +00:00
|
|
|
|
if (TasmotaGlobal.sleep > 0) {
|
2021-01-19 10:54:49 +00:00
|
|
|
|
TasmotaGlobal.sleep = 1; // Speed up loop used for xmodem upload
|
2020-07-27 10:01:20 +01:00
|
|
|
|
}
|
2020-07-26 17:55:31 +01:00
|
|
|
|
XModem.timeout = millis() + (XMODEM_SYNC_TIMEOUT * 1000);
|
|
|
|
|
ZbUpload.ota_step = ZBU_SYNC;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-07-27 14:43:24 +01:00
|
|
|
|
case ZBU_SYNC: { // *** Handle file upload using XModem - sync
|
2020-07-26 17:55:31 +01:00
|
|
|
|
if (millis() > XModem.timeout) {
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: SYNC timeout"));
|
2020-09-01 12:05:46 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_ERROR;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
return true;
|
|
|
|
|
}
|
2020-07-27 14:43:24 +01:00
|
|
|
|
// Wait for either C or NACK as a sync packet. Determines protocol details, checksum algorithm.
|
2020-07-26 17:55:31 +01:00
|
|
|
|
if (ZigbeeSerial->available()) {
|
2021-01-19 10:54:49 +00:00
|
|
|
|
// [cr][lf]
|
|
|
|
|
// begin upload[cr][lf]
|
|
|
|
|
// C
|
2020-07-26 17:55:31 +01:00
|
|
|
|
char xmodem_sync = ZigbeeSerial->read();
|
2021-01-19 10:54:49 +00:00
|
|
|
|
|
|
|
|
|
// AddLog_P(LOG_LEVEL_DEBUG_MORE, PSTR("XMD: Rcvd2 0x%02X"), xmodem_sync);
|
|
|
|
|
|
2020-07-27 14:43:24 +01:00
|
|
|
|
if (('C' == xmodem_sync) || (XM_NAK == xmodem_sync)) {
|
2020-07-26 17:55:31 +01:00
|
|
|
|
// Determine which checksum algorithm to use
|
|
|
|
|
XModem.oldChecksum = (xmodem_sync == XM_NAK);
|
2021-01-19 10:54:49 +00:00
|
|
|
|
XModem.packet_no = 1;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
ZbUpload.byte_counter = 0;
|
|
|
|
|
ZbUpload.ota_step = ZBU_UPLOAD;
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Init packet send"));
|
2020-07-26 17:55:31 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-07-27 14:43:24 +01:00
|
|
|
|
case ZBU_UPLOAD: { // *** Handle file upload using XModem - upload
|
2020-07-26 17:55:31 +01:00
|
|
|
|
if (ZigbeeUploadAvailable()) {
|
2021-01-19 10:54:49 +00:00
|
|
|
|
if (ZbUpload.byte_counter && !(ZbUpload.byte_counter % 10240)) { // Show progress every 10kB
|
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Progress %d kB"), ZbUpload.byte_counter / 1024);
|
|
|
|
|
}
|
|
|
|
|
if (!XModemSendPacket(XModem.packet_no)) {
|
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Packet %d send failed"), XModem.packet_no);
|
2020-09-01 12:05:46 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_ERROR;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
return true;
|
|
|
|
|
}
|
2021-01-19 10:54:49 +00:00
|
|
|
|
XModem.packet_no++;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
} else {
|
2020-07-27 14:43:24 +01:00
|
|
|
|
// Once the last block is ACKed by the target, the transfer should be finalized by an
|
|
|
|
|
// EOT (ASCII 0x04) packet from the source. Once this packet is confirmed via XModem ACK
|
|
|
|
|
// from the target, the device will reboot, causing the new firmware to be launched.
|
|
|
|
|
ZigbeeSerial->write(XM_EOT);
|
|
|
|
|
XModem.timeout = millis() + (30 * 1000); // Allow 30 seconds to receive EOT ACK
|
|
|
|
|
ZbUpload.ota_step = ZBU_EOT;
|
2021-01-19 10:54:49 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Transferred %d bytes"), ZbUpload.ota_size);
|
2020-07-27 14:43:24 +01:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case ZBU_EOT: { // *** Send EOT and wait for ACK
|
|
|
|
|
// The ACK for the last XModem data packet may take much longer (1-3 seconds) than prior
|
|
|
|
|
// data packets to be received. This is due to the CRC32 checksum being performed across
|
|
|
|
|
// the received EBL file data prior to sending the ACK. The source device must ensure that
|
|
|
|
|
// its XModem state machine waits a sufficient amount of time to allow this checksum process
|
|
|
|
|
// to occur without timing out on the response just before the EOT is sent.
|
|
|
|
|
if (millis() > XModem.timeout) {
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: EOT ACK timeout"));
|
2020-09-01 12:05:46 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_ERROR;
|
2020-07-27 14:43:24 +01:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if (ZigbeeSerial->available()) {
|
|
|
|
|
char xmodem_ack = XModemWaitACK();
|
2021-01-19 10:54:49 +00:00
|
|
|
|
if (XM_CAN == xmodem_ack) {
|
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Transfer invalid"));
|
|
|
|
|
ZbUpload.ota_step = ZBU_ERROR;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
else if (XM_ACK == xmodem_ack) {
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: " D_SUCCESSFUL));
|
2020-07-27 14:43:24 +01:00
|
|
|
|
XModem.timeout = millis() + (30 * 1000); // Allow 30 seconds to receive EBL prompt
|
|
|
|
|
ZbUpload.byte_counter = 0;
|
2020-07-27 17:31:14 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_COMPLETE;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
}
|
2020-07-27 14:43:24 +01:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case ZBU_COMPLETE: { // *** Wait for Serial upload complete EBL prompt
|
|
|
|
|
if (millis() > XModem.timeout) {
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: Bootloader timeout"));
|
2020-09-01 12:05:46 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_ERROR;
|
2020-07-27 14:43:24 +01:00
|
|
|
|
return true;
|
|
|
|
|
} else {
|
|
|
|
|
// After an image successfully uploads, the XModem transaction completes and the bootloader displays
|
|
|
|
|
// ‘Serial upload complete’ before redisplaying the menu
|
2021-01-19 10:54:49 +00:00
|
|
|
|
//
|
|
|
|
|
// [cr][lf]
|
|
|
|
|
// Serial upload complete[cr][lf]
|
|
|
|
|
// [cr][lf]
|
|
|
|
|
// Gecko Bootloader v1.A.3 or Gecko Bootloader v1.9.1.04[cr][lf]
|
|
|
|
|
// 1. upload gbl[cr][lf]
|
|
|
|
|
// 2. run[cr][lf]
|
|
|
|
|
// 3. ebl info[cr][lf]
|
2020-07-27 14:43:24 +01:00
|
|
|
|
// BL >
|
|
|
|
|
if (ZigbeeUploadBootloaderPrompt()) {
|
2020-09-01 12:05:46 +01:00
|
|
|
|
ZbUpload.state = ZBU_COMPLETE;
|
2020-07-27 14:43:24 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_DONE;
|
2021-01-19 10:54:49 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: " D_RESTARTING));
|
2020-07-27 14:43:24 +01:00
|
|
|
|
}
|
2020-07-26 17:55:31 +01:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-09-01 12:05:46 +01:00
|
|
|
|
case ZBU_ERROR:
|
|
|
|
|
ZbUpload.state = ZBU_ERROR;
|
2021-01-19 10:54:49 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("XMD: " D_FAILED));
|
2020-07-27 14:43:24 +01:00
|
|
|
|
case ZBU_DONE: { // *** Clean up and restart to disable bootloader and use new firmware
|
|
|
|
|
ZigbeeUploadSetBootloader(1); // Disable bootloader and reset MCU - should happen at restart
|
2020-10-29 15:16:34 +00:00
|
|
|
|
if (1 == TasmotaGlobal.sleep) {
|
|
|
|
|
TasmotaGlobal.sleep = Settings.sleep; // Restore loop sleep
|
2020-07-27 14:43:24 +01:00
|
|
|
|
}
|
2020-10-29 11:58:22 +00:00
|
|
|
|
// TasmotaGlobal.restart_flag = 2; // Restart to disable bootloader and use new firmware
|
2021-01-19 10:54:49 +00:00
|
|
|
|
if (ZbUpload.buffer) { free(ZbUpload.buffer); }
|
2020-07-27 14:43:24 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_FINISH; // Never return to zero without a restart to get a sane Zigbee environment
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case ZBU_FINISH: { // *** Wait for restart making sure not to start Zigbee serial again
|
|
|
|
|
// Wait for restart
|
|
|
|
|
break;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*********************************************************************************************\
|
|
|
|
|
* Step 1 - Upload MCU firmware in ESP8266 flash free space (current size is about 200k)
|
|
|
|
|
\*********************************************************************************************/
|
|
|
|
|
|
2020-11-19 14:43:47 +00:00
|
|
|
|
#ifdef USE_WEBSERVER
|
2020-09-01 12:05:46 +01:00
|
|
|
|
|
2020-11-19 14:43:47 +00:00
|
|
|
|
uint8_t ZigbeeUploadStep1Init(void) {
|
2020-07-25 17:25:06 +01:00
|
|
|
|
if (!PinUsed(GPIO_ZIGBEE_RST) && (ZigbeeSerial == nullptr)) { return 1; } // Wrong pin configuration - No file selected
|
|
|
|
|
|
2020-07-26 17:55:31 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_IDLE;
|
2020-09-01 12:05:46 +01:00
|
|
|
|
ZbUpload.state = ZBU_IDLE;
|
2020-07-25 17:25:06 +01:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-19 14:43:47 +00:00
|
|
|
|
void ZigbeeUploadStep1Done(uint32_t data, size_t size) {
|
|
|
|
|
ZbUpload.sector_counter = data;
|
|
|
|
|
ZbUpload.ota_size = size;
|
2020-07-26 17:55:31 +01:00
|
|
|
|
ZbUpload.ota_step = ZBU_INIT;
|
2020-11-19 14:43:47 +00:00
|
|
|
|
ZbUpload.state = ZBU_UPLOAD; // Signal upload done and ready for delayed upload to MCU EFR32
|
2020-07-25 17:25:06 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-09-01 12:05:46 +01:00
|
|
|
|
#define WEB_HANDLE_ZIGBEE_XFER "zx"
|
|
|
|
|
|
|
|
|
|
const char HTTP_SCRIPT_XFER_STATE[] PROGMEM =
|
|
|
|
|
"function z9(){"
|
|
|
|
|
"if(x!=null){x.abort();}" // Abort if no response within 2 seconds (happens on restart 1)
|
|
|
|
|
"x=new XMLHttpRequest();"
|
|
|
|
|
"x.onreadystatechange=function(){"
|
|
|
|
|
"if(x.readyState==4&&x.status==200){"
|
|
|
|
|
"var s=x.responseText;"
|
|
|
|
|
"if(s!=7){" // ZBU_UPLOAD
|
2020-09-02 10:45:17 +01:00
|
|
|
|
"location.href='/u3';" // Load page HandleUploadDone()
|
2020-09-01 12:05:46 +01:00
|
|
|
|
"}"
|
|
|
|
|
"}"
|
|
|
|
|
"};"
|
2020-09-02 10:45:17 +01:00
|
|
|
|
"x.open('GET','" WEB_HANDLE_ZIGBEE_XFER "?z=1',true);" // ?z related to Webserver->hasArg("z")
|
2020-09-01 12:05:46 +01:00
|
|
|
|
"x.send();"
|
2020-09-02 10:45:17 +01:00
|
|
|
|
"lt=setTimeout(z9,950);" // Poll every 0.95 second
|
2020-09-01 12:05:46 +01:00
|
|
|
|
"}"
|
2020-09-02 10:45:17 +01:00
|
|
|
|
"wl(z9);"; // Execute z9() on page load
|
2020-09-01 12:05:46 +01:00
|
|
|
|
|
|
|
|
|
void HandleZigbeeXfer(void) {
|
|
|
|
|
if (!HttpCheckPriviledgedAccess()) { return; }
|
|
|
|
|
|
2020-09-02 10:45:17 +01:00
|
|
|
|
if (Webserver->hasArg("z")) { // Status refresh requested
|
2020-09-01 12:05:46 +01:00
|
|
|
|
if (ZBU_ERROR == ZbUpload.state) {
|
2020-09-02 10:45:17 +01:00
|
|
|
|
Web.upload_error = 7; // Upload aborted (xmodem transfer failed)
|
2020-09-01 12:05:46 +01:00
|
|
|
|
}
|
2021-01-19 10:54:49 +00:00
|
|
|
|
WSContentBegin(200, CT_PLAIN);
|
|
|
|
|
WSContentSend_P(PSTR("%d"), ZbUpload.state);
|
|
|
|
|
WSContentEnd();
|
2020-09-01 12:05:46 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-06 16:09:13 +00:00
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR(D_LOG_HTTP D_UPLOAD_TRANSFER));
|
2020-09-01 12:05:46 +01:00
|
|
|
|
|
2020-11-06 16:09:13 +00:00
|
|
|
|
WSContentStart_P(PSTR(D_INFORMATION));
|
2020-09-01 12:05:46 +01:00
|
|
|
|
WSContentSend_P(HTTP_SCRIPT_XFER_STATE);
|
|
|
|
|
WSContentSendStyle();
|
2020-09-02 10:45:17 +01:00
|
|
|
|
WSContentSend_P(PSTR("<div style='text-align:center;'><b>" D_UPLOAD_TRANSFER " ...</b></div>"));
|
2020-09-01 12:05:46 +01:00
|
|
|
|
WSContentSpaceButton(BUTTON_MAIN);
|
|
|
|
|
WSContentStop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif // USE_WEBSERVER
|
|
|
|
|
|
|
|
|
|
#endif // USE_ZIGBEE_EZSP
|
2020-07-25 17:25:06 +01:00
|
|
|
|
|
2020-09-01 12:05:46 +01:00
|
|
|
|
#endif // USE_ZIGBEE
|